nostr challenge step much more robust
This commit is contained in:
parent
805ad5fad9
commit
564dcb8083
7 changed files with 99 additions and 50 deletions
|
|
@ -5,6 +5,13 @@ class ChallengedUsedError extends Error {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class AlreadyUsedError extends Error {
|
||||||
|
constructor(message) {
|
||||||
|
super(message);
|
||||||
|
this.name = "AlreadyUsedError"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
class InvalidSignatureError extends Error {
|
class InvalidSignatureError extends Error {
|
||||||
constructor(message) {
|
constructor(message) {
|
||||||
super(message);
|
super(message);
|
||||||
|
|
@ -27,8 +34,7 @@ class NotFoundError extends Error {
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
ChallengedUsedError,
|
AlreadyUsedError,
|
||||||
InvalidSignatureError,
|
InvalidSignatureError,
|
||||||
AppInvitedUsedError,
|
|
||||||
NotFoundError
|
NotFoundError
|
||||||
};
|
};
|
||||||
|
|
@ -10,19 +10,28 @@ window.onload = function () {
|
||||||
|
|
||||||
async function acceptInvite() {
|
async function acceptInvite() {
|
||||||
|
|
||||||
if (!window.nostr) {
|
let challengeResponse;
|
||||||
console.log("No Nostr extension found.");
|
try {
|
||||||
return { success: false, error: "No Nostr extension detected." };
|
challengeResponse = await fetch('/api/signup/nostr-challenge', {
|
||||||
|
method: 'GET',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json'
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.log(`Something went wrong: ${error}`);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
try {
|
|
||||||
const challengeResponse = await fetch("/api/signup/nostr-challenge");
|
|
||||||
if (!challengeResponse.ok) throw new Error("Failed to fetch challenge");
|
|
||||||
const { challenge } = await challengeResponse.json();
|
const { challenge } = await challengeResponse.json();
|
||||||
|
|
||||||
const pubkey = await window.nostr.getPublicKey();
|
let pubkey;
|
||||||
|
try {
|
||||||
|
pubkey = await window.nostr.getPublicKey();
|
||||||
|
} catch (error) {
|
||||||
|
document.querySelector('#rejected-nostr-nudges').style.display = 'block';
|
||||||
|
return;
|
||||||
|
}
|
||||||
const event = {
|
const event = {
|
||||||
kind: 22242,
|
kind: 22242,
|
||||||
created_at: Math.floor(Date.now() / 1000),
|
created_at: Math.floor(Date.now() / 1000),
|
||||||
|
|
@ -31,7 +40,13 @@ async function acceptInvite() {
|
||||||
pubkey: pubkey
|
pubkey: pubkey
|
||||||
};
|
};
|
||||||
|
|
||||||
const signedEvent = await window.nostr.signEvent(event);
|
let signedEvent;
|
||||||
|
try {
|
||||||
|
signedEvent = await window.nostr.signEvent(event);
|
||||||
|
} catch (error) {
|
||||||
|
document.querySelector('#rejected-nostr-nudges').style.display = 'block';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const verifyResponse = await fetch("/api/signup/nostr-verify", {
|
const verifyResponse = await fetch("/api/signup/nostr-verify", {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
|
|
@ -42,8 +57,4 @@ async function acceptInvite() {
|
||||||
if (verifyResponse.status === 200) {
|
if (verifyResponse.status === 200) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
} catch (error) { }
|
|
||||||
}
|
}
|
||||||
|
|
@ -1,5 +1,4 @@
|
||||||
const express = require('express');
|
const express = require('express');
|
||||||
const crypto = require("crypto");
|
|
||||||
|
|
||||||
const invitesService = require('../services/invitesService');
|
const invitesService = require('../services/invitesService');
|
||||||
const nostrService = require('../services/nostrService');
|
const nostrService = require('../services/nostrService');
|
||||||
|
|
@ -9,17 +8,48 @@ const errors = require('../errors');
|
||||||
const router = express.Router();
|
const router = express.Router();
|
||||||
|
|
||||||
router.get('/signup/nostr-challenge', async (req, res) => {
|
router.get('/signup/nostr-challenge', async (req, res) => {
|
||||||
|
console.log("I'm heeeere")
|
||||||
const inviteUuid = req.cookies.inviteUuid;
|
const inviteUuid = req.cookies.inviteUuid;
|
||||||
|
|
||||||
const signUpChallenge = await invitesService.createSignUpChallenge(
|
let signUpChallenge;
|
||||||
|
try {
|
||||||
|
signUpChallenge = await invitesService.createSignUpChallenge(
|
||||||
inviteUuid
|
inviteUuid
|
||||||
)
|
)
|
||||||
|
} catch (error) {
|
||||||
|
if (error instanceof errors.NotFoundError) {
|
||||||
|
return res.status(404).json({
|
||||||
|
success: false,
|
||||||
|
message: 'Could not find invite with that id.'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
const relatedNostrChallenge = await nostrService.getNostrChallenge(
|
if (error instanceof errors.AlreadyUsedError) {
|
||||||
|
return res.status(410).json({
|
||||||
|
success: false,
|
||||||
|
message: 'That invite has already been used.'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return res.status(500).json({
|
||||||
|
success: false,
|
||||||
|
message: 'Unexpected error.'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
let relatedNostrChallenge;
|
||||||
|
try {
|
||||||
|
relatedNostrChallenge = await nostrService.getNostrChallenge(
|
||||||
signUpChallenge.nostr_challenge_uuid
|
signUpChallenge.nostr_challenge_uuid
|
||||||
)
|
)
|
||||||
|
} catch (error) {
|
||||||
|
return res.status(500).json({
|
||||||
|
success: false,
|
||||||
|
message: 'Unexpected error.'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
res.status(200).json({ 'challenge': relatedNostrChallenge.challenge });
|
return res.status(200).json({ 'challenge': relatedNostrChallenge.challenge });
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -34,7 +64,7 @@ router.post("/signup/nostr-verify", async (req, res) => {
|
||||||
if (error instanceof TimeoutError) {
|
if (error instanceof TimeoutError) {
|
||||||
console.error('The challenge is outdated.');
|
console.error('The challenge is outdated.');
|
||||||
}
|
}
|
||||||
if (error instanceof errors.ChallengedUsedError) {
|
if (error instanceof errors.AlreadyUsedError) {
|
||||||
console.error('The challenge was already used, request a new one.');
|
console.error('The challenge was already used, request a new one.');
|
||||||
}
|
}
|
||||||
if (error instanceof errors.InvalidSignatureError) {
|
if (error instanceof errors.InvalidSignatureError) {
|
||||||
|
|
|
||||||
|
|
@ -13,24 +13,23 @@ router.get('/invite/:inviteUuid', async (req, res) => {
|
||||||
|
|
||||||
res.cookie('inviteUuid', inviteUuid, { httpOnly: true, maxAge: 86400000 });
|
res.cookie('inviteUuid', inviteUuid, { httpOnly: true, maxAge: 86400000 });
|
||||||
|
|
||||||
|
let invite;
|
||||||
try {
|
try {
|
||||||
|
invite = await invitesService.getAppInvite(inviteUuid);
|
||||||
if (await !invitesService.appInviteExists(inviteUuid)) {
|
if (!invite) {
|
||||||
return res.status(404).render('error', { message: 'Invite not found' });
|
return res.status(404).render('error', { message: 'Invite not found.' });
|
||||||
}
|
}
|
||||||
|
|
||||||
const invite = await invitesService.getAppInvite(inviteUuid);
|
|
||||||
|
|
||||||
if (await invitesService.isAppInviteSpent(inviteUuid)) {
|
if (await invitesService.isAppInviteSpent(inviteUuid)) {
|
||||||
return res.render('invite_spent', { invite })
|
return res.status(410).render('invite_spent', { invite })
|
||||||
}
|
}
|
||||||
|
|
||||||
return res.render('invite', { invite });
|
|
||||||
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Error fetching invite:', error);
|
console.error('Error fetching invite:', error);
|
||||||
return res.status(500).render('error', { message: 'An error occurred' });
|
return res.status(500).render('error', { message: 'An error occurred' });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return res.render('invite', { invite });
|
||||||
});
|
});
|
||||||
|
|
||||||
router.get('/private', authMiddleware, (req, res) => {
|
router.get('/private', authMiddleware, (req, res) => {
|
||||||
|
|
|
||||||
|
|
@ -49,7 +49,7 @@ async function createSignUpChallenge(appInviteUuid) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (await isAppInviteSpent(appInviteUuid)) {
|
if (await isAppInviteSpent(appInviteUuid)) {
|
||||||
throw new errors.AppInvitedUsedError("Invite has already been used.")
|
throw new errors.AlreadyUsedError("Invite has already been used.")
|
||||||
}
|
}
|
||||||
|
|
||||||
const nostrChallenge = await nostrService.createNostrChallenge()
|
const nostrChallenge = await nostrService.createNostrChallenge()
|
||||||
|
|
@ -84,7 +84,7 @@ async function verifySignUpChallenge(signedEvent) {
|
||||||
console.log(`Found this signup challenge: ${signUpChallenge}`);
|
console.log(`Found this signup challenge: ${signUpChallenge}`);
|
||||||
|
|
||||||
if (await nostrService.hasNostrChallengeBeenCompleted(challenge)) {
|
if (await nostrService.hasNostrChallengeBeenCompleted(challenge)) {
|
||||||
throw new errors.ChallengedUsedError("This challenge has already been used.");
|
throw new errors.AlreadyUsedError("This challenge has already been used.");
|
||||||
}
|
}
|
||||||
console.log(`I'm gonna verify the nostr challenge`);
|
console.log(`I'm gonna verify the nostr challenge`);
|
||||||
const completedNostrChallenge = await nostrService.verifyNostrChallenge(signedEvent);
|
const completedNostrChallenge = await nostrService.verifyNostrChallenge(signedEvent);
|
||||||
|
|
|
||||||
|
|
@ -58,7 +58,7 @@ async function verifyNostrChallenge(signedEvent) {
|
||||||
|
|
||||||
console.log("Checking if completed")
|
console.log("Checking if completed")
|
||||||
if (await hasNostrChallengeBeenCompleted(challenge)) {
|
if (await hasNostrChallengeBeenCompleted(challenge)) {
|
||||||
throw new errors.ChallengedUsedError("Challenge already used, request new one.");
|
throw new errors.AlreadyUsedError("Challenge already used, request new one.");
|
||||||
}
|
}
|
||||||
|
|
||||||
console.log("Checking if valid")
|
console.log("Checking if valid")
|
||||||
|
|
|
||||||
|
|
@ -17,6 +17,9 @@
|
||||||
<form onsubmit="acceptInvite();return false">
|
<form onsubmit="acceptInvite();return false">
|
||||||
<button id="nostr-signup" type="submit">Alta con Nostr</button>
|
<button id="nostr-signup" type="submit">Alta con Nostr</button>
|
||||||
</form>
|
</form>
|
||||||
|
<div id="rejected-nostr-nudges" style="display:none">
|
||||||
|
<p>Ups, parece que no has aceptado que usemos tus claves. Si te has equivocado, puedes intentarlo de nuevo.</p>
|
||||||
|
</div>
|
||||||
<div id="no-extension-nudges" style="display:none">
|
<div id="no-extension-nudges" style="display:none">
|
||||||
<p>¡Atención! No se ha encontrado una extensión de Nostr en tu navegador. Puedes usar: </p>
|
<p>¡Atención! No se ha encontrado una extensión de Nostr en tu navegador. Puedes usar: </p>
|
||||||
<ul>
|
<ul>
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue