2016-01-15 11:03:42 +01:00
|
|
|
const Dnode = require('dnode')
|
2016-04-15 06:22:04 +02:00
|
|
|
const ObjectMultiplex = require('./lib/obj-multiplex')
|
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-01-15 11:03:42 +01:00
|
|
|
const PortStream = require('./lib/port-stream.js')
|
2016-04-01 01:32:35 +02:00
|
|
|
const MetaMaskProvider = require('./lib/zero.js')
|
2016-02-11 02:44:46 +01:00
|
|
|
const IdentityStore = require('./lib/idStore')
|
2016-03-12 02:13:48 +01:00
|
|
|
const createTxNotification = require('./lib/tx-notification.js')
|
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-04-15 21:12:04 +02:00
|
|
|
const setupMultiplex = require('./lib/stream-utils.js').setupMultiplex
|
|
|
|
const HostStore = require('./lib/remote-store.js').HostStore
|
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-04-15 21:12:04 +02:00
|
|
|
setupTrustedCommunication(portStream)
|
2016-01-15 11:03:42 +01:00
|
|
|
} else {
|
|
|
|
// communication with page
|
2016-04-15 21:12:04 +02:00
|
|
|
setupUntrustedCommunication(portStream)
|
2016-01-15 11:03:42 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-15 21:12:04 +02:00
|
|
|
function setupUntrustedCommunication(connectionStream){
|
|
|
|
// setup multiplexing
|
|
|
|
var mx = setupMultiplex(connectionStream)
|
|
|
|
// connect features
|
|
|
|
setupProviderConnection(mx.createStream('provider'))
|
|
|
|
setupPublicConfig(mx.createStream('publicConfig'))
|
|
|
|
}
|
|
|
|
|
|
|
|
function setupTrustedCommunication(connectionStream){
|
|
|
|
// setup multiplexing
|
|
|
|
var mx = setupMultiplex(connectionStream)
|
|
|
|
// connect features
|
|
|
|
setupControllerConnection(mx.createStream('controller'))
|
|
|
|
setupProviderConnection(mx.createStream('provider'))
|
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-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-02-11 02:44:46 +01:00
|
|
|
getAccounts: function(cb){
|
|
|
|
var selectedAddress = idStore.getSelectedAddress()
|
|
|
|
var result = selectedAddress ? [selectedAddress] : []
|
|
|
|
cb(null, result)
|
|
|
|
},
|
2016-03-11 00:39:31 +01:00
|
|
|
approveTransaction: addUnconfirmedTx,
|
2016-02-20 21:13:18 +01:00
|
|
|
signTransaction: idStore.signTransaction.bind(idStore),
|
2016-04-01 01:32:35 +02:00
|
|
|
}
|
|
|
|
var provider = MetaMaskProvider(providerOpts)
|
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-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-01-15 11:03:42 +01:00
|
|
|
// handle rpc requests
|
2016-03-10 03:33:30 +01:00
|
|
|
function onRpcRequest(remoteStream, payload){
|
2016-01-15 11:03:42 +01:00
|
|
|
// console.log('MetaMaskPlugin - incoming payload:', payload)
|
2016-04-01 01:32:35 +02:00
|
|
|
provider.sendAsync(payload, function onPayloadHandled(err, response){
|
2016-03-08 22:27:38 +01:00
|
|
|
// provider engine errors are included in response objects
|
2016-03-11 00:39:31 +01:00
|
|
|
if (!payload.isMetamaskInternal) console.log('MetaMaskPlugin - RPC complete:', payload, '->', response)
|
2016-02-10 20:46:13 +01:00
|
|
|
try {
|
2016-03-10 22:04:45 +01:00
|
|
|
remoteStream.write(response)
|
2016-03-10 03:33:30 +01:00
|
|
|
} catch (err) {
|
|
|
|
console.error(err)
|
2016-02-10 20:46:13 +01:00
|
|
|
}
|
2015-12-19 07:05:16 +01:00
|
|
|
})
|
|
|
|
}
|
2015-08-02 01:33:31 +02:00
|
|
|
|
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)
|
|
|
|
}
|
|
|
|
|
|
|
|
function setupProviderConnection(stream){
|
|
|
|
stream.on('data', onRpcRequest.bind(null, stream))
|
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-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-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-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()
|
|
|
|
var count = Object.keys(unconfTxs).length
|
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
|
|
|
//
|
|
|
|
// Add unconfirmed Tx
|
|
|
|
//
|
|
|
|
|
|
|
|
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-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-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(){}
|