mirror of
https://github.com/kremalicious/metamask-extension.git
synced 2024-12-23 09:52:26 +01:00
Merge branch 'master' into i18n
This commit is contained in:
commit
b7c7083a11
152
.circleci/config.yml
Normal file
152
.circleci/config.yml
Normal file
@ -0,0 +1,152 @@
|
|||||||
|
version: 2
|
||||||
|
|
||||||
|
workflows:
|
||||||
|
version: 2
|
||||||
|
full_test:
|
||||||
|
jobs:
|
||||||
|
- prep-deps-npm
|
||||||
|
- prep-deps-firefox
|
||||||
|
- prep-scss:
|
||||||
|
requires:
|
||||||
|
- prep-deps-npm
|
||||||
|
- test-lint:
|
||||||
|
requires:
|
||||||
|
- prep-deps-npm
|
||||||
|
- test-unit:
|
||||||
|
requires:
|
||||||
|
- prep-deps-npm
|
||||||
|
- test-integration-mascara:
|
||||||
|
requires:
|
||||||
|
- prep-deps-npm
|
||||||
|
- prep-deps-firefox
|
||||||
|
- prep-scss
|
||||||
|
- test-integration-flat:
|
||||||
|
requires:
|
||||||
|
- prep-deps-npm
|
||||||
|
- prep-deps-firefox
|
||||||
|
- prep-scss
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
prep-deps-npm:
|
||||||
|
docker:
|
||||||
|
- image: circleci/node:8-browsers
|
||||||
|
steps:
|
||||||
|
- checkout
|
||||||
|
- restore_cache:
|
||||||
|
key: dependency-cache-{{ checksum "package-lock.json" }}
|
||||||
|
- run:
|
||||||
|
name: Install deps via npm
|
||||||
|
command: npm install
|
||||||
|
- save_cache:
|
||||||
|
key: dependency-cache-{{ checksum "package-lock.json" }}
|
||||||
|
paths:
|
||||||
|
- node_modules
|
||||||
|
|
||||||
|
prep-deps-firefox:
|
||||||
|
docker:
|
||||||
|
- image: circleci/node:8-browsers
|
||||||
|
steps:
|
||||||
|
- checkout
|
||||||
|
- run:
|
||||||
|
name: Download Firefox
|
||||||
|
command: >
|
||||||
|
wget https://ftp.mozilla.org/pub/firefox/releases/58.0/linux-x86_64/en-US/firefox-58.0.tar.bz2
|
||||||
|
&& tar xjf firefox-58.0.tar.bz2
|
||||||
|
- save_cache:
|
||||||
|
key: dependency-cache-firefox-{{ .Revision }}
|
||||||
|
paths:
|
||||||
|
- firefox
|
||||||
|
|
||||||
|
|
||||||
|
prep-scss:
|
||||||
|
docker:
|
||||||
|
- image: circleci/node:8-browsers
|
||||||
|
steps:
|
||||||
|
- checkout
|
||||||
|
- restore_cache:
|
||||||
|
key: dependency-cache-{{ checksum "package-lock.json" }}
|
||||||
|
- run:
|
||||||
|
name: Get Scss Cache key
|
||||||
|
# this allows us to checksum against a whole directory
|
||||||
|
command: find ui/app/css -type f -exec md5sum {} \; | sort -k 2 > scss_checksum
|
||||||
|
- run:
|
||||||
|
name: Build for integration tests
|
||||||
|
command: npm run test:integration:build
|
||||||
|
- save_cache:
|
||||||
|
key: scss-cache-{{ checksum "scss_checksum" }}
|
||||||
|
paths:
|
||||||
|
- ui/app/css/output
|
||||||
|
|
||||||
|
test-lint:
|
||||||
|
docker:
|
||||||
|
- image: circleci/node:8-browsers
|
||||||
|
steps:
|
||||||
|
- checkout
|
||||||
|
- restore_cache:
|
||||||
|
key: dependency-cache-{{ checksum "package-lock.json" }}
|
||||||
|
- run:
|
||||||
|
name: Test
|
||||||
|
command: npm run lint
|
||||||
|
|
||||||
|
test-unit:
|
||||||
|
docker:
|
||||||
|
- image: circleci/node:8-browsers
|
||||||
|
steps:
|
||||||
|
- checkout
|
||||||
|
- restore_cache:
|
||||||
|
key: dependency-cache-{{ checksum "package-lock.json" }}
|
||||||
|
- run:
|
||||||
|
name: test:coverage
|
||||||
|
command: npm run test:coverage
|
||||||
|
|
||||||
|
test-integration-flat:
|
||||||
|
docker:
|
||||||
|
- image: circleci/node:8-browsers
|
||||||
|
steps:
|
||||||
|
- checkout
|
||||||
|
- restore_cache:
|
||||||
|
key: dependency-cache-firefox-{{ .Revision }}
|
||||||
|
- run:
|
||||||
|
name: Install firefox
|
||||||
|
command: >
|
||||||
|
sudo rm -r /opt/firefox
|
||||||
|
&& sudo mv firefox /opt/firefox58
|
||||||
|
&& sudo mv /usr/bin/firefox /usr/bin/firefox-old
|
||||||
|
&& sudo ln -s /opt/firefox58/firefox /usr/bin/firefox
|
||||||
|
- restore_cache:
|
||||||
|
key: dependency-cache-{{ checksum "package-lock.json" }}
|
||||||
|
- run:
|
||||||
|
name: Get Scss Cache key
|
||||||
|
# this allows us to checksum against a whole directory
|
||||||
|
command: find ui/app/css -type f -exec md5sum {} \; | sort -k 2 > scss_checksum
|
||||||
|
- restore_cache:
|
||||||
|
key: scss-cache-{{ checksum "scss_checksum" }}
|
||||||
|
- run:
|
||||||
|
name: test:integration:flat
|
||||||
|
command: npm run test:flat
|
||||||
|
|
||||||
|
test-integration-mascara:
|
||||||
|
docker:
|
||||||
|
- image: circleci/node:8-browsers
|
||||||
|
steps:
|
||||||
|
- checkout
|
||||||
|
- restore_cache:
|
||||||
|
key: dependency-cache-firefox-{{ .Revision }}
|
||||||
|
- run:
|
||||||
|
name: Install firefox
|
||||||
|
command: >
|
||||||
|
sudo rm -r /opt/firefox
|
||||||
|
&& sudo mv firefox /opt/firefox58
|
||||||
|
&& sudo mv /usr/bin/firefox /usr/bin/firefox-old
|
||||||
|
&& sudo ln -s /opt/firefox58/firefox /usr/bin/firefox
|
||||||
|
- restore_cache:
|
||||||
|
key: dependency-cache-{{ checksum "package-lock.json" }}
|
||||||
|
- run:
|
||||||
|
name: Get Scss Cache key
|
||||||
|
# this allows us to checksum against a whole directory
|
||||||
|
command: find ui/app/css -type f -exec md5sum {} \; | sort -k 2 > scss_checksum
|
||||||
|
- restore_cache:
|
||||||
|
key: scss-cache-{{ checksum "scss_checksum" }}
|
||||||
|
- run:
|
||||||
|
name: test:integration:mascara
|
||||||
|
command: npm run test:mascara
|
1
.gitignore
vendored
1
.gitignore
vendored
@ -1,6 +1,5 @@
|
|||||||
npm-debug.log
|
npm-debug.log
|
||||||
node_modules
|
node_modules
|
||||||
package-lock.json
|
|
||||||
|
|
||||||
app/bower_components
|
app/bower_components
|
||||||
test/bower_components
|
test/bower_components
|
||||||
|
3
.nsprc
Normal file
3
.nsprc
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
{
|
||||||
|
"exceptions": ["https://nodesecurity.io/advisories/566"]
|
||||||
|
}
|
@ -1,6 +1,7 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
## Current Master
|
## Current Master
|
||||||
|
- Fix flashing to Log in screen after logging in or restoring from seed phrase.
|
||||||
|
|
||||||
## 4.2.0 Tue Mar 06 2018
|
## 4.2.0 Tue Mar 06 2018
|
||||||
|
|
||||||
|
@ -3,7 +3,7 @@ const ObservableStore = require('obs-store')
|
|||||||
const ethUtil = require('ethereumjs-util')
|
const ethUtil = require('ethereumjs-util')
|
||||||
const Transaction = require('ethereumjs-tx')
|
const Transaction = require('ethereumjs-tx')
|
||||||
const EthQuery = require('ethjs-query')
|
const EthQuery = require('ethjs-query')
|
||||||
const TransactionStateManger = require('../lib/tx-state-manager')
|
const TransactionStateManager = require('../lib/tx-state-manager')
|
||||||
const TxGasUtil = require('../lib/tx-gas-utils')
|
const TxGasUtil = require('../lib/tx-gas-utils')
|
||||||
const PendingTransactionTracker = require('../lib/pending-tx-tracker')
|
const PendingTransactionTracker = require('../lib/pending-tx-tracker')
|
||||||
const createId = require('../lib/random-id')
|
const createId = require('../lib/random-id')
|
||||||
@ -38,7 +38,7 @@ module.exports = class TransactionController extends EventEmitter {
|
|||||||
this.query = new EthQuery(this.provider)
|
this.query = new EthQuery(this.provider)
|
||||||
this.txGasUtil = new TxGasUtil(this.provider)
|
this.txGasUtil = new TxGasUtil(this.provider)
|
||||||
|
|
||||||
this.txStateManager = new TransactionStateManger({
|
this.txStateManager = new TransactionStateManager({
|
||||||
initState: opts.initState,
|
initState: opts.initState,
|
||||||
txHistoryLimit: opts.txHistoryLimit,
|
txHistoryLimit: opts.txHistoryLimit,
|
||||||
getNetwork: this.getNetwork.bind(this),
|
getNetwork: this.getNetwork.bind(this),
|
||||||
|
48
app/scripts/lib/seed-phrase-verifier.js
Normal file
48
app/scripts/lib/seed-phrase-verifier.js
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
const KeyringController = require('eth-keyring-controller')
|
||||||
|
|
||||||
|
const seedPhraseVerifier = {
|
||||||
|
|
||||||
|
// Verifies if the seed words can restore the accounts.
|
||||||
|
//
|
||||||
|
// The seed words can recreate the primary keyring and the accounts belonging to it.
|
||||||
|
// The created accounts in the primary keyring are always the same.
|
||||||
|
// The keyring always creates the accounts in the same sequence.
|
||||||
|
verifyAccounts (createdAccounts, seedWords) {
|
||||||
|
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
|
||||||
|
if (!createdAccounts || createdAccounts.length < 1) {
|
||||||
|
return reject(new Error('No created accounts defined.'))
|
||||||
|
}
|
||||||
|
|
||||||
|
const keyringController = new KeyringController({})
|
||||||
|
const Keyring = keyringController.getKeyringClassForType('HD Key Tree')
|
||||||
|
const opts = {
|
||||||
|
mnemonic: seedWords,
|
||||||
|
numberOfAccounts: createdAccounts.length,
|
||||||
|
}
|
||||||
|
|
||||||
|
const keyring = new Keyring(opts)
|
||||||
|
keyring.getAccounts()
|
||||||
|
.then((restoredAccounts) => {
|
||||||
|
|
||||||
|
log.debug('Created accounts: ' + JSON.stringify(createdAccounts))
|
||||||
|
log.debug('Restored accounts: ' + JSON.stringify(restoredAccounts))
|
||||||
|
|
||||||
|
if (restoredAccounts.length !== createdAccounts.length) {
|
||||||
|
// this should not happen...
|
||||||
|
return reject(new Error('Wrong number of accounts'))
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let i = 0; i < restoredAccounts.length; i++) {
|
||||||
|
if (restoredAccounts[i].toLowerCase() !== createdAccounts[i].toLowerCase()) {
|
||||||
|
return reject(new Error('Not identical accounts! Original: ' + createdAccounts[i] + ', Restored: ' + restoredAccounts[i]))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return resolve()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = seedPhraseVerifier
|
@ -4,7 +4,7 @@ const ObservableStore = require('obs-store')
|
|||||||
const ethUtil = require('ethereumjs-util')
|
const ethUtil = require('ethereumjs-util')
|
||||||
const txStateHistoryHelper = require('./tx-state-history-helper')
|
const txStateHistoryHelper = require('./tx-state-history-helper')
|
||||||
|
|
||||||
module.exports = class TransactionStateManger extends EventEmitter {
|
module.exports = class TransactionStateManager extends EventEmitter {
|
||||||
constructor ({ initState, txHistoryLimit, getNetwork }) {
|
constructor ({ initState, txHistoryLimit, getNetwork }) {
|
||||||
super()
|
super()
|
||||||
|
|
||||||
|
@ -37,6 +37,7 @@ const version = require('../manifest.json').version
|
|||||||
const BN = require('ethereumjs-util').BN
|
const BN = require('ethereumjs-util').BN
|
||||||
const GWEI_BN = new BN('1000000000')
|
const GWEI_BN = new BN('1000000000')
|
||||||
const percentile = require('percentile')
|
const percentile = require('percentile')
|
||||||
|
const seedPhraseVerifier = require('./lib/seed-phrase-verifier')
|
||||||
|
|
||||||
module.exports = class MetamaskController extends EventEmitter {
|
module.exports = class MetamaskController extends EventEmitter {
|
||||||
|
|
||||||
@ -344,6 +345,7 @@ module.exports = class MetamaskController extends EventEmitter {
|
|||||||
// primary HD keyring management
|
// primary HD keyring management
|
||||||
addNewAccount: nodeify(this.addNewAccount, this),
|
addNewAccount: nodeify(this.addNewAccount, this),
|
||||||
placeSeedWords: this.placeSeedWords.bind(this),
|
placeSeedWords: this.placeSeedWords.bind(this),
|
||||||
|
verifySeedPhrase: nodeify(this.verifySeedPhrase, this),
|
||||||
clearSeedWordCache: this.clearSeedWordCache.bind(this),
|
clearSeedWordCache: this.clearSeedWordCache.bind(this),
|
||||||
resetAccount: this.resetAccount.bind(this),
|
resetAccount: this.resetAccount.bind(this),
|
||||||
importAccountWithStrategy: this.importAccountWithStrategy.bind(this),
|
importAccountWithStrategy: this.importAccountWithStrategy.bind(this),
|
||||||
@ -565,14 +567,18 @@ module.exports = class MetamaskController extends EventEmitter {
|
|||||||
// Opinionated Keyring Management
|
// Opinionated Keyring Management
|
||||||
//
|
//
|
||||||
|
|
||||||
async addNewAccount (cb) {
|
async addNewAccount () {
|
||||||
const primaryKeyring = this.keyringController.getKeyringsByType('HD Key Tree')[0]
|
const primaryKeyring = this.keyringController.getKeyringsByType('HD Key Tree')[0]
|
||||||
if (!primaryKeyring) return cb(new Error('MetamaskController - No HD Key Tree found'))
|
if (!primaryKeyring) {
|
||||||
|
throw new Error('MetamaskController - No HD Key Tree found')
|
||||||
|
}
|
||||||
const keyringController = this.keyringController
|
const keyringController = this.keyringController
|
||||||
const oldAccounts = await keyringController.getAccounts()
|
const oldAccounts = await keyringController.getAccounts()
|
||||||
const keyState = await keyringController.addNewAccount(primaryKeyring)
|
const keyState = await keyringController.addNewAccount(primaryKeyring)
|
||||||
const newAccounts = await keyringController.getAccounts()
|
const newAccounts = await keyringController.getAccounts()
|
||||||
|
|
||||||
|
await this.verifySeedPhrase()
|
||||||
|
|
||||||
newAccounts.forEach((address) => {
|
newAccounts.forEach((address) => {
|
||||||
if (!oldAccounts.includes(address)) {
|
if (!oldAccounts.includes(address)) {
|
||||||
this.preferencesController.setSelectedAddress(address)
|
this.preferencesController.setSelectedAddress(address)
|
||||||
@ -587,14 +593,43 @@ module.exports = class MetamaskController extends EventEmitter {
|
|||||||
// Used when creating a first vault, to allow confirmation.
|
// Used when creating a first vault, to allow confirmation.
|
||||||
// Also used when revealing the seed words in the confirmation view.
|
// Also used when revealing the seed words in the confirmation view.
|
||||||
placeSeedWords (cb) {
|
placeSeedWords (cb) {
|
||||||
const primaryKeyring = this.keyringController.getKeyringsByType('HD Key Tree')[0]
|
|
||||||
if (!primaryKeyring) return cb(new Error('MetamaskController - No HD Key Tree found'))
|
this.verifySeedPhrase()
|
||||||
primaryKeyring.serialize()
|
.then((seedWords) => {
|
||||||
.then((serialized) => {
|
|
||||||
const seedWords = serialized.mnemonic
|
|
||||||
this.configManager.setSeedWords(seedWords)
|
this.configManager.setSeedWords(seedWords)
|
||||||
cb(null, seedWords)
|
return cb(null, seedWords)
|
||||||
})
|
})
|
||||||
|
.catch((err) => {
|
||||||
|
return cb(err)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verifies the current vault's seed words if they can restore the
|
||||||
|
// accounts belonging to the current vault.
|
||||||
|
//
|
||||||
|
// Called when the first account is created and on unlocking the vault.
|
||||||
|
async verifySeedPhrase () {
|
||||||
|
|
||||||
|
const primaryKeyring = this.keyringController.getKeyringsByType('HD Key Tree')[0]
|
||||||
|
if (!primaryKeyring) {
|
||||||
|
throw new Error('MetamaskController - No HD Key Tree found')
|
||||||
|
}
|
||||||
|
|
||||||
|
const serialized = await primaryKeyring.serialize()
|
||||||
|
const seedWords = serialized.mnemonic
|
||||||
|
|
||||||
|
const accounts = await primaryKeyring.getAccounts()
|
||||||
|
if (accounts.length < 1) {
|
||||||
|
throw new Error('MetamaskController - No accounts found')
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await seedPhraseVerifier.verifyAccounts(accounts, seedWords)
|
||||||
|
return seedWords
|
||||||
|
} catch (err) {
|
||||||
|
log.error(err.message)
|
||||||
|
throw err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ClearSeedWordCache
|
// ClearSeedWordCache
|
||||||
|
17
circle.yml
17
circle.yml
@ -1,17 +0,0 @@
|
|||||||
machine:
|
|
||||||
node:
|
|
||||||
version: 8.1.4
|
|
||||||
test:
|
|
||||||
override:
|
|
||||||
- "npm test"
|
|
||||||
dependencies:
|
|
||||||
pre:
|
|
||||||
- sudo apt-get update
|
|
||||||
# get latest stable firefox
|
|
||||||
- sudo apt-get install firefox
|
|
||||||
- firefox_cmd=`which firefox`; sudo rm -f $firefox_cmd; sudo ln -s `which firefox.ubuntu` $firefox_cmd
|
|
||||||
# get latest stable chrome
|
|
||||||
- wget -q -O - https://dl-ssl.google.com/linux/linux_signing_key.pub | sudo apt-key add -
|
|
||||||
- sudo sh -c 'echo "deb [arch=amd64] http://dl.google.com/linux/chrome/deb/ stable main" >> /etc/apt/sources.list.d/google.list'
|
|
||||||
- sudo apt-get update
|
|
||||||
- sudo apt-get install google-chrome-stable
|
|
@ -15,7 +15,6 @@
|
|||||||
const extend = require('xtend')
|
const extend = require('xtend')
|
||||||
const render = require('react-dom').render
|
const render = require('react-dom').render
|
||||||
const h = require('react-hyperscript')
|
const h = require('react-hyperscript')
|
||||||
const pipe = require('mississippi').pipe
|
|
||||||
const Root = require('./ui/app/root')
|
const Root = require('./ui/app/root')
|
||||||
const configureStore = require('./ui/app/store')
|
const configureStore = require('./ui/app/store')
|
||||||
const actions = require('./ui/app/actions')
|
const actions = require('./ui/app/actions')
|
||||||
|
@ -1,26 +1,23 @@
|
|||||||
const inherits = require('util').inherits
|
|
||||||
const Component = require('react').Component
|
const Component = require('react').Component
|
||||||
const h = require('react-hyperscript')
|
const h = require('react-hyperscript')
|
||||||
const connect = require('react-redux').connect
|
const connect = require('react-redux').connect
|
||||||
const actions = require('../../../../ui/app/actions')
|
const actions = require('../../../../ui/app/actions')
|
||||||
const FileInput = require('react-simple-file-input').default
|
const FileInput = require('react-simple-file-input').default
|
||||||
|
const PropTypes = require('prop-types')
|
||||||
|
|
||||||
const HELP_LINK = 'https://github.com/MetaMask/faq/blob/master/README.md#q-i-cant-use-the-import-feature-for-uploading-a-json-file-the-window-keeps-closing-when-i-try-to-select-a-file'
|
const HELP_LINK = 'https://github.com/MetaMask/faq/blob/master/README.md#q-i-cant-use-the-import-feature-for-uploading-a-json-file-the-window-keeps-closing-when-i-try-to-select-a-file'
|
||||||
|
|
||||||
module.exports = connect(mapStateToProps)(JsonImportSubview)
|
class JsonImportSubview extends Component {
|
||||||
|
constructor (props) {
|
||||||
|
super(props)
|
||||||
|
|
||||||
function mapStateToProps (state) {
|
this.state = {
|
||||||
return {
|
file: null,
|
||||||
error: state.appState.warning,
|
fileContents: '',
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
inherits(JsonImportSubview, Component)
|
render () {
|
||||||
function JsonImportSubview () {
|
|
||||||
Component.call(this)
|
|
||||||
}
|
|
||||||
|
|
||||||
JsonImportSubview.prototype.render = function () {
|
|
||||||
const { error } = this.props
|
const { error } = this.props
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@ -34,7 +31,10 @@ JsonImportSubview.prototype.render = function () {
|
|||||||
}, [
|
}, [
|
||||||
|
|
||||||
h('p', 'Used by a variety of different clients'),
|
h('p', 'Used by a variety of different clients'),
|
||||||
h('a.warning', { href: HELP_LINK, target: '_blank' }, 'File import not working? Click here!'),
|
h('a.warning', {
|
||||||
|
href: HELP_LINK,
|
||||||
|
target: '_blank',
|
||||||
|
}, 'File import not working? Click here!'),
|
||||||
|
|
||||||
h(FileInput, {
|
h(FileInput, {
|
||||||
readAs: 'text',
|
readAs: 'text',
|
||||||
@ -68,24 +68,23 @@ JsonImportSubview.prototype.render = function () {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
JsonImportSubview.prototype.onLoad = function (event, file) {
|
onLoad (event, file) {
|
||||||
this.setState({file: file, fileContents: event.target.result})
|
this.setState({file: file, fileContents: event.target.result})
|
||||||
}
|
}
|
||||||
|
|
||||||
JsonImportSubview.prototype.createKeyringOnEnter = function (event) {
|
createKeyringOnEnter (event) {
|
||||||
if (event.key === 'Enter') {
|
if (event.key === 'Enter') {
|
||||||
event.preventDefault()
|
event.preventDefault()
|
||||||
this.createNewKeychain()
|
this.createNewKeychain()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
JsonImportSubview.prototype.createNewKeychain = function () {
|
createNewKeychain () {
|
||||||
const state = this.state
|
const { fileContents } = this.state
|
||||||
const { fileContents } = state
|
|
||||||
|
|
||||||
if (!fileContents) {
|
if (!fileContents) {
|
||||||
const message = 'You must select a file to import.'
|
const message = 'You must select a file to import.'
|
||||||
return this.props.dispatch(actions.displayWarning(message))
|
return this.props.displayWarning(message)
|
||||||
}
|
}
|
||||||
|
|
||||||
const passwordInput = document.getElementById('json-password-box')
|
const passwordInput = document.getElementById('json-password-box')
|
||||||
@ -93,8 +92,31 @@ JsonImportSubview.prototype.createNewKeychain = function () {
|
|||||||
|
|
||||||
if (!password) {
|
if (!password) {
|
||||||
const message = 'You must enter a password for the selected file.'
|
const message = 'You must enter a password for the selected file.'
|
||||||
return this.props.dispatch(actions.displayWarning(message))
|
return this.props.displayWarning(message)
|
||||||
}
|
}
|
||||||
|
|
||||||
this.props.dispatch(actions.importNewAccount('JSON File', [ fileContents, password ]))
|
this.props.importNewAccount([ fileContents, password ])
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
JsonImportSubview.propTypes = {
|
||||||
|
error: PropTypes.string,
|
||||||
|
displayWarning: PropTypes.func,
|
||||||
|
importNewAccount: PropTypes.func,
|
||||||
|
}
|
||||||
|
|
||||||
|
const mapStateToProps = state => {
|
||||||
|
return {
|
||||||
|
error: state.appState.warning,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const mapDispatchToProps = dispatch => {
|
||||||
|
return {
|
||||||
|
goHome: () => dispatch(actions.goHome()),
|
||||||
|
displayWarning: warning => dispatch(actions.displayWarning(warning)),
|
||||||
|
importNewAccount: options => dispatch(actions.importNewAccount('JSON File', options)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = connect(mapStateToProps, mapDispatchToProps)(JsonImportSubview)
|
||||||
|
22241
package-lock.json
generated
Normal file
22241
package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load Diff
16
package.json
16
package.json
@ -15,7 +15,8 @@
|
|||||||
"test": "npm run lint && npm run test:coverage && npm run test:integration",
|
"test": "npm run lint && npm run test:coverage && npm run test:integration",
|
||||||
"test:unit": "METAMASK_ENV=test mocha --exit --require babel-core/register --require test/helper.js --recursive \"test/unit/**/*.js\"",
|
"test:unit": "METAMASK_ENV=test mocha --exit --require babel-core/register --require test/helper.js --recursive \"test/unit/**/*.js\"",
|
||||||
"test:single": "METAMASK_ENV=test mocha --require test/helper.js",
|
"test:single": "METAMASK_ENV=test mocha --require test/helper.js",
|
||||||
"test:integration": "gulp build:scss && npm run test:flat && npm run test:mascara",
|
"test:integration": "npm run test:integration:build && npm run test:flat && npm run test:mascara",
|
||||||
|
"test:integration:build": "gulp build:scss",
|
||||||
"test:coverage": "nyc npm run test:unit && npm run test:coveralls-upload",
|
"test:coverage": "nyc npm run test:unit && npm run test:coveralls-upload",
|
||||||
"test:coveralls-upload": "if [ $COVERALLS_REPO_TOKEN ]; then nyc report --reporter=text-lcov | coveralls; fi",
|
"test:coveralls-upload": "if [ $COVERALLS_REPO_TOKEN ]; then nyc report --reporter=text-lcov | coveralls; fi",
|
||||||
"test:flat": "npm run test:flat:build && karma start test/flat.conf.js",
|
"test:flat": "npm run test:flat:build && karma start test/flat.conf.js",
|
||||||
@ -105,7 +106,7 @@
|
|||||||
"fast-levenshtein": "^2.0.6",
|
"fast-levenshtein": "^2.0.6",
|
||||||
"fuse.js": "^3.2.0",
|
"fuse.js": "^3.2.0",
|
||||||
"gulp": "github:gulpjs/gulp#4.0",
|
"gulp": "github:gulpjs/gulp#4.0",
|
||||||
"gulp-autoprefixer": "^4.0.0",
|
"gulp-autoprefixer": "^5.0.0",
|
||||||
"gulp-eslint": "^4.0.0",
|
"gulp-eslint": "^4.0.0",
|
||||||
"gulp-sass": "^3.1.0",
|
"gulp-sass": "^3.1.0",
|
||||||
"hat": "0.0.3",
|
"hat": "0.0.3",
|
||||||
@ -125,7 +126,6 @@
|
|||||||
"loglevel": "^1.4.1",
|
"loglevel": "^1.4.1",
|
||||||
"metamascara": "^2.0.0",
|
"metamascara": "^2.0.0",
|
||||||
"metamask-logo": "^2.1.2",
|
"metamask-logo": "^2.1.2",
|
||||||
"mississippi": "^1.2.0",
|
|
||||||
"mkdirp": "^0.5.1",
|
"mkdirp": "^0.5.1",
|
||||||
"multiplex": "^6.7.0",
|
"multiplex": "^6.7.0",
|
||||||
"number-to-bn": "^1.7.0",
|
"number-to-bn": "^1.7.0",
|
||||||
@ -139,7 +139,7 @@
|
|||||||
"post-message-stream": "^3.0.0",
|
"post-message-stream": "^3.0.0",
|
||||||
"promise-filter": "^1.1.0",
|
"promise-filter": "^1.1.0",
|
||||||
"promise-to-callback": "^1.0.0",
|
"promise-to-callback": "^1.0.0",
|
||||||
"pump": "^1.0.2",
|
"pump": "^3.0.0",
|
||||||
"pumpify": "^1.3.4",
|
"pumpify": "^1.3.4",
|
||||||
"qrcode-npm": "0.0.3",
|
"qrcode-npm": "0.0.3",
|
||||||
"ramda": "^0.24.1",
|
"ramda": "^0.24.1",
|
||||||
@ -189,7 +189,7 @@
|
|||||||
"babelify": "^8.0.0",
|
"babelify": "^8.0.0",
|
||||||
"beefy": "^2.1.5",
|
"beefy": "^2.1.5",
|
||||||
"brfs": "^1.4.3",
|
"brfs": "^1.4.3",
|
||||||
"browserify": "^14.4.0",
|
"browserify": "^16.1.1",
|
||||||
"chai": "^4.1.0",
|
"chai": "^4.1.0",
|
||||||
"compression": "^1.7.1",
|
"compression": "^1.7.1",
|
||||||
"coveralls": "^3.0.0",
|
"coveralls": "^3.0.0",
|
||||||
@ -212,7 +212,7 @@
|
|||||||
"gulp-replace": "^0.6.1",
|
"gulp-replace": "^0.6.1",
|
||||||
"gulp-sourcemaps": "^2.6.0",
|
"gulp-sourcemaps": "^2.6.0",
|
||||||
"gulp-stylefmt": "^1.1.0",
|
"gulp-stylefmt": "^1.1.0",
|
||||||
"gulp-stylelint": "^4.0.0",
|
"gulp-stylelint": "^7.0.0",
|
||||||
"gulp-uglify": "^3.0.0",
|
"gulp-uglify": "^3.0.0",
|
||||||
"gulp-uglify-es": "^1.0.1",
|
"gulp-uglify-es": "^1.0.1",
|
||||||
"gulp-util": "^3.0.7",
|
"gulp-util": "^3.0.7",
|
||||||
@ -222,7 +222,7 @@
|
|||||||
"jsdom": "^11.1.0",
|
"jsdom": "^11.1.0",
|
||||||
"jsdom-global": "^3.0.2",
|
"jsdom-global": "^3.0.2",
|
||||||
"jshint-stylish": "~2.2.1",
|
"jshint-stylish": "~2.2.1",
|
||||||
"karma": "^1.7.1",
|
"karma": "^2.0.0",
|
||||||
"karma-chrome-launcher": "^2.2.0",
|
"karma-chrome-launcher": "^2.2.0",
|
||||||
"karma-cli": "^1.0.1",
|
"karma-cli": "^1.0.1",
|
||||||
"karma-firefox-launcher": "^1.0.1",
|
"karma-firefox-launcher": "^1.0.1",
|
||||||
@ -244,7 +244,7 @@
|
|||||||
"react-testutils-additions": "^15.2.0",
|
"react-testutils-additions": "^15.2.0",
|
||||||
"redux-test-utils": "^0.2.2",
|
"redux-test-utils": "^0.2.2",
|
||||||
"sinon": "^4.0.0",
|
"sinon": "^4.0.0",
|
||||||
"stylelint-config-standard": "^17.0.0",
|
"stylelint-config-standard": "^18.2.0",
|
||||||
"tape": "^4.5.1",
|
"tape": "^4.5.1",
|
||||||
"testem": "^2.0.0",
|
"testem": "^2.0.0",
|
||||||
"uglifyify": "^4.0.2",
|
"uglifyify": "^4.0.2",
|
||||||
|
@ -1,11 +1,11 @@
|
|||||||
const assert = require('assert')
|
const assert = require('assert')
|
||||||
const MessageManger = require('../../app/scripts/lib/message-manager')
|
const MessageManager = require('../../app/scripts/lib/message-manager')
|
||||||
|
|
||||||
describe('Message Manager', function () {
|
describe('Message Manager', function () {
|
||||||
let messageManager
|
let messageManager
|
||||||
|
|
||||||
beforeEach(function () {
|
beforeEach(function () {
|
||||||
messageManager = new MessageManger()
|
messageManager = new MessageManager()
|
||||||
})
|
})
|
||||||
|
|
||||||
describe('#getMsgList', function () {
|
describe('#getMsgList', function () {
|
||||||
|
@ -15,11 +15,8 @@ describe('# Network Controller', function () {
|
|||||||
|
|
||||||
beforeEach(function () {
|
beforeEach(function () {
|
||||||
|
|
||||||
nock('https://api.infura.io')
|
|
||||||
.get('/*/')
|
|
||||||
.reply(200)
|
|
||||||
|
|
||||||
nock('https://rinkeby.infura.io')
|
nock('https://rinkeby.infura.io')
|
||||||
|
.persist()
|
||||||
.post('/metamask')
|
.post('/metamask')
|
||||||
.reply(200)
|
.reply(200)
|
||||||
|
|
||||||
@ -29,6 +26,11 @@ describe('# Network Controller', function () {
|
|||||||
|
|
||||||
networkController.initializeProvider(networkControllerProviderInit, provider)
|
networkController.initializeProvider(networkControllerProviderInit, provider)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
afterEach(function () {
|
||||||
|
nock.cleanAll()
|
||||||
|
})
|
||||||
|
|
||||||
describe('network', function () {
|
describe('network', function () {
|
||||||
describe('#provider', function () {
|
describe('#provider', function () {
|
||||||
it('provider should be updatable without reassignment', function () {
|
it('provider should be updatable without reassignment', function () {
|
||||||
|
133
test/unit/seed-phrase-verifier-test.js
Normal file
133
test/unit/seed-phrase-verifier-test.js
Normal file
@ -0,0 +1,133 @@
|
|||||||
|
const assert = require('assert')
|
||||||
|
const clone = require('clone')
|
||||||
|
const KeyringController = require('eth-keyring-controller')
|
||||||
|
const firstTimeState = require('../../app/scripts/first-time-state')
|
||||||
|
const seedPhraseVerifier = require('../../app/scripts/lib/seed-phrase-verifier')
|
||||||
|
const mockEncryptor = require('../lib/mock-encryptor')
|
||||||
|
|
||||||
|
describe('SeedPhraseVerifier', function () {
|
||||||
|
|
||||||
|
describe('verifyAccounts', function () {
|
||||||
|
|
||||||
|
let password = 'passw0rd1'
|
||||||
|
let hdKeyTree = 'HD Key Tree'
|
||||||
|
|
||||||
|
let keyringController
|
||||||
|
let vault
|
||||||
|
let primaryKeyring
|
||||||
|
|
||||||
|
beforeEach(async function () {
|
||||||
|
keyringController = new KeyringController({
|
||||||
|
initState: clone(firstTimeState),
|
||||||
|
encryptor: mockEncryptor,
|
||||||
|
})
|
||||||
|
|
||||||
|
assert(keyringController)
|
||||||
|
|
||||||
|
vault = await keyringController.createNewVaultAndKeychain(password)
|
||||||
|
primaryKeyring = keyringController.getKeyringsByType(hdKeyTree)[0]
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should be able to verify created account with seed words', async function () {
|
||||||
|
|
||||||
|
let createdAccounts = await primaryKeyring.getAccounts()
|
||||||
|
assert.equal(createdAccounts.length, 1)
|
||||||
|
|
||||||
|
let serialized = await primaryKeyring.serialize()
|
||||||
|
let seedWords = serialized.mnemonic
|
||||||
|
assert.notEqual(seedWords.length, 0)
|
||||||
|
|
||||||
|
let result = await seedPhraseVerifier.verifyAccounts(createdAccounts, seedWords)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should be able to verify created account (upper case) with seed words', async function () {
|
||||||
|
|
||||||
|
let createdAccounts = await primaryKeyring.getAccounts()
|
||||||
|
assert.equal(createdAccounts.length, 1)
|
||||||
|
|
||||||
|
let upperCaseAccounts = [createdAccounts[0].toUpperCase()]
|
||||||
|
|
||||||
|
let serialized = await primaryKeyring.serialize()
|
||||||
|
let seedWords = serialized.mnemonic
|
||||||
|
assert.notEqual(seedWords.length, 0)
|
||||||
|
|
||||||
|
let result = await seedPhraseVerifier.verifyAccounts(upperCaseAccounts, seedWords)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should be able to verify created account (lower case) with seed words', async function () {
|
||||||
|
|
||||||
|
let createdAccounts = await primaryKeyring.getAccounts()
|
||||||
|
assert.equal(createdAccounts.length, 1)
|
||||||
|
let lowerCaseAccounts = [createdAccounts[0].toLowerCase()]
|
||||||
|
|
||||||
|
let serialized = await primaryKeyring.serialize()
|
||||||
|
let seedWords = serialized.mnemonic
|
||||||
|
assert.notEqual(seedWords.length, 0)
|
||||||
|
|
||||||
|
let result = await seedPhraseVerifier.verifyAccounts(lowerCaseAccounts, seedWords)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return error with good but different seed words', async function () {
|
||||||
|
|
||||||
|
let createdAccounts = await primaryKeyring.getAccounts()
|
||||||
|
assert.equal(createdAccounts.length, 1)
|
||||||
|
|
||||||
|
let serialized = await primaryKeyring.serialize()
|
||||||
|
let seedWords = 'debris dizzy just program just float decrease vacant alarm reduce speak stadium'
|
||||||
|
|
||||||
|
try {
|
||||||
|
let result = await seedPhraseVerifier.verifyAccounts(createdAccounts, seedWords)
|
||||||
|
assert.fail("Should reject")
|
||||||
|
} catch (err) {
|
||||||
|
assert.ok(err.message.indexOf('Not identical accounts!') >= 0, 'Wrong error message')
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return error with undefined existing accounts', async function () {
|
||||||
|
|
||||||
|
let createdAccounts = await primaryKeyring.getAccounts()
|
||||||
|
assert.equal(createdAccounts.length, 1)
|
||||||
|
|
||||||
|
let serialized = await primaryKeyring.serialize()
|
||||||
|
let seedWords = 'debris dizzy just program just float decrease vacant alarm reduce speak stadium'
|
||||||
|
|
||||||
|
try {
|
||||||
|
let result = await seedPhraseVerifier.verifyAccounts(undefined, seedWords)
|
||||||
|
assert.fail("Should reject")
|
||||||
|
} catch (err) {
|
||||||
|
assert.equal(err.message, 'No created accounts defined.')
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return error with empty accounts array', async function () {
|
||||||
|
|
||||||
|
let createdAccounts = await primaryKeyring.getAccounts()
|
||||||
|
assert.equal(createdAccounts.length, 1)
|
||||||
|
|
||||||
|
let serialized = await primaryKeyring.serialize()
|
||||||
|
let seedWords = 'debris dizzy just program just float decrease vacant alarm reduce speak stadium'
|
||||||
|
|
||||||
|
try {
|
||||||
|
let result = await seedPhraseVerifier.verifyAccounts([], seedWords)
|
||||||
|
assert.fail("Should reject")
|
||||||
|
} catch (err) {
|
||||||
|
assert.equal(err.message, 'No created accounts defined.')
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should be able to verify more than one created account with seed words', async function () {
|
||||||
|
|
||||||
|
const keyState = await keyringController.addNewAccount(primaryKeyring)
|
||||||
|
const keyState2 = await keyringController.addNewAccount(primaryKeyring)
|
||||||
|
|
||||||
|
let createdAccounts = await primaryKeyring.getAccounts()
|
||||||
|
assert.equal(createdAccounts.length, 3)
|
||||||
|
|
||||||
|
let serialized = await primaryKeyring.serialize()
|
||||||
|
let seedWords = serialized.mnemonic
|
||||||
|
assert.notEqual(seedWords.length, 0)
|
||||||
|
|
||||||
|
let result = await seedPhraseVerifier.verifyAccounts(createdAccounts, seedWords)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
@ -5,7 +5,7 @@ const TxStateManager = require('../../app/scripts/lib/tx-state-manager')
|
|||||||
const txStateHistoryHelper = require('../../app/scripts/lib/tx-state-history-helper')
|
const txStateHistoryHelper = require('../../app/scripts/lib/tx-state-history-helper')
|
||||||
const noop = () => true
|
const noop = () => true
|
||||||
|
|
||||||
describe('TransactionStateManger', function () {
|
describe('TransactionStateManager', function () {
|
||||||
let txStateManager
|
let txStateManager
|
||||||
const currentNetworkId = 42
|
const currentNetworkId = 42
|
||||||
const otherNetworkId = 2
|
const otherNetworkId = 2
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
const inherits = require('util').inherits
|
|
||||||
const Component = require('react').Component
|
const Component = require('react').Component
|
||||||
|
const PropTypes = require('prop-types')
|
||||||
const h = require('react-hyperscript')
|
const h = require('react-hyperscript')
|
||||||
const connect = require('react-redux').connect
|
const connect = require('react-redux').connect
|
||||||
const actions = require('../../actions')
|
const actions = require('../../actions')
|
||||||
@ -8,27 +8,27 @@ const t = require('../../../i18n')
|
|||||||
|
|
||||||
const HELP_LINK = 'https://support.metamask.io/kb/article/7-importing-accounts'
|
const HELP_LINK = 'https://support.metamask.io/kb/article/7-importing-accounts'
|
||||||
|
|
||||||
module.exports = connect(mapStateToProps)(JsonImportSubview)
|
class JsonImportSubview extends Component {
|
||||||
|
constructor (props) {
|
||||||
|
super(props)
|
||||||
|
|
||||||
function mapStateToProps (state) {
|
this.state = {
|
||||||
return {
|
file: null,
|
||||||
error: state.appState.warning,
|
fileContents: '',
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
inherits(JsonImportSubview, Component)
|
render () {
|
||||||
function JsonImportSubview () {
|
|
||||||
Component.call(this)
|
|
||||||
}
|
|
||||||
|
|
||||||
JsonImportSubview.prototype.render = function () {
|
|
||||||
const { error } = this.props
|
const { error } = this.props
|
||||||
|
|
||||||
return (
|
return (
|
||||||
h('div.new-account-import-form__json', [
|
h('div.new-account-import-form__json', [
|
||||||
|
|
||||||
h('p', t('usedByClients')),
|
h('p', 'Used by a variety of different clients'),
|
||||||
h('a.warning', { href: HELP_LINK, target: '_blank' }, t('fileImportFail')),
|
h('a.warning', {
|
||||||
|
href: HELP_LINK,
|
||||||
|
target: '_blank',
|
||||||
|
}, 'File import not working? Click here!'),
|
||||||
|
|
||||||
h(FileInput, {
|
h(FileInput, {
|
||||||
readAs: 'text',
|
readAs: 'text',
|
||||||
@ -43,23 +43,23 @@ JsonImportSubview.prototype.render = function () {
|
|||||||
|
|
||||||
h('input.new-account-import-form__input-password', {
|
h('input.new-account-import-form__input-password', {
|
||||||
type: 'password',
|
type: 'password',
|
||||||
placeholder: t('enterPassword'),
|
placeholder: 'Enter password',
|
||||||
id: 'json-password-box',
|
id: 'json-password-box',
|
||||||
onKeyPress: this.createKeyringOnEnter.bind(this),
|
onKeyPress: this.createKeyringOnEnter.bind(this),
|
||||||
}),
|
}),
|
||||||
|
|
||||||
h('div.new-account-create-form__buttons', {}, [
|
h('div.new-account-create-form__buttons', {}, [
|
||||||
|
|
||||||
h('button.new-account-create-form__button-cancel.allcaps', {
|
h('button.new-account-create-form__button-cancel', {
|
||||||
onClick: () => this.props.goHome(),
|
onClick: () => this.props.goHome(),
|
||||||
}, [
|
}, [
|
||||||
t('cancel'),
|
'CANCEL',
|
||||||
]),
|
]),
|
||||||
|
|
||||||
h('button.new-account-create-form__button-create.allcaps', {
|
h('button.new-account-create-form__button-create', {
|
||||||
onClick: () => this.createNewKeychain.bind(this),
|
onClick: () => this.createNewKeychain(),
|
||||||
}, [
|
}, [
|
||||||
t('import'),
|
'IMPORT',
|
||||||
]),
|
]),
|
||||||
|
|
||||||
]),
|
]),
|
||||||
@ -69,39 +69,63 @@ JsonImportSubview.prototype.render = function () {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
JsonImportSubview.prototype.onLoad = function (event, file) {
|
onLoad (event, file) {
|
||||||
this.setState({file: file, fileContents: event.target.result})
|
this.setState({file: file, fileContents: event.target.result})
|
||||||
}
|
}
|
||||||
|
|
||||||
JsonImportSubview.prototype.createKeyringOnEnter = function (event) {
|
createKeyringOnEnter (event) {
|
||||||
if (event.key === 'Enter') {
|
if (event.key === 'Enter') {
|
||||||
event.preventDefault()
|
event.preventDefault()
|
||||||
this.createNewKeychain()
|
this.createNewKeychain()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
JsonImportSubview.prototype.createNewKeychain = function () {
|
createNewKeychain () {
|
||||||
const state = this.state
|
const state = this.state
|
||||||
|
|
||||||
if (!state) {
|
if (!state) {
|
||||||
const message = 'You must select a valid file to import.'
|
const message = 'You must select a valid file to import.'
|
||||||
return this.props.dispatch(actions.displayWarning(message))
|
return this.props.displayWarning(message)
|
||||||
}
|
}
|
||||||
|
|
||||||
const { fileContents } = state
|
const { fileContents } = state
|
||||||
|
|
||||||
if (!fileContents) {
|
if (!fileContents) {
|
||||||
const message = t('needImportFile')
|
const message = 'You must select a file to import.'
|
||||||
return this.props.dispatch(actions.displayWarning(message))
|
return this.props.displayWarning(message)
|
||||||
}
|
}
|
||||||
|
|
||||||
const passwordInput = document.getElementById('json-password-box')
|
const passwordInput = document.getElementById('json-password-box')
|
||||||
const password = passwordInput.value
|
const password = passwordInput.value
|
||||||
|
|
||||||
if (!password) {
|
if (!password) {
|
||||||
const message = t('needImportPassword')
|
const message = 'You must enter a password for the selected file.'
|
||||||
return this.props.dispatch(actions.displayWarning(message))
|
return this.props.displayWarning(message)
|
||||||
}
|
}
|
||||||
|
|
||||||
this.props.dispatch(actions.importNewAccount('JSON File', [ fileContents, password ]))
|
this.props.importNewJsonAccount([ fileContents, password ])
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
JsonImportSubview.propTypes = {
|
||||||
|
error: PropTypes.string,
|
||||||
|
goHome: PropTypes.func,
|
||||||
|
displayWarning: PropTypes.func,
|
||||||
|
importNewJsonAccount: PropTypes.func,
|
||||||
|
}
|
||||||
|
|
||||||
|
const mapStateToProps = state => {
|
||||||
|
return {
|
||||||
|
error: state.appState.warning,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const mapDispatchToProps = dispatch => {
|
||||||
|
return {
|
||||||
|
goHome: () => dispatch(actions.goHome()),
|
||||||
|
displayWarning: warning => dispatch(actions.displayWarning(warning)),
|
||||||
|
importNewJsonAccount: options => dispatch(actions.importNewAccount('JSON File', options)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = connect(mapStateToProps, mapDispatchToProps)(JsonImportSubview)
|
||||||
|
@ -284,19 +284,42 @@ function goHome () {
|
|||||||
// async actions
|
// async actions
|
||||||
|
|
||||||
function tryUnlockMetamask (password) {
|
function tryUnlockMetamask (password) {
|
||||||
return (dispatch) => {
|
return dispatch => {
|
||||||
dispatch(actions.showLoadingIndication())
|
dispatch(actions.showLoadingIndication())
|
||||||
dispatch(actions.unlockInProgress())
|
dispatch(actions.unlockInProgress())
|
||||||
log.debug(`background.submitPassword`)
|
log.debug(`background.submitPassword`)
|
||||||
background.submitPassword(password, (err) => {
|
|
||||||
dispatch(actions.hideLoadingIndication())
|
return new Promise((resolve, reject) => {
|
||||||
if (err) {
|
background.submitPassword(password, error => {
|
||||||
dispatch(actions.unlockFailed(err.message))
|
if (error) {
|
||||||
} else {
|
return reject(error)
|
||||||
dispatch(actions.unlockSucceeded())
|
|
||||||
dispatch(actions.transitionForward())
|
|
||||||
forceUpdateMetamaskState(dispatch)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
resolve()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.then(() => {
|
||||||
|
dispatch(actions.unlockSucceeded())
|
||||||
|
return forceUpdateMetamaskState(dispatch)
|
||||||
|
})
|
||||||
|
.then(() => {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
background.verifySeedPhrase(err => {
|
||||||
|
if (err) {
|
||||||
|
dispatch(actions.displayWarning(err.message))
|
||||||
|
}
|
||||||
|
|
||||||
|
resolve()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.then(() => {
|
||||||
|
dispatch(actions.transitionForward())
|
||||||
|
dispatch(actions.hideLoadingIndication())
|
||||||
|
})
|
||||||
|
.catch(err => {
|
||||||
|
dispatch(actions.unlockFailed(err.message))
|
||||||
|
dispatch(actions.hideLoadingIndication())
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -339,46 +362,53 @@ function createNewVaultAndRestore (password, seed) {
|
|||||||
log.debug(`background.createNewVaultAndRestore`)
|
log.debug(`background.createNewVaultAndRestore`)
|
||||||
|
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
background.createNewVaultAndRestore(password, seed, (err) => {
|
background.createNewVaultAndRestore(password, seed, err => {
|
||||||
|
|
||||||
dispatch(actions.hideLoadingIndication())
|
|
||||||
|
|
||||||
if (err) {
|
if (err) {
|
||||||
dispatch(actions.displayWarning(err.message))
|
|
||||||
return reject(err)
|
return reject(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
dispatch(actions.showAccountsPage())
|
|
||||||
resolve()
|
resolve()
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
.then(() => dispatch(actions.unMarkPasswordForgotten()))
|
||||||
|
.then(() => {
|
||||||
|
dispatch(actions.showAccountsPage())
|
||||||
|
dispatch(actions.hideLoadingIndication())
|
||||||
|
})
|
||||||
|
.catch(err => {
|
||||||
|
dispatch(actions.displayWarning(err.message))
|
||||||
|
dispatch(actions.hideLoadingIndication())
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function createNewVaultAndKeychain (password) {
|
function createNewVaultAndKeychain (password) {
|
||||||
return (dispatch) => {
|
return dispatch => {
|
||||||
dispatch(actions.showLoadingIndication())
|
dispatch(actions.showLoadingIndication())
|
||||||
log.debug(`background.createNewVaultAndKeychain`)
|
log.debug(`background.createNewVaultAndKeychain`)
|
||||||
|
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
background.createNewVaultAndKeychain(password, (err) => {
|
background.createNewVaultAndKeychain(password, err => {
|
||||||
if (err) {
|
if (err) {
|
||||||
dispatch(actions.displayWarning(err.message))
|
dispatch(actions.displayWarning(err.message))
|
||||||
return reject(err)
|
return reject(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
log.debug(`background.placeSeedWords`)
|
log.debug(`background.placeSeedWords`)
|
||||||
|
|
||||||
background.placeSeedWords((err) => {
|
background.placeSeedWords((err) => {
|
||||||
if (err) {
|
if (err) {
|
||||||
dispatch(actions.displayWarning(err.message))
|
dispatch(actions.displayWarning(err.message))
|
||||||
return reject(err)
|
return reject(err)
|
||||||
}
|
}
|
||||||
dispatch(actions.hideLoadingIndication())
|
|
||||||
forceUpdateMetamaskState(dispatch)
|
|
||||||
resolve()
|
resolve()
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
.then(() => forceUpdateMetamaskState(dispatch))
|
||||||
|
.then(() => dispatch(actions.hideLoadingIndication()))
|
||||||
|
.catch(() => dispatch(actions.hideLoadingIndication()))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -389,18 +419,27 @@ function revealSeedConfirmation () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function requestRevealSeed (password) {
|
function requestRevealSeed (password) {
|
||||||
return (dispatch) => {
|
return dispatch => {
|
||||||
dispatch(actions.showLoadingIndication())
|
dispatch(actions.showLoadingIndication())
|
||||||
log.debug(`background.submitPassword`)
|
log.debug(`background.submitPassword`)
|
||||||
background.submitPassword(password, (err) => {
|
return new Promise((resolve, reject) => {
|
||||||
|
background.submitPassword(password, err => {
|
||||||
if (err) {
|
if (err) {
|
||||||
return dispatch(actions.displayWarning(err.message))
|
dispatch(actions.displayWarning(err.message))
|
||||||
|
return reject(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
log.debug(`background.placeSeedWords`)
|
log.debug(`background.placeSeedWords`)
|
||||||
background.placeSeedWords((err, result) => {
|
background.placeSeedWords((err, result) => {
|
||||||
if (err) return dispatch(actions.displayWarning(err.message))
|
if (err) {
|
||||||
dispatch(actions.hideLoadingIndication())
|
dispatch(actions.displayWarning(err.message))
|
||||||
|
return reject(err)
|
||||||
|
}
|
||||||
|
|
||||||
dispatch(actions.showNewVaultSeed(result))
|
dispatch(actions.showNewVaultSeed(result))
|
||||||
|
dispatch(actions.hideLoadingIndication())
|
||||||
|
resolve()
|
||||||
|
})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@ -851,11 +890,14 @@ function markPasswordForgotten () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function unMarkPasswordForgotten () {
|
function unMarkPasswordForgotten () {
|
||||||
return (dispatch) => {
|
return dispatch => {
|
||||||
return background.unMarkPasswordForgotten(() => {
|
return new Promise(resolve => {
|
||||||
|
background.unMarkPasswordForgotten(() => {
|
||||||
dispatch(actions.forgotPassword(false))
|
dispatch(actions.forgotPassword(false))
|
||||||
forceUpdateMetamaskState(dispatch)
|
resolve()
|
||||||
})
|
})
|
||||||
|
})
|
||||||
|
.then(() => forceUpdateMetamaskState(dispatch))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1704,11 +1746,16 @@ function callBackgroundThenUpdate (method, ...args) {
|
|||||||
|
|
||||||
function forceUpdateMetamaskState (dispatch) {
|
function forceUpdateMetamaskState (dispatch) {
|
||||||
log.debug(`background.getState`)
|
log.debug(`background.getState`)
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
background.getState((err, newState) => {
|
background.getState((err, newState) => {
|
||||||
if (err) {
|
if (err) {
|
||||||
return dispatch(actions.displayWarning(err.message))
|
dispatch(actions.displayWarning(err.message))
|
||||||
|
return reject(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
dispatch(actions.updateMetamaskState(newState))
|
dispatch(actions.updateMetamaskState(newState))
|
||||||
|
resolve()
|
||||||
|
})
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -71,13 +71,17 @@ AddTokenScreen.prototype.componentWillMount = function () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
AddTokenScreen.prototype.toggleToken = function (address, token) {
|
AddTokenScreen.prototype.toggleToken = function (address, token) {
|
||||||
const { selectedTokens, errors } = this.state
|
const { selectedTokens = {}, errors } = this.state
|
||||||
const { [address]: selectedToken } = selectedTokens
|
const selectedTokensCopy = { ...selectedTokens }
|
||||||
|
|
||||||
|
if (address in selectedTokensCopy) {
|
||||||
|
delete selectedTokensCopy[address]
|
||||||
|
} else {
|
||||||
|
selectedTokensCopy[address] = token
|
||||||
|
}
|
||||||
|
|
||||||
this.setState({
|
this.setState({
|
||||||
selectedTokens: {
|
selectedTokens: selectedTokensCopy,
|
||||||
...selectedTokens,
|
|
||||||
[address]: selectedToken ? null : token,
|
|
||||||
},
|
|
||||||
errors: {
|
errors: {
|
||||||
...errors,
|
...errors,
|
||||||
tokenSelector: null,
|
tokenSelector: null,
|
||||||
|
@ -115,7 +115,7 @@ NetworkDropdown.prototype.render = function () {
|
|||||||
[
|
[
|
||||||
providerType === 'mainnet' ? h('i.fa.fa-check') : h('.network-check__transparent', '✓'),
|
providerType === 'mainnet' ? h('i.fa.fa-check') : h('.network-check__transparent', '✓'),
|
||||||
h(NetworkDropdownIcon, {
|
h(NetworkDropdownIcon, {
|
||||||
backgroundColor: '#038789', // $blue-lagoon
|
backgroundColor: '#29B6AF', // $java
|
||||||
isSelected: providerType === 'mainnet',
|
isSelected: providerType === 'mainnet',
|
||||||
}),
|
}),
|
||||||
h('span.network-name-item', {
|
h('span.network-name-item', {
|
||||||
@ -137,7 +137,7 @@ NetworkDropdown.prototype.render = function () {
|
|||||||
[
|
[
|
||||||
providerType === 'ropsten' ? h('i.fa.fa-check') : h('.network-check__transparent', '✓'),
|
providerType === 'ropsten' ? h('i.fa.fa-check') : h('.network-check__transparent', '✓'),
|
||||||
h(NetworkDropdownIcon, {
|
h(NetworkDropdownIcon, {
|
||||||
backgroundColor: '#e91550', // $crimson
|
backgroundColor: '#ff4a8d', // $wild-strawberry
|
||||||
isSelected: providerType === 'ropsten',
|
isSelected: providerType === 'ropsten',
|
||||||
}),
|
}),
|
||||||
h('span.network-name-item', {
|
h('span.network-name-item', {
|
||||||
@ -159,7 +159,7 @@ NetworkDropdown.prototype.render = function () {
|
|||||||
[
|
[
|
||||||
providerType === 'kovan' ? h('i.fa.fa-check') : h('.network-check__transparent', '✓'),
|
providerType === 'kovan' ? h('i.fa.fa-check') : h('.network-check__transparent', '✓'),
|
||||||
h(NetworkDropdownIcon, {
|
h(NetworkDropdownIcon, {
|
||||||
backgroundColor: '#690496', // $purple
|
backgroundColor: '#7057ff', // $cornflower-blue
|
||||||
isSelected: providerType === 'kovan',
|
isSelected: providerType === 'kovan',
|
||||||
}),
|
}),
|
||||||
h('span.network-name-item', {
|
h('span.network-name-item', {
|
||||||
@ -181,7 +181,7 @@ NetworkDropdown.prototype.render = function () {
|
|||||||
[
|
[
|
||||||
providerType === 'rinkeby' ? h('i.fa.fa-check') : h('.network-check__transparent', '✓'),
|
providerType === 'rinkeby' ? h('i.fa.fa-check') : h('.network-check__transparent', '✓'),
|
||||||
h(NetworkDropdownIcon, {
|
h(NetworkDropdownIcon, {
|
||||||
backgroundColor: '#ebb33f', // $tulip-tree
|
backgroundColor: '#f6c343', // $saffron
|
||||||
isSelected: providerType === 'rinkeby',
|
isSelected: providerType === 'rinkeby',
|
||||||
}),
|
}),
|
||||||
h('span.network-name-item', {
|
h('span.network-name-item', {
|
||||||
|
@ -64,8 +64,8 @@ function mapDispatchToProps (dispatch, ownProps) {
|
|||||||
updateTokenExchangeRate: () => dispatch(actions.updateTokenExchangeRate(symbol)),
|
updateTokenExchangeRate: () => dispatch(actions.updateTokenExchangeRate(symbol)),
|
||||||
editTransaction: txMeta => {
|
editTransaction: txMeta => {
|
||||||
const { token: { address } } = ownProps
|
const { token: { address } } = ownProps
|
||||||
const { txParams, id } = txMeta
|
const { txParams = {}, id } = txMeta
|
||||||
const tokenData = txParams.data && abiDecoder.decodeMethod(txParams.data)
|
const tokenData = txParams.data && abiDecoder.decodeMethod(txParams.data) || {}
|
||||||
const { params = [] } = tokenData
|
const { params = [] } = tokenData
|
||||||
const { value: to } = params[0] || {}
|
const { value: to } = params[0] || {}
|
||||||
const { value: tokenAmountInDec } = params[1] || {}
|
const { value: tokenAmountInDec } = params[1] || {}
|
||||||
|
@ -190,7 +190,7 @@
|
|||||||
margin-top: 39px;
|
margin-top: 39px;
|
||||||
display: flex;
|
display: flex;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
justify-content: space-evenly;
|
justify-content: space-between;
|
||||||
}
|
}
|
||||||
|
|
||||||
&__button-cancel,
|
&__button-cancel,
|
||||||
|
@ -46,6 +46,10 @@ $manatee: #93949d;
|
|||||||
$spindle: #c7ddec;
|
$spindle: #c7ddec;
|
||||||
$mid-gray: #5b5d67;
|
$mid-gray: #5b5d67;
|
||||||
$cape-cod: #38393a;
|
$cape-cod: #38393a;
|
||||||
|
$java: #29b6af;
|
||||||
|
$wild-strawberry: #ff4a8d;
|
||||||
|
$cornflower-blue: #7057ff;
|
||||||
|
$saffron: #f6c343;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Z-Indicies
|
Z-Indicies
|
||||||
|
@ -107,12 +107,15 @@ RestoreVaultScreen.prototype.render = function () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
RestoreVaultScreen.prototype.showInitializeMenu = function () {
|
RestoreVaultScreen.prototype.showInitializeMenu = function () {
|
||||||
this.props.dispatch(actions.unMarkPasswordForgotten())
|
const { dispatch, forgottenPassword } = this.props
|
||||||
if (this.props.forgottenPassword) {
|
dispatch(actions.unMarkPasswordForgotten())
|
||||||
this.props.dispatch(actions.backToUnlockView())
|
.then(() => {
|
||||||
|
if (forgottenPassword) {
|
||||||
|
dispatch(actions.backToUnlockView())
|
||||||
} else {
|
} else {
|
||||||
this.props.dispatch(actions.showInitializeMenu())
|
dispatch(actions.showInitializeMenu())
|
||||||
}
|
}
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
RestoreVaultScreen.prototype.createOnEnter = function (event) {
|
RestoreVaultScreen.prototype.createOnEnter = function (event) {
|
||||||
@ -150,11 +153,5 @@ RestoreVaultScreen.prototype.createNewVaultAndRestore = function () {
|
|||||||
this.warning = null
|
this.warning = null
|
||||||
this.props.dispatch(actions.displayWarning(this.warning))
|
this.props.dispatch(actions.displayWarning(this.warning))
|
||||||
this.props.dispatch(actions.createNewVaultAndRestore(password, seed))
|
this.props.dispatch(actions.createNewVaultAndRestore(password, seed))
|
||||||
.then(() => {
|
.catch(err => log.error(err.message))
|
||||||
this.props.dispatch(actions.unMarkPasswordForgotten())
|
|
||||||
})
|
|
||||||
.catch((err) => {
|
|
||||||
log.error(err.message)
|
|
||||||
})
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -138,14 +138,18 @@ function reduceApp (state, action) {
|
|||||||
})
|
})
|
||||||
|
|
||||||
case actions.FORGOT_PASSWORD:
|
case actions.FORGOT_PASSWORD:
|
||||||
return extend(appState, {
|
const newState = extend(appState, {
|
||||||
currentView: {
|
|
||||||
name: action.value ? 'restoreVault' : 'accountDetail',
|
|
||||||
},
|
|
||||||
transForward: false,
|
|
||||||
forgottenPassword: action.value,
|
forgottenPassword: action.value,
|
||||||
})
|
})
|
||||||
|
|
||||||
|
if (action.value) {
|
||||||
|
newState.currentView = {
|
||||||
|
name: 'restoreVault',
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return newState
|
||||||
|
|
||||||
case actions.SHOW_INIT_MENU:
|
case actions.SHOW_INIT_MENU:
|
||||||
return extend(appState, {
|
return extend(appState, {
|
||||||
currentView: defaultView,
|
currentView: defaultView,
|
||||||
|
@ -130,8 +130,6 @@ function reduceMetamask (state, action) {
|
|||||||
|
|
||||||
case actions.SHOW_NEW_VAULT_SEED:
|
case actions.SHOW_NEW_VAULT_SEED:
|
||||||
return extend(metamaskState, {
|
return extend(metamaskState, {
|
||||||
isUnlocked: true,
|
|
||||||
isInitialized: false,
|
|
||||||
isRevealingSeedWords: true,
|
isRevealingSeedWords: true,
|
||||||
seedWords: action.value,
|
seedWords: action.value,
|
||||||
})
|
})
|
||||||
|
Loading…
Reference in New Issue
Block a user