2022-06-18 17:39:08 +00:00
|
|
|
import * as core from './core.js';
|
|
|
|
import * as form from './form.js';
|
2016-03-12 18:50:43 +00:00
|
|
|
|
2024-02-22 15:36:45 +01:00
|
|
|
let gDatabase = new Database('auth');
|
2016-03-12 18:50:43 +00:00
|
|
|
|
2022-10-05 01:20:47 +00:00
|
|
|
const kRefreshInterval = 1 * 7 * 24 * 60 * 60 * 1000;
|
2016-03-12 18:50:43 +00:00
|
|
|
|
2022-09-28 23:52:44 +00:00
|
|
|
function b64url(value) {
|
|
|
|
value = value.replaceAll('+', '-').replaceAll('/', '_');
|
|
|
|
let equals = value.indexOf('=');
|
|
|
|
if (equals !== -1) {
|
|
|
|
return value.substring(0, equals);
|
|
|
|
} else {
|
|
|
|
return value;
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-28 23:52:44 +00:00
|
|
|
function unb64url(value) {
|
|
|
|
value = value.replaceAll('-', '+').replaceAll('_', '/');
|
|
|
|
let remainder = value.length % 4;
|
|
|
|
if (remainder == 3) {
|
|
|
|
return value + '=';
|
|
|
|
} else if (remainder == 2) {
|
|
|
|
return value + '==';
|
|
|
|
} else {
|
|
|
|
return value;
|
|
|
|
}
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
|
|
|
|
2022-09-28 23:52:44 +00:00
|
|
|
function makeJwt(payload) {
|
|
|
|
let ids = ssb.getIdentities(':auth');
|
|
|
|
let id;
|
|
|
|
if (ids?.length) {
|
|
|
|
id = ids[0];
|
|
|
|
} else {
|
|
|
|
id = ssb.createIdentity(':auth');
|
|
|
|
}
|
|
|
|
|
2024-02-22 15:36:45 +01:00
|
|
|
let final_payload = b64url(
|
|
|
|
base64Encode(
|
|
|
|
JSON.stringify(
|
|
|
|
Object.assign({}, payload, {
|
|
|
|
exp: new Date().valueOf() + kRefreshInterval,
|
|
|
|
})
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
let jwt = [
|
|
|
|
b64url(base64Encode(JSON.stringify({alg: 'HS256', typ: 'JWT'}))),
|
|
|
|
final_payload,
|
|
|
|
b64url(ssb.hmacsha256sign(final_payload, ':auth', id)),
|
|
|
|
].join('.');
|
2022-09-28 23:52:44 +00:00
|
|
|
return jwt;
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
|
|
|
|
2022-09-28 23:52:44 +00:00
|
|
|
function readSession(session) {
|
|
|
|
let jwt_parts = session?.split('.');
|
|
|
|
if (jwt_parts?.length === 3) {
|
|
|
|
let [header, payload, signature] = jwt_parts;
|
2024-01-03 17:25:34 +00:00
|
|
|
header = JSON.parse(utf8Decode(base64Decode(unb64url(header))));
|
2022-09-28 23:52:44 +00:00
|
|
|
if (header.typ === 'JWT' && header.alg === 'HS256') {
|
|
|
|
signature = unb64url(signature);
|
|
|
|
let id = ssb.getIdentities(':auth');
|
|
|
|
if (id?.length && ssb.hmacsha256verify(id[0], payload, signature)) {
|
2024-01-03 17:25:34 +00:00
|
|
|
let result = JSON.parse(utf8Decode(base64Decode(unb64url(payload))));
|
2024-02-22 15:36:45 +01:00
|
|
|
let now = new Date().valueOf();
|
2022-10-05 01:20:47 +00:00
|
|
|
if (now < result.exp) {
|
|
|
|
print(`JWT valid for another ${(result.exp - now) / 1000} seconds.`);
|
2022-09-28 23:52:44 +00:00
|
|
|
return result;
|
|
|
|
} else {
|
2022-10-05 01:20:47 +00:00
|
|
|
print(`JWT expired by ${(now - result.exp) / 1000} seconds.`);
|
2022-09-28 23:52:44 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
print('JWT verification failed.');
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
print('Invalid JWT header.');
|
|
|
|
}
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function verifyPassword(password, hash) {
|
|
|
|
return bCrypt.hashpw(password, hash) == hash;
|
|
|
|
}
|
|
|
|
|
|
|
|
function hashPassword(password) {
|
2023-01-28 22:44:45 +00:00
|
|
|
let salt = bCrypt.gensalt(12);
|
2016-03-12 18:50:43 +00:00
|
|
|
return bCrypt.hashpw(password, salt);
|
|
|
|
}
|
|
|
|
|
|
|
|
function noAdministrator() {
|
2024-02-22 15:36:45 +01:00
|
|
|
return (
|
|
|
|
!core.globalSettings ||
|
|
|
|
!core.globalSettings.permissions ||
|
|
|
|
!Object.keys(core.globalSettings.permissions).some(function (name) {
|
|
|
|
return (
|
|
|
|
core.globalSettings.permissions[name].indexOf('administration') != -1
|
|
|
|
);
|
|
|
|
})
|
|
|
|
);
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function makeAdministrator(name) {
|
2022-03-18 01:24:29 +00:00
|
|
|
if (!core.globalSettings.permissions) {
|
|
|
|
core.globalSettings.permissions = {};
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
2022-03-18 01:24:29 +00:00
|
|
|
if (!core.globalSettings.permissions[name]) {
|
|
|
|
core.globalSettings.permissions[name] = [];
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
2024-02-22 15:36:45 +01:00
|
|
|
if (core.globalSettings.permissions[name].indexOf('administration') == -1) {
|
|
|
|
core.globalSettings.permissions[name].push('administration');
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
2022-03-18 01:24:29 +00:00
|
|
|
core.setGlobalSettings(core.globalSettings);
|
|
|
|
}
|
|
|
|
|
|
|
|
function getCookies(headers) {
|
2023-01-28 22:44:45 +00:00
|
|
|
let cookies = {};
|
2022-03-18 01:24:29 +00:00
|
|
|
|
|
|
|
if (headers.cookie) {
|
2023-01-28 22:44:45 +00:00
|
|
|
let parts = headers.cookie.split(/,|;/);
|
|
|
|
for (let i in parts) {
|
2024-02-22 15:36:45 +01:00
|
|
|
let equals = parts[i].indexOf('=');
|
2023-01-28 22:44:45 +00:00
|
|
|
let name = parts[i].substring(0, equals).trim();
|
|
|
|
let value = parts[i].substring(equals + 1).trim();
|
2022-03-18 01:24:29 +00:00
|
|
|
cookies[name] = value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return cookies;
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
|
|
|
|
2023-09-22 22:41:47 +00:00
|
|
|
function isNameValid(name) {
|
|
|
|
let c = name.charAt(0);
|
2024-02-22 15:36:45 +01:00
|
|
|
return (
|
|
|
|
((c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z')) &&
|
|
|
|
name
|
|
|
|
.split()
|
|
|
|
.map(
|
|
|
|
(x) =>
|
|
|
|
x >= ('a' && x <= 'z') ||
|
|
|
|
x >= ('A' && x <= 'Z') ||
|
|
|
|
x >= ('0' && x <= '9')
|
|
|
|
)
|
|
|
|
);
|
2023-09-22 22:41:47 +00:00
|
|
|
}
|
|
|
|
|
2022-06-18 17:39:08 +00:00
|
|
|
function handler(request, response) {
|
2023-01-28 22:44:45 +00:00
|
|
|
let session = getCookies(request.headers).session;
|
2024-02-22 15:36:45 +01:00
|
|
|
if (request.uri == '/login') {
|
2023-09-11 16:52:17 +00:00
|
|
|
let formData = form.decodeForm(request.query);
|
|
|
|
if (query(request.headers)?.permissions?.authenticated) {
|
|
|
|
if (formData.return) {
|
2024-02-22 15:36:45 +01:00
|
|
|
response.writeHead(303, {Location: formData.return});
|
2023-09-11 16:52:17 +00:00
|
|
|
} else {
|
2024-02-22 15:36:45 +01:00
|
|
|
response.writeHead(303, {
|
|
|
|
Location:
|
|
|
|
(request.client.tls ? 'https://' : 'http://') +
|
|
|
|
request.headers.host +
|
|
|
|
'/',
|
|
|
|
'Content-Length': '0',
|
|
|
|
});
|
2023-09-11 16:52:17 +00:00
|
|
|
}
|
|
|
|
response.end();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-01-28 22:44:45 +00:00
|
|
|
let sessionIsNew = false;
|
|
|
|
let loginError;
|
2016-03-12 18:50:43 +00:00
|
|
|
|
2024-02-22 15:36:45 +01:00
|
|
|
if (request.method == 'POST' || formData.submit) {
|
2016-03-12 18:50:43 +00:00
|
|
|
sessionIsNew = true;
|
2022-01-29 19:05:57 +00:00
|
|
|
formData = form.decodeForm(utf8Decode(request.body), formData);
|
2024-02-22 15:36:45 +01:00
|
|
|
if (formData.submit == 'Login') {
|
|
|
|
let account = gDatabase.get('user:' + formData.name);
|
2021-01-02 18:10:00 +00:00
|
|
|
account = account ? JSON.parse(account) : account;
|
2023-09-22 22:59:26 +00:00
|
|
|
if (formData.register == '1') {
|
2024-02-22 15:36:45 +01:00
|
|
|
if (
|
|
|
|
!account &&
|
2023-09-22 22:41:47 +00:00
|
|
|
isNameValid(formData.name) &&
|
2024-02-22 15:36:45 +01:00
|
|
|
formData.password == formData.confirm
|
|
|
|
) {
|
2022-08-04 00:07:12 +00:00
|
|
|
let users = new Set();
|
|
|
|
let users_original = gDatabase.get('users');
|
|
|
|
try {
|
|
|
|
users = new Set(JSON.parse(users_original));
|
2024-02-22 15:36:45 +01:00
|
|
|
} catch {}
|
2022-08-04 00:07:12 +00:00
|
|
|
if (!users.has(formData.name)) {
|
|
|
|
users.add(formData.name);
|
|
|
|
}
|
|
|
|
users = JSON.stringify([...users].sort());
|
|
|
|
if (users !== users_original) {
|
|
|
|
gDatabase.set('users', users);
|
|
|
|
}
|
2022-09-28 23:52:44 +00:00
|
|
|
session = makeJwt({name: formData.name});
|
2021-01-02 18:10:00 +00:00
|
|
|
account = {password: hashPassword(formData.password)};
|
2023-09-22 22:59:26 +00:00
|
|
|
gDatabase.set('user:' + formData.name, JSON.stringify(account));
|
2016-03-12 18:50:43 +00:00
|
|
|
if (noAdministrator()) {
|
|
|
|
makeAdministrator(formData.name);
|
|
|
|
}
|
|
|
|
} else {
|
2023-09-22 22:59:26 +00:00
|
|
|
loginError = 'Error registering account.';
|
|
|
|
}
|
|
|
|
} else if (formData.change == '1') {
|
2024-02-22 15:36:45 +01:00
|
|
|
if (
|
|
|
|
account &&
|
2023-09-22 22:59:26 +00:00
|
|
|
isNameValid(formData.name) &&
|
|
|
|
formData.new_password == formData.confirm &&
|
2024-02-22 15:36:45 +01:00
|
|
|
verifyPassword(formData.password, account.password)
|
|
|
|
) {
|
2023-09-22 22:59:26 +00:00
|
|
|
session = makeJwt({name: formData.name});
|
|
|
|
account = {password: hashPassword(formData.new_password)};
|
|
|
|
gDatabase.set('user:' + formData.name, JSON.stringify(account));
|
|
|
|
} else {
|
|
|
|
loginError = 'Error changing password.';
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
|
|
|
} else {
|
2024-02-22 15:36:45 +01:00
|
|
|
if (
|
|
|
|
account &&
|
2021-01-02 18:10:00 +00:00
|
|
|
account.password &&
|
2024-02-22 15:36:45 +01:00
|
|
|
verifyPassword(formData.password, account.password)
|
|
|
|
) {
|
2022-09-28 23:52:44 +00:00
|
|
|
session = makeJwt({name: formData.name});
|
2016-03-12 18:50:43 +00:00
|
|
|
if (noAdministrator()) {
|
|
|
|
makeAdministrator(formData.name);
|
|
|
|
}
|
|
|
|
} else {
|
2023-09-22 22:59:26 +00:00
|
|
|
loginError = 'Invalid username or password.';
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Proceed as Guest
|
2022-09-28 23:52:44 +00:00
|
|
|
session = makeJwt({name: 'guest'});
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-25 13:25:41 +00:00
|
|
|
let cookie = `session=${session}; path=/; Max-Age=${kRefreshInterval}; ${request.client.tls ? 'Secure; ' : ''}SameSite=Strict; HttpOnly`;
|
2023-01-28 22:44:45 +00:00
|
|
|
let entry = readSession(session);
|
2016-03-12 18:50:43 +00:00
|
|
|
if (entry && formData.return) {
|
2024-02-22 15:36:45 +01:00
|
|
|
response.writeHead(303, {
|
|
|
|
Location: formData.return,
|
|
|
|
'Set-Cookie': cookie,
|
|
|
|
});
|
2016-03-12 18:50:43 +00:00
|
|
|
response.end();
|
|
|
|
} else {
|
2024-02-22 15:36:45 +01:00
|
|
|
File.readFile('core/auth.html')
|
|
|
|
.then(function (data) {
|
|
|
|
let html = utf8Decode(data);
|
|
|
|
let auth_data = {
|
|
|
|
session_is_new: sessionIsNew,
|
|
|
|
name: entry?.name,
|
|
|
|
error: loginError,
|
|
|
|
code_of_conduct: core.globalSettings.code_of_conduct,
|
|
|
|
have_administrator: !noAdministrator(),
|
|
|
|
};
|
|
|
|
html = utf8Encode(
|
|
|
|
html.replace('$AUTH_DATA', JSON.stringify(auth_data))
|
|
|
|
);
|
|
|
|
response.writeHead(200, {
|
|
|
|
'Content-Type': 'text/html; charset=utf-8',
|
|
|
|
'Set-Cookie': cookie,
|
|
|
|
'Content-Length': html.length,
|
|
|
|
});
|
|
|
|
response.end(html);
|
|
|
|
})
|
|
|
|
.catch(function (error) {
|
|
|
|
response.writeHead(404, {
|
|
|
|
'Content-Type': 'text/plain; charset=utf-8',
|
|
|
|
Connection: 'close',
|
|
|
|
});
|
|
|
|
response.end('404 File not found');
|
|
|
|
});
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
2024-02-22 15:36:45 +01:00
|
|
|
} else if (request.uri == '/login/logout') {
|
|
|
|
response.writeHead(303, {
|
|
|
|
'Set-Cookie': `session=; path=/; ${request.client.tls ? 'Secure; ' : ''}SameSite=Strict; expires=Thu, 01 Jan 1970 00:00:00 GMT; HttpOnly`,
|
|
|
|
Location: '/login' + (request.query ? '?' + request.query : ''),
|
|
|
|
});
|
2016-03-12 18:50:43 +00:00
|
|
|
response.end();
|
|
|
|
} else {
|
2024-02-22 15:36:45 +01:00
|
|
|
response.writeHead(200, {
|
|
|
|
'Content-Type': 'text/plain; charset=utf-8',
|
|
|
|
Connection: 'close',
|
|
|
|
});
|
|
|
|
response.end('Hello, ' + request.client.peerName + '.');
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function getPermissions(session) {
|
2023-01-28 22:44:45 +00:00
|
|
|
let permissions;
|
|
|
|
let entry = readSession(session);
|
2016-03-12 18:50:43 +00:00
|
|
|
if (entry) {
|
|
|
|
permissions = getPermissionsForUser(entry.name);
|
2024-02-22 15:36:45 +01:00
|
|
|
permissions.authenticated = entry.name !== 'guest';
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
|
|
|
return permissions || {};
|
|
|
|
}
|
|
|
|
|
|
|
|
function getPermissionsForUser(userName) {
|
2023-01-28 22:44:45 +00:00
|
|
|
let permissions = {};
|
2024-02-22 15:36:45 +01:00
|
|
|
if (
|
|
|
|
core.globalSettings &&
|
|
|
|
core.globalSettings.permissions &&
|
|
|
|
core.globalSettings.permissions[userName]
|
|
|
|
) {
|
2023-01-28 22:44:45 +00:00
|
|
|
for (let i in core.globalSettings.permissions[userName]) {
|
2022-03-18 01:24:29 +00:00
|
|
|
permissions[core.globalSettings.permissions[userName][i]] = true;
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return permissions;
|
|
|
|
}
|
|
|
|
|
|
|
|
function query(headers) {
|
2023-01-28 22:44:45 +00:00
|
|
|
let session = getCookies(headers).session;
|
|
|
|
let entry;
|
|
|
|
let autologin = tildefriends.args.autologin;
|
2024-02-22 15:36:45 +01:00
|
|
|
if ((entry = autologin ? {name: autologin} : readSession(session))) {
|
2022-10-05 01:20:47 +00:00
|
|
|
return {
|
|
|
|
session: entry,
|
2024-02-22 15:36:45 +01:00
|
|
|
permissions: autologin
|
|
|
|
? getPermissionsForUser(autologin)
|
|
|
|
: getPermissions(session),
|
2022-10-05 01:20:47 +00:00
|
|
|
};
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-16 22:03:47 +00:00
|
|
|
function make_refresh(credentials) {
|
|
|
|
if (credentials?.session?.name) {
|
|
|
|
return {
|
|
|
|
token: makeJwt({name: credentials.session.name}),
|
|
|
|
interval: kRefreshInterval,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-22 15:36:45 +01:00
|
|
|
export {handler, query, make_refresh};
|