nifty-wallet/app/scripts/lib/inpage-provider.js

120 lines
3.6 KiB
JavaScript
Raw Normal View History

2016-05-22 15:23:16 -07:00
const Streams = require('mississippi')
const ObjectMultiplex = require('./obj-multiplex')
2016-05-22 18:02:27 -07:00
const StreamProvider = require('web3-stream-provider')
2016-05-22 15:23:16 -07:00
const RemoteStore = require('./remote-store.js').RemoteStore
module.exports = MetamaskInpageProvider
2016-06-21 13:18:32 -07:00
function MetamaskInpageProvider (connectionStream) {
2016-05-22 15:23:16 -07:00
const self = this
2016-06-21 13:18:32 -07:00
// setup connectionStream multiplexing
2016-05-22 15:23:16 -07:00
var multiStream = ObjectMultiplex()
2016-06-21 13:18:32 -07:00
Streams.pipe(connectionStream, multiStream, connectionStream, function (err) {
2016-05-22 15:23:16 -07: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 13:18:32 -07:00
Streams.pipe(storeStream, multiStream.createStream('publicConfig'), storeStream, function (err) {
2016-05-22 15:23:16 -07: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 13:18:32 -07:00
Streams.pipe(asyncProvider, multiStream.createStream('provider'), asyncProvider, function (err) {
2016-05-22 15:23:16 -07:00
console.warn('MetamaskInpageProvider - lost connection to MetaMask provider')
if (err) throw err
})
asyncProvider.on('error', console.error.bind(console))
self.asyncProvider = asyncProvider
// handle sendAsync requests via asyncProvider
self.sendAsync = function(payload, cb){
// rewrite request ids
var request = jsonrpcMessageTransform(payload, (message) => {
message.id = createRandomId()
return message
})
// forward to asyncProvider
asyncProvider.sendAsync(request, cb)
}
2016-05-22 15:23:16 -07:00
}
2016-06-21 13:18:32 -07:00
MetamaskInpageProvider.prototype.send = function (payload) {
2016-05-22 15:23:16 -07:00
const self = this
2016-08-11 13:31:00 -07:00
2016-06-21 13:56:04 -07:00
let selectedAddress
2016-08-11 13:31:00 -07:00
let result = null
2016-05-22 15:23:16 -07:00
switch (payload.method) {
case 'eth_accounts':
// read from localStorage
2016-06-21 13:56:04 -07:00
selectedAddress = self.publicConfigStore.get('selectedAddress')
2016-05-22 15:23:16 -07:00
result = selectedAddress ? [selectedAddress] : []
break
case 'eth_coinbase':
// read from localStorage
2016-06-21 13:56:04 -07:00
selectedAddress = self.publicConfigStore.get('selectedAddress')
2016-05-22 15:23:16 -07:00
result = selectedAddress || '0x0000000000000000000000000000000000000000'
break
// throw not-supported Error
2016-05-22 15:23:16 -07:00
default:
2016-08-11 14:35:55 -07: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.'
throw new Error(message)
2016-05-22 15:23:16 -07:00
}
// return the result
return {
id: payload.id,
jsonrpc: payload.jsonrpc,
result: result,
}
}
2016-06-21 13:18:32 -07:00
MetamaskInpageProvider.prototype.sendAsync = function () {
2016-05-22 15:23:16 -07:00
throw new Error('MetamaskInpageProvider - sendAsync not overwritten')
}
2016-06-21 13:18:32 -07:00
MetamaskInpageProvider.prototype.isConnected = function () {
2016-05-22 15:23:16 -07:00
return true
}
// util
2016-06-21 13:18:32 -07:00
function remoteStoreWithLocalStorageCache (storageKey) {
2016-05-22 15:23:16 -07:00
// read local cache
var initState = JSON.parse(localStorage[storageKey] || '{}')
var store = new RemoteStore(initState)
// cache the latest state locally
2016-06-21 13:18:32 -07:00
store.subscribe(function (state) {
2016-05-22 15:23:16 -07:00
localStorage[storageKey] = JSON.stringify(state)
})
return store
2016-06-21 13:18:32 -07:00
}
function createRandomId(){
const extraDigits = 3
// 13 time digits
const datePart = new Date().getTime() * Math.pow(10, extraDigits)
// 3 random digits
const extraPart = Math.floor(Math.random() * Math.pow(10, extraDigits))
// 16 digits
return datePart + extraPart
}
function jsonrpcMessageTransform(payload, transformFn){
if (Array.isArray(payload)) {
return payload.map(transformFn)
} else {
return transformFn(payload)
}
}