aboutsummaryrefslogblamecommitdiffstats
path: root/app/scripts/lib/idStore.js
blob: c4547b07fff1cdade2960e0642adfa5696442e8c (plain) (tree)
1
2
3
4
5


                                                   

                                                               















                                                               



                          













                                                               

                                                                         
                                           
                                                        
                           



                                                 

                       

 

                                                                       
                                                         





                           
 




                                                   



                                                           
 

                                              
                                                      
                                           
                                                                        
                                   
                           













                                                               
                                                 

                       
                     
      



                                                                
                                            
                           





                                          


                                                                           
 










                                                    
 

                                                                      
 

                   

             

 

                                                                


                                                 

                                                  
 
              
      



                                           


                   
                         



                                                           




                                                  
                                           
                                
                   

 

                                                                 

                   


                                             




                 



          






                                                 
                                                               


               



                                                                             
                                  





                                                                                             
                         

















                                                              
                                              

 
                                                                              
                     
                                                                            

                                                               
 
               

                                                       

                                                   

                                                                               
                                                                              

                     
            
                                                             
     






                                     

        

 















                                                                                          
                                               





























                                                                                              

       
                 
const EventEmitter = require('events').EventEmitter
const inherits = require('util').inherits
const Transaction = require('ethereumjs-tx')
const LightwalletKeyStore = require('eth-lightwallet').keystore
const LightwalletSigner = require('eth-lightwallet').signing
const async = require('async')
const clone = require('clone')
const extend = require('xtend')
const createId = require('web3-provider-engine/util/random-id')


module.exports = IdentityStore


inherits(IdentityStore, EventEmitter)
function IdentityStore(ethStore) {
  const self = this
  EventEmitter.call(self)

  // we just use the ethStore to auto-add accounts
  self._ethStore = ethStore
  // lightwallet key store
  self._keyStore = null
  // lightwallet wrapper
  self._idmgmt = null

  self._currentState = {
    selectedAddress: null,
    identities: {},
    unconfTxs: {},
  }
  // not part of serilized metamask state - only kept in memory
  self._unconfTxCbs = {}
}

//
// public
//

IdentityStore.prototype.createNewVault = function(password, entropy, cb){
  delete this._keyStore
  delete window.localStorage['lightwallet']
  this._createIdmgmt(password, null, entropy, (err) => {
    if (err) return cb(err)
    var seedWords = this._idmgmt.getSeed()
    this._cacheSeedWordsUntilConfirmed(seedWords)
    this._loadIdentities()
    this._didUpdate()
    cb(null, seedWords)
  })
}

IdentityStore.prototype.recoverFromSeed = function(password, seed, cb){
  const self = this
  self._createIdmgmt(password, seed, null, function(err){
    if (err) return cb(err)
    self._loadIdentities()
    self._didUpdate()
    cb()
  })
}

IdentityStore.prototype.setStore = function(store){
  const self = this
  self._ethStore = store
}

IdentityStore.prototype.clearSeedWordCache = function(cb) {
  delete window.localStorage['seedWords']
  cb()
}

IdentityStore.prototype.getState = function(){
  const self = this
  const cachedSeeds = window.localStorage['seedWords']
  return clone(extend(self._currentState, {
    isInitialized: !!window.localStorage['lightwallet'] && !cachedSeeds,
    isUnlocked: self._isUnlocked(),
    seedWords: cachedSeeds,
  }))
}

IdentityStore.prototype.getSelectedAddress = function(){
  const self = this
  return self._currentState.selectedAddress
}

IdentityStore.prototype.setSelectedAddress = function(address){
  const self = this
  self._currentState.selectedAddress = address
  self._didUpdate()
}

IdentityStore.prototype.setLocked = function(cb){
  const self = this
  delete self._keyStore
  delete self._idmgmt
  cb()
}

IdentityStore.prototype.submitPassword = function(password, cb){
  const self = this
  self._tryPassword(password, function(err){
    if (err) return cb(err)
    // load identities before returning...
    self._loadIdentities()
    cb()
  })
}

// comes from dapp via zero-client hooked-wallet provider
IdentityStore.prototype.addUnconfirmedTransaction = function(txParams, cb){
  var self = this

  // create txData obj with parameters and meta data
  var time = (new Date()).getTime()
  var txId = createId()
  var txData = {
    id: txId,
    txParams: txParams,
    time: time,
    status: 'unconfirmed',
  }
  self._currentState.unconfTxs[txId] = txData
  console.log('addUnconfirmedTransaction:', txData)

  // keep the cb around for after approval (requires user interaction)
  self._unconfTxCbs[txId] = cb

  // signal update
  self._didUpdate()

  return txId
}

// comes from metamask ui
IdentityStore.prototype.approveTransaction = function(txId, cb){
  const self = this

  var txData = self._currentState.unconfTxs[txId]
  var txParams = txData.txParams
  var approvalCb = self._unconfTxCbs[txId] || noop

  // accept tx
  cb()
  approvalCb(null, true)
  // clean up
  delete self._currentState.unconfTxs[txId]
  delete self._unconfTxCbs[txId]
  self._didUpdate()
}

// comes from metamask ui
IdentityStore.prototype.cancelTransaction = function(txId){
  const self = this

  var txData = self._currentState.unconfTxs[txId]
  var approvalCb = self._unconfTxCbs[txId] || noop

  // reject tx
  approvalCb(null, false)
  // clean up
  delete self._currentState.unconfTxs[txId]
  delete self._unconfTxCbs[txId]
  self._didUpdate()
}

// performs the actual signing, no autofill of params
IdentityStore.prototype.signTransaction = function(txParams, cb){
  const self = this
  try {
    console.log('signing tx...', txParams)
    var rawTx = self._idmgmt.signTx(txParams)
    cb(null, rawTx)
  } catch (err) {
    cb(err)
  }
}

//
// private
//

IdentityStore.prototype._didUpdate = function(){
  const self = this
  self.emit('update', self.getState())
}

IdentityStore.prototype._isUnlocked = function(){
  const self = this
  var result = Boolean(self._keyStore) && Boolean(self._idmgmt)
  return result
}

IdentityStore.prototype._cacheSeedWordsUntilConfirmed = function(seedWords) {
  window.localStorage['seedWords'] = seedWords
}

// load identities from keyStoreet
IdentityStore.prototype._loadIdentities = function(){
  const self = this
  if (!self._isUnlocked()) throw new Error('not unlocked')
  // get addresses and normalize address hexString
  var addresses = self._keyStore.getAddresses().map(function(address){ return '0x'+address })
  addresses.forEach(function(address){
    // // add to ethStore
    self._ethStore.addAccount(address)
    // add to identities
    var identity = {
      name: 'Wally',
      img: 'QmW6hcwYzXrNkuHrpvo58YeZvbZxUddv69ATSHY3BHpPdd',
      address: address,
    }
    self._currentState.identities[address] = identity
  })
  self._didUpdate()
}

//
// keyStore managment - unlocking + deserialization
//

IdentityStore.prototype._tryPassword = function(password, cb){
  const self = this
  self._createIdmgmt(password, null, null, cb)
}

IdentityStore.prototype._createIdmgmt = function(password, seed, entropy, cb){
  var keyStore = null
  LightwalletKeyStore.deriveKeyFromPassword(password, (err, derivedKey) => {
    if (err) return cb(err)
    var serializedKeystore = window.localStorage['lightwallet']

    if (seed) {
      this._restoreFromSeed(keyStore, seed, derivedKey)

    // returning user, recovering from localStorage
    } else if (serializedKeystore) {
      keyStore = this._loadFromLocalStorage(serializedKeystore, derivedKey, cb)
      var isCorrect = keyStore.isDerivedKeyCorrect(derivedKey)
      if (!isCorrect) return cb(new Error('Lightwallet - password incorrect'))

   // first time here
    } else {
      keyStore = this._createFirstWallet(entropy, derivedKey)
    }

    this._keyStore = keyStore
    this._idmgmt = new IdManagement({
      keyStore: keyStore,
      derivedKey: derivedKey,
    })

    cb()
  })
}

IdentityStore.prototype._restoreFromSeed = function(keyStore, seed, derivedKey) {
  keyStore = new LightwalletKeyStore(seed, derivedKey)
  keyStore.generateNewAddress(derivedKey, 3)
  window.localStorage['lightwallet'] = keyStore.serialize()
  console.log('restored from seed. saved to keystore localStorage')
}

IdentityStore.prototype._loadFromLocalStorage = function(serializedKeystore, derivedKey) {
  return LightwalletKeyStore.deserialize(serializedKeystore)
}

IdentityStore.prototype._createFirstWallet = function(entropy, derivedKey) {
  var secretSeed = LightwalletKeyStore.generateRandomSeed(entropy)
  var keyStore = new LightwalletKeyStore(secretSeed, derivedKey)
  keyStore.generateNewAddress(derivedKey, 3)
  window.localStorage['lightwallet'] = keyStore.serialize()
  console.log('saved to keystore localStorage')
  return keyStore
}

function IdManagement( opts = { keyStore: null, derivedKey: null } ) {
  this.keyStore = opts.keyStore
  this.derivedKey = opts.derivedKey

  this.getAddresses =  function(){
    return keyStore.getAddresses().map(function(address){ return '0x'+address })
  }

  this.signTx = function(txParams){
    // normalize values
    txParams.to = ethUtil.addHexPrefix(txParams.to)
    txParams.from = ethUtil.addHexPrefix(txParams.from)
    txParams.value = ethUtil.addHexPrefix(txParams.value)
    txParams.data = ethUtil.addHexPrefix(txParams.data)
    txParams.gasLimit = ethUtil.addHexPrefix(txParams.gasLimit || txParams.gas)
    txParams.nonce = ethUtil.addHexPrefix(txParams.nonce)
    var tx = new Transaction(txParams)
    var rawTx = '0x'+tx.serialize().toString('hex')
    return '0x'+LightwalletSigner.signTx(this.keyStore, this.derivedKey, rawTx, txParams.from)
  }

  this.getSeed = function(){
    return this.keyStore.getSeed(this.derivedKey)
  }
}


// util

function noop(){}