snappymail/dev/View/Popup/ComposeOpenPgp.js

388 lines
8.5 KiB
JavaScript
Raw Normal View History

import ko from 'ko';
2014-03-20 06:39:36 +08:00
import { pString, defaultOptionsAfterRender } from 'Common/Utils';
2020-08-14 04:58:41 +08:00
import { KeyState } from 'Common/Enums';
2019-07-05 03:19:24 +08:00
import { i18n } from 'Common/Translator';
2014-03-21 07:47:13 +08:00
import PgpStore from 'Stores/User/Pgp';
2014-08-25 15:10:51 +08:00
2019-07-05 03:19:24 +08:00
import { EmailModel } from 'Model/Email';
2014-08-25 15:10:51 +08:00
import { decorateKoCommands } from 'Knoin/Knoin';
import { AbstractViewPopup } from 'Knoin/AbstractViews';
2014-03-21 07:47:13 +08:00
2021-01-26 18:46:30 +08:00
const KEY_NAME_SUBSTR = -8,
i18nPGP = (key, params) => i18n('PGP_NOTIFICATIONS/' + key, params);
2016-08-24 06:17:50 +08:00
class ComposeOpenPgpPopupView extends AbstractViewPopup {
constructor() {
super('ComposeOpenPgp');
2015-07-30 01:21:24 +08:00
2021-01-26 18:46:30 +08:00
this.publicKeysOptionsCaption = i18nPGP('ADD_A_PUBLICK_KEY');
this.privateKeysOptionsCaption = i18nPGP('SELECT_A_PRIVATE_KEY');
2015-07-30 01:21:24 +08:00
this.addObservables({
notification: '',
2014-03-21 07:47:13 +08:00
sign: false,
encrypt: false,
2014-03-21 07:47:13 +08:00
password: '',
2014-03-21 07:47:13 +08:00
text: '',
selectedPrivateKey: null,
selectedPublicKey: null,
2015-07-30 01:21:24 +08:00
signKey: null,
2015-07-30 01:21:24 +08:00
submitRequest: false
});
this.encryptKeys = ko.observableArray();
this.addComputables({
encryptKeysView: () => this.encryptKeys.map(oKey => (oKey ? oKey.key : null)).filter(v => v),
privateKeysOptions: () => {
2021-03-09 21:50:09 +08:00
const opts = PgpStore.openpgpkeysPrivate().map(oKey => {
if (this.signKey() && this.signKey().key.id === oKey.id) {
return null;
}
return oKey.users.map(user => ({
'id': oKey.guid,
'name': '(' + oKey.id.substr(KEY_NAME_SUBSTR).toUpperCase() + ') ' + user,
2021-03-09 21:50:09 +08:00
'key': oKey
}));
});
return opts.flat().filter(v => v);
},
publicKeysOptions: () => {
2021-03-09 21:50:09 +08:00
const opts = PgpStore.openpgpkeysPublic().map(oKey => {
if (this.encryptKeysView().includes(oKey)) {
return null;
}
return oKey.users.map(user => ({
'id': oKey.guid,
'name': '(' + oKey.id.substr(KEY_NAME_SUBSTR).toUpperCase() + ') ' + user,
2021-03-09 21:50:09 +08:00
'key': oKey
}));
});
return opts.flat().filter(v => v);
}
});
2014-08-21 23:08:34 +08:00
this.resultCallback = null;
2015-07-30 01:21:24 +08:00
2016-09-10 06:38:16 +08:00
this.selectedPrivateKey.subscribe((value) => {
2019-07-05 03:19:24 +08:00
if (value) {
2016-09-10 06:38:16 +08:00
this.selectCommand();
this.updateCommand();
}
});
2014-08-21 23:08:34 +08:00
2016-09-10 06:38:16 +08:00
this.selectedPublicKey.subscribe((value) => {
2019-07-05 03:19:24 +08:00
if (value) {
2016-09-10 06:38:16 +08:00
this.addCommand();
}
});
this.sDefaultKeyScope = KeyState.PopupComposeOpenPGP;
this.defaultOptionsAfterRender = defaultOptionsAfterRender;
2016-09-10 06:38:16 +08:00
this.deletePublickKey = this.deletePublickKey.bind(this);
decorateKoCommands(this, {
doCommand: self => !self.submitRequest() && (self.sign() || self.encrypt()),
selectCommand: 1,
addCommand: 1,
updateCommand: 1,
});
2016-09-10 06:38:16 +08:00
}
doCommand() {
2019-07-05 03:19:24 +08:00
let result = true,
2016-09-10 06:38:16 +08:00
privateKey = null,
aPublicKeys = [];
this.submitRequest(true);
2019-07-05 03:19:24 +08:00
if (result && this.sign()) {
if (!this.signKey()) {
2021-01-26 18:46:30 +08:00
this.notification(i18nPGP('NO_PRIVATE_KEY_FOUND'));
2016-09-10 06:38:16 +08:00
result = false;
2019-07-05 03:19:24 +08:00
} else if (!this.signKey().key) {
this.notification(
2021-01-26 18:46:30 +08:00
i18nPGP('NO_PRIVATE_KEY_FOUND_FOR', {
2019-07-05 03:19:24 +08:00
'EMAIL': this.signKey().email
})
);
2014-08-21 23:08:34 +08:00
2016-09-10 06:38:16 +08:00
result = false;
}
2014-08-21 23:08:34 +08:00
2019-07-05 03:19:24 +08:00
if (result) {
2016-09-10 06:38:16 +08:00
const privateKeys = this.signKey().key.getNativeKeys();
privateKey = privateKeys[0] || null;
2019-07-05 03:19:24 +08:00
try {
if (privateKey) {
2016-09-10 06:38:16 +08:00
privateKey.decrypt(pString(this.password()));
}
2019-07-05 03:19:24 +08:00
} catch (e) {
2016-09-10 06:38:16 +08:00
privateKey = null;
}
2014-08-21 23:08:34 +08:00
2019-07-05 03:19:24 +08:00
if (!privateKey) {
2021-01-26 18:46:30 +08:00
this.notification(i18nPGP('NO_PRIVATE_KEY_FOUND'));
result = false;
2014-03-21 07:47:13 +08:00
}
2016-09-10 06:38:16 +08:00
}
}
2014-03-21 07:47:13 +08:00
2019-07-05 03:19:24 +08:00
if (result && this.encrypt()) {
if (this.encryptKeys.length) {
2016-09-10 06:38:16 +08:00
aPublicKeys = [];
2014-03-21 07:47:13 +08:00
this.encryptKeys.forEach(oKey => {
2019-07-05 03:19:24 +08:00
if (oKey && oKey.key) {
aPublicKeys = aPublicKeys.concat(oKey.key.getNativeKeys().flat(Infinity).filter(v => v));
2019-07-05 03:19:24 +08:00
} else if (oKey && oKey.email) {
this.notification(
2021-01-26 18:46:30 +08:00
i18nPGP('NO_PUBLIC_KEYS_FOUND_FOR', {
2019-07-05 03:19:24 +08:00
'EMAIL': oKey.email
})
);
2014-04-18 06:42:30 +08:00
result = false;
}
2016-09-10 06:38:16 +08:00
});
2015-07-30 01:21:24 +08:00
if (result && (!aPublicKeys.length || this.encryptKeys.length !== aPublicKeys.length)) {
result = false;
}
} else {
2021-01-26 18:46:30 +08:00
this.notification(i18nPGP('NO_PUBLIC_KEYS_FOUND'));
result = false;
2016-09-10 06:38:16 +08:00
}
}
2019-07-05 03:19:24 +08:00
if (result && this.resultCallback) {
setTimeout(() => {
2016-09-10 06:38:16 +08:00
let pgpPromise = null;
2016-06-30 08:02:45 +08:00
2019-07-05 03:19:24 +08:00
try {
if (aPublicKeys.length) {
if (privateKey) {
pgpPromise = PgpStore.openpgp.encrypt({
data: this.text(),
publicKeys: aPublicKeys,
privateKeys: [privateKey]
});
} else {
pgpPromise = PgpStore.openpgp.encrypt({
data: this.text(),
publicKeys: aPublicKeys
});
}
} else if (privateKey) {
2016-09-10 06:38:16 +08:00
pgpPromise = PgpStore.openpgp.sign({
data: this.text(),
privateKeys: [privateKey]
});
2015-07-30 01:21:24 +08:00
}
2019-07-05 03:19:24 +08:00
} catch (e) {
console.log(e);
2019-07-05 03:19:24 +08:00
this.notification(
2021-01-26 18:46:30 +08:00
i18nPGP('PGP_ERROR', {
2019-07-05 03:19:24 +08:00
'ERROR': '' + e
})
);
2016-09-10 06:38:16 +08:00
}
2015-07-30 01:21:24 +08:00
2019-07-05 03:19:24 +08:00
if (pgpPromise) {
try {
pgpPromise
.then((mData) => {
this.resultCallback(mData.data);
this.cancelCommand();
})
.catch((e) => {
this.notification(
2021-01-26 18:46:30 +08:00
i18nPGP('PGP_ERROR', {
2019-07-05 03:19:24 +08:00
'ERROR': '' + e
})
);
});
} catch (e) {
this.notification(
2021-01-26 18:46:30 +08:00
i18nPGP('PGP_ERROR', {'ERROR': '' + e})
2019-07-05 03:19:24 +08:00
);
2014-08-21 23:08:34 +08:00
}
2016-09-10 06:38:16 +08:00
}
2014-03-20 06:39:36 +08:00
this.submitRequest(false);
2020-08-14 04:58:41 +08:00
}, 20);
2019-07-05 03:19:24 +08:00
} else {
2016-09-10 06:38:16 +08:00
this.submitRequest(false);
}
2016-09-10 06:38:16 +08:00
return result;
}
2016-09-10 06:38:16 +08:00
selectCommand() {
2019-07-05 03:19:24 +08:00
const keyId = this.selectedPrivateKey(),
option = keyId ? this.privateKeysOptions().find(item => item && keyId === item.id) : null;
2016-06-30 08:02:45 +08:00
2019-07-05 03:19:24 +08:00
if (option) {
2016-09-10 06:38:16 +08:00
this.signKey({
'empty': !option.key,
'selected': ko.observable(!!option.key),
'users': option.key.users,
'hash': option.key.id.substr(KEY_NAME_SUBSTR).toUpperCase(),
'key': option.key
});
}
}
2016-06-30 08:02:45 +08:00
2016-09-10 06:38:16 +08:00
addCommand() {
2019-07-05 03:19:24 +08:00
const keyId = this.selectedPublicKey(),
option = keyId ? this.publicKeysOptions().find(item => item && keyId === item.id) : null;
2019-07-05 03:19:24 +08:00
if (option) {
this.encryptKeys.push({
2016-09-10 06:38:16 +08:00
'empty': !option.key,
'selected': ko.observable(!!option.key),
'removable': ko.observable(!this.sign() || !this.signKey() || this.signKey().key.id !== option.key.id),
'users': option.key.users,
'hash': option.key.id.substr(KEY_NAME_SUBSTR).toUpperCase(),
'key': option.key
});
2016-09-10 06:38:16 +08:00
}
}
2015-07-30 01:21:24 +08:00
2016-09-10 06:38:16 +08:00
updateCommand() {
this.encryptKeys.forEach(oKey =>
oKey.removable(!this.sign() || !this.signKey() || this.signKey().key.id !== oKey.key.id)
);
}
2015-07-30 01:21:24 +08:00
deletePublickKey(publicKey) {
this.encryptKeys.remove(publicKey);
}
2016-04-21 06:42:16 +08:00
clearPopup() {
this.notification('');
2016-04-21 06:42:16 +08:00
this.sign(false);
this.encrypt(false);
2016-04-21 06:42:16 +08:00
this.password('');
2014-03-21 07:47:13 +08:00
this.signKey(null);
this.encryptKeys([]);
this.text('');
2014-03-21 07:47:13 +08:00
this.resultCallback = null;
}
2014-03-21 07:47:13 +08:00
onBuild() {
// shortcuts.add('tab', 'shift', KeyState.PopupComposeOpenPGP, () => {
shortcuts.add('tab', '', KeyState.PopupComposeOpenPGP, () => {
let btn = this.querySelector('.inputPassword');
if (btn.matches(':focus')) {
btn = this.querySelector('.buttonDo');
}
btn.focus();
return false;
});
}
2015-07-30 01:21:24 +08:00
onHideWithDelay() {
this.clearPopup();
}
2014-03-20 06:39:36 +08:00
onShowWithDelay() {
this.querySelector(this.sign() ? '.inputPassword' : '.buttonDo').focus();
}
2015-07-30 01:21:24 +08:00
onShow(fCallback, sText, identity, sTo, sCc, sBcc) {
this.clearPopup();
2014-03-20 06:39:36 +08:00
2019-07-05 03:19:24 +08:00
let rec = [],
emailLine = '';
const email = new EmailModel();
2014-03-20 06:39:36 +08:00
this.resultCallback = fCallback;
2014-03-21 00:05:35 +08:00
if (sTo) {
rec.push(sTo);
}
2014-03-21 00:05:35 +08:00
if (sCc) {
rec.push(sCc);
}
2014-03-20 06:39:36 +08:00
if (sBcc) {
rec.push(sBcc);
}
2014-03-21 00:05:35 +08:00
rec = rec.join(', ').split(',');
rec = rec.map(value => {
2019-07-05 03:19:24 +08:00
email.clear();
email.parse(value.trim());
return email.email || false;
}).filter(v => v);
2019-07-05 03:19:24 +08:00
if (identity && identity.email()) {
emailLine = identity.email();
rec.unshift(emailLine);
2014-08-21 23:08:34 +08:00
const keys = PgpStore.findAllPrivateKeysByEmailNotNative(emailLine);
2019-07-05 03:19:24 +08:00
if (keys && keys[0]) {
this.signKey({
'users': keys[0].users || [emailLine],
2016-08-24 06:17:50 +08:00
'hash': keys[0].id.substr(KEY_NAME_SUBSTR).toUpperCase(),
'key': keys[0]
});
}
}
2016-08-10 03:52:30 +08:00
2019-07-05 03:19:24 +08:00
if (this.signKey()) {
this.sign(true);
2014-08-21 23:08:34 +08:00
}
if (rec.length) {
2019-07-05 03:19:24 +08:00
this.encryptKeys(
rec.map(recEmail => {
const keys = PgpStore.findAllPublicKeysByEmailNotNative(recEmail);
return keys
? keys.map(publicKey => ({
'empty': !publicKey,
'selected': ko.observable(!!publicKey),
'removable': ko.observable(
!this.sign() || !this.signKey() || this.signKey().key.id !== publicKey.id
),
'users': publicKey ? publicKey.users || [recEmail] : [recEmail],
'hash': publicKey ? publicKey.id.substr(KEY_NAME_SUBSTR).toUpperCase() : '',
'key': publicKey
}))
: [];
}).flat().validUnique(encryptKey => encryptKey.hash)
2019-07-05 03:19:24 +08:00
);
if (this.encryptKeys.length) {
this.encrypt(true);
}
2015-07-30 01:21:24 +08:00
}
this.text(sText);
}
}
2014-08-21 23:08:34 +08:00
2019-07-05 03:19:24 +08:00
export { ComposeOpenPgpPopupView, ComposeOpenPgpPopupView as default };