1
0
mirror of https://github.com/kremalicious/metamask-extension.git synced 2024-12-20 00:13:22 +01:00
metamask-extension/app/scripts/lib/idStore.js

508 lines
14 KiB
JavaScript
Raw Normal View History

2016-02-11 02:44:46 +01:00
const EventEmitter = require('events').EventEmitter
const inherits = require('util').inherits
const async = require('async')
2016-06-17 04:51:34 +02:00
const ethUtil = require('ethereumjs-util')
const EthQuery = require('eth-query')
2016-03-03 07:58:23 +01:00
const LightwalletKeyStore = require('eth-lightwallet').keystore
2016-02-11 02:44:46 +01:00
const clone = require('clone')
const extend = require('xtend')
const createId = require('web3-provider-engine/util/random-id')
2016-06-17 04:51:34 +02:00
const ethBinToOps = require('eth-bin-to-ops')
const autoFaucet = require('./auto-faucet')
const messageManager = require('./message-manager')
const DEFAULT_RPC = 'https://testrpc.metamask.io/'
const IdManagement = require('./id-management')
2016-02-11 02:44:46 +01:00
module.exports = IdentityStore
inherits(IdentityStore, EventEmitter)
2016-06-21 22:18:32 +02:00
function IdentityStore (opts = {}) {
EventEmitter.call(this)
2016-02-11 02:44:46 +01:00
// we just use the ethStore to auto-add accounts
this._ethStore = opts.ethStore
this.configManager = opts.configManager
2016-03-03 07:58:23 +01:00
// lightwallet key store
this._keyStore = null
2016-03-03 07:58:23 +01:00
// lightwallet wrapper
this._idmgmt = null
this.hdPathString = "m/44'/60'/0'/0"
2016-02-11 02:44:46 +01:00
this._currentState = {
2016-02-11 02:44:46 +01:00
selectedAddress: null,
identities: {},
}
2016-02-11 02:44:46 +01:00
// not part of serilized metamask state - only kept in memory
this._unconfTxCbs = {}
this._unconfMsgCbs = {}
2016-02-11 02:44:46 +01:00
}
//
// public
//
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.createNewVault = function (password, entropy, cb) {
delete this._keyStore
2016-06-25 01:13:27 +02:00
this._createIdmgmt(password, null, entropy, (err) => {
2016-03-03 07:58:23 +01:00
if (err) return cb(err)
this._loadIdentities()
this._didUpdate()
this._autoFaucet()
this.configManager.setShowSeedWords(true)
var seedWords = this._idmgmt.getSeed()
2016-03-03 07:58:23 +01:00
cb(null, seedWords)
})
2016-02-17 09:55:57 +01:00
}
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.recoverSeed = function (cb) {
this.configManager.setShowSeedWords(true)
2016-06-03 02:11:10 +02:00
if (!this._idmgmt) return cb(new Error('Unauthenticated. Please sign in.'))
var seedWords = this._idmgmt.getSeed()
cb(null, seedWords)
}
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.recoverFromSeed = function (password, seed, cb) {
this._createIdmgmt(password, seed, null, (err) => {
2016-03-15 21:39:12 +01:00
if (err) return cb(err)
this._loadIdentities()
this._didUpdate()
cb(null, this.getState())
2016-03-15 21:39:12 +01:00
})
}
2016-02-17 09:55:57 +01:00
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.setStore = function (store) {
this._ethStore = store
2016-02-11 02:44:46 +01:00
}
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.clearSeedWordCache = function (cb) {
const configManager = this.configManager
configManager.setShowSeedWords(false)
cb(null, configManager.getSelectedAccount())
}
2016-02-17 09:55:57 +01:00
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.getState = function () {
const configManager = this.configManager
var seedWords = this.getSeedIfUnlocked()
return clone(extend(this._currentState, {
isInitialized: !!configManager.getWallet() && !seedWords,
isUnlocked: this._isUnlocked(),
seedWords: seedWords,
isConfirmed: configManager.getConfirmed(),
isEthConfirmed: configManager.getShouldntShowWarning(),
unconfTxs: configManager.unconfirmedTxs(),
2016-04-19 02:19:58 +02:00
transactions: configManager.getTxList(),
unconfMsgs: messageManager.unconfirmedMsgs(),
messages: messageManager.getMsgList(),
selectedAddress: configManager.getSelectedAccount(),
2016-02-11 02:44:46 +01:00
}))
}
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.getSeedIfUnlocked = function () {
const configManager = this.configManager
var showSeed = configManager.getShouldShowSeedWords()
var idmgmt = this._idmgmt
var shouldShow = showSeed && !!idmgmt
var seedWords = shouldShow ? idmgmt.getSeed() : null
return seedWords
}
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.getSelectedAddress = function () {
const configManager = this.configManager
return configManager.getSelectedAccount()
2016-02-11 02:44:46 +01:00
}
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.setSelectedAddress = function (address, cb) {
const configManager = this.configManager
if (!address) {
var addresses = this._getAddresses()
address = addresses[0]
}
configManager.setSelectedAccount(address)
if (cb) return cb(null, address)
2016-02-11 02:44:46 +01:00
}
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.revealAccount = function (cb) {
const derivedKey = this._idmgmt.derivedKey
const keyStore = this._keyStore
const configManager = this.configManager
keyStore.setDefaultHdDerivationPath(this.hdPathString)
keyStore.generateNewAddress(derivedKey, 1)
configManager.setWallet(keyStore.serialize())
this._loadIdentities()
this._didUpdate()
cb(null)
}
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.getNetwork = function (err) {
if (err) {
this._currentState.network = 'loading'
2016-06-03 22:58:09 +02:00
this._didUpdate()
2016-05-30 20:22:19 +02:00
}
this.web3.version.getNetwork((err, network) => {
if (err) {
this._currentState.network = 'loading'
return this._didUpdate()
}
console.log('web3.getNetwork returned ' + network)
this._currentState.network = network
2016-06-03 22:58:09 +02:00
this._didUpdate()
})
}
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.setLocked = function (cb) {
delete this._keyStore
delete this._idmgmt
cb()
2016-02-11 02:44:46 +01:00
}
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.submitPassword = function (password, cb) {
const configManager = this.configManager
this.tryPassword(password, (err) => {
2016-02-11 02:44:46 +01:00
if (err) return cb(err)
// load identities before returning...
this._loadIdentities()
cb(null, configManager.getSelectedAccount())
2016-02-11 02:44:46 +01:00
})
}
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.exportAccount = function (address, cb) {
2016-04-06 21:01:10 +02:00
var privateKey = this._idmgmt.exportPrivateKey(address)
cb(null, privateKey)
}
//
// Transactions
//
2016-03-03 07:58:23 +01:00
// comes from dapp via zero-client hooked-wallet provider
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.addUnconfirmedTransaction = function (txParams, onTxDoneCb, cb) {
const configManager = this.configManager
2016-06-17 04:51:34 +02:00
var self = this
2016-03-03 07:58:23 +01:00
// create txData obj with parameters and meta data
var time = (new Date()).getTime()
var txId = createId()
txParams.metamaskId = txId
2016-06-17 04:51:34 +02:00
txParams.metamaskNetworkId = self._currentState.network
2016-03-03 07:58:23 +01:00
var txData = {
id: txId,
txParams: txParams,
time: time,
status: 'unconfirmed',
}
console.log('addUnconfirmedTransaction:', txData)
2016-02-11 02:44:46 +01:00
2016-06-17 04:51:34 +02:00
// keep the onTxDoneCb around for after approval/denial (requires user interaction)
// This onTxDoneCb fires completion to the Dapp's write operation.
self._unconfTxCbs[txId] = onTxDoneCb
var provider = self._ethStore._query.currentProvider
var query = new EthQuery(provider)
// calculate metadata for tx
async.parallel([
analyzeForDelegateCall,
estimateGas,
], didComplete)
// perform static analyis on the target contract code
function analyzeForDelegateCall(cb){
if (txParams.to) {
query.getCode(txParams.to, function (err, result) {
if (err) return cb(err)
var code = ethUtil.toBuffer(result)
if (code !== '0x') {
var ops = ethBinToOps(code)
var containsDelegateCall = ops.some((op) => op.name === 'DELEGATECALL')
txData.containsDelegateCall = containsDelegateCall
cb()
} else {
cb()
}
})
} else {
cb()
}
}
function estimateGas(cb){
query.estimateGas(txParams, function(err, result){
if (err) return cb(err)
txData.estimatedGas = result
cb()
2016-06-17 04:51:34 +02:00
})
}
2016-02-11 02:44:46 +01:00
2016-06-21 22:18:32 +02:00
function didComplete (err) {
2016-06-17 04:51:34 +02:00
if (err) return cb(err)
configManager.addTx(txData)
2016-06-17 04:51:34 +02:00
// signal update
self._didUpdate()
// signal completion of add tx
cb(null, txData)
}
2016-02-11 02:44:46 +01:00
}
2016-03-03 07:58:23 +01:00
// comes from metamask ui
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.approveTransaction = function (txId, cb) {
const configManager = this.configManager
var approvalCb = this._unconfTxCbs[txId] || noop
2016-03-03 07:58:23 +01:00
// accept tx
2016-02-13 02:57:10 +01:00
cb()
2016-03-03 07:58:23 +01:00
approvalCb(null, true)
// clean up
configManager.confirmTx(txId)
delete this._unconfTxCbs[txId]
this._didUpdate()
2016-02-13 02:57:10 +01:00
}
2016-03-03 07:58:23 +01:00
// comes from metamask ui
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.cancelTransaction = function (txId) {
const configManager = this.configManager
var approvalCb = this._unconfTxCbs[txId] || noop
2016-03-03 07:58:23 +01:00
// reject tx
approvalCb(null, false)
// clean up
configManager.rejectTx(txId)
delete this._unconfTxCbs[txId]
this._didUpdate()
2016-02-12 21:55:20 +01:00
}
2016-03-03 07:58:23 +01:00
// performs the actual signing, no autofill of params
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.signTransaction = function (txParams, cb) {
2016-02-11 02:44:46 +01:00
try {
2016-03-03 07:58:23 +01:00
console.log('signing tx...', txParams)
var rawTx = this._idmgmt.signTx(txParams)
2016-03-03 07:58:23 +01:00
cb(null, rawTx)
2016-02-11 02:44:46 +01:00
} catch (err) {
cb(err)
}
}
//
// Messages
//
// comes from dapp via zero-client hooked-wallet provider
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.addUnconfirmedMessage = function (msgParams, cb) {
// create txData obj with parameters and meta data
var time = (new Date()).getTime()
var msgId = createId()
var msgData = {
id: msgId,
msgParams: msgParams,
time: time,
status: 'unconfirmed',
}
messageManager.addMsg(msgData)
console.log('addUnconfirmedMessage:', msgData)
// keep the cb around for after approval (requires user interaction)
// This cb fires completion to the Dapp's write operation.
this._unconfMsgCbs[msgId] = cb
// signal update
this._didUpdate()
return msgId
}
// comes from metamask ui
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.approveMessage = function (msgId, cb) {
var approvalCb = this._unconfMsgCbs[msgId] || noop
// accept msg
cb()
approvalCb(null, true)
// clean up
messageManager.confirmMsg(msgId)
delete this._unconfMsgCbs[msgId]
this._didUpdate()
}
// comes from metamask ui
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.cancelMessage = function (msgId) {
var approvalCb = this._unconfMsgCbs[msgId] || noop
// reject tx
approvalCb(null, false)
// clean up
messageManager.rejectMsg(msgId)
delete this._unconfTxCbs[msgId]
this._didUpdate()
}
// performs the actual signing, no autofill of params
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.signMessage = function (msgParams, cb) {
try {
console.log('signing msg...', msgParams.data)
var rawMsg = this._idmgmt.signMsg(msgParams.from, msgParams.data)
if ('metamaskId' in msgParams) {
var id = msgParams.metamaskId
delete msgParams.metamaskId
this.approveMessage(id, cb)
} else {
cb(null, rawMsg)
}
} catch (err) {
cb(err)
}
}
2016-03-03 07:58:23 +01:00
//
// private
//
2016-06-21 22:18:32 +02:00
IdentityStore.prototype._didUpdate = function () {
this.emit('update', this.getState())
2016-02-11 02:44:46 +01:00
}
2016-06-21 22:18:32 +02:00
IdentityStore.prototype._isUnlocked = function () {
var result = Boolean(this._keyStore) && Boolean(this._idmgmt)
2016-02-11 02:44:46 +01:00
return result
}
2016-02-17 09:55:57 +01:00
// load identities from keyStoreet
2016-06-21 22:18:32 +02:00
IdentityStore.prototype._loadIdentities = function () {
const configManager = this.configManager
if (!this._isUnlocked()) throw new Error('not unlocked')
var addresses = this._getAddresses()
addresses.forEach((address, i) => {
2016-02-17 09:55:57 +01:00
// // add to ethStore
this._ethStore.addAccount(address)
2016-02-11 02:44:46 +01:00
// add to identities
2016-06-21 22:18:32 +02:00
const defaultLabel = 'Wallet ' + (i + 1)
const nickname = configManager.nicknameForWallet(address)
2016-02-11 02:44:46 +01:00
var identity = {
name: nickname || defaultLabel,
2016-02-11 02:44:46 +01:00
address: address,
mayBeFauceting: this._mayBeFauceting(i),
2016-02-11 02:44:46 +01:00
}
this._currentState.identities[address] = identity
2016-02-11 02:44:46 +01:00
})
this._didUpdate()
2016-02-11 02:44:46 +01:00
}
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.saveAccountLabel = function (account, label, cb) {
const configManager = this.configManager
configManager.setNicknameForWallet(account, label)
this._loadIdentities()
cb(null, label)
this._didUpdate()
}
// mayBeFauceting
// If on testnet, index 0 may be fauceting.
// The UI will have to check the balance to know.
// If there is no balance and it mayBeFauceting,
// then it is in fact fauceting.
2016-06-21 22:18:32 +02:00
IdentityStore.prototype._mayBeFauceting = function (i) {
const configManager = this.configManager
var config = configManager.getProvider()
if (i === 0 &&
config.type === 'rpc' &&
config.rpcTarget === DEFAULT_RPC) {
return true
}
return false
}
2016-02-11 02:44:46 +01:00
//
// keyStore managment - unlocking + deserialization
//
2016-06-21 22:18:32 +02:00
IdentityStore.prototype.tryPassword = function (password, cb) {
this._createIdmgmt(password, null, null, cb)
2016-02-11 02:44:46 +01:00
}
2016-06-21 22:18:32 +02:00
IdentityStore.prototype._createIdmgmt = function (password, seed, entropy, cb) {
const configManager = this.configManager
2016-02-11 02:44:46 +01:00
var keyStore = null
LightwalletKeyStore.deriveKeyFromPassword(password, (err, derivedKey) => {
2016-03-03 07:58:23 +01:00
if (err) return cb(err)
var serializedKeystore = configManager.getWallet()
2016-03-15 21:39:12 +01:00
if (seed) {
2016-04-01 23:02:02 +02:00
try {
keyStore = this._restoreFromSeed(password, seed, derivedKey)
} catch (e) {
return cb(e)
}
// returning user, recovering from storage
2016-03-15 21:39:12 +01:00
} else if (serializedKeystore) {
2016-03-31 19:24:39 +02:00
keyStore = LightwalletKeyStore.deserialize(serializedKeystore)
var isCorrect = keyStore.isDerivedKeyCorrect(derivedKey)
2016-03-03 07:58:23 +01:00
if (!isCorrect) return cb(new Error('Lightwallet - password incorrect'))
// first time here
2016-03-03 07:58:23 +01:00
} else {
keyStore = this._createFirstWallet(entropy, derivedKey)
2016-03-03 07:58:23 +01:00
}
this._keyStore = keyStore
this._idmgmt = new IdManagement({
keyStore: keyStore,
derivedKey: derivedKey,
hdPathSTring: this.hdPathString,
2016-06-25 01:13:27 +02:00
configManager: this.configManager,
})
2016-03-03 07:58:23 +01:00
cb()
})
2016-02-11 02:44:46 +01:00
}
2016-06-21 22:18:32 +02:00
IdentityStore.prototype._restoreFromSeed = function (password, seed, derivedKey) {
const configManager = this.configManager
var keyStore = new LightwalletKeyStore(seed, derivedKey, this.hdPathString)
2016-06-21 22:18:32 +02:00
keyStore.addHdDerivationPath(this.hdPathString, derivedKey, {curve: 'secp256k1', purpose: 'sign'})
keyStore.setDefaultHdDerivationPath(this.hdPathString)
keyStore.generateNewAddress(derivedKey, 3)
configManager.setWallet(keyStore.serialize())
console.log('restored from seed. saved to keystore')
return keyStore
}
2016-06-21 22:18:32 +02:00
IdentityStore.prototype._createFirstWallet = function (entropy, derivedKey) {
const configManager = this.configManager
var secretSeed = LightwalletKeyStore.generateRandomSeed(entropy)
var keyStore = new LightwalletKeyStore(secretSeed, derivedKey, this.hdPathString)
2016-06-21 22:18:32 +02:00
keyStore.addHdDerivationPath(this.hdPathString, derivedKey, {curve: 'secp256k1', purpose: 'sign'})
keyStore.setDefaultHdDerivationPath(this.hdPathString)
keyStore.generateNewAddress(derivedKey, 1)
configManager.setWallet(keyStore.serialize())
console.log('saved to keystore')
return keyStore
}
// get addresses and normalize address hexString
2016-06-21 22:18:32 +02:00
IdentityStore.prototype._getAddresses = function () {
return this._keyStore.getAddresses(this.hdPathString).map((address) => { return '0x' + address })
}
2016-06-21 22:18:32 +02:00
IdentityStore.prototype._autoFaucet = function () {
var addresses = this._getAddresses()
autoFaucet(addresses[0])
}
2016-02-11 02:44:46 +01:00
// util
2016-06-21 22:18:32 +02:00
function noop () {}