embark/packages/embark-utils/src/accountParser.js

155 lines
5.7 KiB
JavaScript
Raw Normal View History

import { __ } from 'i18n';
2018-05-17 17:48:39 +00:00
const bip39 = require("bip39");
const hdkey = require('ethereumjs-wallet/hdkey');
const ethereumjsWallet = require('ethereumjs-wallet');
const fs = require('fs');
import {getHexBalanceFromString} from './web3Utils';
const {utils} = require('web3');
2018-05-17 17:48:39 +00:00
const path = require('path');
2018-05-17 17:48:39 +00:00
class AccountParser {
static parseAccountsConfig(accountsConfig, web3, dappPath, logger, nodeAccounts) {
2018-05-17 17:48:39 +00:00
let accounts = [];
if (accountsConfig && accountsConfig.length) {
accountsConfig.forEach(accountConfig => {
let account = AccountParser.getAccount(accountConfig, web3, dappPath, logger, nodeAccounts);
2018-05-17 17:48:39 +00:00
if (!account) {
return;
}
if (Array.isArray(account)) {
accounts = accounts.concat(account);
return;
}
accounts.push(account);
});
}
return accounts;
}
/*eslint complexity: ["error", 30]*/
static getAccount(accountConfig, web3, dappPath, logger = console, nodeAccounts) {
const returnAddress = web3 === false;
2018-06-07 14:53:20 +00:00
let hexBalance = null;
if (accountConfig.balance && web3) {
2018-07-16 16:48:32 +00:00
hexBalance = getHexBalanceFromString(accountConfig.balance, web3);
2018-06-07 14:53:20 +00:00
}
if (accountConfig.privateKey === 'random') {
if (!web3) {
logger.warn('Cannot use random in this context');
return null;
}
let randomAccount = web3.eth.accounts.create();
accountConfig.privateKey = randomAccount.privateKey;
}
2018-10-04 19:18:00 +00:00
if (accountConfig.nodeAccounts) {
if (!nodeAccounts && !returnAddress) {
2018-10-04 19:18:00 +00:00
logger.warn('Cannot use nodeAccounts in this context');
return null;
}
if (!nodeAccounts || !nodeAccounts.length) {
2018-10-04 19:18:00 +00:00
return null;
}
return nodeAccounts.map(account => {
return (typeof account === 'string') ? { address: account } : account;
2018-10-04 19:18:00 +00:00
});
}
2018-05-17 17:48:39 +00:00
if (accountConfig.privateKey) {
if (!accountConfig.privateKey.startsWith('0x')) {
accountConfig.privateKey = '0x' + accountConfig.privateKey;
}
if (!utils.isHexStrict(accountConfig.privateKey)) {
2018-05-18 17:27:01 +00:00
logger.warn(`Private key ending with ${accountConfig.privateKey.substr(accountConfig.privateKey.length - 5)} is not a HEX string`);
return null;
}
const key = Buffer.from(accountConfig.privateKey.substr(2), 'hex');
if (returnAddress) {
return ethereumjsWallet.fromPrivateKey(key).getChecksumAddressString();
}
return Object.assign(web3.eth.accounts.privateKeyToAccount(key), {hexBalance});
} else if (Object.hasOwnProperty('privateKey')) {
logger.error(__('accounts error: privateKey field is specified but its value is undefined'));
2018-05-17 17:48:39 +00:00
}
2018-05-17 17:48:39 +00:00
if (accountConfig.privateKeyFile) {
let privateKeyFile = path.resolve(dappPath, accountConfig.privateKeyFile);
let fileContent = fs.readFileSync(privateKeyFile).toString();
if (accountConfig.password) {
try {
fileContent = JSON.parse(fileContent);
if (!ethereumjsWallet['fromV' + fileContent.version]) {
logger.error(`Key file ${accountConfig.privateKeyFile} is not a valid keystore file`);
return null;
}
const wallet = ethereumjsWallet['fromV' + fileContent.version](fileContent, accountConfig.password);
if (returnAddress) {
return wallet.getChecksumAddressString();
}
return Object.assign(web3.eth.accounts.privateKeyToAccount('0x' + wallet.getPrivateKey().toString('hex')), {hexBalance});
} catch (e) {
logger.error('Private key file is not a keystore JSON file but a password was provided');
logger.error(e.message || e);
return null;
}
}
2018-05-17 17:48:39 +00:00
fileContent = fileContent.trim().split(/[,;]/);
2018-05-18 17:27:01 +00:00
return fileContent.map((key, index) => {
2018-05-17 17:48:39 +00:00
if (!key.startsWith('0x')) {
key = '0x' + key;
}
if (!utils.isHexStrict(key)) {
2018-05-18 17:27:01 +00:00
logger.warn(`Private key is not a HEX string in file ${accountConfig.privateKeyFile} at index ${index}`);
return null;
}
key = Buffer.from(key.substr(2), 'hex');
if (returnAddress) {
return ethereumjsWallet.fromPrivateKey(key).getChecksumAddressString();
}
2018-06-07 14:53:20 +00:00
return Object.assign(web3.eth.accounts.privateKeyToAccount(key), {hexBalance});
2018-05-17 17:48:39 +00:00
});
} else if (Object.hasOwnProperty('privateKeyFile')) {
logger.error(__('accounts error: privateKeyFile field is specified but its value is undefined'));
2018-05-17 17:48:39 +00:00
}
2018-05-17 17:48:39 +00:00
if (accountConfig.mnemonic) {
const hdwallet = hdkey.fromMasterSeed(bip39.mnemonicToSeed(accountConfig.mnemonic.trim()));
const addressIndex = accountConfig.addressIndex || 0;
const numAddresses = accountConfig.numAddresses || 1;
const wallet_hdpath = accountConfig.hdpath || "m/44'/60'/0'/0/";
const accounts = [];
for (let i = addressIndex; i < addressIndex + numAddresses; i++) {
const wallet = hdwallet.derivePath(wallet_hdpath + i).getWallet();
if (returnAddress) {
accounts.push(wallet.getAddressString());
} else {
accounts.push(Object.assign(web3.eth.accounts.privateKeyToAccount('0x' + wallet.getPrivateKey().toString('hex')), {hexBalance}));
}
2018-05-17 17:48:39 +00:00
}
return accounts;
} else if (Object.hasOwnProperty('mnemonic')) {
logger.error(__('accounts error: mnemonic field is specified but its value is undefined'));
2018-05-17 17:48:39 +00:00
}
if (accountConfig.secretKey) {
// Ignore simulator configs
return null;
}
logger.warn(__('Unsupported account configuration: %s' ,JSON.stringify(accountConfig)));
logger.warn(__('Check the docs at %s', 'https://embark.status.im/docs/contracts_deployment.html#Using-accounts-in-a-wallet'.underline));
2018-05-17 17:48:39 +00:00
return null;
}
}
module.exports = AccountParser;