2016-05-26 02:27:49 +02:00
|
|
|
const urlUtil = require('url')
|
2016-01-15 11:03:42 +01:00
|
|
|
const Dnode = require('dnode')
|
2016-02-11 02:44:46 +01:00
|
|
|
const eos = require('end-of-stream')
|
2016-03-10 22:04:45 +01:00
|
|
|
const combineStreams = require('pumpify')
|
2016-02-11 02:44:46 +01:00
|
|
|
const extend = require('xtend')
|
|
|
|
const EthStore = require('eth-store')
|
2016-04-26 20:36:05 +02:00
|
|
|
const MetaMaskProvider = require('web3-provider-engine/zero.js')
|
2016-05-23 03:02:27 +02:00
|
|
|
const ObjectMultiplex = require('./lib/obj-multiplex')
|
|
|
|
const PortStream = require('./lib/port-stream.js')
|
2016-02-11 02:44:46 +01:00
|
|
|
const IdentityStore = require('./lib/idStore')
|
2016-05-24 23:59:33 +02:00
|
|
|
const createUnlockRequestNotification = require('./lib/notifications.js').createUnlockRequestNotification
|
2016-04-28 23:16:24 +02:00
|
|
|
const createTxNotification = require('./lib/notifications.js').createTxNotification
|
|
|
|
const createMsgNotification = require('./lib/notifications.js').createMsgNotification
|
Made configuration migrateable
Abstract all configuration data into a singleton called `configManager`, who is responsible for reading and writing to the persisted storage (localStorage, in our case).
Uses my new module [pojo-migrator](https://www.npmjs.com/package/pojo-migrator), and wraps it with the `ConfigManager` class, which we can hang any state setting or getting methods we need.
By keeping all the persisted state in one place, we can stabilize its outward-facing API, making the interactions increasingly atomic, which will allow us to add features that require restructuring the persisted data in the long term without having to rewrite UI or even `background.js` code.
All the restructuring and data-type management is kept in one neat little place.
This should make it very easy to add new configuration options like user-configured providers, per-domain vaults, and more!
I know this doesn't seem like a big user-facing feature, but we have a big laundry list of features that I think this will really help streamline.
2016-03-31 04:15:49 +02:00
|
|
|
const configManager = require('./lib/config-manager-singleton')
|
2016-05-03 23:32:22 +02:00
|
|
|
const messageManager = require('./lib/message-manager')
|
2016-04-15 21:12:04 +02:00
|
|
|
const setupMultiplex = require('./lib/stream-utils.js').setupMultiplex
|
|
|
|
const HostStore = require('./lib/remote-store.js').HostStore
|
2016-04-28 00:29:10 +02:00
|
|
|
const Web3 = require('web3')
|
2015-12-19 07:05:16 +01:00
|
|
|
|
2016-02-11 02:44:46 +01:00
|
|
|
//
|
|
|
|
// connect to other contexts
|
|
|
|
//
|
2016-01-17 01:22:54 +01:00
|
|
|
|
2015-08-02 08:36:03 +02:00
|
|
|
chrome.runtime.onConnect.addListener(connectRemote)
|
2015-12-19 07:05:16 +01:00
|
|
|
function connectRemote(remotePort){
|
2016-01-15 11:03:42 +01:00
|
|
|
var isMetaMaskInternalProcess = (remotePort.name === 'popup')
|
2016-03-10 03:33:30 +01:00
|
|
|
var portStream = new PortStream(remotePort)
|
2016-01-15 11:03:42 +01:00
|
|
|
if (isMetaMaskInternalProcess) {
|
|
|
|
// communication with popup
|
2016-05-26 02:27:49 +02:00
|
|
|
setupTrustedCommunication(portStream, 'MetaMask')
|
2016-01-15 11:03:42 +01:00
|
|
|
} else {
|
|
|
|
// communication with page
|
2016-05-26 02:27:49 +02:00
|
|
|
var originDomain = urlUtil.parse(remotePort.sender.url).hostname
|
|
|
|
setupUntrustedCommunication(portStream, originDomain)
|
2016-01-15 11:03:42 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-26 02:27:49 +02:00
|
|
|
function setupUntrustedCommunication(connectionStream, originDomain){
|
2016-04-15 21:12:04 +02:00
|
|
|
// setup multiplexing
|
|
|
|
var mx = setupMultiplex(connectionStream)
|
|
|
|
// connect features
|
2016-05-26 02:27:49 +02:00
|
|
|
setupProviderConnection(mx.createStream('provider'), originDomain)
|
2016-04-15 21:12:04 +02:00
|
|
|
setupPublicConfig(mx.createStream('publicConfig'))
|
|
|
|
}
|
|
|
|
|
2016-05-26 02:27:49 +02:00
|
|
|
function setupTrustedCommunication(connectionStream, originDomain){
|
2016-04-15 21:12:04 +02:00
|
|
|
// setup multiplexing
|
|
|
|
var mx = setupMultiplex(connectionStream)
|
|
|
|
// connect features
|
|
|
|
setupControllerConnection(mx.createStream('controller'))
|
2016-05-26 02:27:49 +02:00
|
|
|
setupProviderConnection(mx.createStream('provider'), originDomain)
|
2016-02-11 02:44:46 +01:00
|
|
|
}
|
2016-01-17 10:27:25 +01:00
|
|
|
|
2016-02-11 02:44:46 +01:00
|
|
|
//
|
|
|
|
// state and network
|
|
|
|
//
|
2016-01-17 10:27:25 +01:00
|
|
|
|
2016-04-01 01:32:35 +02:00
|
|
|
var providerConfig = configManager.getProvider()
|
2016-02-11 02:44:46 +01:00
|
|
|
var idStore = new IdentityStore()
|
2016-04-28 03:04:33 +02:00
|
|
|
|
2016-04-01 01:32:35 +02:00
|
|
|
var providerOpts = {
|
Made configuration migrateable
Abstract all configuration data into a singleton called `configManager`, who is responsible for reading and writing to the persisted storage (localStorage, in our case).
Uses my new module [pojo-migrator](https://www.npmjs.com/package/pojo-migrator), and wraps it with the `ConfigManager` class, which we can hang any state setting or getting methods we need.
By keeping all the persisted state in one place, we can stabilize its outward-facing API, making the interactions increasingly atomic, which will allow us to add features that require restructuring the persisted data in the long term without having to rewrite UI or even `background.js` code.
All the restructuring and data-type management is kept in one neat little place.
This should make it very easy to add new configuration options like user-configured providers, per-domain vaults, and more!
I know this doesn't seem like a big user-facing feature, but we have a big laundry list of features that I think this will really help streamline.
2016-03-31 04:15:49 +02:00
|
|
|
rpcUrl: configManager.getCurrentRpcAddress(),
|
2016-04-28 23:16:24 +02:00
|
|
|
// account mgmt
|
2016-02-11 02:44:46 +01:00
|
|
|
getAccounts: function(cb){
|
|
|
|
var selectedAddress = idStore.getSelectedAddress()
|
|
|
|
var result = selectedAddress ? [selectedAddress] : []
|
|
|
|
cb(null, result)
|
|
|
|
},
|
2016-04-28 23:16:24 +02:00
|
|
|
// tx signing
|
2016-05-26 02:27:49 +02:00
|
|
|
approveTransaction: newUnsignedTransaction,
|
2016-02-20 21:13:18 +01:00
|
|
|
signTransaction: idStore.signTransaction.bind(idStore),
|
2016-04-28 23:16:24 +02:00
|
|
|
// msg signing
|
2016-05-26 02:27:49 +02:00
|
|
|
approveMessage: newUnsignedMessage,
|
2016-04-28 23:16:24 +02:00
|
|
|
signMessage: idStore.signMessage.bind(idStore),
|
2016-04-01 01:32:35 +02:00
|
|
|
}
|
|
|
|
var provider = MetaMaskProvider(providerOpts)
|
2016-04-28 00:29:10 +02:00
|
|
|
var web3 = new Web3(provider)
|
2016-04-28 03:04:33 +02:00
|
|
|
idStore.web3 = web3
|
2016-06-04 00:18:20 +02:00
|
|
|
idStore.getNetwork()
|
2016-03-03 01:58:47 +01:00
|
|
|
|
|
|
|
// log new blocks
|
2016-04-01 01:32:35 +02:00
|
|
|
provider.on('block', function(block){
|
2016-03-03 01:58:47 +01:00
|
|
|
console.log('BLOCK CHANGED:', '#'+block.number.toString('hex'), '0x'+block.hash.toString('hex'))
|
2016-06-04 00:18:20 +02:00
|
|
|
|
|
|
|
// Check network when restoring connectivity:
|
|
|
|
if (idStore._currentState.network === 'loading') {
|
|
|
|
idStore.getNetwork()
|
|
|
|
}
|
2016-03-03 01:58:47 +01:00
|
|
|
})
|
|
|
|
|
2016-06-04 00:18:20 +02:00
|
|
|
provider.on('error', idStore.getNetwork.bind(idStore))
|
|
|
|
|
2016-04-01 01:32:35 +02:00
|
|
|
var ethStore = new EthStore(provider)
|
2016-02-11 02:44:46 +01:00
|
|
|
idStore.setStore(ethStore)
|
2016-01-15 11:03:42 +01:00
|
|
|
|
2016-02-11 02:44:46 +01:00
|
|
|
function getState(){
|
2016-03-08 23:33:01 +01:00
|
|
|
var state = extend(
|
|
|
|
ethStore.getState(),
|
|
|
|
idStore.getState(),
|
Made configuration migrateable
Abstract all configuration data into a singleton called `configManager`, who is responsible for reading and writing to the persisted storage (localStorage, in our case).
Uses my new module [pojo-migrator](https://www.npmjs.com/package/pojo-migrator), and wraps it with the `ConfigManager` class, which we can hang any state setting or getting methods we need.
By keeping all the persisted state in one place, we can stabilize its outward-facing API, making the interactions increasingly atomic, which will allow us to add features that require restructuring the persisted data in the long term without having to rewrite UI or even `background.js` code.
All the restructuring and data-type management is kept in one neat little place.
This should make it very easy to add new configuration options like user-configured providers, per-domain vaults, and more!
I know this doesn't seem like a big user-facing feature, but we have a big laundry list of features that I think this will really help streamline.
2016-03-31 04:15:49 +02:00
|
|
|
configManager.getConfig()
|
2016-03-08 23:33:01 +01:00
|
|
|
)
|
2016-02-11 02:44:46 +01:00
|
|
|
return state
|
2015-08-02 08:36:03 +02:00
|
|
|
}
|
|
|
|
|
2016-04-15 22:04:17 +02:00
|
|
|
//
|
|
|
|
// public store
|
|
|
|
//
|
|
|
|
|
|
|
|
// get init state
|
|
|
|
var initPublicState = extend(
|
|
|
|
idStoreToPublic(idStore.getState()),
|
|
|
|
configToPublic(configManager.getConfig())
|
|
|
|
)
|
|
|
|
|
|
|
|
var publicConfigStore = new HostStore(initPublicState)
|
|
|
|
|
|
|
|
// subscribe to changes
|
|
|
|
configManager.subscribe(function(state){
|
|
|
|
storeSetFromObj(publicConfigStore, configToPublic(state))
|
|
|
|
})
|
|
|
|
idStore.on('update', function(state){
|
|
|
|
storeSetFromObj(publicConfigStore, idStoreToPublic(state))
|
|
|
|
})
|
|
|
|
|
|
|
|
// idStore substate
|
|
|
|
function idStoreToPublic(state){
|
|
|
|
return {
|
|
|
|
selectedAddress: state.selectedAddress,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// config substate
|
|
|
|
function configToPublic(state){
|
|
|
|
return {
|
|
|
|
provider: state.provider,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// dump obj into store
|
|
|
|
function storeSetFromObj(store, obj){
|
|
|
|
Object.keys(obj).forEach(function(key){
|
|
|
|
store.set(key, obj[key])
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-02-11 02:44:46 +01:00
|
|
|
//
|
2016-04-15 21:12:04 +02:00
|
|
|
// remote features
|
2016-02-11 02:44:46 +01:00
|
|
|
//
|
|
|
|
|
2016-04-15 21:12:04 +02:00
|
|
|
function setupPublicConfig(stream){
|
|
|
|
var storeStream = publicConfigStore.createStream()
|
|
|
|
stream.pipe(storeStream).pipe(stream)
|
|
|
|
}
|
|
|
|
|
2016-05-26 02:27:49 +02:00
|
|
|
function setupProviderConnection(stream, originDomain){
|
2016-06-04 00:18:20 +02:00
|
|
|
|
2016-05-26 02:27:49 +02:00
|
|
|
stream.on('data', function onRpcRequest(payload){
|
|
|
|
// Append origin to rpc payload
|
|
|
|
payload.origin = originDomain
|
|
|
|
// Append origin to signature request
|
|
|
|
if (payload.method === 'eth_sendTransaction') {
|
|
|
|
payload.params[0].origin = originDomain
|
|
|
|
} else if (payload.method === 'eth_sign') {
|
2016-05-26 03:03:16 +02:00
|
|
|
payload.params.push({ origin: originDomain })
|
2016-05-26 02:27:49 +02:00
|
|
|
}
|
|
|
|
// handle rpc request
|
|
|
|
provider.sendAsync(payload, function onPayloadHandled(err, response){
|
|
|
|
logger(null, payload, response)
|
|
|
|
try {
|
|
|
|
stream.write(response)
|
|
|
|
} catch (err) {
|
|
|
|
logger(err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
2016-05-23 03:02:27 +02:00
|
|
|
|
|
|
|
function logger(err, request, response){
|
|
|
|
if (err) return console.error(err.stack)
|
|
|
|
if (!request.isMetamaskInternal) {
|
2016-05-26 02:27:49 +02:00
|
|
|
console.log(`RPC (${originDomain}):`, request, '->', response)
|
2016-05-23 03:02:27 +02:00
|
|
|
if (response.error) console.error('Error in RPC response:\n'+response.error.message)
|
|
|
|
}
|
|
|
|
}
|
2016-03-10 03:33:30 +01:00
|
|
|
}
|
|
|
|
|
2016-04-15 21:12:04 +02:00
|
|
|
function setupControllerConnection(stream){
|
|
|
|
var dnode = Dnode({
|
2016-02-11 02:44:46 +01:00
|
|
|
getState: function(cb){ cb(null, getState()) },
|
2016-03-08 23:33:01 +01:00
|
|
|
setRpcTarget: setRpcTarget,
|
2016-05-11 00:37:13 +02:00
|
|
|
setProviderType: setProviderType,
|
2016-04-01 01:32:35 +02:00
|
|
|
useEtherscanProvider: useEtherscanProvider,
|
2016-02-11 02:44:46 +01:00
|
|
|
// forward directly to idStore
|
2016-02-17 09:55:57 +01:00
|
|
|
createNewVault: idStore.createNewVault.bind(idStore),
|
2016-03-15 21:39:12 +01:00
|
|
|
recoverFromSeed: idStore.recoverFromSeed.bind(idStore),
|
2016-02-11 02:44:46 +01:00
|
|
|
submitPassword: idStore.submitPassword.bind(idStore),
|
|
|
|
setSelectedAddress: idStore.setSelectedAddress.bind(idStore),
|
2016-03-03 07:58:23 +01:00
|
|
|
approveTransaction: idStore.approveTransaction.bind(idStore),
|
2016-02-12 21:55:20 +01:00
|
|
|
cancelTransaction: idStore.cancelTransaction.bind(idStore),
|
2016-05-03 23:32:22 +02:00
|
|
|
signMessage: idStore.signMessage.bind(idStore),
|
|
|
|
cancelMessage: idStore.cancelMessage.bind(idStore),
|
2016-02-11 02:44:46 +01:00
|
|
|
setLocked: idStore.setLocked.bind(idStore),
|
2016-03-24 18:32:50 +01:00
|
|
|
clearSeedWordCache: idStore.clearSeedWordCache.bind(idStore),
|
2016-04-06 21:01:10 +02:00
|
|
|
exportAccount: idStore.exportAccount.bind(idStore),
|
2016-05-20 21:40:44 +02:00
|
|
|
revealAccount: idStore.revealAccount.bind(idStore),
|
2016-05-21 01:18:54 +02:00
|
|
|
saveAccountLabel: idStore.saveAccountLabel.bind(idStore),
|
2016-06-03 01:52:18 +02:00
|
|
|
tryPassword: idStore.tryPassword.bind(idStore),
|
|
|
|
recoverSeed: idStore.recoverSeed.bind(idStore),
|
2016-02-11 02:44:46 +01:00
|
|
|
})
|
2016-04-15 21:12:04 +02:00
|
|
|
stream.pipe(dnode).pipe(stream)
|
|
|
|
dnode.on('remote', function(remote){
|
2016-03-24 18:32:50 +01:00
|
|
|
|
2016-02-11 02:44:46 +01:00
|
|
|
// push updates to popup
|
|
|
|
ethStore.on('update', sendUpdate)
|
|
|
|
idStore.on('update', sendUpdate)
|
|
|
|
// teardown on disconnect
|
2016-03-10 03:33:30 +01:00
|
|
|
eos(stream, function unsubscribe(){
|
2016-02-11 02:44:46 +01:00
|
|
|
ethStore.removeListener('update', sendUpdate)
|
|
|
|
})
|
|
|
|
function sendUpdate(){
|
|
|
|
var state = getState()
|
|
|
|
remote.sendUpdate(state)
|
|
|
|
}
|
|
|
|
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// plugin badge text
|
|
|
|
//
|
|
|
|
|
|
|
|
idStore.on('update', updateBadge)
|
2015-12-19 07:05:16 +01:00
|
|
|
|
2016-01-19 02:05:46 +01:00
|
|
|
function updateBadge(state){
|
|
|
|
var label = ''
|
2016-04-19 01:39:35 +02:00
|
|
|
var unconfTxs = configManager.unconfirmedTxs()
|
2016-05-03 23:32:22 +02:00
|
|
|
var unconfTxLen = Object.keys(unconfTxs).length
|
|
|
|
var unconfMsgs = messageManager.unconfirmedMsgs()
|
|
|
|
var unconfMsgLen = Object.keys(unconfMsgs).length
|
|
|
|
var count = unconfTxLen + unconfMsgLen
|
2016-01-19 02:05:46 +01:00
|
|
|
if (count) {
|
|
|
|
label = String(count)
|
|
|
|
}
|
2016-03-08 23:33:01 +01:00
|
|
|
chrome.browserAction.setBadgeText({ text: label })
|
|
|
|
chrome.browserAction.setBadgeBackgroundColor({ color: '#506F8B' })
|
2016-01-19 02:05:46 +01:00
|
|
|
}
|
2015-12-19 07:05:16 +01:00
|
|
|
|
2016-03-11 00:39:31 +01:00
|
|
|
//
|
2016-04-28 23:16:24 +02:00
|
|
|
// Add unconfirmed Tx + Msg
|
2016-03-11 00:39:31 +01:00
|
|
|
//
|
|
|
|
|
2016-05-26 02:27:49 +02:00
|
|
|
function newUnsignedTransaction(txParams, cb){
|
2016-05-24 23:59:33 +02:00
|
|
|
var state = idStore.getState()
|
|
|
|
if (!state.isUnlocked) {
|
|
|
|
createUnlockRequestNotification({
|
|
|
|
title: 'Account Unlock Request',
|
|
|
|
})
|
|
|
|
var txId = idStore.addUnconfirmedTransaction(txParams, cb)
|
|
|
|
} else {
|
2016-06-04 00:18:20 +02:00
|
|
|
addUnconfirmedTx(txParams, cb)
|
2016-05-24 23:59:33 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-26 02:27:49 +02:00
|
|
|
function newUnsignedMessage(msgParams, cb){
|
2016-05-24 23:59:33 +02:00
|
|
|
var state = idStore.getState()
|
|
|
|
if (!state.isUnlocked) {
|
|
|
|
createUnlockRequestNotification({
|
|
|
|
title: 'Account Unlock Request',
|
|
|
|
})
|
|
|
|
var msgId = idStore.addUnconfirmedMessage(msgParams, cb)
|
|
|
|
} else {
|
2016-06-04 00:18:20 +02:00
|
|
|
addUnconfirmedMsg(msgParams, cb)
|
2016-05-24 23:59:33 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-11 00:39:31 +01:00
|
|
|
function addUnconfirmedTx(txParams, cb){
|
2016-03-12 02:13:48 +01:00
|
|
|
var txId = idStore.addUnconfirmedTransaction(txParams, cb)
|
|
|
|
createTxNotification({
|
|
|
|
title: 'New Unsigned Transaction',
|
|
|
|
txParams: txParams,
|
|
|
|
confirm: idStore.approveTransaction.bind(idStore, txId, noop),
|
|
|
|
cancel: idStore.cancelTransaction.bind(idStore, txId),
|
2016-03-11 00:39:31 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2016-04-28 23:16:24 +02:00
|
|
|
function addUnconfirmedMsg(msgParams, cb){
|
|
|
|
var msgId = idStore.addUnconfirmedMessage(msgParams, cb)
|
|
|
|
createMsgNotification({
|
|
|
|
title: 'New Unsigned Message',
|
|
|
|
msgParams: msgParams,
|
|
|
|
confirm: idStore.approveMessage.bind(idStore, msgId, noop),
|
|
|
|
cancel: idStore.cancelMessage.bind(idStore, msgId),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2016-03-08 23:33:01 +01:00
|
|
|
//
|
|
|
|
// config
|
|
|
|
//
|
|
|
|
|
|
|
|
// called from popup
|
|
|
|
function setRpcTarget(rpcTarget){
|
Made configuration migrateable
Abstract all configuration data into a singleton called `configManager`, who is responsible for reading and writing to the persisted storage (localStorage, in our case).
Uses my new module [pojo-migrator](https://www.npmjs.com/package/pojo-migrator), and wraps it with the `ConfigManager` class, which we can hang any state setting or getting methods we need.
By keeping all the persisted state in one place, we can stabilize its outward-facing API, making the interactions increasingly atomic, which will allow us to add features that require restructuring the persisted data in the long term without having to rewrite UI or even `background.js` code.
All the restructuring and data-type management is kept in one neat little place.
This should make it very easy to add new configuration options like user-configured providers, per-domain vaults, and more!
I know this doesn't seem like a big user-facing feature, but we have a big laundry list of features that I think this will really help streamline.
2016-03-31 04:15:49 +02:00
|
|
|
configManager.setRpcTarget(rpcTarget)
|
2016-03-08 23:33:01 +01:00
|
|
|
chrome.runtime.reload()
|
2016-06-04 00:18:20 +02:00
|
|
|
idStore.getNetwork()
|
2016-03-08 23:33:01 +01:00
|
|
|
}
|
|
|
|
|
2016-05-11 00:37:13 +02:00
|
|
|
function setProviderType(type) {
|
|
|
|
configManager.setProviderType(type)
|
|
|
|
chrome.runtime.reload()
|
2016-06-04 00:18:20 +02:00
|
|
|
idStore.getNetwork()
|
2016-05-11 00:37:13 +02:00
|
|
|
}
|
|
|
|
|
2016-04-01 01:32:35 +02:00
|
|
|
function useEtherscanProvider() {
|
|
|
|
configManager.useEtherscanProvider()
|
|
|
|
chrome.runtime.reload()
|
|
|
|
}
|
|
|
|
|
2016-03-10 03:33:30 +01:00
|
|
|
// util
|
|
|
|
|
2016-03-24 18:32:50 +01:00
|
|
|
function noop(){}
|