aboutsummaryrefslogblamecommitdiffstats
path: root/app/scripts/background.js
blob: 2a8efd8448e484f77e8368c753254c4bebadc200 (plain) (tree)
1
2
3
4
5
6
7
8
9
                              
                                            
                            
                                           

                                          
                                                               
                                               
                                                         
                                                  
                                                          
                                           
                                           
                                                  
                                                                    
                                                           

                                                    
 
                                     
                                            
 


                                                      



                                                     

                                     
                                     
 
                       
 

                                                                    
                                   
                 

                      
                             
 
                              

                                                    
                                                
 
 



                        
                                            
               
                                               
                   
                                                                                       
                                                 
 
                               






                                                              
                                            

                               

   

                                                           
 
                  




                                                            

                           

 
                                      


                        
 






                                             

                            
    
                                        
 









                                                                     
                            

                               
                  
                                   
                                          


                                        

   
                                   

                              
   
 
                                               
                                 
           
                                   


                                                             
                                                     


                





                                                        

                                                                                                       

                                    
                                                                
                                                                  





                                        

                                
                                                                        
                                                                      

     
 



                         
               
                                                         
                                                          
                                                                  



                           
                                                                          
                                                                         
                                                                                             

                                                                                                     





                                                                         
 
                          


  
         

  

                       
                                                   
 
 




                                                                       
  
const urlUtil = require('url')
const endOfStream = require('end-of-stream')
const pump = require('pump')
const debounce = require('debounce-stream')
const log = require('loglevel')
const extension = require('extensionizer')
const LocalStorageStore = require('obs-store/lib/localStorage')
const LocalStore = require('./lib/local-store')
const storeTransform = require('obs-store/lib/transform')
const asStream = require('obs-store/lib/asStream')
const ExtensionPlatform = require('./platforms/extension')
const Migrator = require('./lib/migrator/')
const migrations = require('./migrations/')
const PortStream = require('./lib/port-stream.js')
const NotificationManager = require('./lib/notification-manager.js')
const MetamaskController = require('./metamask-controller')
const firstTimeState = require('./first-time-state')
const setupRaven = require('./setupRaven')

const STORAGE_KEY = 'metamask-config'
const METAMASK_DEBUG = 'GULP_METAMASK_DEBUG'

window.log = log
log.setDefaultLevel(METAMASK_DEBUG ? 'debug' : 'warn')

const platform = new ExtensionPlatform()
const notificationManager = new NotificationManager()
global.METAMASK_NOTIFIER = notificationManager

// setup sentry error reporting
const release = platform.getVersion()
const raven = setupRaven({ release })

let popupIsOpen = false

// state persistence
const diskStore = new LocalStorageStore({ storageKey: STORAGE_KEY })
const localStore = new LocalStore()
let versionedData

// initialization flow
initialize().catch(log.error)

async function initialize () {
  const initState = await loadStateFromPersistence()
  await setupController(initState)
  log.debug('MetaMask initialization complete.')
}

//
// State and Persistence
//

async function loadStateFromPersistence () {
  // migrations
  const migrator = new Migrator({ migrations })
  // read from disk
  versionedData = diskStore.getState() || migrator.generateInitialState(firstTimeState)
  // fetch from extension store and merge in data

  if (localStore.isSupported) {
    let localData
    try {
      localData = await localStore.get()
    } catch (err) {
      log.error('error fetching state from local store:', err)
    }

    if (Object.keys(localData).length > 0) {
      versionedData = localData
    }
  }

  // migrate data
  versionedData = await migrator.migrateData(versionedData)

  // write to disk
  localStore.set(versionedData)
  .catch((reason) => {
    log.error('Problem saving migrated data', versionedData)
  })

  // return just the data
  return versionedData.data
}

function setupController (initState) {
  //
  // MetaMask Controller
  //

  const controller = new MetamaskController({
    // User confirmation callbacks:
    showUnconfirmedMessage: triggerUi,
    unlockAccountMessage: triggerUi,
    showUnapprovedTx: triggerUi,
    // initial state
    initState,
    // platform specific api
    platform,
  })
  global.metamaskController = controller

  // report failed transactions to Sentry
  controller.txController.on(`tx:status-update`, (txId, status) => {
    if (status !== 'failed') return
    const txMeta = controller.txController.txStateManager.getTx(txId)
    raven.captureMessage('Transaction Failed', {
      // "extra" key is required by Sentry
      extra: txMeta,
    })
  })

  // setup state persistence
  pump(
    asStream(controller.store),
    debounce(200),
    storeTransform(versionifyData),
    storeTransform(syncDataWithExtension),
    (error) => {
      log.error('pump hit error', error)
    }
  )

  function versionifyData (state) {
    versionedData.data = state
    return versionedData
  }

  async function syncDataWithExtension(state) {
    if (localStore.isSupported) {
      try {
        await localStore.set(state)
      } catch (err) {
        log.error('error setting state in local store:', err)
      }
    } else { log.error('local store not supported') }
    return state
  }

  //
  // connect to other contexts
  //

  extension.runtime.onConnect.addListener(connectRemote)
  function connectRemote (remotePort) {
    const isMetaMaskInternalProcess = remotePort.name === 'popup' || remotePort.name === 'notification'
    const portStream = new PortStream(remotePort)
    if (isMetaMaskInternalProcess) {
      // communication with popup
      popupIsOpen = popupIsOpen || (remotePort.name === 'popup')
      controller.setupTrustedCommunication(portStream, 'MetaMask')
      // record popup as closed
      if (remotePort.name === 'popup') {
        endOfStream(portStream, () => {
          popupIsOpen = false
        })
      }
    } else {
      // communication with page
      const originDomain = urlUtil.parse(remotePort.sender.url).hostname
      controller.setupUntrustedCommunication(portStream, originDomain)
    }
  }

  //
  // User Interface setup
  //

  updateBadge()
  controller.txController.on('update:badge', updateBadge)
  controller.messageManager.on('updateBadge', updateBadge)
  controller.personalMessageManager.on('updateBadge', updateBadge)

  // plugin badge text
  function updateBadge () {
    var label = ''
    var unapprovedTxCount = controller.txController.getUnapprovedTxCount()
    var unapprovedMsgCount = controller.messageManager.unapprovedMsgCount
    var unapprovedPersonalMsgs = controller.personalMessageManager.unapprovedPersonalMsgCount
    var unapprovedTypedMsgs = controller.typedMessageManager.unapprovedTypedMessagesCount
    var count = unapprovedTxCount + unapprovedMsgCount + unapprovedPersonalMsgs + unapprovedTypedMsgs
    if (count) {
      label = String(count)
    }
    extension.browserAction.setBadgeText({ text: label })
    extension.browserAction.setBadgeBackgroundColor({ color: '#506F8B' })
  }

  return Promise.resolve()
}

//
// Etc...
//

// popup trigger
function triggerUi () {
  if (!popupIsOpen) notificationManager.showPopup()
}

// On first install, open a window to MetaMask website to how-it-works.
extension.runtime.onInstalled.addListener(function (details) {
  if ((details.reason === 'install') && (!METAMASK_DEBUG)) {
    extension.tabs.create({url: 'https://metamask.io/#how-it-works'})
  }
})