mirror of
https://github.com/kremalicious/metamask-extension.git
synced 2024-12-23 09:52:26 +01:00
Merge branch 'master' into uiFixes
This commit is contained in:
commit
ab55fefa1a
@ -1,19 +1,51 @@
|
|||||||
const urlUtil = require('url')
|
const urlUtil = require('url')
|
||||||
|
const extend = require('xtend')
|
||||||
const Dnode = require('dnode')
|
const Dnode = require('dnode')
|
||||||
const eos = require('end-of-stream')
|
const eos = require('end-of-stream')
|
||||||
const extend = require('xtend')
|
|
||||||
const EthStore = require('eth-store')
|
|
||||||
const MetaMaskProvider = require('web3-provider-engine/zero.js')
|
|
||||||
const PortStream = require('./lib/port-stream.js')
|
const PortStream = require('./lib/port-stream.js')
|
||||||
const IdentityStore = require('./lib/idStore')
|
|
||||||
const createUnlockRequestNotification = require('./lib/notifications.js').createUnlockRequestNotification
|
const createUnlockRequestNotification = require('./lib/notifications.js').createUnlockRequestNotification
|
||||||
const createTxNotification = require('./lib/notifications.js').createTxNotification
|
const createTxNotification = require('./lib/notifications.js').createTxNotification
|
||||||
const createMsgNotification = require('./lib/notifications.js').createMsgNotification
|
const createMsgNotification = require('./lib/notifications.js').createMsgNotification
|
||||||
const configManager = require('./lib/config-manager-singleton')
|
|
||||||
const messageManager = require('./lib/message-manager')
|
const messageManager = require('./lib/message-manager')
|
||||||
const setupMultiplex = require('./lib/stream-utils.js').setupMultiplex
|
const setupMultiplex = require('./lib/stream-utils.js').setupMultiplex
|
||||||
const HostStore = require('./lib/remote-store.js').HostStore
|
const MetamaskController = require('./metamask-controller')
|
||||||
const Web3 = require('web3')
|
|
||||||
|
const STORAGE_KEY = 'metamask-config'
|
||||||
|
|
||||||
|
const controller = new MetamaskController({
|
||||||
|
// User confirmation callbacks:
|
||||||
|
showUnconfirmedMessage,
|
||||||
|
unlockAccountMessage,
|
||||||
|
showUnconfirmedTx,
|
||||||
|
// Persistence Methods:
|
||||||
|
setData,
|
||||||
|
loadData,
|
||||||
|
})
|
||||||
|
const idStore = controller.idStore
|
||||||
|
|
||||||
|
function unlockAccountMessage () {
|
||||||
|
createUnlockRequestNotification({
|
||||||
|
title: 'Account Unlock Request',
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function showUnconfirmedMessage (msgParams, msgId) {
|
||||||
|
createMsgNotification({
|
||||||
|
title: 'New Unsigned Message',
|
||||||
|
msgParams: msgParams,
|
||||||
|
confirm: idStore.approveMessage.bind(idStore, msgId, noop),
|
||||||
|
cancel: idStore.cancelMessage.bind(idStore, msgId),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function showUnconfirmedTx (txParams, txData, onTxDoneCb) {
|
||||||
|
createTxNotification({
|
||||||
|
title: 'New Unsigned Transaction',
|
||||||
|
txParams: txParams,
|
||||||
|
confirm: idStore.approveTransaction.bind(idStore, txData.id, noop),
|
||||||
|
cancel: idStore.cancelTransaction.bind(idStore, txData.id),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
// connect to other contexts
|
// connect to other contexts
|
||||||
@ -37,8 +69,8 @@ function setupUntrustedCommunication (connectionStream, originDomain) {
|
|||||||
// setup multiplexing
|
// setup multiplexing
|
||||||
var mx = setupMultiplex(connectionStream)
|
var mx = setupMultiplex(connectionStream)
|
||||||
// connect features
|
// connect features
|
||||||
setupProviderConnection(mx.createStream('provider'), originDomain)
|
controller.setupProviderConnection(mx.createStream('provider'), originDomain)
|
||||||
setupPublicConfig(mx.createStream('publicConfig'))
|
controller.setupPublicConfig(mx.createStream('publicConfig'))
|
||||||
}
|
}
|
||||||
|
|
||||||
function setupTrustedCommunication (connectionStream, originDomain) {
|
function setupTrustedCommunication (connectionStream, originDomain) {
|
||||||
@ -46,181 +78,28 @@ function setupTrustedCommunication (connectionStream, originDomain) {
|
|||||||
var mx = setupMultiplex(connectionStream)
|
var mx = setupMultiplex(connectionStream)
|
||||||
// connect features
|
// connect features
|
||||||
setupControllerConnection(mx.createStream('controller'))
|
setupControllerConnection(mx.createStream('controller'))
|
||||||
setupProviderConnection(mx.createStream('provider'), originDomain)
|
controller.setupProviderConnection(mx.createStream('provider'), originDomain)
|
||||||
}
|
|
||||||
|
|
||||||
//
|
|
||||||
// state and network
|
|
||||||
//
|
|
||||||
|
|
||||||
var idStore = new IdentityStore()
|
|
||||||
|
|
||||||
var providerOpts = {
|
|
||||||
rpcUrl: configManager.getCurrentRpcAddress(),
|
|
||||||
// account mgmt
|
|
||||||
getAccounts: function (cb) {
|
|
||||||
var selectedAddress = idStore.getSelectedAddress()
|
|
||||||
var result = selectedAddress ? [selectedAddress] : []
|
|
||||||
cb(null, result)
|
|
||||||
},
|
|
||||||
// tx signing
|
|
||||||
approveTransaction: newUnsignedTransaction,
|
|
||||||
signTransaction: idStore.signTransaction.bind(idStore),
|
|
||||||
// msg signing
|
|
||||||
approveMessage: newUnsignedMessage,
|
|
||||||
signMessage: idStore.signMessage.bind(idStore),
|
|
||||||
}
|
|
||||||
var provider = MetaMaskProvider(providerOpts)
|
|
||||||
var web3 = new Web3(provider)
|
|
||||||
idStore.web3 = web3
|
|
||||||
idStore.getNetwork()
|
|
||||||
|
|
||||||
// log new blocks
|
|
||||||
provider.on('block', function (block) {
|
|
||||||
console.log('BLOCK CHANGED:', '#' + block.number.toString('hex'), '0x' + block.hash.toString('hex'))
|
|
||||||
|
|
||||||
// Check network when restoring connectivity:
|
|
||||||
if (idStore._currentState.network === 'loading') {
|
|
||||||
idStore.getNetwork()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
provider.on('error', idStore.getNetwork.bind(idStore))
|
|
||||||
|
|
||||||
var ethStore = new EthStore(provider)
|
|
||||||
idStore.setStore(ethStore)
|
|
||||||
|
|
||||||
function getState () {
|
|
||||||
var state = extend(
|
|
||||||
ethStore.getState(),
|
|
||||||
idStore.getState(),
|
|
||||||
configManager.getConfig()
|
|
||||||
)
|
|
||||||
return state
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
|
||||||
// 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])
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
// remote features
|
// remote features
|
||||||
//
|
//
|
||||||
|
|
||||||
function setupPublicConfig (stream) {
|
|
||||||
var storeStream = publicConfigStore.createStream()
|
|
||||||
stream.pipe(storeStream).pipe(stream)
|
|
||||||
}
|
|
||||||
|
|
||||||
function setupProviderConnection (stream, originDomain) {
|
|
||||||
// decorate all payloads with origin domain
|
|
||||||
stream.on('data', function onRpcRequest (request) {
|
|
||||||
var payloads = Array.isArray(request) ? request : [request]
|
|
||||||
payloads.forEach(function (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') {
|
|
||||||
payload.params.push({ origin: originDomain })
|
|
||||||
}
|
|
||||||
})
|
|
||||||
// handle rpc request
|
|
||||||
provider.sendAsync(request, function onPayloadHandled (err, response) {
|
|
||||||
if (err) {
|
|
||||||
return logger(err)
|
|
||||||
}
|
|
||||||
logger(null, request, response)
|
|
||||||
try {
|
|
||||||
stream.write(response)
|
|
||||||
} catch (err) {
|
|
||||||
logger(err)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
function logger (err, request, response) {
|
|
||||||
if (err) return console.error(err.stack)
|
|
||||||
if (!request.isMetamaskInternal) {
|
|
||||||
console.log(`RPC (${originDomain}):`, request, '->', response)
|
|
||||||
if (response.error) console.error('Error in RPC response:\n' + response.error.message)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function setupControllerConnection (stream) {
|
function setupControllerConnection (stream) {
|
||||||
var dnode = Dnode({
|
controller.stream = stream
|
||||||
getState: function (cb) { cb(null, getState()) },
|
var api = controller.getApi()
|
||||||
setRpcTarget: setRpcTarget,
|
var dnode = Dnode(api)
|
||||||
setProviderType: setProviderType,
|
|
||||||
useEtherscanProvider: useEtherscanProvider,
|
|
||||||
agreeToDisclaimer: agreeToDisclaimer,
|
|
||||||
// forward directly to idStore
|
|
||||||
createNewVault: idStore.createNewVault.bind(idStore),
|
|
||||||
recoverFromSeed: idStore.recoverFromSeed.bind(idStore),
|
|
||||||
submitPassword: idStore.submitPassword.bind(idStore),
|
|
||||||
setSelectedAddress: idStore.setSelectedAddress.bind(idStore),
|
|
||||||
approveTransaction: idStore.approveTransaction.bind(idStore),
|
|
||||||
cancelTransaction: idStore.cancelTransaction.bind(idStore),
|
|
||||||
signMessage: idStore.signMessage.bind(idStore),
|
|
||||||
cancelMessage: idStore.cancelMessage.bind(idStore),
|
|
||||||
setLocked: idStore.setLocked.bind(idStore),
|
|
||||||
clearSeedWordCache: idStore.clearSeedWordCache.bind(idStore),
|
|
||||||
exportAccount: idStore.exportAccount.bind(idStore),
|
|
||||||
revealAccount: idStore.revealAccount.bind(idStore),
|
|
||||||
saveAccountLabel: idStore.saveAccountLabel.bind(idStore),
|
|
||||||
tryPassword: idStore.tryPassword.bind(idStore),
|
|
||||||
recoverSeed: idStore.recoverSeed.bind(idStore),
|
|
||||||
})
|
|
||||||
stream.pipe(dnode).pipe(stream)
|
stream.pipe(dnode).pipe(stream)
|
||||||
dnode.on('remote', function (remote) {
|
dnode.on('remote', (remote) => {
|
||||||
// push updates to popup
|
// push updates to popup
|
||||||
ethStore.on('update', sendUpdate)
|
controller.ethStore.on('update', controller.sendUpdate.bind(controller))
|
||||||
idStore.on('update', sendUpdate)
|
controller.remote = remote
|
||||||
|
idStore.on('update', controller.sendUpdate.bind(controller))
|
||||||
|
|
||||||
// teardown on disconnect
|
// teardown on disconnect
|
||||||
eos(stream, function unsubscribe () {
|
eos(stream, () => {
|
||||||
ethStore.removeListener('update', sendUpdate)
|
controller.ethStore.removeListener('update', controller.sendUpdate.bind(controller))
|
||||||
})
|
})
|
||||||
function sendUpdate () {
|
|
||||||
var state = getState()
|
|
||||||
remote.sendUpdate(state)
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -232,7 +111,7 @@ idStore.on('update', updateBadge)
|
|||||||
|
|
||||||
function updateBadge (state) {
|
function updateBadge (state) {
|
||||||
var label = ''
|
var label = ''
|
||||||
var unconfTxs = configManager.unconfirmedTxs()
|
var unconfTxs = controller.configManager.unconfirmedTxs()
|
||||||
var unconfTxLen = Object.keys(unconfTxs).length
|
var unconfTxLen = Object.keys(unconfTxs).length
|
||||||
var unconfMsgs = messageManager.unconfirmedMsgs()
|
var unconfMsgs = messageManager.unconfirmedMsgs()
|
||||||
var unconfMsgLen = Object.keys(unconfMsgs).length
|
var unconfMsgLen = Object.keys(unconfMsgs).length
|
||||||
@ -244,86 +123,54 @@ function updateBadge (state) {
|
|||||||
chrome.browserAction.setBadgeBackgroundColor({ color: '#506F8B' })
|
chrome.browserAction.setBadgeBackgroundColor({ color: '#506F8B' })
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
function loadData () {
|
||||||
// Add unconfirmed Tx + Msg
|
var oldData = getOldStyleData()
|
||||||
//
|
var newData
|
||||||
|
|
||||||
function newUnsignedTransaction (txParams, onTxDoneCb) {
|
|
||||||
var state = idStore.getState()
|
|
||||||
if (!state.isUnlocked) {
|
|
||||||
createUnlockRequestNotification({
|
|
||||||
title: 'Account Unlock Request',
|
|
||||||
})
|
|
||||||
idStore.addUnconfirmedTransaction(txParams, onTxDoneCb, noop)
|
|
||||||
} else {
|
|
||||||
addUnconfirmedTx(txParams, onTxDoneCb)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function newUnsignedMessage (msgParams, cb) {
|
|
||||||
var state = idStore.getState()
|
|
||||||
if (!state.isUnlocked) {
|
|
||||||
createUnlockRequestNotification({
|
|
||||||
title: 'Account Unlock Request',
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
addUnconfirmedMsg(msgParams, cb)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function addUnconfirmedTx (txParams, onTxDoneCb) {
|
|
||||||
idStore.addUnconfirmedTransaction(txParams, onTxDoneCb, function (err, txData) {
|
|
||||||
if (err) return onTxDoneCb(err)
|
|
||||||
createTxNotification({
|
|
||||||
title: 'New Unsigned Transaction',
|
|
||||||
txParams: txParams,
|
|
||||||
confirm: idStore.approveTransaction.bind(idStore, txData.id, noop),
|
|
||||||
cancel: idStore.cancelTransaction.bind(idStore, txData.id),
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
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),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
|
||||||
// config
|
|
||||||
//
|
|
||||||
|
|
||||||
function agreeToDisclaimer (cb) {
|
|
||||||
try {
|
try {
|
||||||
configManager.setConfirmed(true)
|
newData = JSON.parse(window.localStorage[STORAGE_KEY])
|
||||||
cb()
|
} catch (e) {}
|
||||||
} catch (e) {
|
|
||||||
cb(e)
|
var data = extend({
|
||||||
|
meta: {
|
||||||
|
version: 0,
|
||||||
|
},
|
||||||
|
data: {
|
||||||
|
config: {
|
||||||
|
provider: {
|
||||||
|
type: 'testnet',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}, oldData || null, newData || null)
|
||||||
|
return data
|
||||||
|
}
|
||||||
|
|
||||||
|
function getOldStyleData () {
|
||||||
|
var config, wallet, seedWords
|
||||||
|
|
||||||
|
var result = {
|
||||||
|
meta: { version: 0 },
|
||||||
|
data: {},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
config = JSON.parse(window.localStorage['config'])
|
||||||
|
result.data.config = config
|
||||||
|
} catch (e) {}
|
||||||
|
try {
|
||||||
|
wallet = JSON.parse(window.localStorage['lightwallet'])
|
||||||
|
result.data.wallet = wallet
|
||||||
|
} catch (e) {}
|
||||||
|
try {
|
||||||
|
seedWords = window.localStorage['seedWords']
|
||||||
|
result.data.seedWords = seedWords
|
||||||
|
} catch (e) {}
|
||||||
|
|
||||||
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
// called from popup
|
function setData (data) {
|
||||||
function setRpcTarget (rpcTarget) {
|
window.localStorage[STORAGE_KEY] = JSON.stringify(data)
|
||||||
configManager.setRpcTarget(rpcTarget)
|
|
||||||
chrome.runtime.reload()
|
|
||||||
idStore.getNetwork()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function setProviderType (type) {
|
|
||||||
configManager.setProviderType(type)
|
|
||||||
chrome.runtime.reload()
|
|
||||||
idStore.getNetwork()
|
|
||||||
}
|
|
||||||
|
|
||||||
function useEtherscanProvider () {
|
|
||||||
configManager.useEtherscanProvider()
|
|
||||||
chrome.runtime.reload()
|
|
||||||
}
|
|
||||||
|
|
||||||
// util
|
|
||||||
|
|
||||||
function noop () {}
|
function noop () {}
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
const LocalMessageDuplexStream = require('./lib/local-message-stream.js')
|
const LocalMessageDuplexStream = require('./lib/local-message-stream.js')
|
||||||
const PortStream = require('./lib/port-stream.js')
|
const PortStream = require('./lib/port-stream.js')
|
||||||
const ObjectMultiplex = require('./lib/obj-multiplex')
|
const ObjectMultiplex = require('./lib/obj-multiplex')
|
||||||
const urlUtil = require('url')
|
// const urlUtil = require('url')
|
||||||
|
|
||||||
if (shouldInjectWeb3()) {
|
if (shouldInjectWeb3()) {
|
||||||
setupInjection()
|
setupInjection()
|
||||||
@ -45,8 +45,6 @@ function setupInjection(){
|
|||||||
}
|
}
|
||||||
|
|
||||||
function shouldInjectWeb3(){
|
function shouldInjectWeb3(){
|
||||||
var urlData = urlUtil.parse(window.location.href)
|
var shouldInject = (window.location.href.indexOf('.pdf') === -1)
|
||||||
var extension = urlData.pathname.split('.').slice(-1)[0]
|
|
||||||
var shouldInject = (extension !== 'pdf')
|
|
||||||
return shouldInject
|
return shouldInject
|
||||||
}
|
}
|
@ -1,3 +0,0 @@
|
|||||||
var ConfigManager = require('./config-manager')
|
|
||||||
|
|
||||||
module.exports = new ConfigManager()
|
|
@ -1,9 +1,7 @@
|
|||||||
const Migrator = require('pojo-migrator')
|
const Migrator = require('pojo-migrator')
|
||||||
const extend = require('xtend')
|
|
||||||
const MetamaskConfig = require('../config.js')
|
const MetamaskConfig = require('../config.js')
|
||||||
const migrations = require('./migrations')
|
const migrations = require('./migrations')
|
||||||
|
|
||||||
const STORAGE_KEY = 'metamask-config'
|
|
||||||
const TESTNET_RPC = MetamaskConfig.network.testnet
|
const TESTNET_RPC = MetamaskConfig.network.testnet
|
||||||
const MAINNET_RPC = MetamaskConfig.network.mainnet
|
const MAINNET_RPC = MetamaskConfig.network.mainnet
|
||||||
|
|
||||||
@ -15,7 +13,7 @@ const MAINNET_RPC = MetamaskConfig.network.mainnet
|
|||||||
* particular portions of the state.
|
* particular portions of the state.
|
||||||
*/
|
*/
|
||||||
module.exports = ConfigManager
|
module.exports = ConfigManager
|
||||||
function ConfigManager () {
|
function ConfigManager (opts) {
|
||||||
// ConfigManager is observable and will emit updates
|
// ConfigManager is observable and will emit updates
|
||||||
this._subs = []
|
this._subs = []
|
||||||
|
|
||||||
@ -37,12 +35,10 @@ function ConfigManager () {
|
|||||||
|
|
||||||
// How to load initial config.
|
// How to load initial config.
|
||||||
// Includes step on migrating pre-pojo-migrator data.
|
// Includes step on migrating pre-pojo-migrator data.
|
||||||
loadData: loadData,
|
loadData: opts.loadData,
|
||||||
|
|
||||||
// How to persist migrated config.
|
// How to persist migrated config.
|
||||||
setData: function (data) {
|
setData: opts.setData,
|
||||||
window.localStorage[STORAGE_KEY] = JSON.stringify(data)
|
|
||||||
},
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -280,49 +276,3 @@ ConfigManager.prototype.getConfirmed = function () {
|
|||||||
return ('isConfirmed' in data) && data.isConfirmed
|
return ('isConfirmed' in data) && data.isConfirmed
|
||||||
}
|
}
|
||||||
|
|
||||||
function loadData () {
|
|
||||||
var oldData = getOldStyleData()
|
|
||||||
var newData
|
|
||||||
try {
|
|
||||||
newData = JSON.parse(window.localStorage[STORAGE_KEY])
|
|
||||||
} catch (e) {}
|
|
||||||
|
|
||||||
var data = extend({
|
|
||||||
meta: {
|
|
||||||
version: 0,
|
|
||||||
},
|
|
||||||
data: {
|
|
||||||
config: {
|
|
||||||
provider: {
|
|
||||||
type: 'testnet',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}, oldData || null, newData || null)
|
|
||||||
return data
|
|
||||||
}
|
|
||||||
|
|
||||||
function getOldStyleData () {
|
|
||||||
var config, wallet, seedWords
|
|
||||||
|
|
||||||
var result = {
|
|
||||||
meta: { version: 0 },
|
|
||||||
data: {},
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
config = JSON.parse(window.localStorage['config'])
|
|
||||||
result.data.config = config
|
|
||||||
} catch (e) {}
|
|
||||||
try {
|
|
||||||
wallet = JSON.parse(window.localStorage['lightwallet'])
|
|
||||||
result.data.wallet = wallet
|
|
||||||
} catch (e) {}
|
|
||||||
try {
|
|
||||||
seedWords = window.localStorage['seedWords']
|
|
||||||
result.data.seedWords = seedWords
|
|
||||||
} catch (e) {}
|
|
||||||
|
|
||||||
return result
|
|
||||||
}
|
|
||||||
|
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
const ethUtil = require('ethereumjs-util')
|
const ethUtil = require('ethereumjs-util')
|
||||||
const Transaction = require('ethereumjs-tx')
|
const Transaction = require('ethereumjs-tx')
|
||||||
const configManager = require('./config-manager-singleton')
|
|
||||||
|
|
||||||
module.exports = IdManagement
|
module.exports = IdManagement
|
||||||
|
|
||||||
@ -9,6 +8,7 @@ function IdManagement (opts) {
|
|||||||
|
|
||||||
this.keyStore = opts.keyStore
|
this.keyStore = opts.keyStore
|
||||||
this.derivedKey = opts.derivedKey
|
this.derivedKey = opts.derivedKey
|
||||||
|
this.configManager = opts.configManager
|
||||||
this.hdPathString = "m/44'/60'/0'/0"
|
this.hdPathString = "m/44'/60'/0'/0"
|
||||||
|
|
||||||
this.getAddresses = function () {
|
this.getAddresses = function () {
|
||||||
@ -32,9 +32,9 @@ function IdManagement (opts) {
|
|||||||
|
|
||||||
// Add the tx hash to the persisted meta-tx object
|
// Add the tx hash to the persisted meta-tx object
|
||||||
var txHash = ethUtil.bufferToHex(tx.hash())
|
var txHash = ethUtil.bufferToHex(tx.hash())
|
||||||
var metaTx = configManager.getTx(txParams.metamaskId)
|
var metaTx = this.configManager.getTx(txParams.metamaskId)
|
||||||
metaTx.hash = txHash
|
metaTx.hash = txHash
|
||||||
configManager.updateTx(metaTx)
|
this.configManager.updateTx(metaTx)
|
||||||
|
|
||||||
// return raw serialized tx
|
// return raw serialized tx
|
||||||
var rawTx = ethUtil.bufferToHex(tx.serialize())
|
var rawTx = ethUtil.bufferToHex(tx.serialize())
|
||||||
|
@ -7,7 +7,6 @@ const extend = require('xtend')
|
|||||||
const createId = require('web3-provider-engine/util/random-id')
|
const createId = require('web3-provider-engine/util/random-id')
|
||||||
const ethBinToOps = require('eth-bin-to-ops')
|
const ethBinToOps = require('eth-bin-to-ops')
|
||||||
const autoFaucet = require('./auto-faucet')
|
const autoFaucet = require('./auto-faucet')
|
||||||
const configManager = require('./config-manager-singleton')
|
|
||||||
const messageManager = require('./message-manager')
|
const messageManager = require('./message-manager')
|
||||||
const DEFAULT_RPC = 'https://testrpc.metamask.io/'
|
const DEFAULT_RPC = 'https://testrpc.metamask.io/'
|
||||||
const IdManagement = require('./id-management')
|
const IdManagement = require('./id-management')
|
||||||
@ -20,6 +19,7 @@ function IdentityStore (opts = {}) {
|
|||||||
|
|
||||||
// we just use the ethStore to auto-add accounts
|
// we just use the ethStore to auto-add accounts
|
||||||
this._ethStore = opts.ethStore
|
this._ethStore = opts.ethStore
|
||||||
|
this.configManager = opts.configManager
|
||||||
// lightwallet key store
|
// lightwallet key store
|
||||||
this._keyStore = null
|
this._keyStore = null
|
||||||
// lightwallet wrapper
|
// lightwallet wrapper
|
||||||
@ -43,7 +43,10 @@ function IdentityStore (opts = {}) {
|
|||||||
|
|
||||||
IdentityStore.prototype.createNewVault = function (password, entropy, cb) {
|
IdentityStore.prototype.createNewVault = function (password, entropy, cb) {
|
||||||
delete this._keyStore
|
delete this._keyStore
|
||||||
configManager.clearWallet()
|
if (this.configManager) {
|
||||||
|
this.configManager.clearWallet()
|
||||||
|
}
|
||||||
|
|
||||||
this._createIdmgmt(password, null, entropy, (err) => {
|
this._createIdmgmt(password, null, entropy, (err) => {
|
||||||
if (err) return cb(err)
|
if (err) return cb(err)
|
||||||
|
|
||||||
@ -51,14 +54,14 @@ IdentityStore.prototype.createNewVault = function (password, entropy, cb) {
|
|||||||
this._didUpdate()
|
this._didUpdate()
|
||||||
this._autoFaucet()
|
this._autoFaucet()
|
||||||
|
|
||||||
configManager.setShowSeedWords(true)
|
this.configManager.setShowSeedWords(true)
|
||||||
var seedWords = this._idmgmt.getSeed()
|
var seedWords = this._idmgmt.getSeed()
|
||||||
cb(null, seedWords)
|
cb(null, seedWords)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
IdentityStore.prototype.recoverSeed = function (cb) {
|
IdentityStore.prototype.recoverSeed = function (cb) {
|
||||||
configManager.setShowSeedWords(true)
|
this.configManager.setShowSeedWords(true)
|
||||||
if (!this._idmgmt) return cb(new Error('Unauthenticated. Please sign in.'))
|
if (!this._idmgmt) return cb(new Error('Unauthenticated. Please sign in.'))
|
||||||
var seedWords = this._idmgmt.getSeed()
|
var seedWords = this._idmgmt.getSeed()
|
||||||
cb(null, seedWords)
|
cb(null, seedWords)
|
||||||
@ -79,11 +82,13 @@ IdentityStore.prototype.setStore = function (store) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
IdentityStore.prototype.clearSeedWordCache = function (cb) {
|
IdentityStore.prototype.clearSeedWordCache = function (cb) {
|
||||||
|
const configManager = this.configManager
|
||||||
configManager.setShowSeedWords(false)
|
configManager.setShowSeedWords(false)
|
||||||
cb(null, configManager.getSelectedAccount())
|
cb(null, configManager.getSelectedAccount())
|
||||||
}
|
}
|
||||||
|
|
||||||
IdentityStore.prototype.getState = function () {
|
IdentityStore.prototype.getState = function () {
|
||||||
|
const configManager = this.configManager
|
||||||
var seedWords = this.getSeedIfUnlocked()
|
var seedWords = this.getSeedIfUnlocked()
|
||||||
return clone(extend(this._currentState, {
|
return clone(extend(this._currentState, {
|
||||||
isInitialized: !!configManager.getWallet() && !seedWords,
|
isInitialized: !!configManager.getWallet() && !seedWords,
|
||||||
@ -99,6 +104,7 @@ IdentityStore.prototype.getState = function () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
IdentityStore.prototype.getSeedIfUnlocked = function () {
|
IdentityStore.prototype.getSeedIfUnlocked = function () {
|
||||||
|
const configManager = this.configManager
|
||||||
var showSeed = configManager.getShouldShowSeedWords()
|
var showSeed = configManager.getShouldShowSeedWords()
|
||||||
var idmgmt = this._idmgmt
|
var idmgmt = this._idmgmt
|
||||||
var shouldShow = showSeed && !!idmgmt
|
var shouldShow = showSeed && !!idmgmt
|
||||||
@ -107,10 +113,12 @@ IdentityStore.prototype.getSeedIfUnlocked = function () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
IdentityStore.prototype.getSelectedAddress = function () {
|
IdentityStore.prototype.getSelectedAddress = function () {
|
||||||
|
const configManager = this.configManager
|
||||||
return configManager.getSelectedAccount()
|
return configManager.getSelectedAccount()
|
||||||
}
|
}
|
||||||
|
|
||||||
IdentityStore.prototype.setSelectedAddress = function (address, cb) {
|
IdentityStore.prototype.setSelectedAddress = function (address, cb) {
|
||||||
|
const configManager = this.configManager
|
||||||
if (!address) {
|
if (!address) {
|
||||||
var addresses = this._getAddresses()
|
var addresses = this._getAddresses()
|
||||||
address = addresses[0]
|
address = addresses[0]
|
||||||
@ -123,6 +131,7 @@ IdentityStore.prototype.setSelectedAddress = function (address, cb) {
|
|||||||
IdentityStore.prototype.revealAccount = function (cb) {
|
IdentityStore.prototype.revealAccount = function (cb) {
|
||||||
const derivedKey = this._idmgmt.derivedKey
|
const derivedKey = this._idmgmt.derivedKey
|
||||||
const keyStore = this._keyStore
|
const keyStore = this._keyStore
|
||||||
|
const configManager = this.configManager
|
||||||
|
|
||||||
keyStore.setDefaultHdDerivationPath(this.hdPathString)
|
keyStore.setDefaultHdDerivationPath(this.hdPathString)
|
||||||
keyStore.generateNewAddress(derivedKey, 1)
|
keyStore.generateNewAddress(derivedKey, 1)
|
||||||
@ -158,6 +167,7 @@ IdentityStore.prototype.setLocked = function (cb) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
IdentityStore.prototype.submitPassword = function (password, cb) {
|
IdentityStore.prototype.submitPassword = function (password, cb) {
|
||||||
|
const configManager = this.configManager
|
||||||
this.tryPassword(password, (err) => {
|
this.tryPassword(password, (err) => {
|
||||||
if (err) return cb(err)
|
if (err) return cb(err)
|
||||||
// load identities before returning...
|
// load identities before returning...
|
||||||
@ -177,6 +187,7 @@ IdentityStore.prototype.exportAccount = function (address, cb) {
|
|||||||
|
|
||||||
// comes from dapp via zero-client hooked-wallet provider
|
// comes from dapp via zero-client hooked-wallet provider
|
||||||
IdentityStore.prototype.addUnconfirmedTransaction = function (txParams, onTxDoneCb, cb) {
|
IdentityStore.prototype.addUnconfirmedTransaction = function (txParams, onTxDoneCb, cb) {
|
||||||
|
const configManager = this.configManager
|
||||||
var self = this
|
var self = this
|
||||||
// create txData obj with parameters and meta data
|
// create txData obj with parameters and meta data
|
||||||
var time = (new Date()).getTime()
|
var time = (new Date()).getTime()
|
||||||
@ -227,6 +238,7 @@ IdentityStore.prototype.addUnconfirmedTransaction = function (txParams, onTxDone
|
|||||||
|
|
||||||
// comes from metamask ui
|
// comes from metamask ui
|
||||||
IdentityStore.prototype.approveTransaction = function (txId, cb) {
|
IdentityStore.prototype.approveTransaction = function (txId, cb) {
|
||||||
|
const configManager = this.configManager
|
||||||
var approvalCb = this._unconfTxCbs[txId] || noop
|
var approvalCb = this._unconfTxCbs[txId] || noop
|
||||||
|
|
||||||
// accept tx
|
// accept tx
|
||||||
@ -240,6 +252,7 @@ IdentityStore.prototype.approveTransaction = function (txId, cb) {
|
|||||||
|
|
||||||
// comes from metamask ui
|
// comes from metamask ui
|
||||||
IdentityStore.prototype.cancelTransaction = function (txId) {
|
IdentityStore.prototype.cancelTransaction = function (txId) {
|
||||||
|
const configManager = this.configManager
|
||||||
var approvalCb = this._unconfTxCbs[txId] || noop
|
var approvalCb = this._unconfTxCbs[txId] || noop
|
||||||
|
|
||||||
// reject tx
|
// reject tx
|
||||||
@ -347,6 +360,7 @@ IdentityStore.prototype._isUnlocked = function () {
|
|||||||
|
|
||||||
// load identities from keyStoreet
|
// load identities from keyStoreet
|
||||||
IdentityStore.prototype._loadIdentities = function () {
|
IdentityStore.prototype._loadIdentities = function () {
|
||||||
|
const configManager = this.configManager
|
||||||
if (!this._isUnlocked()) throw new Error('not unlocked')
|
if (!this._isUnlocked()) throw new Error('not unlocked')
|
||||||
|
|
||||||
var addresses = this._getAddresses()
|
var addresses = this._getAddresses()
|
||||||
@ -367,6 +381,7 @@ IdentityStore.prototype._loadIdentities = function () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
IdentityStore.prototype.saveAccountLabel = function (account, label, cb) {
|
IdentityStore.prototype.saveAccountLabel = function (account, label, cb) {
|
||||||
|
const configManager = this.configManager
|
||||||
configManager.setNicknameForWallet(account, label)
|
configManager.setNicknameForWallet(account, label)
|
||||||
this._loadIdentities()
|
this._loadIdentities()
|
||||||
cb(null, label)
|
cb(null, label)
|
||||||
@ -379,6 +394,7 @@ IdentityStore.prototype.saveAccountLabel = function (account, label, cb) {
|
|||||||
// If there is no balance and it mayBeFauceting,
|
// If there is no balance and it mayBeFauceting,
|
||||||
// then it is in fact fauceting.
|
// then it is in fact fauceting.
|
||||||
IdentityStore.prototype._mayBeFauceting = function (i) {
|
IdentityStore.prototype._mayBeFauceting = function (i) {
|
||||||
|
const configManager = this.configManager
|
||||||
var config = configManager.getProvider()
|
var config = configManager.getProvider()
|
||||||
if (i === 0 &&
|
if (i === 0 &&
|
||||||
config.type === 'rpc' &&
|
config.type === 'rpc' &&
|
||||||
@ -397,6 +413,7 @@ IdentityStore.prototype.tryPassword = function (password, cb) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
IdentityStore.prototype._createIdmgmt = function (password, seed, entropy, cb) {
|
IdentityStore.prototype._createIdmgmt = function (password, seed, entropy, cb) {
|
||||||
|
const configManager = this.configManager
|
||||||
var keyStore = null
|
var keyStore = null
|
||||||
LightwalletKeyStore.deriveKeyFromPassword(password, (err, derivedKey) => {
|
LightwalletKeyStore.deriveKeyFromPassword(password, (err, derivedKey) => {
|
||||||
if (err) return cb(err)
|
if (err) return cb(err)
|
||||||
@ -425,6 +442,7 @@ IdentityStore.prototype._createIdmgmt = function (password, seed, entropy, cb) {
|
|||||||
keyStore: keyStore,
|
keyStore: keyStore,
|
||||||
derivedKey: derivedKey,
|
derivedKey: derivedKey,
|
||||||
hdPathSTring: this.hdPathString,
|
hdPathSTring: this.hdPathString,
|
||||||
|
configManager: this.configManager,
|
||||||
})
|
})
|
||||||
|
|
||||||
cb()
|
cb()
|
||||||
@ -432,6 +450,7 @@ IdentityStore.prototype._createIdmgmt = function (password, seed, entropy, cb) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
IdentityStore.prototype._restoreFromSeed = function (password, seed, derivedKey) {
|
IdentityStore.prototype._restoreFromSeed = function (password, seed, derivedKey) {
|
||||||
|
const configManager = this.configManager
|
||||||
var keyStore = new LightwalletKeyStore(seed, derivedKey, this.hdPathString)
|
var keyStore = new LightwalletKeyStore(seed, derivedKey, this.hdPathString)
|
||||||
keyStore.addHdDerivationPath(this.hdPathString, derivedKey, {curve: 'secp256k1', purpose: 'sign'})
|
keyStore.addHdDerivationPath(this.hdPathString, derivedKey, {curve: 'secp256k1', purpose: 'sign'})
|
||||||
keyStore.setDefaultHdDerivationPath(this.hdPathString)
|
keyStore.setDefaultHdDerivationPath(this.hdPathString)
|
||||||
@ -443,6 +462,7 @@ IdentityStore.prototype._restoreFromSeed = function (password, seed, derivedKey)
|
|||||||
}
|
}
|
||||||
|
|
||||||
IdentityStore.prototype._createFirstWallet = function (entropy, derivedKey) {
|
IdentityStore.prototype._createFirstWallet = function (entropy, derivedKey) {
|
||||||
|
const configManager = this.configManager
|
||||||
var secretSeed = LightwalletKeyStore.generateRandomSeed(entropy)
|
var secretSeed = LightwalletKeyStore.generateRandomSeed(entropy)
|
||||||
var keyStore = new LightwalletKeyStore(secretSeed, derivedKey, this.hdPathString)
|
var keyStore = new LightwalletKeyStore(secretSeed, derivedKey, this.hdPathString)
|
||||||
keyStore.addHdDerivationPath(this.hdPathString, derivedKey, {curve: 'secp256k1', purpose: 'sign'})
|
keyStore.addHdDerivationPath(this.hdPathString, derivedKey, {curve: 'secp256k1', purpose: 'sign'})
|
||||||
|
@ -1,10 +1,11 @@
|
|||||||
const createId = require('hat')
|
const createId = require('hat')
|
||||||
|
const extend = require('xtend')
|
||||||
const unmountComponentAtNode = require('react-dom').unmountComponentAtNode
|
const unmountComponentAtNode = require('react-dom').unmountComponentAtNode
|
||||||
const findDOMNode = require('react-dom').findDOMNode
|
const findDOMNode = require('react-dom').findDOMNode
|
||||||
const render = require('react-dom').render
|
const render = require('react-dom').render
|
||||||
const h = require('react-hyperscript')
|
const h = require('react-hyperscript')
|
||||||
const uiUtils = require('../../../ui/app/util')
|
const PendingTxDetails = require('../../../ui/app/components/pending-tx-details')
|
||||||
const renderPendingTx = require('../../../ui/app/components/pending-tx').prototype.renderGeneric
|
const PendingMsgDetails = require('../../../ui/app/components/pending-msg-details')
|
||||||
const MetaMaskUiCss = require('../../../ui/css')
|
const MetaMaskUiCss = require('../../../ui/css')
|
||||||
var notificationHandlers = {}
|
var notificationHandlers = {}
|
||||||
|
|
||||||
@ -56,65 +57,9 @@ function createTxNotification (opts) {
|
|||||||
// guard for chrome bug https://github.com/MetaMask/metamask-plugin/issues/236
|
// guard for chrome bug https://github.com/MetaMask/metamask-plugin/issues/236
|
||||||
if (!chrome.notifications) return console.error('Chrome notifications API missing...')
|
if (!chrome.notifications) return console.error('Chrome notifications API missing...')
|
||||||
|
|
||||||
renderTransactionNotificationSVG(opts, function(err, source){
|
|
||||||
if (err) throw err
|
|
||||||
|
|
||||||
var imageUrl = 'data:image/svg+xml;utf8,' + encodeURIComponent(source)
|
|
||||||
|
|
||||||
var id = createId()
|
|
||||||
chrome.notifications.create(id, {
|
|
||||||
type: 'image',
|
|
||||||
// requireInteraction: true,
|
|
||||||
iconUrl: '/images/icon-128.png',
|
|
||||||
imageUrl: imageUrl,
|
|
||||||
title: opts.title,
|
|
||||||
message: '',
|
|
||||||
buttons: [{
|
|
||||||
title: 'confirm',
|
|
||||||
}, {
|
|
||||||
title: 'cancel',
|
|
||||||
}],
|
|
||||||
})
|
|
||||||
notificationHandlers[id] = {
|
|
||||||
confirm: opts.confirm,
|
|
||||||
cancel: opts.cancel,
|
|
||||||
}
|
|
||||||
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
function createMsgNotification (opts) {
|
|
||||||
// guard for chrome bug https://github.com/MetaMask/metamask-plugin/issues/236
|
|
||||||
if (!chrome.notifications) return console.error('Chrome notifications API missing...')
|
|
||||||
var message = [
|
|
||||||
'Submitted by ' + opts.msgParams.origin,
|
|
||||||
'to be signed by: ' + uiUtils.addressSummary(opts.msgParams.from),
|
|
||||||
'message:\n' + opts.msgParams.data,
|
|
||||||
].join('\n')
|
|
||||||
|
|
||||||
var id = createId()
|
|
||||||
chrome.notifications.create(id, {
|
|
||||||
type: 'basic',
|
|
||||||
requireInteraction: true,
|
|
||||||
iconUrl: '/images/icon-128.png',
|
|
||||||
title: opts.title,
|
|
||||||
message: message,
|
|
||||||
buttons: [{
|
|
||||||
title: 'confirm',
|
|
||||||
}, {
|
|
||||||
title: 'cancel',
|
|
||||||
}],
|
|
||||||
})
|
|
||||||
notificationHandlers[id] = {
|
|
||||||
confirm: opts.confirm,
|
|
||||||
cancel: opts.cancel,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function renderTransactionNotificationSVG(opts, cb){
|
|
||||||
var state = {
|
var state = {
|
||||||
nonInteractive: true,
|
title: 'New Unsigned Transaction',
|
||||||
inlineIdenticons: true,
|
imageifyIdenticons: false,
|
||||||
txData: {
|
txData: {
|
||||||
txParams: opts.txParams,
|
txParams: opts.txParams,
|
||||||
time: (new Date()).getTime(),
|
time: (new Date()).getTime(),
|
||||||
@ -125,8 +70,87 @@ function renderTransactionNotificationSVG(opts, cb){
|
|||||||
accounts: {
|
accounts: {
|
||||||
|
|
||||||
},
|
},
|
||||||
|
onConfirm: opts.confirm,
|
||||||
|
onCancel: opts.cancel,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
renderTxNotificationSVG(state, function(err, notificationSvgSource){
|
||||||
|
if (err) throw err
|
||||||
|
|
||||||
|
showNotification(extend(state, {
|
||||||
|
imageUrl: toSvgUri(notificationSvgSource),
|
||||||
|
}))
|
||||||
|
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function createMsgNotification (opts) {
|
||||||
|
// guard for chrome bug https://github.com/MetaMask/metamask-plugin/issues/236
|
||||||
|
if (!chrome.notifications) return console.error('Chrome notifications API missing...')
|
||||||
|
|
||||||
|
var state = {
|
||||||
|
title: 'New Unsigned Message',
|
||||||
|
imageifyIdenticons: false,
|
||||||
|
txData: {
|
||||||
|
msgParams: opts.msgParams,
|
||||||
|
time: (new Date()).getTime(),
|
||||||
|
},
|
||||||
|
identities: {
|
||||||
|
|
||||||
|
},
|
||||||
|
accounts: {
|
||||||
|
|
||||||
|
},
|
||||||
|
onConfirm: opts.confirm,
|
||||||
|
onCancel: opts.cancel,
|
||||||
|
}
|
||||||
|
|
||||||
|
renderMsgNotificationSVG(state, function(err, notificationSvgSource){
|
||||||
|
if (err) throw err
|
||||||
|
|
||||||
|
showNotification(extend(state, {
|
||||||
|
imageUrl: toSvgUri(notificationSvgSource),
|
||||||
|
}))
|
||||||
|
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function showNotification (state) {
|
||||||
|
// guard for chrome bug https://github.com/MetaMask/metamask-plugin/issues/236
|
||||||
|
if (!chrome.notifications) return console.error('Chrome notifications API missing...')
|
||||||
|
|
||||||
|
var id = createId()
|
||||||
|
chrome.notifications.create(id, {
|
||||||
|
type: 'image',
|
||||||
|
requireInteraction: true,
|
||||||
|
iconUrl: '/images/icon-128.png',
|
||||||
|
imageUrl: state.imageUrl,
|
||||||
|
title: state.title,
|
||||||
|
message: '',
|
||||||
|
buttons: [{
|
||||||
|
title: 'confirm',
|
||||||
|
}, {
|
||||||
|
title: 'cancel',
|
||||||
|
}],
|
||||||
|
})
|
||||||
|
notificationHandlers[id] = {
|
||||||
|
confirm: state.onConfirm,
|
||||||
|
cancel: state.onCancel,
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderTxNotificationSVG(state, cb){
|
||||||
|
var content = h(PendingTxDetails, state)
|
||||||
|
renderNotificationSVG(content, cb)
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderMsgNotificationSVG(state, cb){
|
||||||
|
var content = h(PendingMsgDetails, state)
|
||||||
|
renderNotificationSVG(content, cb)
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderNotificationSVG(content, cb){
|
||||||
var container = document.createElement('div')
|
var container = document.createElement('div')
|
||||||
var confirmView = h('div.app-primary', {
|
var confirmView = h('div.app-primary', {
|
||||||
style: {
|
style: {
|
||||||
@ -138,7 +162,7 @@ function renderTransactionNotificationSVG(opts, cb){
|
|||||||
},
|
},
|
||||||
}, [
|
}, [
|
||||||
h('style', MetaMaskUiCss()),
|
h('style', MetaMaskUiCss()),
|
||||||
renderPendingTx(h, state),
|
content,
|
||||||
])
|
])
|
||||||
|
|
||||||
render(confirmView, container, function ready(){
|
render(confirmView, container, function ready(){
|
||||||
@ -160,4 +184,8 @@ function svgWrapper(content){
|
|||||||
</svg>
|
</svg>
|
||||||
`
|
`
|
||||||
return wrapperSource.split('{{content}}').join(content)
|
return wrapperSource.split('{{content}}').join(content)
|
||||||
|
}
|
||||||
|
|
||||||
|
function toSvgUri(content){
|
||||||
|
return 'data:image/svg+xml;utf8,' + encodeURIComponent(content)
|
||||||
}
|
}
|
257
app/scripts/metamask-controller.js
Normal file
257
app/scripts/metamask-controller.js
Normal file
@ -0,0 +1,257 @@
|
|||||||
|
const extend = require('xtend')
|
||||||
|
const EthStore = require('eth-store')
|
||||||
|
const MetaMaskProvider = require('web3-provider-engine/zero.js')
|
||||||
|
const IdentityStore = require('./lib/idStore')
|
||||||
|
const messageManager = require('./lib/message-manager')
|
||||||
|
const HostStore = require('./lib/remote-store.js').HostStore
|
||||||
|
const Web3 = require('web3')
|
||||||
|
const ConfigManager = require('./lib/config-manager')
|
||||||
|
|
||||||
|
module.exports = class MetamaskController {
|
||||||
|
|
||||||
|
constructor (opts) {
|
||||||
|
this.opts = opts
|
||||||
|
this.configManager = new ConfigManager(opts)
|
||||||
|
this.idStore = new IdentityStore({
|
||||||
|
configManager: this.configManager,
|
||||||
|
})
|
||||||
|
this.provider = this.initializeProvider(opts)
|
||||||
|
this.ethStore = new EthStore(this.provider)
|
||||||
|
this.idStore.setStore(this.ethStore)
|
||||||
|
this.messageManager = messageManager
|
||||||
|
this.publicConfigStore = this.initPublicConfigStore()
|
||||||
|
}
|
||||||
|
|
||||||
|
getState () {
|
||||||
|
return extend(
|
||||||
|
this.ethStore.getState(),
|
||||||
|
this.idStore.getState(),
|
||||||
|
this.configManager.getConfig()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
getApi () {
|
||||||
|
const idStore = this.idStore
|
||||||
|
|
||||||
|
return {
|
||||||
|
getState: (cb) => { cb(null, this.getState()) },
|
||||||
|
setRpcTarget: this.setRpcTarget.bind(this),
|
||||||
|
setProviderType: this.setProviderType.bind(this),
|
||||||
|
useEtherscanProvider: this.useEtherscanProvider.bind(this),
|
||||||
|
agreeToDisclaimer: this.agreeToDisclaimer.bind(this),
|
||||||
|
// forward directly to idStore
|
||||||
|
createNewVault: idStore.createNewVault.bind(idStore),
|
||||||
|
recoverFromSeed: idStore.recoverFromSeed.bind(idStore),
|
||||||
|
submitPassword: idStore.submitPassword.bind(idStore),
|
||||||
|
setSelectedAddress: idStore.setSelectedAddress.bind(idStore),
|
||||||
|
approveTransaction: idStore.approveTransaction.bind(idStore),
|
||||||
|
cancelTransaction: idStore.cancelTransaction.bind(idStore),
|
||||||
|
signMessage: idStore.signMessage.bind(idStore),
|
||||||
|
cancelMessage: idStore.cancelMessage.bind(idStore),
|
||||||
|
setLocked: idStore.setLocked.bind(idStore),
|
||||||
|
clearSeedWordCache: idStore.clearSeedWordCache.bind(idStore),
|
||||||
|
exportAccount: idStore.exportAccount.bind(idStore),
|
||||||
|
revealAccount: idStore.revealAccount.bind(idStore),
|
||||||
|
saveAccountLabel: idStore.saveAccountLabel.bind(idStore),
|
||||||
|
tryPassword: idStore.tryPassword.bind(idStore),
|
||||||
|
recoverSeed: idStore.recoverSeed.bind(idStore),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setupProviderConnection (stream, originDomain) {
|
||||||
|
stream.on('data', this.onRpcRequest.bind(this, stream, originDomain))
|
||||||
|
}
|
||||||
|
|
||||||
|
onRpcRequest (stream, originDomain, request) {
|
||||||
|
var payloads = Array.isArray(request) ? request : [request]
|
||||||
|
payloads.forEach(function (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') {
|
||||||
|
payload.params.push({ origin: originDomain })
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
// handle rpc request
|
||||||
|
this.provider.sendAsync(request, function onPayloadHandled (err, response) {
|
||||||
|
if (err) {
|
||||||
|
return logger(err)
|
||||||
|
}
|
||||||
|
logger(null, request, response)
|
||||||
|
try {
|
||||||
|
stream.write(response)
|
||||||
|
} catch (err) {
|
||||||
|
logger(err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
function logger (err, request, response) {
|
||||||
|
if (err) return console.error(err.stack)
|
||||||
|
if (!request.isMetamaskInternal) {
|
||||||
|
console.log(`RPC (${originDomain}):`, request, '->', response)
|
||||||
|
if (response.error) console.error('Error in RPC response:\n' + response.error.message)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sendUpdate () {
|
||||||
|
if (this.remote) {
|
||||||
|
this.remote.sendUpdate(this.getState())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
initializeProvider (opts) {
|
||||||
|
const idStore = this.idStore
|
||||||
|
|
||||||
|
var providerOpts = {
|
||||||
|
rpcUrl: this.configManager.getCurrentRpcAddress(),
|
||||||
|
// account mgmt
|
||||||
|
getAccounts: (cb) => {
|
||||||
|
var selectedAddress = idStore.getSelectedAddress()
|
||||||
|
var result = selectedAddress ? [selectedAddress] : []
|
||||||
|
cb(null, result)
|
||||||
|
},
|
||||||
|
// tx signing
|
||||||
|
approveTransaction: this.newUnsignedTransaction.bind(this),
|
||||||
|
signTransaction: idStore.signTransaction.bind(idStore),
|
||||||
|
// msg signing
|
||||||
|
approveMessage: this.newUnsignedMessage.bind(this),
|
||||||
|
signMessage: idStore.signMessage.bind(idStore),
|
||||||
|
}
|
||||||
|
|
||||||
|
var provider = MetaMaskProvider(providerOpts)
|
||||||
|
var web3 = new Web3(provider)
|
||||||
|
idStore.web3 = web3
|
||||||
|
idStore.getNetwork()
|
||||||
|
|
||||||
|
provider.on('block', this.processBlock.bind(this))
|
||||||
|
provider.on('error', idStore.getNetwork.bind(idStore))
|
||||||
|
|
||||||
|
return provider
|
||||||
|
}
|
||||||
|
|
||||||
|
initPublicConfigStore () {
|
||||||
|
// get init state
|
||||||
|
var initPublicState = extend(
|
||||||
|
idStoreToPublic(this.idStore.getState()),
|
||||||
|
configToPublic(this.configManager.getConfig())
|
||||||
|
)
|
||||||
|
|
||||||
|
var publicConfigStore = new HostStore(initPublicState)
|
||||||
|
|
||||||
|
// subscribe to changes
|
||||||
|
this.configManager.subscribe(function (state) {
|
||||||
|
storeSetFromObj(publicConfigStore, configToPublic(state))
|
||||||
|
})
|
||||||
|
this.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])
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return publicConfigStore
|
||||||
|
}
|
||||||
|
|
||||||
|
newUnsignedTransaction (txParams, onTxDoneCb) {
|
||||||
|
const idStore = this.idStore
|
||||||
|
var state = idStore.getState()
|
||||||
|
|
||||||
|
// It's locked
|
||||||
|
if (!state.isUnlocked) {
|
||||||
|
this.opts.unlockAccountMessage()
|
||||||
|
idStore.addUnconfirmedTransaction(txParams, onTxDoneCb, noop)
|
||||||
|
|
||||||
|
// It's unlocked
|
||||||
|
} else {
|
||||||
|
idStore.addUnconfirmedTransaction(txParams, onTxDoneCb, (err, txData) => {
|
||||||
|
if (err) return onTxDoneCb(err)
|
||||||
|
this.opts.showUnconfirmedTx(txParams, txData, onTxDoneCb)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
newUnsignedMessage (msgParams, cb) {
|
||||||
|
var state = this.idStore.getState()
|
||||||
|
if (!state.isUnlocked) {
|
||||||
|
this.opts.unlockAccountMessage()
|
||||||
|
} else {
|
||||||
|
this.addUnconfirmedMsg(msgParams, cb)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
addUnconfirmedMessage (msgParams, cb) {
|
||||||
|
const idStore = this.idStore
|
||||||
|
const msgId = idStore.addUnconfirmedMessage(msgParams, cb)
|
||||||
|
this.opts.showUnconfirmedMessage(msgParams, msgId)
|
||||||
|
}
|
||||||
|
|
||||||
|
setupPublicConfig (stream) {
|
||||||
|
var storeStream = this.publicConfigStore.createStream()
|
||||||
|
stream.pipe(storeStream).pipe(stream)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Log blocks
|
||||||
|
processBlock (block) {
|
||||||
|
console.log(`BLOCK CHANGED: #${block.number.toString('hex')} 0x${block.hash.toString('hex')}`)
|
||||||
|
this.verifyNetwork()
|
||||||
|
}
|
||||||
|
|
||||||
|
verifyNetwork () {
|
||||||
|
// Check network when restoring connectivity:
|
||||||
|
if (this.idStore._currentState.network === 'loading') {
|
||||||
|
this.idStore.getNetwork()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// config
|
||||||
|
//
|
||||||
|
|
||||||
|
agreeToDisclaimer (cb) {
|
||||||
|
try {
|
||||||
|
this.configManager.setConfirmed(true)
|
||||||
|
cb()
|
||||||
|
} catch (e) {
|
||||||
|
cb(e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// called from popup
|
||||||
|
setRpcTarget (rpcTarget) {
|
||||||
|
this.configManager.setRpcTarget(rpcTarget)
|
||||||
|
chrome.runtime.reload()
|
||||||
|
this.idStore.getNetwork()
|
||||||
|
}
|
||||||
|
|
||||||
|
setProviderType (type) {
|
||||||
|
this.configManager.setProviderType(type)
|
||||||
|
chrome.runtime.reload()
|
||||||
|
this.idStore.getNetwork()
|
||||||
|
}
|
||||||
|
|
||||||
|
useEtherscanProvider () {
|
||||||
|
this.configManager.useEtherscanProvider()
|
||||||
|
chrome.runtime.reload()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function noop () {}
|
57
test/lib/mock-config-manager.js
Normal file
57
test/lib/mock-config-manager.js
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
var ConfigManager = require('../../app/scripts/lib/config-manager')
|
||||||
|
const STORAGE_KEY = 'metamask-persistance-key'
|
||||||
|
const extend = require('xtend')
|
||||||
|
|
||||||
|
module.exports = function() {
|
||||||
|
return new ConfigManager({ loadData, setData })
|
||||||
|
}
|
||||||
|
|
||||||
|
function loadData () {
|
||||||
|
var oldData = getOldStyleData()
|
||||||
|
var newData
|
||||||
|
try {
|
||||||
|
newData = JSON.parse(window.localStorage[STORAGE_KEY])
|
||||||
|
} catch (e) {}
|
||||||
|
|
||||||
|
var data = extend({
|
||||||
|
meta: {
|
||||||
|
version: 0,
|
||||||
|
},
|
||||||
|
data: {
|
||||||
|
config: {
|
||||||
|
provider: {
|
||||||
|
type: 'testnet',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}, oldData || null, newData || null)
|
||||||
|
return data
|
||||||
|
}
|
||||||
|
|
||||||
|
function getOldStyleData () {
|
||||||
|
var config, wallet, seedWords
|
||||||
|
|
||||||
|
var result = {
|
||||||
|
meta: { version: 0 },
|
||||||
|
data: {},
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
config = JSON.parse(window.localStorage['config'])
|
||||||
|
result.data.config = config
|
||||||
|
} catch (e) {}
|
||||||
|
try {
|
||||||
|
wallet = JSON.parse(window.localStorage['lightwallet'])
|
||||||
|
result.data.wallet = wallet
|
||||||
|
} catch (e) {}
|
||||||
|
try {
|
||||||
|
seedWords = window.localStorage['seedWords']
|
||||||
|
result.data.seedWords = seedWords
|
||||||
|
} catch (e) {}
|
||||||
|
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
function setData (data) {
|
||||||
|
window.localStorage[STORAGE_KEY] = JSON.stringify(data)
|
||||||
|
}
|
@ -1,12 +1,14 @@
|
|||||||
var assert = require('assert')
|
var assert = require('assert')
|
||||||
var ConfigManager = require('../../app/scripts/lib/config-manager')
|
const extend = require('xtend')
|
||||||
|
const STORAGE_KEY = 'metamask-persistance-key'
|
||||||
|
var configManagerGen = require('../lib/mock-config-manager')
|
||||||
var configManager
|
var configManager
|
||||||
|
|
||||||
describe('config-manager', function() {
|
describe('config-manager', function() {
|
||||||
|
|
||||||
beforeEach(function() {
|
beforeEach(function() {
|
||||||
window.localStorage = {} // Hacking localStorage support into JSDom
|
window.localStorage = {} // Hacking localStorage support into JSDom
|
||||||
configManager = new ConfigManager()
|
configManager = configManagerGen()
|
||||||
})
|
})
|
||||||
|
|
||||||
describe('confirmation', function() {
|
describe('confirmation', function() {
|
||||||
@ -209,3 +211,4 @@ describe('config-manager', function() {
|
|||||||
})
|
})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
var assert = require('assert')
|
var assert = require('assert')
|
||||||
var IdentityStore = require('../../app/scripts/lib/idStore')
|
var IdentityStore = require('../../app/scripts/lib/idStore')
|
||||||
|
var configManagerGen = require('../lib/mock-config-manager')
|
||||||
|
|
||||||
describe('IdentityStore', function() {
|
describe('IdentityStore', function() {
|
||||||
|
|
||||||
@ -15,6 +16,7 @@ describe('IdentityStore', function() {
|
|||||||
window.localStorage = {} // Hacking localStorage support into JSDom
|
window.localStorage = {} // Hacking localStorage support into JSDom
|
||||||
|
|
||||||
idStore = new IdentityStore({
|
idStore = new IdentityStore({
|
||||||
|
configManager: configManagerGen(),
|
||||||
ethStore: {
|
ethStore: {
|
||||||
addAccount(acct) { accounts.push(acct) },
|
addAccount(acct) { accounts.push(acct) },
|
||||||
},
|
},
|
||||||
@ -34,6 +36,7 @@ describe('IdentityStore', function() {
|
|||||||
window.localStorage = {} // Hacking localStorage support into JSDom
|
window.localStorage = {} // Hacking localStorage support into JSDom
|
||||||
|
|
||||||
idStore = new IdentityStore({
|
idStore = new IdentityStore({
|
||||||
|
configManager: configManagerGen(),
|
||||||
ethStore: {
|
ethStore: {
|
||||||
addAccount(acct) { newAccounts.push(acct) },
|
addAccount(acct) { newAccounts.push(acct) },
|
||||||
},
|
},
|
||||||
@ -65,6 +68,7 @@ describe('IdentityStore', function() {
|
|||||||
window.localStorage = {} // Hacking localStorage support into JSDom
|
window.localStorage = {} // Hacking localStorage support into JSDom
|
||||||
|
|
||||||
idStore = new IdentityStore({
|
idStore = new IdentityStore({
|
||||||
|
configManager: configManagerGen(),
|
||||||
ethStore: {
|
ethStore: {
|
||||||
addAccount(acct) { accounts.push(acct) },
|
addAccount(acct) { accounts.push(acct) },
|
||||||
},
|
},
|
||||||
|
@ -46,7 +46,7 @@ AccountPanel.prototype.render = function () {
|
|||||||
h('.identicon-wrapper.flex-column.select-none', [
|
h('.identicon-wrapper.flex-column.select-none', [
|
||||||
h(Identicon, {
|
h(Identicon, {
|
||||||
address: panelState.identiconKey,
|
address: panelState.identiconKey,
|
||||||
imageify: !state.inlineIdenticons,
|
imageify: state.imageifyIdenticons,
|
||||||
}),
|
}),
|
||||||
h('span.font-small', panelState.identiconLabel),
|
h('span.font-small', panelState.identiconLabel),
|
||||||
]),
|
]),
|
||||||
|
53
ui/app/components/pending-msg-details.js
Normal file
53
ui/app/components/pending-msg-details.js
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
const Component = require('react').Component
|
||||||
|
const h = require('react-hyperscript')
|
||||||
|
const inherits = require('util').inherits
|
||||||
|
|
||||||
|
const AccountPanel = require('./account-panel')
|
||||||
|
const readableDate = require('../util').readableDate
|
||||||
|
|
||||||
|
module.exports = PendingMsgDetails
|
||||||
|
|
||||||
|
inherits(PendingMsgDetails, Component)
|
||||||
|
function PendingMsgDetails () {
|
||||||
|
Component.call(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
PendingMsgDetails.prototype.render = function () {
|
||||||
|
var state = this.props
|
||||||
|
var msgData = state.txData
|
||||||
|
|
||||||
|
var msgParams = msgData.msgParams || {}
|
||||||
|
var address = msgParams.from || state.selectedAddress
|
||||||
|
var identity = state.identities[address] || { address: address }
|
||||||
|
var account = state.accounts[address] || { address: address }
|
||||||
|
|
||||||
|
return (
|
||||||
|
h('div', {
|
||||||
|
key: msgData.id,
|
||||||
|
}, [
|
||||||
|
|
||||||
|
// account that will sign
|
||||||
|
h(AccountPanel, {
|
||||||
|
showFullAddress: true,
|
||||||
|
identity: identity,
|
||||||
|
account: account,
|
||||||
|
imageifyIdenticons: state.imageifyIdenticons,
|
||||||
|
}),
|
||||||
|
|
||||||
|
// message data
|
||||||
|
h('.tx-data.flex-column.flex-justify-center.flex-grow.select-none', [
|
||||||
|
h('.flex-row.flex-space-between', [
|
||||||
|
h('label.font-small', 'DATE'),
|
||||||
|
h('span.font-small', readableDate(msgData.time)),
|
||||||
|
]),
|
||||||
|
|
||||||
|
h('.flex-row.flex-space-between', [
|
||||||
|
h('label.font-small', 'MESSAGE'),
|
||||||
|
h('span.font-small', msgParams.data),
|
||||||
|
]),
|
||||||
|
]),
|
||||||
|
|
||||||
|
])
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
@ -1,9 +1,7 @@
|
|||||||
const Component = require('react').Component
|
const Component = require('react').Component
|
||||||
const h = require('react-hyperscript')
|
const h = require('react-hyperscript')
|
||||||
const inherits = require('util').inherits
|
const inherits = require('util').inherits
|
||||||
|
const PendingTxDetails = require('./pending-msg-details')
|
||||||
const AccountPanel = require('./account-panel')
|
|
||||||
const readableDate = require('../util').readableDate
|
|
||||||
|
|
||||||
module.exports = PendingMsg
|
module.exports = PendingMsg
|
||||||
|
|
||||||
@ -16,16 +14,13 @@ PendingMsg.prototype.render = function () {
|
|||||||
var state = this.props
|
var state = this.props
|
||||||
var msgData = state.txData
|
var msgData = state.txData
|
||||||
|
|
||||||
var msgParams = msgData.msgParams || {}
|
|
||||||
var address = msgParams.from || state.selectedAddress
|
|
||||||
var identity = state.identities[address] || { address: address }
|
|
||||||
var account = state.accounts[address] || { address: address }
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
h('.transaction', {
|
|
||||||
|
h('div', {
|
||||||
key: msgData.id,
|
key: msgData.id,
|
||||||
}, [
|
}, [
|
||||||
|
|
||||||
|
// header
|
||||||
h('h3', {
|
h('h3', {
|
||||||
style: {
|
style: {
|
||||||
fontWeight: 'bold',
|
fontWeight: 'bold',
|
||||||
@ -33,27 +28,10 @@ PendingMsg.prototype.render = function () {
|
|||||||
},
|
},
|
||||||
}, 'Sign Message'),
|
}, 'Sign Message'),
|
||||||
|
|
||||||
// account that will sign
|
// message details
|
||||||
h(AccountPanel, {
|
h(PendingTxDetails, state),
|
||||||
showFullAddress: true,
|
|
||||||
identity: identity,
|
|
||||||
account: account,
|
|
||||||
}),
|
|
||||||
|
|
||||||
// tx data
|
// sign + cancel
|
||||||
h('.tx-data.flex-column.flex-justify-center.flex-grow.select-none', [
|
|
||||||
h('.flex-row.flex-space-between', [
|
|
||||||
h('label.font-small', 'DATE'),
|
|
||||||
h('span.font-small', readableDate(msgData.time)),
|
|
||||||
]),
|
|
||||||
|
|
||||||
h('.flex-row.flex-space-between', [
|
|
||||||
h('label.font-small', 'MESSAGE'),
|
|
||||||
h('span.font-small', msgParams.data),
|
|
||||||
]),
|
|
||||||
]),
|
|
||||||
|
|
||||||
// send + cancel
|
|
||||||
h('.flex-row.flex-space-around', [
|
h('.flex-row.flex-space-around', [
|
||||||
h('button', {
|
h('button', {
|
||||||
onClick: state.cancelMessage,
|
onClick: state.cancelMessage,
|
||||||
@ -63,6 +41,7 @@ PendingMsg.prototype.render = function () {
|
|||||||
}, 'Sign'),
|
}, 'Sign'),
|
||||||
]),
|
]),
|
||||||
])
|
])
|
||||||
|
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
65
ui/app/components/pending-tx-details.js
Normal file
65
ui/app/components/pending-tx-details.js
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
const Component = require('react').Component
|
||||||
|
const h = require('react-hyperscript')
|
||||||
|
const inherits = require('util').inherits
|
||||||
|
|
||||||
|
const AccountPanel = require('./account-panel')
|
||||||
|
const addressSummary = require('../util').addressSummary
|
||||||
|
const readableDate = require('../util').readableDate
|
||||||
|
const formatBalance = require('../util').formatBalance
|
||||||
|
|
||||||
|
module.exports = PendingTxDetails
|
||||||
|
|
||||||
|
inherits(PendingTxDetails, Component)
|
||||||
|
function PendingTxDetails () {
|
||||||
|
Component.call(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
PendingTxDetails.prototype.render = function () {
|
||||||
|
var state = this.props
|
||||||
|
return this.renderGeneric(h, state)
|
||||||
|
}
|
||||||
|
|
||||||
|
PendingTxDetails.prototype.renderGeneric = function (h, state) {
|
||||||
|
var txData = state.txData
|
||||||
|
|
||||||
|
var txParams = txData.txParams || {}
|
||||||
|
var address = txParams.from || state.selectedAddress
|
||||||
|
var identity = state.identities[address] || { address: address }
|
||||||
|
var account = state.accounts[address] || { address: address }
|
||||||
|
|
||||||
|
return (
|
||||||
|
|
||||||
|
h('div', [
|
||||||
|
|
||||||
|
// account that will sign
|
||||||
|
h(AccountPanel, {
|
||||||
|
showFullAddress: true,
|
||||||
|
identity: identity,
|
||||||
|
account: account,
|
||||||
|
imageifyIdenticons: state.imageifyIdenticons,
|
||||||
|
}),
|
||||||
|
|
||||||
|
// tx data
|
||||||
|
h('.tx-data.flex-column.flex-justify-center.flex-grow.select-none', [
|
||||||
|
|
||||||
|
h('.flex-row.flex-space-between', [
|
||||||
|
h('label.font-small', 'TO ADDRESS'),
|
||||||
|
h('span.font-small', addressSummary(txParams.to)),
|
||||||
|
]),
|
||||||
|
|
||||||
|
h('.flex-row.flex-space-between', [
|
||||||
|
h('label.font-small', 'DATE'),
|
||||||
|
h('span.font-small', readableDate(txData.time)),
|
||||||
|
]),
|
||||||
|
|
||||||
|
h('.flex-row.flex-space-between', [
|
||||||
|
h('label.font-small', 'AMOUNT'),
|
||||||
|
h('span.font-small', formatBalance(txParams.value)),
|
||||||
|
]),
|
||||||
|
]),
|
||||||
|
|
||||||
|
])
|
||||||
|
|
||||||
|
)
|
||||||
|
|
||||||
|
}
|
@ -1,11 +1,8 @@
|
|||||||
const Component = require('react').Component
|
const Component = require('react').Component
|
||||||
const h = require('react-hyperscript')
|
const h = require('react-hyperscript')
|
||||||
const inherits = require('util').inherits
|
const inherits = require('util').inherits
|
||||||
|
const PendingTxDetails = require('./pending-tx-details')
|
||||||
|
|
||||||
const AccountPanel = require('./account-panel')
|
|
||||||
const addressSummary = require('../util').addressSummary
|
|
||||||
const readableDate = require('../util').readableDate
|
|
||||||
const formatBalance = require('../util').formatBalance
|
|
||||||
|
|
||||||
module.exports = PendingTx
|
module.exports = PendingTx
|
||||||
|
|
||||||
@ -16,23 +13,15 @@ function PendingTx () {
|
|||||||
|
|
||||||
PendingTx.prototype.render = function () {
|
PendingTx.prototype.render = function () {
|
||||||
var state = this.props
|
var state = this.props
|
||||||
return this.renderGeneric(h, state)
|
|
||||||
}
|
|
||||||
|
|
||||||
PendingTx.prototype.renderGeneric = function (h, state) {
|
|
||||||
var txData = state.txData
|
var txData = state.txData
|
||||||
|
|
||||||
var txParams = txData.txParams || {}
|
|
||||||
var address = txParams.from || state.selectedAddress
|
|
||||||
var identity = state.identities[address] || { address: address }
|
|
||||||
var account = state.accounts[address] || { address: address }
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|
||||||
h('.transaction', {
|
h('div', {
|
||||||
key: txData.id,
|
key: txData.id,
|
||||||
}, [
|
}, [
|
||||||
|
|
||||||
|
// header
|
||||||
h('h3', {
|
h('h3', {
|
||||||
style: {
|
style: {
|
||||||
fontWeight: 'bold',
|
fontWeight: 'bold',
|
||||||
@ -40,53 +29,21 @@ PendingTx.prototype.renderGeneric = function (h, state) {
|
|||||||
},
|
},
|
||||||
}, 'Submit Transaction'),
|
}, 'Submit Transaction'),
|
||||||
|
|
||||||
// account that will sign
|
// tx info
|
||||||
h(AccountPanel, {
|
h(PendingTxDetails, state),
|
||||||
showFullAddress: true,
|
|
||||||
identity: identity,
|
|
||||||
account: account,
|
|
||||||
inlineIdenticons: state.inlineIdenticons,
|
|
||||||
}),
|
|
||||||
|
|
||||||
// tx data
|
|
||||||
h('.tx-data.flex-column.flex-justify-center.flex-grow.select-none', [
|
|
||||||
|
|
||||||
h('.flex-row.flex-space-between', [
|
|
||||||
h('label.font-small', 'TO ADDRESS'),
|
|
||||||
h('span.font-small', addressSummary(txParams.to)),
|
|
||||||
]),
|
|
||||||
|
|
||||||
h('.flex-row.flex-space-between', [
|
|
||||||
h('label.font-small', 'DATE'),
|
|
||||||
h('span.font-small', readableDate(txData.time)),
|
|
||||||
]),
|
|
||||||
|
|
||||||
h('.flex-row.flex-space-between', [
|
|
||||||
h('label.font-small', 'AMOUNT'),
|
|
||||||
h('span.font-small', formatBalance(txParams.value)),
|
|
||||||
]),
|
|
||||||
]),
|
|
||||||
|
|
||||||
// send + cancel
|
// send + cancel
|
||||||
state.nonInteractive ? null : actionButtons(state),
|
h('.flex-row.flex-space-around', [
|
||||||
|
h('button', {
|
||||||
|
onClick: state.cancelTransaction,
|
||||||
|
}, 'Cancel'),
|
||||||
|
h('button', {
|
||||||
|
onClick: state.sendTransaction,
|
||||||
|
}, 'Send'),
|
||||||
|
]),
|
||||||
|
|
||||||
])
|
])
|
||||||
|
|
||||||
)
|
)
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function actionButtons(state){
|
|
||||||
return (
|
|
||||||
|
|
||||||
h('.flex-row.flex-space-around', [
|
|
||||||
h('button', {
|
|
||||||
onClick: state.cancelTransaction,
|
|
||||||
}, 'Cancel'),
|
|
||||||
h('button', {
|
|
||||||
onClick: state.sendTransaction,
|
|
||||||
}, 'Send'),
|
|
||||||
])
|
|
||||||
|
|
||||||
)
|
|
||||||
}
|
|
@ -6,7 +6,7 @@ const connect = require('react-redux').connect
|
|||||||
const actions = require('./actions')
|
const actions = require('./actions')
|
||||||
const txHelper = require('../lib/tx-helper')
|
const txHelper = require('../lib/tx-helper')
|
||||||
|
|
||||||
const ConfirmTx = require('./components/pending-tx')
|
const PendingTx = require('./components/pending-tx')
|
||||||
const PendingMsg = require('./components/pending-msg')
|
const PendingMsg = require('./components/pending-msg')
|
||||||
|
|
||||||
module.exports = connect(mapStateToProps)(ConfirmTxScreen)
|
module.exports = connect(mapStateToProps)(ConfirmTxScreen)
|
||||||
@ -101,7 +101,7 @@ ConfirmTxScreen.prototype.render = function () {
|
|||||||
function currentTxView (opts) {
|
function currentTxView (opts) {
|
||||||
if ('txParams' in opts.txData) {
|
if ('txParams' in opts.txData) {
|
||||||
// This is a pending transaction
|
// This is a pending transaction
|
||||||
return h(ConfirmTx, opts)
|
return h(PendingTx, opts)
|
||||||
} else if ('msgParams' in opts.txData) {
|
} else if ('msgParams' in opts.txData) {
|
||||||
// This is a pending message to sign
|
// This is a pending message to sign
|
||||||
return h(PendingMsg, opts)
|
return h(PendingMsg, opts)
|
||||||
|
Loading…
Reference in New Issue
Block a user