snappymail/dev/Stores/User/Pgp.js

486 lines
12 KiB
JavaScript
Raw Normal View History

2015-02-01 23:44:44 +08:00
2016-06-30 08:02:45 +08:00
var
_ = require('_'),
ko = require('ko'),
$ = require('$'),
kn = require('Knoin/Knoin'),
Translator = require('Common/Translator'),
Settings = require('Storage/Settings'),
Utils = require('Common/Utils');
/**
* @constructor
*/
function PgpUserStore()
{
this.capaOpenPGP = ko.observable(false);
this.openpgp = null;
this.openpgpkeys = ko.observableArray([]);
this.openpgpKeyring = null;
this.openpgpkeysPublic = this.openpgpkeys.filter(function(oItem) {
return !!(oItem && !oItem.isPrivate);
});
this.openpgpkeysPrivate = this.openpgpkeys.filter(function(oItem) {
return !!(oItem && oItem.isPrivate);
});
}
/**
* @returns {boolean}
*/
PgpUserStore.prototype.isSupported = function()
{
return !!this.openpgp;
};
PgpUserStore.prototype.findKeyByHex = function(aKeys, sHash)
{
return _.find(aKeys, function(oItem) {
return sHash && oItem && (sHash === oItem.id || -1 < oItem.ids.indexOf(sHash));
});
};
PgpUserStore.prototype.findPublicKeyByHex = function(sHash)
{
return this.findKeyByHex(this.openpgpkeysPublic(), sHash);
};
PgpUserStore.prototype.findPrivateKeyByHex = function(sHash)
{
return this.findKeyByHex(this.openpgpkeysPrivate(), sHash);
};
PgpUserStore.prototype.findPublicKeysByEmail = function(sEmail)
{
return _.compact(_.flatten(_.map(this.openpgpkeysPublic(), function(oItem) {
var oKey = oItem && -1 < oItem.emails.indexOf(sEmail) ? oItem : null;
return oKey ? oKey.getNativeKeys() : [null];
}), true));
};
PgpUserStore.prototype.findPublicKeysBySigningKeyIds = function(aSigningKeyIds)
{
var self = this;
return _.compact(_.flatten(_.map(aSigningKeyIds, function(oId) {
var oKey = oId && oId.toHex ? self.findPublicKeyByHex(oId.toHex()) : null;
return oKey ? oKey.getNativeKeys() : [null];
}), true));
};
PgpUserStore.prototype.findPrivateKeysByEncryptionKeyIds = function(aEncryptionKeyIds, aRecipients, bReturnWrapKeys)
{
var self = this, aResult = [];
aResult = Utils.isArray(aEncryptionKeyIds) ? _.compact(_.flatten(_.map(aEncryptionKeyIds, function(oId) {
var oKey = oId && oId.toHex ? self.findPrivateKeyByHex(oId.toHex()) : null;
return oKey ? (bReturnWrapKeys ? [oKey] : oKey.getNativeKeys()) : [null];
}), true)) : [];
if (0 === aResult.length && Utils.isNonEmptyArray(aRecipients))
2015-02-01 23:44:44 +08:00
{
2016-06-30 08:02:45 +08:00
aResult = _.uniq(_.compact(_.flatten(_.map(aRecipients, function(sEmail) {
var aKeys = sEmail ? self.findAllPrivateKeysByEmailNotNative(sEmail) : null;
2016-07-01 06:50:11 +08:00
return aKeys ? (bReturnWrapKeys ? aKeys : _.flatten(_.map(aKeys, function(oKey) {
return oKey.getNativeKeys();
}), true)) : [null];
2016-06-30 08:02:45 +08:00
}), true)), function(oKey) {return oKey.id;});
2015-02-03 07:58:58 +08:00
}
2015-02-01 23:44:44 +08:00
2016-06-30 08:02:45 +08:00
return aResult;
};
/**
* @param {string} sEmail
* @returns {?}
*/
PgpUserStore.prototype.findPublicKeyByEmailNotNative = function(sEmail)
{
return _.find(this.openpgpkeysPublic(), function(oItem) {
return oItem && -1 < oItem.emails.indexOf(sEmail);
}) || null;
};
/**
* @param {string} sEmail
* @returns {?}
*/
PgpUserStore.prototype.findPrivateKeyByEmailNotNative = function(sEmail)
{
return _.find(this.openpgpkeysPrivate(), function(oItem) {
return oItem && -1 < oItem.emails.indexOf(sEmail);
}) || null;
};
/**
* @param {string} sEmail
* @returns {?}
*/
PgpUserStore.prototype.findAllPublicKeysByEmailNotNative = function(sEmail)
{
return _.filter(this.openpgpkeysPublic(), function(oItem) {
return oItem && -1 < oItem.emails.indexOf(sEmail);
}) || null;
};
/**
* @param {string} sEmail
* @returns {?}
*/
PgpUserStore.prototype.findAllPrivateKeysByEmailNotNative = function(sEmail)
{
return _.filter(this.openpgpkeysPrivate(), function(oItem) {
return oItem && -1 < oItem.emails.indexOf(sEmail);
}) || null;
};
/**
* @param {string} sEmail
* @param {string=} sPassword
* @returns {?}
*/
PgpUserStore.prototype.findPrivateKeyByEmail = function(sEmail, sPassword)
{
var
oPrivateKeys = [],
oPrivateKey = null,
oKey = _.find(this.openpgpkeysPrivate(), function(oItem) {
return oItem && -1 < oItem.emails.indexOf(sEmail);
2016-06-30 08:02:45 +08:00
});
2016-06-30 08:02:45 +08:00
if (oKey)
{
2016-06-30 08:02:45 +08:00
oPrivateKeys = oKey.getNativeKeys();
oPrivateKey = oPrivateKeys[0] || null;
2015-07-30 01:21:24 +08:00
2016-06-30 08:02:45 +08:00
try
2015-02-22 06:00:51 +08:00
{
2016-06-30 08:02:45 +08:00
if (oPrivateKey)
2015-02-22 06:00:51 +08:00
{
2016-06-30 08:02:45 +08:00
oPrivateKey.decrypt(Utils.pString(sPassword));
2015-02-22 06:00:51 +08:00
}
}
2016-06-30 08:02:45 +08:00
catch (e)
{
oPrivateKey = null;
}
}
2015-02-22 06:00:51 +08:00
2016-06-30 08:02:45 +08:00
return oPrivateKey;
};
/**
* @param {string=} sPassword
* @returns {?}
*/
PgpUserStore.prototype.findSelfPrivateKey = function(sPassword)
{
return this.findPrivateKeyByEmail(require('Stores/User/Account').email(), sPassword);
};
PgpUserStore.prototype.decryptMessage = function(oMessage, aRecipients, fCallback)
{
var self = this, aPrivateKeys = [];
if (oMessage && oMessage.getEncryptionKeyIds)
{
2016-06-30 08:02:45 +08:00
aPrivateKeys = this.findPrivateKeysByEncryptionKeyIds(oMessage.getEncryptionKeyIds(), aRecipients, true);
if (aPrivateKeys && 0 < aPrivateKeys.length)
{
2016-06-30 08:02:45 +08:00
kn.showScreenPopup(require('View/Popup/MessageOpenPgp'), [function(oDecryptedKey) {
2016-06-30 08:02:45 +08:00
if (oDecryptedKey)
{
oMessage.decrypt(oDecryptedKey).then(function(oDecryptedMessage) {
var oPrivateKey = null;
if (oDecryptedMessage)
{
oPrivateKey = self.findPrivateKeyByHex(oDecryptedKey.primaryKey.keyid.toHex());
if (oPrivateKey)
{
2016-06-30 08:02:45 +08:00
self.verifyMessage(oDecryptedMessage, function(oValidKey, aSigningKeyIds) {
fCallback(oPrivateKey, oDecryptedMessage, oValidKey || null, aSigningKeyIds || null);
});
}
else
{
fCallback(oPrivateKey, oDecryptedMessage);
}
2016-06-30 08:02:45 +08:00
}
else
{
fCallback(oPrivateKey, oDecryptedMessage);
}
2016-06-30 08:02:45 +08:00
}, function() {
fCallback(null, null);
2016-06-30 08:02:45 +08:00
});
}
else
{
fCallback(null, null);
}
2016-06-30 08:02:45 +08:00
}, aPrivateKeys]);
2016-06-30 08:02:45 +08:00
return false;
}
2016-06-30 08:02:45 +08:00
}
2016-06-30 08:02:45 +08:00
fCallback(null, null);
2016-06-30 08:02:45 +08:00
return false;
};
2016-06-30 08:02:45 +08:00
PgpUserStore.prototype.findKeyExternal = function(sEmail, fCallback)
{
if (this.openpgp.HKP && Settings.appSettingsGet('openpgpPublicKeyServer'))
2016-05-24 01:33:01 +08:00
{
2016-06-30 08:02:45 +08:00
var oHkp = new this.openpgp.HKP(Settings.appSettingsGet('openpgpPublicKeyServer').replace(/\/$/, ''));
oHkp.lookup({query: sEmail}).then(function(sKey) {
fCallback(sKey);
}, function() {
2016-05-24 01:33:01 +08:00
fCallback(null);
2016-06-30 08:02:45 +08:00
});
}
else
{
fCallback(null);
}
};
2016-05-24 01:33:01 +08:00
2016-06-30 08:02:45 +08:00
PgpUserStore.prototype.verifyMessage = function(oMessage, fCallback)
{
var oValid = null, aResult = [], aPublicKeys = [], aSigningKeyIds = [];
if (oMessage && oMessage.getSigningKeyIds)
{
2016-06-30 08:02:45 +08:00
aSigningKeyIds = oMessage.getSigningKeyIds();
if (aSigningKeyIds && 0 < aSigningKeyIds.length)
{
2016-05-24 01:33:01 +08:00
// this.findKeyExternal('support@rainloop.net', function(key) {
// console.log(key);
// });
2016-06-30 08:02:45 +08:00
aPublicKeys = this.findPublicKeysBySigningKeyIds(aSigningKeyIds);
if (aPublicKeys && 0 < aPublicKeys.length)
{
try
{
2016-06-30 08:02:45 +08:00
aResult = oMessage.verify(aPublicKeys);
oValid = _.find(_.isArray(aResult) ? aResult : [], function(oItem) {
return oItem && oItem.valid && oItem.keyid;
});
2016-06-30 08:02:45 +08:00
if (oValid && oValid.keyid && oValid.keyid && oValid.keyid.toHex)
2015-10-14 00:36:43 +08:00
{
2016-06-30 08:02:45 +08:00
fCallback(this.findPublicKeyByHex(oValid.keyid.toHex()));
return true;
2015-10-14 00:36:43 +08:00
}
}
2016-06-30 08:02:45 +08:00
catch (e)
{
Utils.log(e);
}
}
2016-06-30 08:02:45 +08:00
fCallback(null, aSigningKeyIds);
return false;
}
2016-06-30 08:02:45 +08:00
}
2016-06-30 08:02:45 +08:00
fCallback(null);
return false;
};
2016-06-30 08:02:45 +08:00
/**
* @param {*} mDom
*/
PgpUserStore.prototype.controlsHelper = function(mDom, oVerControl, bSuccess, sTitle, sText)
{
if (bSuccess)
{
2016-06-30 08:02:45 +08:00
mDom.removeClass('error').addClass('success').attr('title', sTitle);
oVerControl.removeClass('error').addClass('success').attr('title', sTitle);
}
else
{
mDom.removeClass('success').addClass('error').attr('title', sTitle);
oVerControl.removeClass('success').addClass('error').attr('title', sTitle);
}
2016-06-30 08:02:45 +08:00
if (!Utils.isUnd(sText))
{
mDom.text(Utils.trim(sText.replace(/(\u200C|\u0002)/g, '')));
}
};
2016-06-30 08:02:45 +08:00
/**
* @static
*/
PgpUserStore.domControlEncryptedClickHelper = function(store, mDom, sArmoredMessage, aRecipients)
{
return function() {
2016-06-30 08:02:45 +08:00
var oMessage = null, $this = $(this);
2016-06-30 08:02:45 +08:00
if ($this.hasClass('success'))
{
return false;
2016-06-30 08:02:45 +08:00
}
2016-06-30 08:02:45 +08:00
try
{
oMessage = store.openpgp.message.readArmored(sArmoredMessage);
}
catch (e)
{
Utils.log(e);
}
2016-06-30 08:02:45 +08:00
if (oMessage && oMessage.getText && oMessage.verify && oMessage.decrypt)
{
store.decryptMessage(oMessage, aRecipients, function(oValidPrivateKey, oDecryptedMessage, oValidPublicKey, aSigningKeyIds) {
2016-06-30 08:02:45 +08:00
if (oDecryptedMessage)
{
if (oValidPublicKey)
{
2016-06-30 08:02:45 +08:00
store.controlsHelper(mDom, $this, true, Translator.i18n('PGP_NOTIFICATIONS/GOOD_SIGNATURE', {
'USER': oValidPublicKey.user + ' (' + oValidPublicKey.id + ')'
}), oDecryptedMessage.getText());
}
2016-06-30 08:02:45 +08:00
else if (oValidPrivateKey)
{
var
aKeyIds = Utils.isNonEmptyArray(aSigningKeyIds) ? aSigningKeyIds : null,
2016-06-30 08:02:45 +08:00
sAdditional = aKeyIds ? _.compact(_.map(aKeyIds, function(oItem) {
return oItem && oItem.toHex ? oItem.toHex() : null;
2016-06-30 08:02:45 +08:00
})).join(', ') : '';
2016-06-30 08:02:45 +08:00
store.controlsHelper(mDom, $this, false,
Translator.i18n('PGP_NOTIFICATIONS/UNVERIFIRED_SIGNATURE') +
2016-06-30 08:02:45 +08:00
(sAdditional ? ' (' + sAdditional + ')' : ''),
oDecryptedMessage.getText());
}
2016-06-30 08:02:45 +08:00
else
{
store.controlsHelper(mDom, $this, false,
Translator.i18n('PGP_NOTIFICATIONS/DECRYPTION_ERROR'));
}
}
else
{
store.controlsHelper(mDom, $this, false,
Translator.i18n('PGP_NOTIFICATIONS/DECRYPTION_ERROR'));
}
});
return false;
2016-06-30 08:02:45 +08:00
}
store.controlsHelper(mDom, $this, false, Translator.i18n('PGP_NOTIFICATIONS/DECRYPTION_ERROR'));
return false;
};
2016-06-30 08:02:45 +08:00
};
2016-06-30 08:02:45 +08:00
/**
* @static
*/
PgpUserStore.domControlSignedClickHelper = function(store, mDom, sArmoredMessage)
{
return function() {
2016-06-30 08:02:45 +08:00
var oMessage = null, $this = $(this);
if ($this.hasClass('success') || $this.hasClass('error'))
{
return false;
}
2016-06-30 08:02:45 +08:00
try
{
oMessage = store.openpgp.cleartext.readArmored(sArmoredMessage);
}
catch (e)
{
Utils.log(e);
}
2016-06-30 08:02:45 +08:00
if (oMessage && oMessage.getText && oMessage.verify)
{
store.verifyMessage(oMessage, function(oValidKey, aSigningKeyIds) {
if (oValidKey)
{
2016-06-30 08:02:45 +08:00
store.controlsHelper(mDom, $this, true, Translator.i18n('PGP_NOTIFICATIONS/GOOD_SIGNATURE', {
'USER': oValidKey.user + ' (' + oValidKey.id + ')'
}), oMessage.getText());
}
2016-06-30 08:02:45 +08:00
else
{
2016-06-30 08:02:45 +08:00
var
aKeyIds = Utils.isNonEmptyArray(aSigningKeyIds) ? aSigningKeyIds : null,
sAdditional = aKeyIds ? _.compact(_.map(aKeyIds, function(oItem) {
return oItem && oItem.toHex ? oItem.toHex() : null;
})).join(', ') : '';
store.controlsHelper(mDom, $this, false,
Translator.i18n('PGP_NOTIFICATIONS/UNVERIFIRED_SIGNATURE') +
(sAdditional ? ' (' + sAdditional + ')' : ''));
}
2016-06-30 08:02:45 +08:00
});
2016-06-30 08:02:45 +08:00
return false;
}
2016-06-30 08:02:45 +08:00
store.controlsHelper(mDom, $this, false, Translator.i18n('PGP_NOTIFICATIONS/DECRYPTION_ERROR'));
return false;
};
2016-06-30 08:02:45 +08:00
};
/**
* @param {*} mDom
* @param {MessageModel} oRainLoopMessage
*/
PgpUserStore.prototype.initMessageBodyControls = function(mDom, oRainLoopMessage)
{
if (mDom && !mDom.hasClass('inited'))
{
mDom.addClass('inited');
var
bEncrypted = mDom.hasClass('encrypted'),
bSigned = mDom.hasClass('signed'),
oVerControl = null,
aRecipients = oRainLoopMessage ? oRainLoopMessage.getEmails(['from', 'to', 'cc']) : [],
sData = '';
2016-06-30 08:02:45 +08:00
if (bEncrypted || bSigned)
{
sData = mDom.text();
mDom.data('openpgp-original', sData);
2015-02-01 23:44:44 +08:00
2016-06-30 08:02:45 +08:00
if (bEncrypted)
{
oVerControl = $('<div class="b-openpgp-control"><i class="icon-lock"></i></div>')
.attr('title', Translator.i18n('MESSAGE/PGP_ENCRYPTED_MESSAGE_DESC'))
.on('click', PgpUserStore.domControlEncryptedClickHelper(this, mDom, sData, aRecipients));
}
else if (bSigned)
{
oVerControl = $('<div class="b-openpgp-control"><i class="icon-lock"></i></div>')
.attr('title', Translator.i18n('MESSAGE/PGP_SIGNED_MESSAGE_DESC'))
.on('click', PgpUserStore.domControlSignedClickHelper(this, mDom, sData));
}
if (oVerControl)
{
mDom.before(oVerControl).before('<div></div>');
}
}
}
};
2015-02-01 23:44:44 +08:00
2016-06-30 08:02:45 +08:00
module.exports = new PgpUserStore();