158 lines
4.3 KiB
JavaScript
158 lines
4.3 KiB
JavaScript
import jwt from './jwt';
|
|
import { makeId } from '../src/helpers';
|
|
const dbConnection = require('./db');
|
|
const SQL = require('sql-template-strings');
|
|
import { ulid } from 'ulid';
|
|
import translations from "./translations";
|
|
const mailer = require('./mailer');
|
|
|
|
const now = Math.floor(Date.now() / 1000);
|
|
|
|
const getUser = (authorization) => {
|
|
if (!authorization || !authorization.startsWith('Bearer ')) {
|
|
return null;
|
|
}
|
|
|
|
return jwt.validate(authorization.substring(7));
|
|
}
|
|
|
|
const saveAuthenticator = async (db, type, user, payload, validForMinutes = null) => {
|
|
const id = ulid();
|
|
await db.get(SQL`INSERT INTO authenticators (id, userId, type, payload, validUntil) VALUES (
|
|
${id},
|
|
${user ? user.id : null},
|
|
${type},
|
|
${JSON.stringify(payload)},
|
|
${validForMinutes ? (now + validForMinutes * 60) : null}
|
|
)`);
|
|
return id;
|
|
}
|
|
|
|
const findAuthenticator = async (db, id, type) => {
|
|
const authenticator = await db.get(SQL`SELECT * FROM authenticators
|
|
WHERE id = ${id}
|
|
AND type = ${type}
|
|
AND (validUntil IS NULL OR validUntil > ${now})
|
|
`);
|
|
|
|
if (authenticator) {
|
|
authenticator.payload = JSON.parse(authenticator.payload);
|
|
}
|
|
|
|
return authenticator
|
|
}
|
|
|
|
const invalidateAuthenticator = async (db, id) => {
|
|
await db.get(SQL`UPDATE authenticators
|
|
SET validUntil = ${now}
|
|
WHERE id = ${id}
|
|
`);
|
|
}
|
|
|
|
const init = async (db, usernameOrEmail) => {
|
|
let user = undefined;
|
|
|
|
const isEmail = usernameOrEmail.indexOf('@') > -1;
|
|
|
|
if (isEmail) {
|
|
user = await db.get(SQL`SELECT * FROM users WHERE email = ${usernameOrEmail}`);
|
|
} else {
|
|
user = await db.get(SQL`SELECT * FROM users WHERE username = ${usernameOrEmail}`);
|
|
}
|
|
|
|
if (!user && !isEmail) {
|
|
return {error: 'user.login.userNotFound'}
|
|
}
|
|
|
|
const payload = {
|
|
username: isEmail ? (user ? user.username : null) : usernameOrEmail,
|
|
email: isEmail ? usernameOrEmail : user.email,
|
|
code: makeId(6, '0123456789'),
|
|
}
|
|
|
|
const codeKey = await saveAuthenticator(db, 'email', user, payload, 15);
|
|
|
|
mailer(
|
|
payload.email,
|
|
`[${translations.title}] ${translations.user.login.email.subject.replace('%code%', payload.code)}`,
|
|
translations.user.login.email.content.replace('%code%', payload.code),
|
|
)
|
|
|
|
return {
|
|
token: jwt.sign({...payload, code: null, codeKey}, '15m'),
|
|
};
|
|
}
|
|
|
|
const validate = async (db, user, code) => {
|
|
if (!user || !user.codeKey) {
|
|
return {error: 'user.tokenExpired'};
|
|
}
|
|
|
|
const authenticator = await findAuthenticator(db, user.codeKey, 'email');
|
|
if (!authenticator) {
|
|
return {error: 'user.tokenExpired'};
|
|
}
|
|
|
|
if (authenticator.payload.code !== code) {
|
|
return {error: 'user.code.invalid'};
|
|
}
|
|
|
|
return await authenticate(db, user, authenticator);
|
|
}
|
|
|
|
const defaultUsername = async (db, email) => {
|
|
const base = email.substring(0, email.indexOf('@'))
|
|
.padEnd(4, '0')
|
|
.replace(/[^A-Za-z0-9._-]/g, '_');
|
|
|
|
let c = 0;
|
|
while (true) {
|
|
let proposal = base + (c || '');
|
|
let dbUser = await db.get(SQL`SELECT id FROM users WHERE username = ${proposal}`);
|
|
if (!dbUser) {
|
|
return proposal;
|
|
}
|
|
c++;
|
|
}
|
|
}
|
|
|
|
const authenticate = async (db, user, authenticator) => {
|
|
let dbUser = await db.get(SQL`SELECT * FROM users WHERE email = ${user.email}`);
|
|
if (!dbUser) {
|
|
dbUser = {
|
|
id: ulid(),
|
|
username: await defaultUsername(db, user.email),
|
|
email: user.email,
|
|
roles: 'user',
|
|
avatarSource: null,
|
|
}
|
|
}
|
|
|
|
invalidateAuthenticator(db, authenticator);
|
|
|
|
return {
|
|
token: jwt.sign({
|
|
...dbUser,
|
|
authenticated: true,
|
|
}),
|
|
};
|
|
}
|
|
|
|
export default async function (req, res, next) {
|
|
const db = await dbConnection();
|
|
|
|
let result = {error: 'notfound'}
|
|
|
|
const user = getUser(req.headers.authorization);
|
|
|
|
if (req.method === 'POST' && req.url === '/init' && req.body.usernameOrEmail) {
|
|
result = await init(db, req.body.usernameOrEmail)
|
|
} else if (req.method === 'POST' && req.url === '/validate' && req.body.code) {
|
|
result = await validate(db, user, req.body.code);
|
|
}
|
|
|
|
res.setHeader('content-type', 'application/json');
|
|
res.write(JSON.stringify(result));
|
|
res.end();
|
|
}
|