2016-05-23 00:23:16 +02:00
const Streams = require ( 'mississippi' )
const ObjectMultiplex = require ( './obj-multiplex' )
2016-05-23 03:02:27 +02:00
const StreamProvider = require ( 'web3-stream-provider' )
2016-05-23 00:23:16 +02:00
const RemoteStore = require ( './remote-store.js' ) . RemoteStore
module . exports = MetamaskInpageProvider
2016-06-21 22:18:32 +02:00
function MetamaskInpageProvider ( connectionStream ) {
2016-05-23 00:23:16 +02:00
const self = this
2016-06-21 22:18:32 +02:00
// setup connectionStream multiplexing
2016-05-23 00:23:16 +02:00
var multiStream = ObjectMultiplex ( )
2016-06-21 22:18:32 +02:00
Streams . pipe ( connectionStream , multiStream , connectionStream , function ( err ) {
2016-05-23 00:23:16 +02:00
console . warn ( 'MetamaskInpageProvider - lost connection to MetaMask' )
if ( err ) throw err
} )
self . multiStream = multiStream
// subscribe to metamask public config
var publicConfigStore = remoteStoreWithLocalStorageCache ( 'MetaMask-Config' )
var storeStream = publicConfigStore . createStream ( )
2016-06-21 22:18:32 +02:00
Streams . pipe ( storeStream , multiStream . createStream ( 'publicConfig' ) , storeStream , function ( err ) {
2016-05-23 00:23:16 +02:00
console . warn ( 'MetamaskInpageProvider - lost connection to MetaMask publicConfig' )
if ( err ) throw err
} )
self . publicConfigStore = publicConfigStore
// connect to async provider
var asyncProvider = new StreamProvider ( )
2016-06-21 22:18:32 +02:00
Streams . pipe ( asyncProvider , multiStream . createStream ( 'provider' ) , asyncProvider , function ( err ) {
2016-05-23 00:23:16 +02:00
console . warn ( 'MetamaskInpageProvider - lost connection to MetaMask provider' )
if ( err ) throw err
} )
asyncProvider . on ( 'error' , console . error . bind ( console ) )
self . asyncProvider = asyncProvider
// overwrite own sendAsync method
self . sendAsync = asyncProvider . sendAsync . bind ( asyncProvider )
}
2016-06-21 22:18:32 +02:00
MetamaskInpageProvider . prototype . send = function ( payload ) {
2016-05-23 00:23:16 +02:00
const self = this
2016-08-11 22:31:00 +02:00
2016-06-21 22:56:04 +02:00
let selectedAddress
2016-08-11 22:31:00 +02:00
let result = null
2016-05-23 00:23:16 +02:00
switch ( payload . method ) {
case 'eth_accounts' :
// read from localStorage
2016-06-21 22:56:04 +02:00
selectedAddress = self . publicConfigStore . get ( 'selectedAddress' )
2016-05-23 00:23:16 +02:00
result = selectedAddress ? [ selectedAddress ] : [ ]
break
case 'eth_coinbase' :
// read from localStorage
2016-06-21 22:56:04 +02:00
selectedAddress = self . publicConfigStore . get ( 'selectedAddress' )
2016-05-23 00:23:16 +02:00
result = selectedAddress || '0x0000000000000000000000000000000000000000'
break
2016-08-11 22:23:49 +02:00
// throw not-supported Error
2016-05-23 00:23:16 +02:00
default :
2016-08-11 23:35:55 +02:00
var message = 'The MetaMask Web3 object does not support synchronous methods. See https://github.com/MetaMask/faq/blob/master/DEVELOPERS.md#all-async---think-of-metamask-as-a-light-client for details.'
2016-08-11 22:23:49 +02:00
throw new Error ( message )
2016-05-23 00:23:16 +02:00
}
// return the result
return {
id : payload . id ,
jsonrpc : payload . jsonrpc ,
result : result ,
}
}
2016-06-21 22:18:32 +02:00
MetamaskInpageProvider . prototype . sendAsync = function ( ) {
2016-05-23 00:23:16 +02:00
throw new Error ( 'MetamaskInpageProvider - sendAsync not overwritten' )
}
2016-06-21 22:18:32 +02:00
MetamaskInpageProvider . prototype . isConnected = function ( ) {
2016-05-23 00:23:16 +02:00
return true
}
// util
2016-06-21 22:18:32 +02:00
function remoteStoreWithLocalStorageCache ( storageKey ) {
2016-05-23 00:23:16 +02:00
// read local cache
var initState = JSON . parse ( localStorage [ storageKey ] || '{}' )
var store = new RemoteStore ( initState )
// cache the latest state locally
2016-06-21 22:18:32 +02:00
store . subscribe ( function ( state ) {
2016-05-23 00:23:16 +02:00
localStorage [ storageKey ] = JSON . stringify ( state )
} )
return store
2016-06-21 22:18:32 +02:00
}