-
Notifications
You must be signed in to change notification settings - Fork 13
/
Copy pathhelpers.js
100 lines (92 loc) · 3.78 KB
/
helpers.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
const bip39 = require('bip39')
const hdkey = require('ethereumjs-wallet/hdkey')
const ProviderEngine = require('web3-provider-engine')
const FiltersSubprovider = require('web3-provider-engine/subproviders/filters.js')
const HookedSubprovider = require('web3-provider-engine/subproviders/hooked-wallet.js')
const RpcSubprovider = require('web3-provider-engine/subproviders/rpc.js')
// const ProviderSubprovider = require('web3-provider-engine/subproviders/provider.js')
// const Web3 = require('web3')
const Transaction = require('ethereumjs-tx')
const ethUtil = require('ethereumjs-util')
/* eslint-disable */
function HDWalletProvider (
mnemonic,
provider_url,
address_index = 0,
num_addresses = 1,
wallet_hdpath = "m/44'/889'/0'/0/"
) {
this.mnemonic = mnemonic
this.hdwallet = hdkey.fromMasterSeed(bip39.mnemonicToSeedSync(mnemonic))
this.wallet_hdpath = wallet_hdpath
this.wallets = {}
this.addresses = []
for (let i = address_index; i < address_index + num_addresses; i++) {
const wallet = this.hdwallet.derivePath(this.wallet_hdpath + '/' + i).getWallet()
const addr = '0x' + wallet.getAddress().toString('hex')
this.addresses.push(addr)
this.wallets[addr] = wallet
}
const tmp_accounts = this.addresses
const tmp_wallets = this.wallets
this.engine = new ProviderEngine()
this.engine.addProvider(new HookedSubprovider({
getAccounts: function (cb) { cb(null, tmp_accounts) },
getPrivateKey: function (address, cb) {
if (!tmp_wallets[address]) { return cb('Account not found') }
else { cb(null, tmp_wallets[address].getPrivateKey().toString('hex')) }
},
signTransaction: function (txParams, cb) {
let pkey
if (tmp_wallets[txParams.from]) { pkey = tmp_wallets[txParams.from].getPrivateKey() }
else { cb('Account not found') }
const tx = new Transaction(txParams)
tx.sign(pkey)
const rawTx = '0x' + tx.serialize().toString('hex')
cb(null, rawTx)
},
signMessage(message, cb) {
const dataIfExists = message.data;
if (!dataIfExists) {
cb('No data to sign');
}
if (!tmp_wallets[message.from]) {
cb('Account not found');
}
let pkey = tmp_wallets[message.from].getPrivateKey();
const dataBuff = ethUtil.toBuffer(dataIfExists);
const msgHashBuff = ethUtil.hashPersonalMessage(dataBuff);
const sig = ethUtil.ecsign(msgHashBuff, pkey);
const rpcSig = ethUtil.toRpcSig(sig.v, sig.r, sig.s);
cb(null, rpcSig);
}
}))
this.engine.addProvider(new FiltersSubprovider())
// Web3.providers.HttpProvider.prototype.sendAsync = Web3.providers.HttpProvider.prototype.send
// this.engine.addProvider(new ProviderSubprovider(new Web3.providers.HttpProvider(provider_url)))
// if (typeof provider === 'string') {
// this.engine.addProvider(new ProviderSubprovider(new Web3.providers.HttpProvider(provider_url)));
// } else {
// this.engine.addProvider(new ProviderSubprovider(provider_url));
// }
this.engine.addProvider(new RpcSubprovider({
rpcUrl: provider_url,
}))
this.engine.start() // Required by the provider engine.
}
HDWalletProvider.prototype.sendAsync = function () {
this.engine.sendAsync.apply(this.engine, arguments)
}
HDWalletProvider.prototype.send = function () {
return this.engine.send.apply(this.engine, arguments)
}
// returns the address of the given address_index, first checking the cache
HDWalletProvider.prototype.getAddress = function (idx) {
if (!idx) { return this.addresses[0] }
else { return this.addresses[idx] }
}
// returns the addresses cache
HDWalletProvider.prototype.getAddresses = function () {
return this.addresses
}
module.exports = { HDWalletProvider }