keeweb/app/scripts/comp/extension/protocol-impl.js

412 lines
11 KiB
JavaScript
Raw Normal View History

2021-04-06 20:10:42 +02:00
import kdbxweb from 'kdbxweb';
2021-04-06 21:27:58 +02:00
import { Events } from 'framework/events';
2021-04-06 20:10:42 +02:00
import { Launcher } from 'comp/launcher';
import { box as tweetnaclBox } from 'tweetnacl';
import { PasswordGenerator } from 'util/generators/password-generator';
import { GeneratorPresets } from 'comp/app/generator-presets';
import { Alerts } from 'comp/ui/alerts';
import { Locale } from 'util/locale';
import { RuntimeInfo } from 'const/runtime-info';
import { KnownAppVersions } from 'const/known-app-versions';
2021-04-11 10:17:02 +02:00
2021-04-11 17:22:58 +02:00
const KeeWebAssociationId = 'KeeWeb';
const KeeWebHash = '398d9c782ec76ae9e9877c2321cbda2b31fc6d18ccf0fed5ca4bd746bab4d64a'; // sha256('KeeWeb')
2021-04-06 20:10:42 +02:00
const Errors = {
noOpenFiles: {
message: Locale.extensionErrorNoOpenFiles,
code: '1'
},
userRejected: {
message: Locale.extensionErrorUserRejected,
code: '6'
}
};
const connectedClients = new Map();
let logger;
let appModel;
let sendEvent;
2021-04-19 23:33:16 +02:00
const ProtocolImpl = {
init(vars) {
appModel = vars.appModel;
logger = vars.logger;
sendEvent = vars.sendEvent;
2021-04-12 23:42:42 +02:00
2021-04-19 23:33:16 +02:00
setupListeners();
},
2021-04-12 23:42:42 +02:00
2021-04-19 23:33:16 +02:00
cleanup() {
connectedClients.clear();
},
2021-04-19 23:33:16 +02:00
deleteConnection(connectionId) {
for (const client of connectedClients.values()) {
if (client.connection.connectionId === connectionId) {
connectedClients.delete(client);
}
}
}
2021-04-19 23:33:16 +02:00
};
2021-04-12 23:42:42 +02:00
function setupListeners() {
Events.on('file-opened', () => {
sendEvent({ action: 'database-unlocked' });
});
Events.on('one-file-closed', () => {
if (!appModel.files.hasOpenFiles()) {
sendEvent({ action: 'database-locked' });
}
});
Events.on('all-files-closed', () => {
sendEvent({ action: 'database-locked' });
});
}
2021-04-06 20:10:42 +02:00
function incrementNonce(nonce) {
// from libsodium/utils.c, like it is in KeePassXC
let i = 0;
let c = 1;
for (; i < nonce.length; ++i) {
c += nonce[i];
nonce[i] = c;
c >>= 8;
}
}
function getClient(request) {
if (!request.clientID) {
throw new Error('Empty clientID');
}
2021-04-11 12:48:58 +02:00
const client = connectedClients.get(request.clientID);
2021-04-06 20:10:42 +02:00
if (!client) {
throw new Error(`Client not connected: ${request.clientID}`);
}
return client;
}
function decryptRequest(request) {
const client = getClient(request);
if (!request.nonce) {
throw new Error('Empty nonce');
}
if (!request.message) {
throw new Error('Empty message');
}
const nonce = kdbxweb.ByteUtils.base64ToBytes(request.nonce);
const message = kdbxweb.ByteUtils.base64ToBytes(request.message);
const data = tweetnaclBox.open(message, nonce, client.publicKey, client.keys.secretKey);
const json = new TextDecoder().decode(data);
const payload = JSON.parse(json);
logger.debug('Extension -> KeeWeb -> (decrypted)', payload);
2021-04-08 17:36:42 +02:00
if (!payload) {
throw new Error('Empty request payload');
}
if (payload.action !== request.action) {
2021-04-06 20:10:42 +02:00
throw new Error(`Bad action in decrypted payload`);
}
return payload;
}
function encryptResponse(request, payload) {
logger.debug('KeeWeb -> Extension (decrypted)', payload);
2021-04-08 17:36:42 +02:00
const nonceBytes = kdbxweb.ByteUtils.base64ToBytes(request.nonce);
incrementNonce(nonceBytes);
const nonce = kdbxweb.ByteUtils.bytesToBase64(nonceBytes);
2021-04-06 20:10:42 +02:00
const client = getClient(request);
2021-04-08 17:36:42 +02:00
payload.nonce = nonce;
2021-04-06 20:10:42 +02:00
const json = JSON.stringify(payload);
const data = new TextEncoder().encode(json);
2021-04-08 17:36:42 +02:00
const encrypted = tweetnaclBox(data, nonceBytes, client.publicKey, client.keys.secretKey);
2021-04-06 20:10:42 +02:00
const message = kdbxweb.ByteUtils.bytesToBase64(encrypted);
return {
action: request.action,
message,
nonce
};
}
function makeError(def) {
const e = new Error(def.message);
e.code = def.code;
return e;
}
2021-04-11 17:22:58 +02:00
function ensureAtLeastOneFileIsOpen() {
if (!appModel.files.hasOpenFiles()) {
throw makeError(Errors.noOpenFiles);
2021-04-11 17:22:58 +02:00
}
}
function checkContentRequestPermissions(request) {
ensureAtLeastOneFileIsOpen();
const client = getClient(request);
if (client.authorized) {
return;
2021-04-11 17:22:58 +02:00
}
return new Promise((resolve, reject) => {
if (Alerts.alertDisplayed) {
return reject(new Error(Locale.extensionErrorAlertDisplayed));
}
focusKeeWeb();
// TODO: make a proper dialog here instead of a simple question
if (Launcher) {
Alerts.yesno({
header: 'Extension connection',
body: 'Allow this extension to connect?',
success: () => {
resolve();
},
cancel: () => reject(makeError(Errors.userRejected))
});
} else {
// it's 'confirm' here because other browser extensions can't interact with browser alerts
// while they can easily press a button on our alert
// eslint-disable-next-line no-alert
const allowed = confirm('Allow this extension to connect?');
if (allowed) {
resolve();
} else {
reject(makeError(Errors.userRejected));
}
}
})
.then(() => {
client.authorized = true;
Launcher.hideApp();
})
.catch((e) => {
Launcher.hideApp();
throw e;
});
2021-04-11 17:22:58 +02:00
}
2021-04-10 23:35:42 +02:00
function getVersion(request) {
const extensionName = getClient(request).connection.extensionName;
2021-04-10 23:35:42 +02:00
return extensionName ? RuntimeInfo.version : KnownAppVersions.KeePassXC;
}
function isKeeWebConnect(request) {
return getClient(request).connection.extensionName === 'keeweb-connect';
2021-04-10 23:35:42 +02:00
}
function focusKeeWeb() {
logger.debug('Focus KeeWeb');
if (Launcher) {
Launcher.showMainWindow();
} else {
sendEvent({ action: 'attention-required' });
}
}
2021-04-11 14:53:56 +02:00
2021-04-06 20:10:42 +02:00
const ProtocolHandlers = {
'ping'({ data }) {
return { data };
},
'change-public-keys'(request, connection) {
let { publicKey, version, clientID: clientId } = request;
2021-04-10 23:35:42 +02:00
2021-04-11 12:48:58 +02:00
if (connectedClients.has(clientId)) {
2021-04-11 10:17:02 +02:00
throw new Error('Changing keys is not allowed');
}
2021-04-11 12:48:58 +02:00
if (!Launcher) {
// on web there can be only one connected client
connectedClients.clear();
}
2021-04-06 20:10:42 +02:00
const keys = tweetnaclBox.keyPair();
publicKey = kdbxweb.ByteUtils.base64ToBytes(publicKey);
connectedClients.set(clientId, { connection, publicKey, version, keys });
2021-04-06 20:10:42 +02:00
logger.info('New client key created', clientId, version);
2021-04-11 10:17:02 +02:00
2021-04-11 14:53:56 +02:00
const nonceBytes = kdbxweb.ByteUtils.base64ToBytes(request.nonce);
incrementNonce(nonceBytes);
const nonce = kdbxweb.ByteUtils.bytesToBase64(nonceBytes);
2021-04-06 20:10:42 +02:00
return {
action: 'change-public-keys',
2021-04-10 23:35:42 +02:00
version: getVersion(request),
2021-04-06 20:10:42 +02:00
publicKey: kdbxweb.ByteUtils.bytesToBase64(keys.publicKey),
2021-04-11 14:53:56 +02:00
nonce,
success: 'true',
...(isKeeWebConnect(request) ? { appName: 'KeeWeb' } : undefined)
2021-04-06 20:10:42 +02:00
};
},
'get-databasehash'(request) {
decryptRequest(request);
2021-04-11 17:22:58 +02:00
ensureAtLeastOneFileIsOpen();
2021-04-11 17:22:58 +02:00
return encryptResponse(request, {
hash: KeeWebHash,
success: 'true',
version: getVersion(request)
});
2021-04-07 22:49:23 +02:00
},
'generate-password'(request) {
const password = PasswordGenerator.generate(GeneratorPresets.browserExtensionPreset);
return encryptResponse(request, {
2021-04-10 23:35:42 +02:00
version: getVersion(request),
success: 'true',
2021-04-11 14:53:56 +02:00
entries: [{ password }]
});
},
2021-04-07 22:49:23 +02:00
'lock-database'(request) {
decryptRequest(request);
2021-04-11 17:22:58 +02:00
ensureAtLeastOneFileIsOpen();
2021-04-07 22:49:23 +02:00
2021-04-11 17:22:58 +02:00
Events.emit('lock-workspace');
2021-04-11 17:22:58 +02:00
if (Alerts.alertDisplayed) {
focusKeeWeb();
2021-04-11 14:53:56 +02:00
}
2021-04-11 17:22:58 +02:00
return encryptResponse(request, {
success: 'true',
version: getVersion(request)
});
},
'associate'(request) {
decryptRequest(request);
ensureAtLeastOneFileIsOpen();
return encryptResponse(request, {
success: 'true',
version: getVersion(request),
hash: KeeWebHash,
id: KeeWebAssociationId
});
},
'test-associate'(request) {
const payload = decryptRequest(request);
ensureAtLeastOneFileIsOpen();
2021-04-12 23:05:57 +02:00
if (payload.id !== KeeWebAssociationId) {
throw makeError(Errors.noOpenFiles);
}
2021-04-11 17:22:58 +02:00
return encryptResponse(request, {
success: 'true',
version: getVersion(request),
hash: KeeWebHash,
id: payload.id
});
},
async 'get-logins'(request) {
decryptRequest(request);
await checkContentRequestPermissions(request);
throw new Error('Not implemented');
},
async 'get-totp'(request) {
decryptRequest(request);
await checkContentRequestPermissions(request);
throw new Error('Not implemented');
},
async 'set-login'(request) {
decryptRequest(request);
await checkContentRequestPermissions(request);
throw new Error('Not implemented');
},
async 'get-database-groups'(request) {
decryptRequest(request);
await checkContentRequestPermissions(request);
2021-04-12 23:05:57 +02:00
const makeGroups = (group) => {
const res = {
name: group.title,
uuid: kdbxweb.ByteUtils.bytesToHex(group.group.uuid.bytes),
children: []
};
for (const subGroup of group.items) {
if (subGroup.matches()) {
res.children.push(makeGroups(subGroup));
}
}
return res;
};
const groups = [];
for (const file of appModel.files.filter((f) => f.active)) {
for (const group of file.groups) {
groups.push(makeGroups(group));
}
}
return encryptResponse(request, {
success: 'true',
version: getVersion(request),
groups: { groups }
});
},
async 'create-new-group'(request) {
2021-04-12 23:23:22 +02:00
const payload = decryptRequest(request);
await checkContentRequestPermissions(request);
2021-04-12 23:23:22 +02:00
if (!payload.groupName) {
throw new Error('No groupName');
}
// TODO: show file selector
// throw makeError(Errors.userRejected);
const groupNames = payload.groupName
.split('/')
.map((g) => g.trim())
.filter((g) => g);
if (!groupNames.length) {
throw new Error('Empty group path');
}
// TODO: create a new group
throw new Error('Not implemented');
2021-04-12 23:23:22 +02:00
// return encryptResponse(request, {
// success: 'true',
// version: getVersion(request),
// name: groupNames[groupNames.length - 1],
// uuid: kdbxweb.ByteUtils.bytesToHex(appModel.files[0].groups[0].group.uuid.bytes)
// });
2021-04-06 20:10:42 +02:00
}
};
2021-04-19 23:33:16 +02:00
export { ProtocolHandlers, ProtocolImpl };