1
0
mirror of https://github.com/kremalicious/metamask-extension.git synced 2024-12-23 09:52:26 +01:00
metamask-extension/app/scripts/inpage.js

195 lines
4.9 KiB
JavaScript
Raw Normal View History

cleanContextForImports()
const createPayload = require('web3-provider-engine/util/create-payload')
2015-12-19 07:05:16 +01:00
const StreamProvider = require('./lib/stream-provider.js')
const LocalMessageDuplexStream = require('./lib/local-message-stream.js')
const setupMultiplex = require('./lib/stream-utils.js').setupMultiplex
const RemoteStore = require('./lib/remote-store.js').RemoteStore
2016-05-20 01:53:16 +02:00
const MetamaskConfig = require('./config.js')
const Web3 = require('web3')
2016-05-06 01:04:43 +02:00
const once = require('once')
restoreContextAfterImports()
// rename on window
delete window.Web3
window.MetamaskWeb3 = Web3
2015-10-10 08:14:18 +02:00
2016-02-15 06:53:54 +01:00
//
2015-12-19 07:05:16 +01:00
// setup plugin communication
2016-02-15 06:53:54 +01:00
//
// setup background connection
2015-12-19 07:05:16 +01:00
var pluginStream = new LocalMessageDuplexStream({
name: 'inpage',
target: 'contentscript',
})
var mx = setupMultiplex(pluginStream)
2016-05-06 01:04:43 +02:00
// connect to provider
2015-12-19 07:05:16 +01:00
var remoteProvider = new StreamProvider()
remoteProvider.pipe(mx.createStream('provider')).pipe(remoteProvider)
2016-02-10 20:46:13 +01:00
remoteProvider.on('error', console.error.bind(console))
2016-05-06 01:04:43 +02:00
// subscribe to metamask public config
var initState = JSON.parse(localStorage['MetaMask-Config'] || '{}')
var publicConfigStore = new RemoteStore(initState)
var storeStream = publicConfigStore.createStream()
storeStream.pipe(mx.createStream('publicConfig')).pipe(storeStream)
publicConfigStore.subscribe(function(state){
localStorage['MetaMask-Config'] = JSON.stringify(state)
})
2016-04-15 06:11:35 +02:00
//
2016-05-06 01:04:43 +02:00
// setup web3
2016-04-15 06:11:35 +02:00
//
var web3 = new Web3(remoteProvider)
web3.setProvider = function(){
console.log('MetaMask - overrode web3.setProvider')
}
console.log('MetaMask - injected web3')
2016-05-06 01:04:43 +02:00
//
// automatic dapp reset
//
// export web3 as a global, checking for usage
var pageIsUsingWeb3 = false
var resetWasRequested = false
window.web3 = ensnare(web3, once(function(){
// if web3 usage happened after a reset request, trigger reset late
if (resetWasRequested) return triggerReset()
// mark web3 as used
pageIsUsingWeb3 = true
// reset web3 reference
window.web3 = web3
}))
// listen for reset requests
mx.createStream('control').once('data', function(){
resetWasRequested = true
// ignore if web3 was not used
if (!pageIsUsingWeb3) return
// reload after short timeout
triggerReset()
})
function triggerReset(){
setTimeout(function(){
window.location.reload()
}, 500)
}
2016-02-15 06:53:54 +01:00
//
// handle synchronous requests
//
global.publicConfigStore = publicConfigStore
// set web3 defaultAcount
publicConfigStore.subscribe(function(state){
web3.eth.defaultAccount = state.selectedAddress
})
// setup sync http provider
2016-05-20 01:53:16 +02:00
updateProvider({ provider: publicConfigStore.get('provider') })
publicConfigStore.subscribe(updateProvider)
var syncProvider = null
var syncProviderUrl = null
function updateProvider(state){
var providerConfig = state.provider || {}
var newSyncProviderUrl = undefined
if (providerConfig.rpcTarget) {
newSyncProviderUrl = providerConfig.rpcTarget
} else {
switch(providerConfig.type) {
case 'testnet':
newSyncProviderUrl = MetamaskConfig.network.testnet
break
case 'mainnet':
newSyncProviderUrl = MetamaskConfig.network.mainnet
break
default:
newSyncProviderUrl = MetamaskConfig.network.default
}
}
if (newSyncProviderUrl === syncProviderUrl) return
syncProvider = new Web3.providers.HttpProvider(newSyncProviderUrl)
}
// handle sync methods
remoteProvider.send = function(payload){
var result = null
switch (payload.method) {
2016-04-12 23:48:48 +02:00
case 'eth_accounts':
// read from localStorage
var selectedAddress = publicConfigStore.get('selectedAddress')
result = selectedAddress ? [selectedAddress] : []
break
case 'eth_coinbase':
// read from localStorage
var selectedAddress = publicConfigStore.get('selectedAddress')
result = selectedAddress || '0x0000000000000000000000000000000000000000'
break
// fallback to normal rpc
default:
return syncProvider.send(payload)
}
// return the result
return {
2016-02-15 06:53:54 +01:00
id: payload.id,
jsonrpc: payload.jsonrpc,
result: result,
}
}
2016-01-31 08:15:38 +01:00
2016-05-06 01:04:43 +02:00
//
// util
//
// creates a proxy object that calls cb everytime the obj's properties/fns are accessed
function ensnare(obj, cb){
var proxy = {}
Object.keys(obj).forEach(function(key){
var val = obj[key]
switch (typeof val) {
case 'function':
proxy[key] = function(){
cb()
val.apply(obj, arguments)
}
return
default:
Object.defineProperty(proxy, key, {
get: function(){ cb(); return obj[key] },
set: function(val){ cb(); return obj[key] = val },
})
return
}
})
return proxy
}
// need to make sure we aren't affected by overlapping namespaces
// and that we dont affect the app with our namespace
// mostly a fix for web3's BigNumber if AMD's "define" is defined...
var __define = undefined
function cleanContextForImports(){
__define = global.define
delete global.define
}
function restoreContextAfterImports(){
global.define = __define
2016-05-06 01:04:43 +02:00
}