mirror of
https://github.com/kremalicious/metamask-extension.git
synced 2024-12-23 09:52:26 +01:00
eth-store - convert to obs-store subclass
This commit is contained in:
parent
f08f40aee2
commit
73edfc9f31
@ -7,140 +7,122 @@
|
|||||||
* on each new block.
|
* on each new block.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const EventEmitter = require('events').EventEmitter
|
|
||||||
const inherits = require('util').inherits
|
|
||||||
const async = require('async')
|
const async = require('async')
|
||||||
const clone = require('clone')
|
|
||||||
const EthQuery = require('eth-query')
|
const EthQuery = require('eth-query')
|
||||||
|
const ObservableStore = require('obs-store')
|
||||||
module.exports = EthereumStore
|
|
||||||
|
|
||||||
|
|
||||||
inherits(EthereumStore, EventEmitter)
|
|
||||||
function EthereumStore(engine) {
|
|
||||||
const self = this
|
|
||||||
EventEmitter.call(self)
|
|
||||||
self._currentState = {
|
|
||||||
accounts: {},
|
|
||||||
transactions: {},
|
|
||||||
}
|
|
||||||
self._query = new EthQuery(engine)
|
|
||||||
|
|
||||||
engine.on('block', self._updateForBlock.bind(self))
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
|
||||||
// public
|
|
||||||
//
|
|
||||||
|
|
||||||
EthereumStore.prototype.getState = function () {
|
|
||||||
const self = this
|
|
||||||
return clone(self._currentState)
|
|
||||||
}
|
|
||||||
|
|
||||||
EthereumStore.prototype.addAccount = function (address) {
|
|
||||||
const self = this
|
|
||||||
self._currentState.accounts[address] = {}
|
|
||||||
self._didUpdate()
|
|
||||||
if (!self.currentBlockNumber) return
|
|
||||||
self._updateAccount(address, () => {
|
|
||||||
self._didUpdate()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
EthereumStore.prototype.removeAccount = function (address) {
|
|
||||||
const self = this
|
|
||||||
delete self._currentState.accounts[address]
|
|
||||||
self._didUpdate()
|
|
||||||
}
|
|
||||||
|
|
||||||
EthereumStore.prototype.addTransaction = function (txHash) {
|
|
||||||
const self = this
|
|
||||||
self._currentState.transactions[txHash] = {}
|
|
||||||
self._didUpdate()
|
|
||||||
if (!self.currentBlockNumber) return
|
|
||||||
self._updateTransaction(self.currentBlockNumber, txHash, noop)
|
|
||||||
}
|
|
||||||
|
|
||||||
EthereumStore.prototype.removeTransaction = function (address) {
|
|
||||||
const self = this
|
|
||||||
delete self._currentState.transactions[address]
|
|
||||||
self._didUpdate()
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
//
|
|
||||||
// private
|
|
||||||
//
|
|
||||||
|
|
||||||
EthereumStore.prototype._didUpdate = function () {
|
|
||||||
const self = this
|
|
||||||
var state = self.getState()
|
|
||||||
self.emit('update', state)
|
|
||||||
}
|
|
||||||
|
|
||||||
EthereumStore.prototype._updateForBlock = function (block) {
|
|
||||||
const self = this
|
|
||||||
var blockNumber = '0x' + block.number.toString('hex')
|
|
||||||
self.currentBlockNumber = blockNumber
|
|
||||||
async.parallel([
|
|
||||||
self._updateAccounts.bind(self),
|
|
||||||
self._updateTransactions.bind(self, blockNumber),
|
|
||||||
], function (err) {
|
|
||||||
if (err) return console.error(err)
|
|
||||||
self.emit('block', self.getState())
|
|
||||||
self._didUpdate()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
EthereumStore.prototype._updateAccounts = function (cb) {
|
|
||||||
var accountsState = this._currentState.accounts
|
|
||||||
var addresses = Object.keys(accountsState)
|
|
||||||
async.each(addresses, this._updateAccount.bind(this), cb)
|
|
||||||
}
|
|
||||||
|
|
||||||
EthereumStore.prototype._updateAccount = function (address, cb) {
|
|
||||||
var accountsState = this._currentState.accounts
|
|
||||||
this.getAccount(address, function (err, result) {
|
|
||||||
if (err) return cb(err)
|
|
||||||
result.address = address
|
|
||||||
// only populate if the entry is still present
|
|
||||||
if (accountsState[address]) {
|
|
||||||
accountsState[address] = result
|
|
||||||
}
|
|
||||||
cb(null, result)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
EthereumStore.prototype.getAccount = function (address, cb) {
|
|
||||||
const query = this._query
|
|
||||||
async.parallel({
|
|
||||||
balance: query.getBalance.bind(query, address),
|
|
||||||
nonce: query.getTransactionCount.bind(query, address),
|
|
||||||
code: query.getCode.bind(query, address),
|
|
||||||
}, cb)
|
|
||||||
}
|
|
||||||
|
|
||||||
EthereumStore.prototype._updateTransactions = function (block, cb) {
|
|
||||||
const self = this
|
|
||||||
var transactionsState = self._currentState.transactions
|
|
||||||
var txHashes = Object.keys(transactionsState)
|
|
||||||
async.each(txHashes, self._updateTransaction.bind(self, block), cb)
|
|
||||||
}
|
|
||||||
|
|
||||||
EthereumStore.prototype._updateTransaction = function (block, txHash, cb) {
|
|
||||||
const self = this
|
|
||||||
// would use the block here to determine how many confirmations the tx has
|
|
||||||
var transactionsState = self._currentState.transactions
|
|
||||||
self._query.getTransaction(txHash, function (err, result) {
|
|
||||||
if (err) return cb(err)
|
|
||||||
// only populate if the entry is still present
|
|
||||||
if (transactionsState[txHash]) {
|
|
||||||
transactionsState[txHash] = result
|
|
||||||
self._didUpdate()
|
|
||||||
}
|
|
||||||
cb(null, result)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
function noop() {}
|
function noop() {}
|
||||||
|
|
||||||
|
|
||||||
|
class EthereumStore extends ObservableStore {
|
||||||
|
|
||||||
|
constructor (opts = {}) {
|
||||||
|
super({
|
||||||
|
accounts: {},
|
||||||
|
transactions: {},
|
||||||
|
})
|
||||||
|
this._provider = opts.provider
|
||||||
|
this._query = new EthQuery(this._provider)
|
||||||
|
this._blockTracker = opts.blockTracker
|
||||||
|
// subscribe to latest block
|
||||||
|
this._blockTracker.on('block', this._updateForBlock.bind(this))
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// public
|
||||||
|
//
|
||||||
|
|
||||||
|
addAccount (address) {
|
||||||
|
const accounts = this.getState().accounts
|
||||||
|
accounts[address] = {}
|
||||||
|
this.updateState({ accounts })
|
||||||
|
if (!this._currentBlockNumber) return
|
||||||
|
this._updateAccount(address)
|
||||||
|
}
|
||||||
|
|
||||||
|
removeAccount (address) {
|
||||||
|
const accounts = this.getState().accounts
|
||||||
|
delete accounts[address]
|
||||||
|
this.updateState({ accounts })
|
||||||
|
}
|
||||||
|
|
||||||
|
addTransaction (txHash) {
|
||||||
|
const transactions = this.getState().transactions
|
||||||
|
transactions[txHash] = {}
|
||||||
|
this.updateState({ transactions })
|
||||||
|
if (!this._currentBlockNumber) return
|
||||||
|
this._updateTransaction(this._currentBlockNumber, txHash, noop)
|
||||||
|
}
|
||||||
|
|
||||||
|
removeTransaction (txHash) {
|
||||||
|
const transactions = this.getState().transactions
|
||||||
|
delete transactions[txHash]
|
||||||
|
this.updateState({ transactions })
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//
|
||||||
|
// private
|
||||||
|
//
|
||||||
|
|
||||||
|
_updateForBlock (block) {
|
||||||
|
const blockNumber = '0x' + block.number.toString('hex')
|
||||||
|
this._currentBlockNumber = blockNumber
|
||||||
|
async.parallel([
|
||||||
|
this._updateAccounts.bind(this),
|
||||||
|
this._updateTransactions.bind(this, blockNumber),
|
||||||
|
], (err) => {
|
||||||
|
if (err) return console.error(err)
|
||||||
|
this.emit('block', this.getState())
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
_updateAccounts (cb) {
|
||||||
|
const accounts = this.getState().accounts
|
||||||
|
const addresses = Object.keys(accounts)
|
||||||
|
async.each(addresses, this._updateAccount.bind(this), cb)
|
||||||
|
}
|
||||||
|
|
||||||
|
_updateAccount (address, cb) {
|
||||||
|
const accounts = this.getState().accounts
|
||||||
|
this._getAccount(address, (err, result) => {
|
||||||
|
if (err) return cb(err)
|
||||||
|
result.address = address
|
||||||
|
// only populate if the entry is still present
|
||||||
|
if (accounts[address]) {
|
||||||
|
accounts[address] = result
|
||||||
|
}
|
||||||
|
cb(null, result)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
_updateTransactions (block, cb) {
|
||||||
|
const transactions = this.getState().transactions
|
||||||
|
const txHashes = Object.keys(transactions)
|
||||||
|
async.each(txHashes, this._updateTransaction.bind(this, block), cb)
|
||||||
|
}
|
||||||
|
|
||||||
|
_updateTransaction (block, txHash, cb) {
|
||||||
|
// would use the block here to determine how many confirmations the tx has
|
||||||
|
const transactions = this.getState().transactions
|
||||||
|
this._query.getTransaction(txHash, (err, result) => {
|
||||||
|
if (err) return cb(err)
|
||||||
|
// only populate if the entry is still present
|
||||||
|
if (transactions[txHash]) {
|
||||||
|
transactions[txHash] = result
|
||||||
|
}
|
||||||
|
cb(null, result)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
_getAccount (address, cb) {
|
||||||
|
const query = this._query
|
||||||
|
async.parallel({
|
||||||
|
balance: query.getBalance.bind(query, address),
|
||||||
|
nonce: query.getTransactionCount.bind(query, address),
|
||||||
|
code: query.getCode.bind(query, address),
|
||||||
|
}, cb)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = EthereumStore
|
@ -55,7 +55,10 @@ module.exports = class MetamaskController extends EventEmitter {
|
|||||||
|
|
||||||
// eth data query tools
|
// eth data query tools
|
||||||
this.ethQuery = new EthQuery(this.provider)
|
this.ethQuery = new EthQuery(this.provider)
|
||||||
this.ethStore = new EthStore(this.provider)
|
this.ethStore = new EthStore({
|
||||||
|
provider: this.provider,
|
||||||
|
blockTracker: this.provider,
|
||||||
|
})
|
||||||
|
|
||||||
// key mgmt
|
// key mgmt
|
||||||
this.keyringController = new KeyringController({
|
this.keyringController = new KeyringController({
|
||||||
@ -113,7 +116,7 @@ module.exports = class MetamaskController extends EventEmitter {
|
|||||||
})
|
})
|
||||||
|
|
||||||
// manual mem state subscriptions
|
// manual mem state subscriptions
|
||||||
this.ethStore.on('update', this.sendUpdate.bind(this))
|
this.ethStore.subscribe(this.sendUpdate.bind(this))
|
||||||
this.networkStore.subscribe(this.sendUpdate.bind(this))
|
this.networkStore.subscribe(this.sendUpdate.bind(this))
|
||||||
this.keyringController.memStore.subscribe(this.sendUpdate.bind(this))
|
this.keyringController.memStore.subscribe(this.sendUpdate.bind(this))
|
||||||
this.txManager.memStore.subscribe(this.sendUpdate.bind(this))
|
this.txManager.memStore.subscribe(this.sendUpdate.bind(this))
|
||||||
|
@ -11,8 +11,8 @@
|
|||||||
"disc": "gulp disc --debug",
|
"disc": "gulp disc --debug",
|
||||||
"dist": "gulp dist --disableLiveReload",
|
"dist": "gulp dist --disableLiveReload",
|
||||||
"test": "npm run lint && npm run fastTest && npm run ci",
|
"test": "npm run lint && npm run fastTest && npm run ci",
|
||||||
"fastTest": "METAMASK_ENV=test mocha --require test/helper.js --compilers js:babel-register --recursive \"test/unit/**/*.js\"",
|
"fastTest": "METAMASK_ENV=test mocha --require test/helper.js --recursive \"test/unit/**/*.js\"",
|
||||||
"watch": "mocha watch --compilers js:babel-register --recursive \"test/unit/**/*.js\"",
|
"watch": "mocha watch --recursive \"test/unit/**/*.js\"",
|
||||||
"genStates": "node development/genStates.js",
|
"genStates": "node development/genStates.js",
|
||||||
"ui": "npm run genStates && beefy ui-dev.js:bundle.js --live --open --index=./development/index.html --cwd ./",
|
"ui": "npm run genStates && beefy ui-dev.js:bundle.js --live --open --index=./development/index.html --cwd ./",
|
||||||
"mock": "beefy mock-dev.js:bundle.js --live --open --index=./development/index.html --cwd ./",
|
"mock": "beefy mock-dev.js:bundle.js --live --open --index=./development/index.html --cwd ./",
|
||||||
@ -71,7 +71,7 @@
|
|||||||
"mississippi": "^1.2.0",
|
"mississippi": "^1.2.0",
|
||||||
"mkdirp": "^0.5.1",
|
"mkdirp": "^0.5.1",
|
||||||
"multiplex": "^6.7.0",
|
"multiplex": "^6.7.0",
|
||||||
"obs-store": "^2.3.0",
|
"obs-store": "^2.3.1",
|
||||||
"once": "^1.3.3",
|
"once": "^1.3.3",
|
||||||
"ping-pong-stream": "^1.0.0",
|
"ping-pong-stream": "^1.0.0",
|
||||||
"pojo-migrator": "^2.1.0",
|
"pojo-migrator": "^2.1.0",
|
||||||
|
Loading…
x
Reference in New Issue
Block a user