embark/lib/core/provider.js

123 lines
3.9 KiB
JavaScript
Raw Normal View History

2018-05-10 18:52:30 +00:00
const ProviderEngine = require('web3-provider-engine');
const RpcSubprovider = require('web3-provider-engine/subproviders/rpc.js');
2018-05-10 18:52:51 +00:00
const bip39 = require("bip39");
const hdkey = require('ethereumjs-wallet/hdkey');
const fs = require('./fs');
2018-05-10 18:52:30 +00:00
class Provider {
2018-05-10 18:52:51 +00:00
constructor(options) {
2018-05-11 15:41:54 +00:00
const self = this;
2018-05-10 18:52:51 +00:00
this.web3 = options.web3;
this.accountsConfig = options.accountsConfig;
this.logger = options.logger;
2018-05-10 18:52:30 +00:00
this.engine = new ProviderEngine();
this.engine.addProvider(new RpcSubprovider({
2018-05-10 18:52:51 +00:00
rpcUrl: options.web3Endpoint
2018-05-10 18:52:30 +00:00
}));
2018-05-10 18:52:51 +00:00
if (this.accountsConfig && this.accountsConfig.length) {
this.accounts = [];
2018-05-11 15:41:54 +00:00
this.addresses = [];
2018-05-10 18:52:51 +00:00
this.accountsConfig.forEach(accountConfig => {
const account = this.getAccount(accountConfig);
if (!account) {
return;
}
if (Array.isArray(account)) {
this.accounts = this.accounts.concat(account);
2018-05-11 15:41:54 +00:00
account.forEach(acc => {
this.addresses.push(acc.address);
});
2018-05-10 18:52:51 +00:00
return;
}
2018-05-11 15:41:54 +00:00
this.accounts.push(account);
this.addresses.push(account.address);
2018-05-10 18:52:51 +00:00
});
2018-05-11 15:41:54 +00:00
if (this.accounts.length) {
this.asyncMethods = {
eth_accounts: self.eth_accounts.bind(this)
};
this.syncMethods = {
eth_accounts: self.eth_accounts_sync.bind(this)
};
}
2018-05-10 18:52:51 +00:00
}
2018-05-10 18:52:30 +00:00
this.engine.on('block', function (block) {
console.log('================================');
console.log('BLOCK CHANGED:', '#' + block.number.toString('hex'), '0x' + block.hash.toString('hex'));
console.log('================================');
});
// network connectivity error
this.engine.on('error', function (err) {
// report connectivity errors
console.error(err.stack);
});
this.engine.start();
}
2018-05-10 18:52:51 +00:00
getAccount(accountConfig) {
if (accountConfig.privateKey) {
return this.web3.eth.accounts.privateKeyToAccount(accountConfig.privateKey);
}
if (accountConfig.privateKeyFile) {
let fileContent = fs.readFileSync(fs.dappPath(accountConfig.privateKeyFile)).toString();
return this.web3.eth.accounts.privateKeyToAccount(fileContent.trim());
}
if (accountConfig.mnemonic) {
const hdwallet = hdkey.fromMasterSeed(bip39.mnemonicToSeed(accountConfig.mnemonic.trim()));
const addressIndex = accountConfig.addressIndex || 0;
const numAddresses = accountConfig.numAddresses || 1;
2018-05-11 15:41:54 +00:00
const wallet_hdpath = "m/44'/60'/0'/0/"; // TODO check if this can change
2018-05-10 18:52:51 +00:00
const accounts = [];
2018-05-11 15:41:54 +00:00
for (let i = addressIndex; i < addressIndex + numAddresses; i++) {
2018-05-10 18:52:51 +00:00
const wallet = hdwallet.derivePath(wallet_hdpath + i).getWallet();
accounts.push(this.web3.eth.accounts.privateKeyToAccount(wallet.getPrivateKey()));
}
return accounts;
}
this.logger.warn('Unsupported account configuration: ' + JSON.stringify(accountConfig));
this.logger.warn('Try using one of those: ' +
'{ "privateKey": "your-private-key", "privateKeyFile": "path/to/file/containing/key", "mnemonic": "12 word mnemonic" }');
}
2018-05-11 15:41:54 +00:00
eth_accounts(payload, cb) {
return cb(null, this.addresses);
}
eth_accounts_sync() {
return this.addresses;
}
sendAsync(payload, callback) {
// this.engine.sendAsync.apply(this.engine, arguments);
let method = this.asyncMethods[payload.method];
if (method) {
return method.call(method, payload, (err, result) => {
if (err) {
return callback(err);
}
let response = {'id': payload.id, 'jsonrpc': '2.0', 'result': result};
callback(null, response);
});
}
2018-05-10 18:52:30 +00:00
this.engine.sendAsync.apply(this.engine, arguments);
}
2018-05-11 15:41:54 +00:00
send(payload) {
let method = this.syncMethods[payload.method];
if (method) {
return method.call(method, payload); // TODO check if that makes sense
}
2018-05-10 18:52:30 +00:00
return this.engine.send.apply(this.engine, arguments);
}
}
module.exports = Provider;