const ProviderEngine = require('embark-web3-provider-engine'); const RpcSubprovider = require('embark-web3-provider-engine/subproviders/rpc'); const WsSubprovider = require('embark-web3-provider-engine/subproviders/websocket'); const async = require('async'); const AccountParser = require('./accountParser'); const fundAccount = require('./fundAccount'); const NO_ACCOUNTS = 'noAccounts'; class Provider { constructor(options) { this.web3 = options.web3; this.accountsConfig = options.accountsConfig; this.blockchainConfig = options.blockchainConfig; this.type = options.type; this.web3Endpoint = options.web3Endpoint; this.logger = options.logger; this.isDev = options.isDev; this.engine = new ProviderEngine(); this.asyncMethods = {}; } startWeb3Provider(callback) { const self = this; if (this.type === 'rpc') { self.engine.addProvider(new RpcSubprovider({ rpcUrl: self.web3Endpoint })); } else if (this.type === 'ws') { self.engine.addProvider(new WsSubprovider({ rpcUrl: self.web3Endpoint, origin: this.blockchainConfig.wsOrigins.split(',')[0] })); } else { return callback(__("contracts config error: unknown deployment type %s", this.type)); } // network connectivity error self.engine.on('error', (err) => { // report connectivity errors self.logger.error(err.stack); }); self.engine.start(); self.web3.setProvider(self); self.accounts = AccountParser.parseAccountsConfig(self.accountsConfig, self.web3, self.logger); self.addresses = []; async.waterfall([ function populateWeb3Wallet(next) { if (!self.accounts.length) { return next(NO_ACCOUNTS); } self.accounts.forEach(account => { self.addresses.push(account.address); self.web3.eth.accounts.wallet.add(account); }); self.asyncMethods = { eth_accounts: self.eth_accounts.bind(self) }; next(); } ], function (err) { if (err && err !== NO_ACCOUNTS) { self.logger.error((err)); } callback(); }); } fundAccounts(callback) { const self = this; if (!self.accounts.length) { return callback(); } if (!self.isDev) { return callback(); } let nonce = 0; async.each(self.accounts, (account, eachCb) => { console.dir('[contracts/provider]: nonce is ' + nonce); fundAccount(self.web3, account.address, account.hexBalance, nonce++, (err) => { eachCb(err); }); }, callback); } stop() { this.engine.stop(); } eth_accounts(payload, cb) { return cb(null, this.addresses); } sendAsync(payload, callback) { 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); }); } this.engine.sendAsync.apply(this.engine, arguments); } send() { return this.engine.send.apply(this.engine, arguments); } } module.exports = Provider;