2022-06-18 17:39:08 +00:00
|
|
|
import * as core from './core.js';
|
|
|
|
import * as http from './http.js';
|
|
|
|
import * as form from './form.js';
|
2016-03-12 18:50:43 +00:00
|
|
|
|
|
|
|
var gTokens = {};
|
2021-01-02 18:10:00 +00:00
|
|
|
var gDatabase = new Database("auth");
|
2016-03-12 18:50:43 +00:00
|
|
|
|
2022-10-02 00:11:57 +00:00
|
|
|
const kRefreshInterval =
|
|
|
|
1 /* w */ *
|
|
|
|
7 /* d */ *
|
|
|
|
24 /* h */ *
|
|
|
|
60 /* m */ *
|
|
|
|
60 /* s */ *
|
|
|
|
1000 /* ms */;
|
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');
|
|
|
|
}
|
|
|
|
|
|
|
|
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('.');
|
|
|
|
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;
|
|
|
|
header = JSON.parse(base64Decode(unb64url(header)));
|
|
|
|
if (header.typ === 'JWT' && header.alg === 'HS256') {
|
|
|
|
signature = unb64url(signature);
|
|
|
|
let id = ssb.getIdentities(':auth');
|
|
|
|
if (id?.length && ssb.hmacsha256verify(id[0], payload, signature)) {
|
|
|
|
let result = JSON.parse(base64Decode(unb64url(payload)));
|
|
|
|
if ((new Date()).valueOf() < result.exp) {
|
|
|
|
return result;
|
|
|
|
} else {
|
|
|
|
print('JWT expired.');
|
|
|
|
}
|
|
|
|
} 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) {
|
|
|
|
var salt = bCrypt.gensalt(12);
|
|
|
|
return bCrypt.hashpw(password, salt);
|
|
|
|
}
|
|
|
|
|
|
|
|
function noAdministrator() {
|
2022-03-18 01:24:29 +00: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
|
|
|
}
|
2022-03-18 01:24:29 +00: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) {
|
|
|
|
var cookies = {};
|
|
|
|
|
|
|
|
if (headers.cookie) {
|
|
|
|
var parts = headers.cookie.split(/,|;/);
|
|
|
|
for (var i in parts) {
|
|
|
|
var equals = parts[i].indexOf("=");
|
|
|
|
var name = parts[i].substring(0, equals).trim();
|
|
|
|
var value = parts[i].substring(equals + 1).trim();
|
|
|
|
cookies[name] = value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return cookies;
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
|
|
|
|
2022-06-18 17:39:08 +00:00
|
|
|
function handler(request, response) {
|
2016-03-12 18:50:43 +00:00
|
|
|
var session = getCookies(request.headers).session;
|
|
|
|
if (request.uri == "/login") {
|
|
|
|
var sessionIsNew = false;
|
|
|
|
var loginError;
|
|
|
|
|
|
|
|
var formData = form.decodeForm(request.query);
|
|
|
|
|
|
|
|
if (request.method == "POST" || formData.submit) {
|
|
|
|
sessionIsNew = true;
|
2022-01-29 19:05:57 +00:00
|
|
|
formData = form.decodeForm(utf8Decode(request.body), formData);
|
2016-03-12 18:50:43 +00:00
|
|
|
if (formData.submit == "Login") {
|
2021-01-02 18:10:00 +00:00
|
|
|
var account = gDatabase.get("user:" + formData.name);
|
|
|
|
account = account ? JSON.parse(account) : account;
|
2016-03-12 18:50:43 +00:00
|
|
|
if (formData.register == "1") {
|
2021-01-02 18:10:00 +00:00
|
|
|
if (!account &&
|
2016-03-12 18:50:43 +00: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));
|
|
|
|
} catch {
|
|
|
|
}
|
|
|
|
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)};
|
|
|
|
gDatabase.set("user:" + formData.name, JSON.stringify(account));
|
2016-03-12 18:50:43 +00:00
|
|
|
if (noAdministrator()) {
|
|
|
|
makeAdministrator(formData.name);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
loginError = "Error registering account.";
|
|
|
|
}
|
|
|
|
} else {
|
2021-01-02 18:10:00 +00:00
|
|
|
if (account &&
|
|
|
|
account.password &&
|
|
|
|
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 {
|
|
|
|
loginError = "Invalid username or password.";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Proceed as Guest
|
2022-09-28 23:52:44 +00:00
|
|
|
session = makeJwt({name: 'guest'});
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-07 18:39:52 +00:00
|
|
|
var cookie = "session=" + session + "; path=/; Max-Age=604800; Secure; SameSite=Strict";
|
2016-03-12 18:50:43 +00:00
|
|
|
var entry = readSession(session);
|
|
|
|
if (entry && formData.return) {
|
|
|
|
response.writeHead(303, {"Location": formData.return, "Set-Cookie": cookie});
|
|
|
|
response.end();
|
|
|
|
} else {
|
2021-10-06 01:25:33 +00:00
|
|
|
File.readFile("core/auth.html").then(function(data) {
|
2022-01-27 01:15:54 +00:00
|
|
|
var html = utf8Decode(data);
|
2021-10-06 01:25:33 +00:00
|
|
|
var contents = "";
|
2016-03-12 18:50:43 +00:00
|
|
|
|
2021-10-06 01:25:33 +00:00
|
|
|
if (entry) {
|
|
|
|
if (sessionIsNew) {
|
|
|
|
contents += '<div>Welcome back, ' + entry.name + '.</div>\n';
|
|
|
|
} else {
|
|
|
|
contents += '<div>You are already logged in, ' + entry.name + '.</div>\n';
|
|
|
|
}
|
|
|
|
contents += '<div><a href="/login/logout">Logout</a></div>\n';
|
2016-03-12 18:50:43 +00:00
|
|
|
} else {
|
2021-10-06 01:25:33 +00:00
|
|
|
contents += '<form method="POST">\n';
|
|
|
|
if (loginError) {
|
|
|
|
contents += "<p>" + loginError + "</p>\n";
|
|
|
|
}
|
|
|
|
contents += '<div id="auth_greeting"><b>Halt. Who goes there?</b></div>\n'
|
|
|
|
contents += '<div id="auth">\n';
|
|
|
|
contents += '<div id="auth_login">\n'
|
|
|
|
if (noAdministrator()) {
|
|
|
|
contents += '<div class="notice">There is currently no administrator. You will be made administrator.</div>\n';
|
|
|
|
}
|
|
|
|
contents += '<div><label for="name">Name:</label> <input type="text" id="name" name="name" value=""></div>\n';
|
|
|
|
contents += '<div><label for="password">Password:</label> <input type="password" id="password" name="password" value=""></div>\n';
|
|
|
|
contents += '<div id="confirmPassword" style="display: none"><label for="confirm">Confirm:</label> <input type="password" id="confirm" name="confirm" value=""></div>\n';
|
|
|
|
contents += '<div><input type="checkbox" id="register" name="register" value="1" onchange="showHideConfirm()"> <label for="register">Register a new account</label></div>\n';
|
|
|
|
contents += '<div><input id="loginButton" type="submit" name="submit" value="Login"></div>\n';
|
|
|
|
contents += '</div>';
|
|
|
|
contents += '<div class="auth_or"> - or - </div>';
|
|
|
|
contents += '<div id="auth_guest">\n';
|
|
|
|
contents += '<input id="guestButton" type="submit" name="submit" value="Proceeed as Guest">\n';
|
|
|
|
contents += '</div>\n';
|
|
|
|
contents += '</div>\n';
|
|
|
|
contents += '</form>';
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
2021-10-06 01:25:33 +00:00
|
|
|
var text = html.replace("<!--SESSION-->", contents);
|
|
|
|
response.writeHead(200, {"Content-Type": "text/html; charset=utf-8", "Set-Cookie": cookie, "Content-Length": text.length});
|
|
|
|
response.end(text);
|
|
|
|
}).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
|
|
|
}
|
|
|
|
} else if (request.uri == "/login/logout") {
|
2022-03-07 18:39:52 +00:00
|
|
|
response.writeHead(303, {"Set-Cookie": "session=; path=/; secure; SameSite=Strict; expires=Thu, 01 Jan 1970 00:00:00 GMT", "Location": "/login" + (request.query ? "?" + request.query : "")});
|
2016-03-12 18:50:43 +00:00
|
|
|
response.end();
|
|
|
|
} else {
|
|
|
|
response.writeHead(200, {"Content-Type": "text/plain; charset=utf-8", "Connection": "close"});
|
|
|
|
response.end("Hello, " + request.client.peerName + ".");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function getPermissions(session) {
|
|
|
|
var permissions;
|
|
|
|
var entry = readSession(session);
|
|
|
|
if (entry) {
|
|
|
|
permissions = getPermissionsForUser(entry.name);
|
|
|
|
permissions.authenticated = entry.name !== "guest";
|
|
|
|
}
|
|
|
|
return permissions || {};
|
|
|
|
}
|
|
|
|
|
|
|
|
function getPermissionsForUser(userName) {
|
|
|
|
var permissions = {};
|
2022-03-18 01:24:29 +00:00
|
|
|
if (core.globalSettings && core.globalSettings.permissions && core.globalSettings.permissions[userName]) {
|
|
|
|
for (var i in core.globalSettings.permissions[userName]) {
|
|
|
|
permissions[core.globalSettings.permissions[userName][i]] = true;
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return permissions;
|
|
|
|
}
|
|
|
|
|
|
|
|
function query(headers) {
|
|
|
|
var session = getCookies(headers).session;
|
|
|
|
var entry;
|
2021-12-01 23:29:53 +00:00
|
|
|
var autologin = tildefriends.args.autologin;
|
|
|
|
if (entry = autologin ? {name: autologin} : readSession(session)) {
|
|
|
|
return {session: entry, permissions: autologin ? getPermissionsForUser(autologin) : getPermissions(session)};
|
2016-03-12 18:50:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-18 17:39:08 +00:00
|
|
|
export { handler, query };
|