2015-02-01 23:44:44 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
import ko from 'ko';
|
|
|
|
import _ from '_';
|
|
|
|
import $ from '$';
|
2016-06-30 08:02:45 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
import {i18n} from 'Common/Translator';
|
|
|
|
import {log, isArray, isNonEmptyArray, pString, isUnd, trim} from 'Common/Utils';
|
2016-06-30 08:02:45 +08:00
|
|
|
|
2016-09-13 04:50:21 +08:00
|
|
|
import AccountStore from 'Stores/User/Account';
|
|
|
|
|
|
|
|
import {showScreenPopup} from 'Knoin/Knoin';
|
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
class PgpUserStore
|
2016-06-30 08:02:45 +08:00
|
|
|
{
|
2016-08-17 06:01:20 +08:00
|
|
|
constructor() {
|
|
|
|
this.capaOpenPGP = ko.observable(false);
|
2016-06-30 08:02:45 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
this.openpgp = null;
|
2016-06-30 08:02:45 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
this.openpgpkeys = ko.observableArray([]);
|
|
|
|
this.openpgpKeyring = null;
|
2016-06-30 08:02:45 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
this.openpgpkeysPublic = this.openpgpkeys.filter((item) => !!(item && !item.isPrivate));
|
|
|
|
this.openpgpkeysPrivate = this.openpgpkeys.filter((item) => !!(item && item.isPrivate));
|
|
|
|
}
|
2016-06-30 08:02:45 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
/**
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
|
|
|
isSupported() {
|
|
|
|
return !!this.openpgp;
|
|
|
|
}
|
2016-06-30 08:02:45 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
findKeyByHex(keys, hash) {
|
|
|
|
return _.find(keys, (item) => (hash && item && (hash === item.id || -1 < item.ids.indexOf(hash))));
|
|
|
|
}
|
2016-06-30 08:02:45 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
findPublicKeyByHex(hash) {
|
|
|
|
return this.findKeyByHex(this.openpgpkeysPublic(), hash);
|
|
|
|
}
|
2016-06-30 08:02:45 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
findPrivateKeyByHex(hash) {
|
|
|
|
return this.findKeyByHex(this.openpgpkeysPrivate(), hash);
|
|
|
|
}
|
2016-06-30 08:02:45 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
findPublicKeysByEmail(email) {
|
|
|
|
return _.compact(_.flatten(_.map(this.openpgpkeysPublic(), (item) => {
|
|
|
|
const key = item && -1 < item.emails.indexOf(email) ? item : null;
|
|
|
|
return key ? key.getNativeKeys() : [null];
|
|
|
|
}), true));
|
|
|
|
}
|
2016-06-30 08:02:45 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
findPublicKeysBySigningKeyIds(signingKeyIds) {
|
|
|
|
return _.compact(_.flatten(_.map(signingKeyIds, (id) => {
|
|
|
|
const key = id && id.toHex ? this.findPublicKeyByHex(id.toHex()) : null;
|
|
|
|
return key ? key.getNativeKeys() : [null];
|
|
|
|
}), true));
|
|
|
|
}
|
2016-06-30 08:02:45 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
findPrivateKeysByEncryptionKeyIds(encryptionKeyIds, recipients, returnWrapKeys) {
|
|
|
|
let
|
|
|
|
result = isArray(encryptionKeyIds) ? _.compact(_.flatten(_.map(encryptionKeyIds, (id) => {
|
|
|
|
const key = id && id.toHex ? this.findPrivateKeyByHex(id.toHex()) : null;
|
|
|
|
return key ? (returnWrapKeys ? [key] : key.getNativeKeys()) : [null];
|
|
|
|
}), true)) : [];
|
|
|
|
|
|
|
|
if (0 === result.length && isNonEmptyArray(recipients))
|
|
|
|
{
|
|
|
|
result = _.uniq(_.compact(_.flatten(_.map(recipients, (sEmail) => {
|
|
|
|
const keys = sEmail ? this.findAllPrivateKeysByEmailNotNative(sEmail) : null;
|
|
|
|
return keys ? (returnWrapKeys ? keys : _.flatten(_.map(keys, (key) => key.getNativeKeys()), true)) : [null];
|
|
|
|
}), true)), (key) => key.id);
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
2015-02-03 07:58:58 +08:00
|
|
|
}
|
2015-02-01 23:44:44 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
/**
|
|
|
|
* @param {string} email
|
|
|
|
* @returns {?}
|
|
|
|
*/
|
|
|
|
findPublicKeyByEmailNotNative(email) {
|
|
|
|
return _.find(this.openpgpkeysPublic(), (item) => (item && -1 < item.emails.indexOf(email))) || null;
|
|
|
|
}
|
2016-06-30 08:02:45 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
/**
|
|
|
|
* @param {string} email
|
|
|
|
* @returns {?}
|
|
|
|
*/
|
|
|
|
findPrivateKeyByEmailNotNative(email) {
|
|
|
|
return _.find(this.openpgpkeysPrivate(), (item) => (item && -1 < item.emails.indexOf(email))) || null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} email
|
|
|
|
* @returns {?}
|
|
|
|
*/
|
|
|
|
findAllPublicKeysByEmailNotNative(email) {
|
|
|
|
return _.filter(this.openpgpkeysPublic(), (item) => (item && -1 < item.emails.indexOf(email))) || null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} email
|
|
|
|
* @returns {?}
|
|
|
|
*/
|
|
|
|
findAllPrivateKeysByEmailNotNative(email) {
|
|
|
|
return _.filter(this.openpgpkeysPrivate(), (item) => (item && -1 < item.emails.indexOf(email))) || null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} email
|
|
|
|
* @param {string=} password
|
|
|
|
* @returns {?}
|
|
|
|
*/
|
|
|
|
findPrivateKeyByEmail(email, password) {
|
|
|
|
|
|
|
|
let privateKey = null;
|
|
|
|
const key = _.find(this.openpgpkeysPrivate(), (item) => (item && -1 < item.emails.indexOf(email)));
|
|
|
|
|
|
|
|
if (key)
|
2015-02-22 06:00:51 +08:00
|
|
|
{
|
2016-08-17 06:01:20 +08:00
|
|
|
try
|
2015-02-22 06:00:51 +08:00
|
|
|
{
|
2016-08-17 06:01:20 +08:00
|
|
|
privateKey = key.getNativeKeys()[0] || null;
|
|
|
|
if (privateKey)
|
|
|
|
{
|
|
|
|
privateKey.decrypt(pString(password));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (e)
|
|
|
|
{
|
|
|
|
privateKey = null;
|
2015-02-22 06:00:51 +08:00
|
|
|
}
|
|
|
|
}
|
2016-06-30 08:02:45 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
return privateKey;
|
|
|
|
}
|
2016-06-30 08:02:45 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
/**
|
|
|
|
* @param {string=} password
|
|
|
|
* @returns {?}
|
|
|
|
*/
|
|
|
|
findSelfPrivateKey(password) {
|
2016-09-13 04:50:21 +08:00
|
|
|
return this.findPrivateKeyByEmail(AccountStore.email(), password);
|
2016-08-17 06:01:20 +08:00
|
|
|
}
|
2016-08-10 02:58:34 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
decryptMessage(message, recipients, fCallback) {
|
|
|
|
if (message && message.getEncryptionKeyIds)
|
2015-06-23 05:33:27 +08:00
|
|
|
{
|
2016-08-17 06:01:20 +08:00
|
|
|
const privateKeys = this.findPrivateKeysByEncryptionKeyIds(message.getEncryptionKeyIds(), recipients, true);
|
|
|
|
if (privateKeys && 0 < privateKeys.length)
|
|
|
|
{
|
|
|
|
showScreenPopup(require('View/Popup/MessageOpenPgp'), [(decryptedKey) => {
|
2015-06-23 05:33:27 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
if (decryptedKey)
|
|
|
|
{
|
|
|
|
message.decrypt(decryptedKey).then((decryptedMessage) => {
|
|
|
|
let privateKey = null;
|
|
|
|
if (decryptedMessage)
|
2015-06-23 05:33:27 +08:00
|
|
|
{
|
2016-08-17 06:01:20 +08:00
|
|
|
privateKey = this.findPrivateKeyByHex(decryptedKey.primaryKey.keyid.toHex());
|
|
|
|
if (privateKey)
|
|
|
|
{
|
|
|
|
this.verifyMessage(decryptedMessage, (oValidKey, aSigningKeyIds) => {
|
|
|
|
fCallback(privateKey, decryptedMessage, oValidKey || null, aSigningKeyIds || null);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fCallback(privateKey, decryptedMessage);
|
|
|
|
}
|
2015-06-23 05:33:27 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-08-17 06:01:20 +08:00
|
|
|
fCallback(privateKey, decryptedMessage);
|
2015-06-23 05:33:27 +08:00
|
|
|
}
|
2016-05-05 08:14:38 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
}, () => {
|
|
|
|
fCallback(null, null);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-09-18 03:48:52 +08:00
|
|
|
fCallback(null, null);
|
2016-08-17 06:01:20 +08:00
|
|
|
}
|
2015-06-23 05:33:27 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
}, privateKeys]);
|
2015-06-23 05:33:27 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
return false;
|
|
|
|
}
|
2015-06-23 05:33:27 +08:00
|
|
|
}
|
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
fCallback(null, null);
|
2015-06-23 05:33:27 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
return false;
|
2016-06-30 08:02:45 +08:00
|
|
|
}
|
2016-05-24 01:33:01 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
verifyMessage(message, fCallback) {
|
|
|
|
if (message && message.getSigningKeyIds)
|
2015-06-23 05:33:27 +08:00
|
|
|
{
|
2016-08-17 06:01:20 +08:00
|
|
|
const signingKeyIds = message.getSigningKeyIds();
|
|
|
|
if (signingKeyIds && 0 < signingKeyIds.length)
|
2016-06-30 08:02:45 +08:00
|
|
|
{
|
2016-08-17 06:01:20 +08:00
|
|
|
const publicKeys = this.findPublicKeysBySigningKeyIds(signingKeyIds);
|
|
|
|
if (publicKeys && 0 < publicKeys.length)
|
2015-06-23 05:33:27 +08:00
|
|
|
{
|
2016-08-17 06:01:20 +08:00
|
|
|
try
|
|
|
|
{
|
|
|
|
const
|
|
|
|
result = message.verify(publicKeys),
|
|
|
|
valid = _.find(_.isArray(result) ? result : [], (item) => (item && item.valid && item.keyid));
|
2015-06-23 05:33:27 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
if (valid && valid.keyid && valid.keyid && valid.keyid.toHex)
|
|
|
|
{
|
|
|
|
fCallback(this.findPublicKeyByHex(valid.keyid.toHex()));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (e)
|
2015-10-14 00:36:43 +08:00
|
|
|
{
|
2016-08-17 06:01:20 +08:00
|
|
|
log(e);
|
2015-10-14 00:36:43 +08:00
|
|
|
}
|
2015-06-23 05:33:27 +08:00
|
|
|
}
|
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
fCallback(null, signingKeyIds);
|
|
|
|
return false;
|
|
|
|
}
|
2015-06-23 05:33:27 +08:00
|
|
|
}
|
2016-05-05 08:14:38 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
fCallback(null);
|
|
|
|
return false;
|
2016-06-30 08:02:45 +08:00
|
|
|
}
|
2016-05-05 08:14:38 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
controlsHelper(dom, verControl, success, title, text) {
|
|
|
|
if (success)
|
2016-06-30 08:02:45 +08:00
|
|
|
{
|
2016-08-17 06:01:20 +08:00
|
|
|
dom.removeClass('error').addClass('success').attr('title', title);
|
|
|
|
verControl.removeClass('error').addClass('success').attr('title', title);
|
2016-06-30 08:02:45 +08:00
|
|
|
}
|
2016-08-17 06:01:20 +08:00
|
|
|
else
|
2016-06-30 08:02:45 +08:00
|
|
|
{
|
2016-08-17 06:01:20 +08:00
|
|
|
dom.removeClass('success').addClass('error').attr('title', title);
|
|
|
|
verControl.removeClass('success').addClass('error').attr('title', title);
|
2016-06-30 08:02:45 +08:00
|
|
|
}
|
2016-08-17 06:01:20 +08:00
|
|
|
|
|
|
|
if (!isUnd(text))
|
2016-06-30 08:02:45 +08:00
|
|
|
{
|
2016-08-17 06:01:20 +08:00
|
|
|
dom.text(trim(text));
|
2016-06-30 08:02:45 +08:00
|
|
|
}
|
2016-08-17 06:01:20 +08:00
|
|
|
}
|
2016-05-05 08:14:38 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
static domControlEncryptedClickHelper(store, dom, armoredMessage, recipients) {
|
2016-05-05 08:14:38 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
return function() {
|
|
|
|
|
|
|
|
let message = null;
|
|
|
|
const $this = $(this); // eslint-disable-line no-invalid-this
|
|
|
|
|
|
|
|
if ($this.hasClass('success'))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
message = store.openpgp.message.readArmored(armoredMessage);
|
|
|
|
}
|
|
|
|
catch (e)
|
|
|
|
{
|
|
|
|
log(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (message && message.getText && message.verify && message.decrypt)
|
|
|
|
{
|
|
|
|
store.decryptMessage(message, recipients, (validPrivateKey, decryptedMessage, validPublicKey, signingKeyIds) => {
|
|
|
|
if (decryptedMessage)
|
2016-05-05 08:14:38 +08:00
|
|
|
{
|
2016-08-17 06:01:20 +08:00
|
|
|
if (validPublicKey)
|
|
|
|
{
|
|
|
|
store.controlsHelper(dom, $this, true, i18n('PGP_NOTIFICATIONS/GOOD_SIGNATURE', {
|
|
|
|
'USER': validPublicKey.user + ' (' + validPublicKey.id + ')'
|
|
|
|
}), decryptedMessage.getText());
|
|
|
|
}
|
|
|
|
else if (validPrivateKey)
|
|
|
|
{
|
|
|
|
const
|
|
|
|
keyIds = isNonEmptyArray(signingKeyIds) ? signingKeyIds : null,
|
|
|
|
additional = keyIds ? _.compact(_.map(keyIds,
|
|
|
|
(item) => (item && item.toHex ? item.toHex() : null))).join(', ') : '';
|
|
|
|
|
|
|
|
store.controlsHelper(dom, $this, false, i18n('PGP_NOTIFICATIONS/UNVERIFIRED_SIGNATURE') +
|
|
|
|
(additional ? ' (' + additional + ')' : ''), decryptedMessage.getText());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
store.controlsHelper(dom, $this, false, i18n('PGP_NOTIFICATIONS/DECRYPTION_ERROR'));
|
|
|
|
}
|
2016-05-05 08:14:38 +08:00
|
|
|
}
|
2016-06-30 08:02:45 +08:00
|
|
|
else
|
|
|
|
{
|
2016-08-17 06:01:20 +08:00
|
|
|
store.controlsHelper(dom, $this, false, i18n('PGP_NOTIFICATIONS/DECRYPTION_ERROR'));
|
2016-06-30 08:02:45 +08:00
|
|
|
}
|
2016-08-17 06:01:20 +08:00
|
|
|
});
|
2016-05-05 08:14:38 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
store.controlsHelper(dom, $this, false, i18n('PGP_NOTIFICATIONS/DECRYPTION_ERROR'));
|
2016-05-05 08:14:38 +08:00
|
|
|
return false;
|
2016-08-17 06:01:20 +08:00
|
|
|
};
|
|
|
|
}
|
2016-06-30 08:02:45 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
static domControlSignedClickHelper(store, dom, armoredMessage) {
|
2016-05-05 08:14:38 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
return function() {
|
2015-06-23 05:33:27 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
let message = null;
|
|
|
|
const $this = $(this); // eslint-disable-line no-invalid-this
|
2016-07-16 05:29:42 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
if ($this.hasClass('success') || $this.hasClass('error'))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2015-06-23 05:33:27 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
try
|
|
|
|
{
|
|
|
|
message = store.openpgp.cleartext.readArmored(armoredMessage);
|
|
|
|
}
|
|
|
|
catch (e)
|
|
|
|
{
|
|
|
|
log(e);
|
|
|
|
}
|
2015-06-23 05:33:27 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
if (message && message.getText && message.verify)
|
|
|
|
{
|
|
|
|
store.verifyMessage(message, (validKey, signingKeyIds) => {
|
|
|
|
if (validKey)
|
|
|
|
{
|
|
|
|
store.controlsHelper(dom, $this, true, i18n('PGP_NOTIFICATIONS/GOOD_SIGNATURE', {
|
|
|
|
'USER': validKey.user + ' (' + validKey.id + ')'
|
|
|
|
}), message.getText());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
const
|
|
|
|
keyIds = isNonEmptyArray(signingKeyIds) ? signingKeyIds : null,
|
|
|
|
additional = keyIds ? _.compact(_.map(keyIds, (item) => (item && item.toHex ? item.toHex() : null))).join(', ') : '';
|
2015-06-23 05:33:27 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
store.controlsHelper(dom, $this, false,
|
|
|
|
i18n('PGP_NOTIFICATIONS/UNVERIFIRED_SIGNATURE') + (additional ? ' (' + additional + ')' : ''));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
store.controlsHelper(dom, $this, false, i18n('PGP_NOTIFICATIONS/DECRYPTION_ERROR'));
|
2016-06-30 08:02:45 +08:00
|
|
|
return false;
|
2016-08-17 06:01:20 +08:00
|
|
|
};
|
|
|
|
}
|
2016-06-30 08:02:45 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
/**
|
|
|
|
* @param {*} dom
|
|
|
|
* @param {MessageModel} rainLoopMessage
|
|
|
|
*/
|
|
|
|
initMessageBodyControls(dom, rainLoopMessage) {
|
|
|
|
if (dom && !dom.hasClass('inited'))
|
2016-06-30 08:02:45 +08:00
|
|
|
{
|
2016-08-17 06:01:20 +08:00
|
|
|
dom.addClass('inited');
|
2015-02-01 23:44:44 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
const
|
|
|
|
encrypted = dom.hasClass('encrypted'),
|
|
|
|
signed = dom.hasClass('signed'),
|
|
|
|
recipients = rainLoopMessage ? rainLoopMessage.getEmails(['from', 'to', 'cc']) : [];
|
2016-06-30 08:02:45 +08:00
|
|
|
|
2016-08-17 06:01:20 +08:00
|
|
|
let verControl = null;
|
|
|
|
|
|
|
|
if (encrypted || signed)
|
2016-06-30 08:02:45 +08:00
|
|
|
{
|
2016-08-17 06:01:20 +08:00
|
|
|
const domText = dom.text();
|
|
|
|
dom.data('openpgp-original', domText);
|
|
|
|
|
|
|
|
if (encrypted)
|
|
|
|
{
|
|
|
|
verControl = $('<div class="b-openpgp-control"><i class="icon-lock"></i></div>')
|
|
|
|
.attr('title', i18n('MESSAGE/PGP_ENCRYPTED_MESSAGE_DESC'))
|
|
|
|
.on('click', PgpUserStore.domControlEncryptedClickHelper(this, dom, domText, recipients));
|
|
|
|
}
|
|
|
|
else if (signed)
|
|
|
|
{
|
|
|
|
verControl = $('<div class="b-openpgp-control"><i class="icon-lock"></i></div>')
|
|
|
|
.attr('title', i18n('MESSAGE/PGP_SIGNED_MESSAGE_DESC'))
|
|
|
|
.on('click', PgpUserStore.domControlSignedClickHelper(this, dom, domText));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (verControl)
|
|
|
|
{
|
|
|
|
dom.before(verControl).before('<div></div>');
|
|
|
|
}
|
2016-06-30 08:02:45 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-08-17 06:01:20 +08:00
|
|
|
}
|
2015-02-01 23:44:44 +08:00
|
|
|
|
2016-09-13 04:50:21 +08:00
|
|
|
export default new PgpUserStore();
|