2017-03-29 17:07:58 +02:00
|
|
|
// Copyright 2015-2017 Parity Technologies (UK) Ltd.
|
|
|
|
// This file is part of Parity.
|
|
|
|
|
|
|
|
// Parity is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
|
|
|
// Parity is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
2017-07-21 15:46:53 +02:00
|
|
|
import { bytesToHex } from '../../util/format';
|
2017-06-13 11:02:38 +02:00
|
|
|
import { extern, slice } from './ethkey.js';
|
2017-04-03 18:50:11 +02:00
|
|
|
|
|
|
|
const isWorker = typeof self !== 'undefined';
|
2017-03-29 17:07:58 +02:00
|
|
|
|
|
|
|
// Stay compatible between environments
|
2017-04-03 18:50:11 +02:00
|
|
|
if (!isWorker) {
|
2017-03-29 17:07:58 +02:00
|
|
|
const scope = typeof global === 'undefined' ? window : global;
|
|
|
|
|
|
|
|
scope.self = scope;
|
|
|
|
}
|
|
|
|
|
2017-04-03 18:50:11 +02:00
|
|
|
// keythereum should never be used outside of the browser
|
2017-04-06 17:13:53 +02:00
|
|
|
let keythereum = require('keythereum');
|
2017-03-29 17:07:58 +02:00
|
|
|
|
2017-04-03 18:50:11 +02:00
|
|
|
if (isWorker) {
|
|
|
|
keythereum = self.keythereum;
|
|
|
|
}
|
|
|
|
|
|
|
|
function route ({ action, payload }) {
|
|
|
|
if (action in actions) {
|
|
|
|
return actions[action](payload);
|
2017-03-29 17:07:58 +02:00
|
|
|
}
|
|
|
|
|
2017-04-03 18:50:11 +02:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2017-06-13 11:02:38 +02:00
|
|
|
const input = slice(extern._input_ptr(), 1024);
|
|
|
|
const secret = slice(extern._secret_ptr(), 32);
|
|
|
|
const publicKey = slice(extern._public_ptr(), 64);
|
|
|
|
const address = slice(extern._address_ptr(), 20);
|
|
|
|
|
|
|
|
extern._ecpointg();
|
|
|
|
|
2017-04-03 18:50:11 +02:00
|
|
|
const actions = {
|
|
|
|
phraseToWallet (phrase) {
|
2017-06-13 11:02:38 +02:00
|
|
|
const phraseUtf8 = Buffer.from(phrase, 'utf8');
|
2017-04-03 18:50:11 +02:00
|
|
|
|
2017-06-13 11:02:38 +02:00
|
|
|
if (phraseUtf8.length > input.length) {
|
|
|
|
throw new Error('Phrase is too long!');
|
2017-04-03 18:50:11 +02:00
|
|
|
}
|
|
|
|
|
2017-06-13 11:02:38 +02:00
|
|
|
input.set(phraseUtf8);
|
2017-03-29 17:07:58 +02:00
|
|
|
|
2017-06-13 11:02:38 +02:00
|
|
|
extern._brain(phraseUtf8.length);
|
2017-03-29 17:07:58 +02:00
|
|
|
|
2017-06-13 11:02:38 +02:00
|
|
|
const wallet = {
|
|
|
|
secret: bytesToHex(secret),
|
|
|
|
public: bytesToHex(publicKey),
|
|
|
|
address: bytesToHex(address)
|
|
|
|
};
|
2017-04-03 18:50:11 +02:00
|
|
|
|
2017-06-13 11:02:38 +02:00
|
|
|
return wallet;
|
2017-04-03 18:50:11 +02:00
|
|
|
},
|
|
|
|
|
2017-06-13 11:02:38 +02:00
|
|
|
verifySecret (key) {
|
|
|
|
const keyBuf = Buffer.from(key.slice(2), 'hex');
|
|
|
|
|
|
|
|
secret.set(keyBuf);
|
2017-04-03 18:50:11 +02:00
|
|
|
|
2017-06-13 11:02:38 +02:00
|
|
|
return extern._verify_secret();
|
2017-04-03 18:50:11 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
createKeyObject ({ key, password }) {
|
|
|
|
key = Buffer.from(key);
|
|
|
|
password = Buffer.from(password);
|
|
|
|
|
|
|
|
const iv = keythereum.crypto.randomBytes(16);
|
|
|
|
const salt = keythereum.crypto.randomBytes(32);
|
|
|
|
const keyObject = keythereum.dump(password, key, salt, iv);
|
2017-03-29 17:07:58 +02:00
|
|
|
|
2017-04-03 18:50:11 +02:00
|
|
|
return JSON.stringify(keyObject);
|
|
|
|
},
|
2017-03-29 17:07:58 +02:00
|
|
|
|
2017-04-03 18:50:11 +02:00
|
|
|
decryptPrivateKey ({ keyObject, password }) {
|
|
|
|
password = Buffer.from(password);
|
|
|
|
|
|
|
|
try {
|
|
|
|
const key = keythereum.recover(password, keyObject);
|
|
|
|
|
|
|
|
// Convert to array to safely send from the worker
|
|
|
|
return Array.from(key);
|
|
|
|
} catch (e) {
|
|
|
|
return null;
|
2017-03-29 17:07:58 +02:00
|
|
|
}
|
|
|
|
}
|
2017-04-03 18:50:11 +02:00
|
|
|
};
|
|
|
|
|
2017-03-29 17:07:58 +02:00
|
|
|
self.onmessage = function ({ data }) {
|
2017-04-06 17:13:53 +02:00
|
|
|
try {
|
|
|
|
const result = route(data);
|
2017-03-29 17:07:58 +02:00
|
|
|
|
2017-04-06 17:13:53 +02:00
|
|
|
postMessage([null, result]);
|
|
|
|
} catch (err) {
|
2017-06-13 11:02:38 +02:00
|
|
|
console.error(err);
|
|
|
|
postMessage([err.toString(), null]);
|
2017-04-06 17:13:53 +02:00
|
|
|
}
|
2017-03-29 17:07:58 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
// Emulate a web worker in Node.js
|
|
|
|
class KeyWorker {
|
|
|
|
postMessage (data) {
|
|
|
|
// Force async
|
|
|
|
setTimeout(() => {
|
2017-04-06 17:13:53 +02:00
|
|
|
try {
|
|
|
|
const result = route(data);
|
2017-03-29 17:07:58 +02:00
|
|
|
|
2017-04-06 17:13:53 +02:00
|
|
|
this.onmessage({ data: [null, result] });
|
|
|
|
} catch (err) {
|
|
|
|
this.onmessage({ data: [err, null] });
|
|
|
|
}
|
2017-03-29 17:07:58 +02:00
|
|
|
}, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
onmessage (event) {
|
|
|
|
// no-op to be overriden
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (exports != null) {
|
|
|
|
exports.KeyWorker = KeyWorker;
|
|
|
|
}
|