aboutsummaryrefslogtreecommitdiffstats
path: root/test/e2e
diff options
context:
space:
mode:
Diffstat (limited to 'test/e2e')
-rw-r--r--test/e2e/beta/from-import-beta-ui.spec.js406
-rw-r--r--test/e2e/beta/helpers.js55
-rw-r--r--test/e2e/beta/metamask-beta-ui.spec.js491
-rwxr-xr-xtest/e2e/beta/run-all.sh10
-rw-r--r--test/e2e/func.js55
-rw-r--r--test/e2e/metamask.spec.js320
6 files changed, 1279 insertions, 58 deletions
diff --git a/test/e2e/beta/from-import-beta-ui.spec.js b/test/e2e/beta/from-import-beta-ui.spec.js
new file mode 100644
index 000000000..e07d4a99e
--- /dev/null
+++ b/test/e2e/beta/from-import-beta-ui.spec.js
@@ -0,0 +1,406 @@
+const path = require('path')
+const assert = require('assert')
+const webdriver = require('selenium-webdriver')
+const { By, Key } = webdriver
+const {
+ delay,
+ buildChromeWebDriver,
+ buildFirefoxWebdriver,
+ installWebExt,
+ getExtensionIdChrome,
+ getExtensionIdFirefox,
+} = require('../func')
+const {
+ checkBrowserForConsoleErrors,
+ loadExtension,
+ verboseReportOnFailure,
+} = require('./helpers')
+
+describe('Using MetaMask with an existing account', function () {
+ let extensionId
+ let driver
+ let tokenAddress
+
+ const testSeedPhrase = 'phrase upgrade clock rough situate wedding elder clever doctor stamp excess tent'
+ const testAddress = '0xE18035BF8712672935FDB4e5e431b1a0183d2DFC'
+ const regularDelayMs = 1000
+ const largeDelayMs = regularDelayMs * 2
+ const waitingNewPageDelayMs = regularDelayMs * 10
+
+ this.timeout(0)
+ this.bail(true)
+
+ before(async function () {
+ switch (process.env.SELENIUM_BROWSER) {
+ case 'chrome': {
+ const extensionPath = path.resolve('dist/chrome')
+ driver = buildChromeWebDriver(extensionPath)
+ extensionId = await getExtensionIdChrome(driver)
+ await driver.get(`chrome-extension://${extensionId}/popup.html`)
+ await delay(regularDelayMs)
+ break
+ }
+ case 'firefox': {
+ const extensionPath = path.resolve('dist/firefox')
+ driver = buildFirefoxWebdriver()
+ await installWebExt(driver, extensionPath)
+ await delay(regularDelayMs)
+ extensionId = await getExtensionIdFirefox(driver)
+ await driver.get(`moz-extension://${extensionId}/popup.html`)
+ await delay(regularDelayMs)
+ break
+ }
+ }
+ })
+
+ afterEach(async function () {
+ if (process.env.SELENIUM_BROWSER === 'chrome') {
+ const errors = await checkBrowserForConsoleErrors(driver)
+ if (errors.length) {
+ const errorReports = errors.map(err => err.message)
+ const errorMessage = `Errors found in browser console:\n${errorReports.join('\n')}`
+ console.error(new Error(errorMessage))
+ }
+ }
+ if (this.currentTest.state === 'failed') {
+ await verboseReportOnFailure(driver, this.currentTest)
+ }
+ })
+
+ after(async function () {
+ await driver.quit()
+ })
+
+ describe('New UI setup', async function () {
+ it('switches to first tab', async function () {
+ const [firstTab] = await driver.getAllWindowHandles()
+ await driver.switchTo().window(firstTab)
+ await delay(regularDelayMs)
+ })
+
+ it('use the local network', async function () {
+ const [networkSelector] = await driver.findElements(By.css('#network_component'))
+ await networkSelector.click()
+ await delay(regularDelayMs)
+
+ const [localhost] = await driver.findElements(By.xpath(`//li[contains(text(), 'Localhost')]`))
+ await localhost.click()
+ await delay(regularDelayMs)
+ })
+
+ it('selects the new UI option', async () => {
+ const button = await driver.findElement(By.xpath("//p[contains(text(), 'Try Beta Version')]"))
+ await button.click()
+ await delay(regularDelayMs)
+
+ // Close all other tabs
+ const [oldUi, infoPage, newUi] = await driver.getAllWindowHandles()
+ await driver.switchTo().window(oldUi)
+ await driver.close()
+ await driver.switchTo().window(infoPage)
+ await driver.close()
+ await driver.switchTo().window(newUi)
+ await delay(regularDelayMs)
+
+ const [continueBtn] = await driver.findElements(By.css('.welcome-screen__button'))
+ await continueBtn.click()
+ await delay(regularDelayMs)
+ })
+ })
+
+ describe('First time flow starting from an existing seed phrase', () => {
+ it('imports a seed phrase', async () => {
+ const [seedPhrase] = await driver.findElements(By.xpath(`//a[contains(text(), 'Import with seed phrase')]`))
+ await seedPhrase.click()
+ await delay(regularDelayMs)
+
+ const [seedTextArea] = await driver.findElements(By.css('textarea.import-account__secret-phrase'))
+ await seedTextArea.sendKeys(testSeedPhrase)
+ await delay(regularDelayMs)
+
+ const [password] = await driver.findElements(By.id('password'))
+ await password.sendKeys('correct horse battery staple')
+ const [confirmPassword] = await driver.findElements(By.id('confirm-password'))
+ confirmPassword.sendKeys('correct horse battery staple')
+
+ const [importButton] = await driver.findElements(By.xpath(`//button[contains(text(), 'Import')]`))
+ await importButton.click()
+ await delay(regularDelayMs)
+ })
+
+ it('clicks through the privacy notice', async () => {
+ const [nextScreen] = await driver.findElements(By.css('.tou button'))
+ await nextScreen.click()
+ await delay(regularDelayMs)
+
+ const canClickThrough = await driver.findElement(By.css('.tou button')).isEnabled()
+ assert.equal(canClickThrough, false, 'disabled continue button')
+ const element = await driver.findElement(By.linkText('Attributions'))
+ await driver.executeScript('arguments[0].scrollIntoView(true)', element)
+ await delay(regularDelayMs)
+
+ const [acceptTos] = await driver.findElements(By.css('.tou button'))
+ await acceptTos.click()
+ await delay(regularDelayMs)
+ })
+ })
+
+ describe('Show account information', () => {
+ it('shows the correct account address', async () => {
+ await driver.findElement(By.css('.wallet-view__details-button')).click()
+ await driver.findElement(By.css('.qr-wrapper')).isDisplayed()
+ await delay(regularDelayMs)
+
+ const [address] = await driver.findElements(By.css('input.qr-ellip-address'))
+ assert.equal(await address.getAttribute('value'), testAddress)
+
+ await driver.executeScript("document.querySelector('.account-modal-close').click()")
+ await delay(largeDelayMs)
+ })
+
+ it('shows a QR code for the account', async () => {
+ await driver.findElement(By.css('.wallet-view__details-button')).click()
+ await driver.findElement(By.css('.qr-wrapper')).isDisplayed()
+ await delay(regularDelayMs)
+
+ await driver.executeScript("document.querySelector('.account-modal-close').click()")
+ await delay(regularDelayMs)
+ })
+ })
+
+ describe('Log out and log back in', () => {
+ it('logs out of the account', async () => {
+ await driver.findElement(By.css('.account-menu__icon')).click()
+ await delay(regularDelayMs)
+
+ const [logoutButton] = await driver.findElements(By.css('.account-menu__logout-button'))
+ assert.equal(await logoutButton.getText(), 'Log out')
+ await logoutButton.click()
+ await delay(regularDelayMs)
+ })
+
+ it('accepts the account password after lock', async () => {
+ await driver.findElement(By.id('password')).sendKeys('correct horse battery staple')
+ await driver.findElement(By.id('password')).sendKeys(Key.ENTER)
+ await delay(largeDelayMs)
+ })
+ })
+
+ describe('Add an account', () => {
+ it('choose Create Account from the account menu', async () => {
+ await driver.findElement(By.css('.account-menu__icon')).click()
+ await delay(regularDelayMs)
+
+ const [createAccount] = await driver.findElements(By.xpath(`//div[contains(text(), 'Create Account')]`))
+ await createAccount.click()
+ await delay(regularDelayMs)
+ })
+
+ it('set account name', async () => {
+ const [accountName] = await driver.findElements(By.css('.new-account-create-form input'))
+ await accountName.sendKeys('2nd account')
+ await delay(regularDelayMs)
+
+ const [createButton] = await driver.findElements(By.xpath(`//button[contains(text(), 'Create')]`))
+ await createButton.click()
+ await delay(regularDelayMs)
+ })
+
+ it('should show the correct account name', async () => {
+ const [accountName] = await driver.findElements(By.css('.account-name'))
+ assert.equal(await accountName.getText(), '2nd account')
+ await delay(regularDelayMs)
+ })
+ })
+
+ describe('Switch back to original account', () => {
+ it('chooses the original account from the account menu', async () => {
+ await driver.findElement(By.css('.account-menu__icon')).click()
+ await delay(regularDelayMs)
+
+ const [originalAccountMenuItem] = await driver.findElements(By.css('.account-menu__name'))
+ await originalAccountMenuItem.click()
+ await delay(regularDelayMs)
+ })
+ })
+
+ describe('Send ETH from inside MetaMask', () => {
+ it('starts to send a transaction', async function () {
+ const [sendButton] = await driver.findElements(By.xpath(`//button[contains(text(), 'Send')]`))
+ await sendButton.click()
+ await delay(regularDelayMs)
+
+ const [inputAddress] = await driver.findElements(By.css('input[placeholder="Recipient Address"]'))
+ const [inputAmount] = await driver.findElements(By.css('.currency-display__input'))
+ await inputAddress.sendKeys('0x2f318C334780961FB129D2a6c30D0763d9a5C970')
+ await inputAmount.sendKeys('1')
+
+ // Set the gas limit
+ const [configureGas] = await driver.findElements(By.css('.send-v2__gas-fee-display button'))
+ await configureGas.click()
+ await delay(regularDelayMs)
+
+ const [save] = await driver.findElements(By.xpath(`//button[contains(text(), 'Save')]`))
+ await save.click()
+ await delay(regularDelayMs)
+
+ // Continue to next screen
+ const [nextScreen] = await driver.findElements(By.xpath(`//button[contains(text(), 'Next')]`))
+ await nextScreen.click()
+ await delay(regularDelayMs)
+ })
+
+ it('confirms the transaction', async function () {
+ const [confirmButton] = await driver.findElements(By.xpath(`//button[contains(text(), 'Confirm')]`))
+ await confirmButton.click()
+ await delay(regularDelayMs)
+ })
+
+ it('finds the transaction in the transactions list', async function () {
+ const transactions = await driver.findElements(By.css('.tx-list-item'))
+ assert.equal(transactions.length, 1)
+
+ const txValues = await driver.findElements(By.css('.tx-list-value'))
+ assert.equal(txValues.length, 1)
+ assert.equal(await txValues[0].getText(), '1 ETH')
+ })
+ })
+
+ describe('Send ETH from Faucet', () => {
+ it('starts a send transaction inside Faucet', async () => {
+ await driver.executeScript('window.open("https://faucet.metamask.io")')
+ await delay(waitingNewPageDelayMs)
+
+ const [extension, faucet] = await driver.getAllWindowHandles()
+ await driver.switchTo().window(faucet)
+ await delay(regularDelayMs)
+
+ const [send1eth] = await driver.findElements(By.xpath(`//button[contains(text(), '10 ether')]`))
+ await send1eth.click()
+ await delay(regularDelayMs)
+
+ await driver.switchTo().window(extension)
+ await loadExtension(driver, extensionId)
+ await delay(regularDelayMs)
+
+ const [confirmButton] = await driver.findElements(By.xpath(`//button[contains(text(),'Confirm')]`))
+ await confirmButton.click()
+ await delay(regularDelayMs)
+
+ await driver.switchTo().window(faucet)
+ await delay(regularDelayMs)
+ await driver.close()
+ await delay(regularDelayMs)
+ await driver.switchTo().window(extension)
+ await delay(regularDelayMs)
+ await loadExtension(driver, extensionId)
+ await delay(regularDelayMs)
+ })
+ })
+
+ describe('Add existing token using search', () => {
+ it('clicks on the Add Token button', async () => {
+ const [addToken] = await driver.findElements(By.xpath(`//button[contains(text(), 'Add Token')]`))
+ await addToken.click()
+ await delay(regularDelayMs)
+ })
+
+ it('picks an existing token', async () => {
+ const [tokenSearch] = await driver.findElements(By.css('input.add-token__input'))
+ await tokenSearch.sendKeys('BAT')
+ await delay(regularDelayMs)
+
+ const [token] = await driver.findElements(By.xpath("//div[contains(text(), 'BAT')]"))
+ await token.click()
+ await delay(regularDelayMs)
+
+ const [nextScreen] = await driver.findElements(By.xpath(`//button[contains(text(), 'Next')]`))
+ await nextScreen.click()
+ await delay(regularDelayMs)
+
+ const [addTokens] = await driver.findElements(By.xpath(`//button[contains(text(), 'Add Tokens')]`))
+ await addTokens.click()
+ await delay(largeDelayMs)
+ })
+
+ it('renders the balance for the new token', async () => {
+ const balance = await driver.findElement(By.css('.tx-view .balance-display .token-amount'))
+ const tokenAmount = await balance.getText()
+ assert.equal(tokenAmount, '0BAT')
+ await delay(regularDelayMs)
+ })
+ })
+
+ describe('Add a custom token from TokenFactory', () => {
+ it('creates a new token', async () => {
+ await driver.executeScript('window.open("https://tokenfactory.surge.sh/#/factory")')
+ await delay(waitingNewPageDelayMs)
+
+ const [extension, tokenFactory] = await driver.getAllWindowHandles()
+ await driver.switchTo().window(tokenFactory)
+ const [
+ totalSupply,
+ tokenName,
+ tokenDecimal,
+ tokenSymbol,
+ ] = await driver.findElements(By.css('input'))
+
+ await totalSupply.sendKeys('100')
+ await tokenName.sendKeys('Test')
+ await tokenDecimal.sendKeys('0')
+ await tokenSymbol.sendKeys('TST')
+
+ const [createToken] = await driver.findElements(By.xpath(`//button[contains(text(), 'Create Token')]`))
+ await createToken.click()
+ await delay(regularDelayMs)
+
+ await driver.switchTo().window(extension)
+ await loadExtension(driver, extensionId)
+ await delay(regularDelayMs)
+
+ const [confirmButton] = await driver.findElements(By.xpath(`//button[contains(text(),'Confirm')]`))
+ await confirmButton.click()
+ await delay(regularDelayMs)
+
+ await driver.switchTo().window(tokenFactory)
+ await delay(regularDelayMs)
+ const tokenContactAddress = await driver.findElement(By.css('div > div > div:nth-child(2) > span:nth-child(3)'))
+ tokenAddress = await tokenContactAddress.getText()
+ await driver.close()
+ await driver.switchTo().window(extension)
+ await loadExtension(driver, extensionId)
+ await delay(regularDelayMs)
+ })
+
+ it('clicks on the Add Token button', async () => {
+ const [addToken] = await driver.findElements(By.xpath(`//button[contains(text(), 'Add Token')]`))
+ await addToken.click()
+ await delay(regularDelayMs)
+ })
+
+ it('picks the new Test token', async () => {
+ const [addCustomToken] = await driver.findElements(By.xpath("//div[contains(text(), 'Custom Token')]"))
+ await addCustomToken.click()
+ await delay(regularDelayMs)
+
+ const [newTokenAddress] = await driver.findElements(By.css('.add-token__add-custom-form input'))
+ await newTokenAddress.sendKeys(tokenAddress)
+ await delay(regularDelayMs)
+
+ const [nextScreen] = await driver.findElements(By.xpath(`//button[contains(text(), 'Next')]`))
+ await nextScreen.click()
+ await delay(regularDelayMs)
+
+ const [addTokens] = await driver.findElements(By.xpath(`//button[contains(text(), 'Add Tokens')]`))
+ await addTokens.click()
+ await delay(regularDelayMs)
+ })
+
+ it('renders the balance for the new token', async () => {
+ const [balance] = await driver.findElements(By.css('.tx-view .balance-display .token-amount'))
+ const tokenAmount = await balance.getText()
+ assert.equal(tokenAmount, '100TST')
+ await delay(regularDelayMs)
+ })
+ })
+})
diff --git a/test/e2e/beta/helpers.js b/test/e2e/beta/helpers.js
new file mode 100644
index 000000000..8307fdc50
--- /dev/null
+++ b/test/e2e/beta/helpers.js
@@ -0,0 +1,55 @@
+const fs = require('fs')
+const mkdirp = require('mkdirp')
+const pify = require('pify')
+
+module.exports = {
+ checkBrowserForConsoleErrors,
+ loadExtension,
+ verboseReportOnFailure,
+}
+
+async function loadExtension (driver, extensionId) {
+ switch (process.env.SELENIUM_BROWSER) {
+ case 'chrome': {
+ await driver.get(`chrome-extension://${extensionId}/home.html`)
+ break
+ }
+ case 'firefox': {
+ await driver.get(`moz-extension://${extensionId}/home.html`)
+ break
+ }
+ }
+}
+
+async function checkBrowserForConsoleErrors (driver) {
+ const ignoredLogTypes = ['WARNING']
+ const ignoredErrorMessages = [
+ // React throws error warnings on "dataset", but still sets the data-* properties correctly
+ 'Warning: Unknown prop `dataset` on ',
+ // Third-party Favicon 404s show up as errors
+ 'favicon.ico - Failed to load resource: the server responded with a status of 404 (Not Found)',
+ // React Development build - known issue blocked by test build sys
+ 'Warning: It looks like you\'re using a minified copy of the development build of React.',
+ // Redux Development build - known issue blocked by test build sys
+ 'This means that you are running a slower development build of Redux.',
+ ]
+ const browserLogs = await driver.manage().logs().get('browser')
+ const errorEntries = browserLogs.filter(entry => !ignoredLogTypes.includes(entry.level.toString()))
+ const errorObjects = errorEntries.map(entry => entry.toJSON())
+ return errorObjects.filter(entry => !ignoredErrorMessages.some(message => entry.message.includes(message)))
+}
+
+async function verboseReportOnFailure (driver, test) {
+ let artifactDir
+ if (process.env.SELENIUM_BROWSER === 'chrome') {
+ artifactDir = `./test-artifacts/chrome/${test.title}`
+ } else if (process.env.SELENIUM_BROWSER === 'firefox') {
+ artifactDir = `./test-artifacts/firefox/${test.title}`
+ }
+ const filepathBase = `${artifactDir}/test-failure`
+ await pify(mkdirp)(artifactDir)
+ const screenshot = await driver.takeScreenshot()
+ await pify(fs.writeFile)(`${filepathBase}-screenshot.png`, screenshot, { encoding: 'base64' })
+ const htmlSource = await driver.getPageSource()
+ await pify(fs.writeFile)(`${filepathBase}-dom.html`, htmlSource)
+}
diff --git a/test/e2e/beta/metamask-beta-ui.spec.js b/test/e2e/beta/metamask-beta-ui.spec.js
new file mode 100644
index 000000000..00863e3b3
--- /dev/null
+++ b/test/e2e/beta/metamask-beta-ui.spec.js
@@ -0,0 +1,491 @@
+const path = require('path')
+const assert = require('assert')
+const webdriver = require('selenium-webdriver')
+const { By, Key } = webdriver
+const {
+ delay,
+ buildChromeWebDriver,
+ buildFirefoxWebdriver,
+ installWebExt,
+ getExtensionIdChrome,
+ getExtensionIdFirefox,
+} = require('../func')
+const {
+ checkBrowserForConsoleErrors,
+ loadExtension,
+ verboseReportOnFailure,
+} = require('./helpers')
+
+describe('MetaMask', function () {
+ let extensionId
+ let driver
+ let tokenAddress
+
+ const testSeedPhrase = 'phrase upgrade clock rough situate wedding elder clever doctor stamp excess tent'
+ const tinyDelayMs = 500
+ const regularDelayMs = tinyDelayMs * 2
+ const largeDelayMs = regularDelayMs * 2
+ const waitingNewPageDelayMs = regularDelayMs * 10
+
+ this.timeout(0)
+ this.bail(true)
+
+ before(async function () {
+ switch (process.env.SELENIUM_BROWSER) {
+ case 'chrome': {
+ const extPath = path.resolve('dist/chrome')
+ driver = buildChromeWebDriver(extPath)
+ extensionId = await getExtensionIdChrome(driver)
+ await driver.get(`chrome-extension://${extensionId}/popup.html`)
+ break
+ }
+ case 'firefox': {
+ const extPath = path.resolve('dist/firefox')
+ driver = buildFirefoxWebdriver()
+ await installWebExt(driver, extPath)
+ await delay(700)
+ extensionId = await getExtensionIdFirefox(driver)
+ await driver.get(`moz-extension://${extensionId}/popup.html`)
+ }
+ }
+ })
+
+ afterEach(async function () {
+ if (process.env.SELENIUM_BROWSER === 'chrome') {
+ const errors = await checkBrowserForConsoleErrors(driver)
+ if (errors.length) {
+ const errorReports = errors.map(err => err.message)
+ const errorMessage = `Errors found in browser console:\n${errorReports.join('\n')}`
+ console.error(new Error(errorMessage))
+ }
+ }
+ if (this.currentTest.state === 'failed') {
+ await verboseReportOnFailure(this.currentTest)
+ }
+ })
+
+ after(async function () {
+ await driver.quit()
+ })
+
+ describe('New UI setup', async function () {
+ it('switches to first tab', async function () {
+ const [firstTab] = await driver.getAllWindowHandles()
+ await driver.switchTo().window(firstTab)
+ await delay(regularDelayMs)
+ })
+
+ it('use the local network', async function () {
+ const [networkSelector] = await driver.findElements(By.css('#network_component'))
+ await networkSelector.click()
+ await delay(regularDelayMs)
+
+ const [localhost] = await driver.findElements(By.xpath(`//li[contains(text(), 'Localhost')]`))
+ await localhost.click()
+ await delay(regularDelayMs)
+ })
+
+ it('selects the new UI option', async () => {
+ const button = await driver.findElement(By.xpath("//p[contains(text(), 'Try Beta Version')]"))
+ await button.click()
+ await delay(regularDelayMs)
+
+ // Close all other tabs
+ const [oldUi, infoPage, newUi] = await driver.getAllWindowHandles()
+ await driver.switchTo().window(oldUi)
+ await driver.close()
+ await driver.switchTo().window(infoPage)
+ await driver.close()
+ await driver.switchTo().window(newUi)
+ await delay(regularDelayMs)
+
+ const [continueBtn] = await driver.findElements(By.css('.welcome-screen__button'))
+ await continueBtn.click()
+ await delay(regularDelayMs)
+ })
+ })
+
+ describe('Going through the first time flow', () => {
+ it('accepts a secure password', async () => {
+ const [passwordBox] = await driver.findElements(By.css('.create-password #create-password'))
+ const [passwordBoxConfirm] = await driver.findElements(By.css('.create-password #confirm-password'))
+ const [button] = await driver.findElements(By.css('.create-password button'))
+
+ await passwordBox.sendKeys('correct horse battery staple')
+ await passwordBoxConfirm.sendKeys('correct horse battery staple')
+ await button.click()
+ await delay(regularDelayMs)
+ })
+
+ it('clicks through the unique image screen', async () => {
+ const [nextScreen] = await driver.findElements(By.css('.unique-image button'))
+ await nextScreen.click()
+ await delay(regularDelayMs)
+ })
+
+ it('clicks through the privacy notice', async () => {
+ const [nextScreen] = await driver.findElements(By.css('.tou button'))
+ await nextScreen.click()
+ await delay(regularDelayMs)
+
+ const canClickThrough = await driver.findElement(By.css('.tou button')).isEnabled()
+ assert.equal(canClickThrough, false, 'disabled continue button')
+ const [bottomOfTos] = await driver.findElements(By.linkText('Attributions'))
+ await driver.executeScript('arguments[0].scrollIntoView(true)', bottomOfTos)
+ await delay(regularDelayMs)
+
+ const [acceptTos] = await driver.findElements(By.css('.tou button'))
+ await acceptTos.click()
+ await delay(regularDelayMs)
+ })
+
+ let seedPhrase
+
+ it('reveals the seed phrase', async () => {
+ const [revealSeedPhrase] = await driver.findElements(By.css('.backup-phrase__secret-blocker'))
+ await revealSeedPhrase.click()
+ await delay(regularDelayMs)
+
+ seedPhrase = await driver.findElement(By.css('.backup-phrase__secret-words')).getText()
+ assert.equal(seedPhrase.split(' ').length, 12)
+ await delay(regularDelayMs)
+
+ const [nextScreen] = await driver.findElements(By.css('.backup-phrase button'))
+ await nextScreen.click()
+ await delay(regularDelayMs)
+ })
+
+ it('can retype the seed phrase', async () => {
+ const words = seedPhrase.split(' ')
+
+ const [word0] = await driver.findElements(By.xpath(`//button[contains(text(), '${words[0]}')]`))
+ await word0.click()
+ await delay(tinyDelayMs)
+
+ const [word1] = await driver.findElements(By.xpath(`//button[contains(text(), '${words[1]}')]`))
+ await word1.click()
+ await delay(tinyDelayMs)
+
+ const [word2] = await driver.findElements(By.xpath(`//button[contains(text(), '${words[2]}')]`))
+ await word2.click()
+ await delay(tinyDelayMs)
+
+ const [word3] = await driver.findElements(By.xpath(`//button[contains(text(), '${words[3]}')]`))
+ await word3.click()
+ await delay(tinyDelayMs)
+
+ const [word4] = await driver.findElements(By.xpath(`//button[contains(text(), '${words[4]}')]`))
+ await word4.click()
+ await delay(tinyDelayMs)
+
+ const [word5] = await driver.findElements(By.xpath(`//button[contains(text(), '${words[5]}')]`))
+ await word5.click()
+ await delay(tinyDelayMs)
+
+ const [word6] = await driver.findElements(By.xpath(`//button[contains(text(), '${words[6]}')]`))
+ await word6.click()
+ await delay(tinyDelayMs)
+
+ const [word7] = await driver.findElements(By.xpath(`//button[contains(text(), '${words[7]}')]`))
+ await word7.click()
+ await delay(tinyDelayMs)
+
+ const [word8] = await driver.findElements(By.xpath(`//button[contains(text(), '${words[8]}')]`))
+ await word8.click()
+ await delay(tinyDelayMs)
+
+ const [word9] = await driver.findElements(By.xpath(`//button[contains(text(), '${words[9]}')]`))
+ await word9.click()
+ await delay(tinyDelayMs)
+
+ const [word10] = await driver.findElements(By.xpath(`//button[contains(text(), '${words[10]}')]`))
+ await word10.click()
+ await delay(tinyDelayMs)
+
+ const [word11] = await driver.findElements(By.xpath(`//button[contains(text(), '${words[11]}')]`))
+ await word11.click()
+ await delay(tinyDelayMs)
+
+ const [confirm] = await driver.findElements(By.xpath(`//button[contains(text(), 'Confirm')]`))
+ await confirm.click()
+ await delay(regularDelayMs)
+ })
+
+ it('clicks through the deposit modal', async () => {
+ const [closeModal] = await driver.findElements(By.css('.page-container__header-close'))
+ await closeModal.click()
+ await delay(regularDelayMs)
+ })
+ })
+
+ describe('Show account information', () => {
+ it('shows the QR code for the account', async () => {
+ await driver.findElement(By.css('.wallet-view__details-button')).click()
+ await driver.findElement(By.css('.qr-wrapper')).isDisplayed()
+ await delay(regularDelayMs)
+
+ await driver.executeScript("document.querySelector('.account-modal-close').click()")
+ await delay(regularDelayMs * 4)
+ })
+ })
+
+ describe('Log out an log back in', () => {
+ it('logs out of the account', async () => {
+ await driver.findElement(By.css('.account-menu__icon')).click()
+ await delay(regularDelayMs)
+
+ const [logoutButton] = await driver.findElements(By.css('.account-menu__logout-button'))
+ assert.equal(await logoutButton.getText(), 'Log out')
+ await logoutButton.click()
+ await delay(regularDelayMs)
+ })
+
+ it('accepts the account password after lock', async () => {
+ await driver.findElement(By.id('password')).sendKeys('correct horse battery staple')
+ await driver.findElement(By.id('password')).sendKeys(Key.ENTER)
+ await delay(regularDelayMs * 4)
+ })
+ })
+
+ describe('Add account', () => {
+ it('choose Create Account from the account menu', async () => {
+ await driver.findElement(By.css('.account-menu__icon')).click()
+ await delay(regularDelayMs)
+
+ const [createAccount] = await driver.findElements(By.xpath(`//div[contains(text(), 'Create Account')]`))
+ await createAccount.click()
+ await delay(regularDelayMs)
+ })
+
+ it('set account name', async () => {
+ const [accountName] = await driver.findElements(By.css('.new-account-create-form input'))
+ await accountName.sendKeys('2nd account')
+ await delay(regularDelayMs)
+
+ const [create] = await driver.findElements(By.xpath(`//button[contains(text(), 'Create')]`))
+ await create.click()
+ await delay(regularDelayMs)
+ })
+
+ it('should correct account name', async () => {
+ const [accountName] = await driver.findElements(By.css('.account-name'))
+ assert.equal(await accountName.getText(), '2nd account')
+ await delay(regularDelayMs)
+ })
+ })
+
+ describe('Import seed phrase', () => {
+ it('logs out of the vault', async () => {
+ await driver.findElement(By.css('.account-menu__icon')).click()
+ await delay(regularDelayMs)
+
+ const [logoutButton] = await driver.findElements(By.css('.account-menu__logout-button'))
+ assert.equal(await logoutButton.getText(), 'Log out')
+ await logoutButton.click()
+ await delay(regularDelayMs)
+ })
+
+ it('imports seed phrase', async () => {
+ const [restoreSeedLink] = await driver.findElements(By.css('.unlock-page__link--import'))
+ assert.equal(await restoreSeedLink.getText(), 'Import using account seed phrase')
+ await restoreSeedLink.click()
+ await delay(regularDelayMs)
+
+ const [seedTextArea] = await driver.findElements(By.css('textarea'))
+ await seedTextArea.sendKeys(testSeedPhrase)
+ await delay(regularDelayMs)
+
+ await driver.findElement(By.id('password-box')).sendKeys('correct horse battery staple')
+ await driver.findElement(By.id('password-box-confirm')).sendKeys('correct horse battery staple')
+ await driver.findElement(By.css('button:nth-child(2)')).click()
+ await delay(regularDelayMs)
+ })
+
+ it('balance renders', async () => {
+ const balance = await driver.findElement(By.css('.balance-display .token-amount'))
+ const tokenAmount = await balance.getText()
+ assert.equal(tokenAmount, '100.000 ETH')
+ await delay(regularDelayMs)
+ })
+ })
+
+ describe('Send ETH from inside MetaMask', () => {
+ it('starts to send a transaction', async function () {
+ const [sendButton] = await driver.findElements(By.xpath(`//button[contains(text(), 'Send')]`))
+ await sendButton.click()
+ await delay(regularDelayMs)
+
+ const [inputAddress] = await driver.findElements(By.css('input[placeholder="Recipient Address"]'))
+ const [inputAmount] = await driver.findElements(By.css('.currency-display__input'))
+ await inputAddress.sendKeys('0x2f318C334780961FB129D2a6c30D0763d9a5C970')
+ await inputAmount.sendKeys('1')
+
+ // Set the gas limit
+ const [configureGas] = await driver.findElements(By.css('.send-v2__gas-fee-display button'))
+ await configureGas.click()
+ await delay(regularDelayMs)
+
+ const [save] = await driver.findElements(By.xpath(`//button[contains(text(), 'Save')]`))
+ await save.click()
+ await delay(regularDelayMs)
+
+ // Continue to next screen
+ const [nextScreen] = await driver.findElements(By.xpath(`//button[contains(text(), 'Next')]`))
+ await nextScreen.click()
+ await delay(regularDelayMs)
+ })
+
+ it('confirms the transaction', async function () {
+ const [confirmButton] = await driver.findElements(By.xpath(`//button[contains(text(), 'Confirm')]`))
+ await confirmButton.click()
+ await delay(regularDelayMs)
+ })
+
+ it('finds the transaction in the transactions list', async function () {
+ const transactions = await driver.findElements(By.css('.tx-list-item'))
+ assert.equal(transactions.length, 1)
+
+ const txValues = await driver.findElements(By.css('.tx-list-value'))
+ assert.equal(txValues.length, 1)
+ assert.equal(await txValues[0].getText(), '1 ETH')
+ })
+ })
+
+ describe('Send ETH from Faucet', () => {
+ it('starts a send transaction inside Faucet', async () => {
+ await driver.executeScript('window.open("https://faucet.metamask.io")')
+ await delay(waitingNewPageDelayMs)
+
+ const [extension, faucet] = await driver.getAllWindowHandles()
+ await driver.switchTo().window(faucet)
+ await delay(regularDelayMs)
+
+ const [send1eth] = await driver.findElements(By.xpath(`//button[contains(text(), '10 ether')]`))
+ await send1eth.click()
+ await delay(regularDelayMs)
+
+ await driver.switchTo().window(extension)
+ await loadExtension(driver, extensionId)
+ await delay(regularDelayMs)
+
+ const [confirmButton] = await driver.findElements(By.xpath(`//button[contains(text(), 'Confirm')]`))
+ await confirmButton.click()
+ await delay(regularDelayMs)
+
+ await driver.switchTo().window(faucet)
+ await delay(regularDelayMs)
+ await driver.close()
+ await delay(regularDelayMs)
+ await driver.switchTo().window(extension)
+ await delay(regularDelayMs)
+ await loadExtension(driver, extensionId)
+ await delay(regularDelayMs)
+ })
+ })
+
+ describe('Add existing token using search', () => {
+ it('clicks on the Add Token button', async () => {
+ const [addToken] = await driver.findElements(By.xpath(`//button[contains(text(), 'Add Token')]`))
+ await addToken.click()
+ await delay(regularDelayMs)
+ })
+
+ it('can pick a token from the existing options', async () => {
+ const [tokenSearch] = await driver.findElements(By.css('input.add-token__input'))
+ await tokenSearch.sendKeys('BAT')
+ await delay(regularDelayMs)
+
+ const [token] = await driver.findElements(By.xpath("//div[contains(text(), 'BAT')]"))
+ await token.click()
+ await delay(regularDelayMs)
+
+ const [nextScreen] = await driver.findElements(By.xpath(`//button[contains(text(), 'Next')]`))
+ await nextScreen.click()
+ await delay(regularDelayMs)
+
+ const [addTokens] = await driver.findElements(By.xpath(`//button[contains(text(), 'Add Tokens')]`))
+ await addTokens.click()
+ await delay(largeDelayMs)
+ })
+
+ it('renders the balance for the chosen token', async () => {
+ const balance = await driver.findElement(By.css('.tx-view .balance-display .token-amount'))
+ const tokenAmount = await balance.getText()
+ assert.equal(tokenAmount, '0BAT')
+ await delay(regularDelayMs)
+ })
+ })
+
+ describe('Add a custom token from TokenFactory', () => {
+ it('creates a new token', async () => {
+ await driver.executeScript('window.open("https://tokenfactory.surge.sh/#/factory")')
+ await delay(waitingNewPageDelayMs)
+
+ const [extension, tokenFactory] = await driver.getAllWindowHandles()
+ await driver.switchTo().window(tokenFactory)
+ const [
+ totalSupply,
+ tokenName,
+ tokenDecimal,
+ tokenSymbol,
+ ] = await driver.findElements(By.css('input'))
+
+ await totalSupply.sendKeys('100')
+ await tokenName.sendKeys('Test')
+ await tokenDecimal.sendKeys('0')
+ await tokenSymbol.sendKeys('TST')
+
+ const [createToken] = await driver.findElements(By.xpath(`//button[contains(text(), 'Create Token')]`))
+ await createToken.click()
+ await delay(regularDelayMs)
+
+ await driver.switchTo().window(extension)
+ await loadExtension(driver, extensionId)
+ await delay(regularDelayMs)
+
+ const [confirmButton] = await driver.findElements(By.xpath(`//button[contains(text(), 'Confirm')]`))
+ await confirmButton.click()
+ await delay(regularDelayMs)
+
+ await driver.switchTo().window(tokenFactory)
+ await delay(regularDelayMs)
+ const tokenContactAddress = await driver.findElement(By.css('div > div > div:nth-child(2) > span:nth-child(3)'))
+ tokenAddress = await tokenContactAddress.getText()
+ await driver.close()
+ await driver.switchTo().window(extension)
+ await loadExtension(driver, extensionId)
+ await delay(regularDelayMs)
+ })
+
+ it('clicks on the Add Token button', async () => {
+ const [addToken] = await driver.findElements(By.xpath(`//button[contains(text(), 'Add Token')]`))
+ await addToken.click()
+ await delay(regularDelayMs)
+ })
+
+ it('picks the newly created Test token', async () => {
+ const [addCustomToken] = await driver.findElements(By.xpath("//div[contains(text(), 'Custom Token')]"))
+ await addCustomToken.click()
+ await delay(regularDelayMs)
+
+ const [newTokenAddress] = await driver.findElements(By.css('.add-token__add-custom-form input'))
+ await newTokenAddress.sendKeys(tokenAddress)
+ await delay(regularDelayMs)
+
+ const [nextScreen] = await driver.findElements(By.xpath(`//button[contains(text(), 'Next')]`))
+ await nextScreen.click()
+ await delay(regularDelayMs)
+
+ const [addTokens] = await driver.findElements(By.xpath(`//button[contains(text(), 'Add Tokens')]`))
+ await addTokens.click()
+ await delay(regularDelayMs)
+ })
+
+ it('renders the balance for the new token', async () => {
+ const [balance] = await driver.findElements(By.css('.tx-view .balance-display .token-amount'))
+ const tokenAmount = await balance.getText()
+ assert.equal(tokenAmount, '100TST')
+ await delay(regularDelayMs)
+ })
+ })
+})
diff --git a/test/e2e/beta/run-all.sh b/test/e2e/beta/run-all.sh
new file mode 100755
index 000000000..5916d5614
--- /dev/null
+++ b/test/e2e/beta/run-all.sh
@@ -0,0 +1,10 @@
+#!/usr/bin/env bash
+
+set -e
+set -u
+set -o pipefail
+
+export PATH="$PATH:./node_modules/.bin"
+
+shell-parallel -s 'npm run ganache:start' -x 'sleep 5 && mocha test/e2e/beta/metamask-beta-ui.spec'
+shell-parallel -s 'npm run ganache:start' -x 'sleep 5 && mocha test/e2e/beta/from-import-beta-ui.spec'
diff --git a/test/e2e/func.js b/test/e2e/func.js
index 733225565..9f06e7f37 100644
--- a/test/e2e/func.js
+++ b/test/e2e/func.js
@@ -1,18 +1,63 @@
require('chromedriver')
+require('geckodriver')
+const fs = require('fs')
+const os = require('os')
+const path = require('path')
const webdriver = require('selenium-webdriver')
+const Command = require('selenium-webdriver/lib/command').Command
+const By = webdriver.By
-exports.delay = function delay (time) {
- return new Promise(resolve => setTimeout(resolve, time))
+module.exports = {
+ delay,
+ buildChromeWebDriver,
+ buildFirefoxWebdriver,
+ installWebExt,
+ getExtensionIdChrome,
+ getExtensionIdFirefox,
}
+function delay (time) {
+ return new Promise(resolve => setTimeout(resolve, time))
+}
-exports.buildWebDriver = function buildWebDriver (extPath) {
+function buildChromeWebDriver (extPath) {
+ const tmpProfile = path.join(os.tmpdir(), fs.mkdtempSync('mm-chrome-profile'));
return new webdriver.Builder()
.withCapabilities({
chromeOptions: {
- args: [`load-extension=${extPath}`],
+ args: [
+ `load-extension=${extPath}`,
+ `user-data-dir=${tmpProfile}`,
+ ],
+ binary: process.env.SELENIUM_CHROME_BINARY,
},
})
- .forBrowser('chrome')
.build()
}
+
+function buildFirefoxWebdriver () {
+ return new webdriver.Builder().build()
+}
+
+async function getExtensionIdChrome (driver) {
+ await driver.get('chrome://extensions')
+ const extensionId = await driver.executeScript('return document.querySelector("extensions-manager").shadowRoot.querySelector("extensions-view-manager extensions-item-list").shadowRoot.querySelector("extensions-item:nth-child(2)").getAttribute("id")')
+ return extensionId
+}
+
+async function getExtensionIdFirefox (driver) {
+ await driver.get('about:debugging#addons')
+ const extensionId = await driver.findElement(By.css('dd.addon-target-info-content:nth-child(6) > span:nth-child(1)')).getText()
+ return extensionId
+}
+
+async function installWebExt (driver, extension) {
+ const cmd = await new Command('moz-install-web-ext')
+ .setParameter('path', path.resolve(extension))
+ .setParameter('temporary', true)
+
+ await driver.getExecutor()
+ .defineCommand(cmd.getName(), 'POST', '/session/:sessionId/moz/addon/install')
+
+ return await driver.schedule(cmd, 'installWebExt(' + extension + ')')
+}
diff --git a/test/e2e/metamask.spec.js b/test/e2e/metamask.spec.js
index e0ff2a57e..a08a34d96 100644
--- a/test/e2e/metamask.spec.js
+++ b/test/e2e/metamask.spec.js
@@ -4,26 +4,44 @@ const path = require('path')
const assert = require('assert')
const pify = require('pify')
const webdriver = require('selenium-webdriver')
-const By = webdriver.By
-const { delay, buildWebDriver } = require('./func')
+const { By, Key } = webdriver
+const { delay, buildChromeWebDriver, buildFirefoxWebdriver, installWebExt, getExtensionIdChrome, getExtensionIdFirefox } = require('./func')
describe('Metamask popup page', function () {
- let driver
- this.seedPhase
- this.accountAddress
+ let driver, accountAddress, tokenAddress, extensionId
+
this.timeout(0)
before(async function () {
- const extPath = path.resolve('dist/chrome')
- driver = buildWebDriver(extPath)
- await driver.get('chrome://extensions-frame')
- const elems = await driver.findElements(By.css('.extension-list-item-wrapper'))
- const extensionId = await elems[1].getAttribute('id')
- await driver.get(`chrome-extension://${extensionId}/popup.html`)
- await delay(500)
+ if (process.env.SELENIUM_BROWSER === 'chrome') {
+ const extPath = path.resolve('dist/chrome')
+ driver = buildChromeWebDriver(extPath)
+ extensionId = await getExtensionIdChrome(driver)
+ await driver.get(`chrome-extension://${extensionId}/popup.html`)
+
+ } else if (process.env.SELENIUM_BROWSER === 'firefox') {
+ const extPath = path.resolve('dist/firefox')
+ driver = buildFirefoxWebdriver()
+ await installWebExt(driver, extPath)
+ await delay(700)
+ extensionId = await getExtensionIdFirefox(driver)
+ await driver.get(`moz-extension://${extensionId}/popup.html`)
+ }
})
afterEach(async function () {
+ // logs command not supported in firefox
+ // https://github.com/SeleniumHQ/selenium/issues/2910
+ if (process.env.SELENIUM_BROWSER === 'chrome') {
+ // check for console errors
+ const errors = await checkBrowserForConsoleErrors()
+ if (errors.length) {
+ const errorReports = errors.map(err => err.message)
+ const errorMessage = `Errors found in browser console:\n${errorReports.join('\n')}`
+ this.test.error(new Error(errorMessage))
+ }
+ }
+ // gather extra data if test failed
if (this.currentTest.state === 'failed') {
await verboseReportOnFailure(this.currentTest)
}
@@ -33,105 +51,301 @@ describe('Metamask popup page', function () {
await driver.quit()
})
- describe('#onboarding', () => {
- it('should open Metamask.io', async function () {
- const tabs = await driver.getAllWindowHandles()
- await driver.switchTo().window(tabs[0])
+ describe('Setup', function () {
+
+ it('switches to Chrome extensions list', async function () {
await delay(300)
- await setProviderType('localhost')
+ const windowHandles = await driver.getAllWindowHandles()
+ await driver.switchTo().window(windowHandles[0])
+ })
+
+ it('sets provider type to localhost', async function () {
await delay(300)
+ await setProviderType('localhost')
})
- it('should match title', async () => {
+ })
+
+ describe('Account Creation', () => {
+
+ it('matches MetaMask title', async () => {
const title = await driver.getTitle()
assert.equal(title, 'MetaMask', 'title matches MetaMask')
})
- it('should show privacy notice', async () => {
+ it('shows privacy notice', async () => {
+ await delay(300)
const privacy = await driver.findElement(By.css('.terms-header')).getText()
assert.equal(privacy, 'PRIVACY NOTICE', 'shows privacy notice')
- driver.findElement(By.css('button')).click()
+ await driver.findElement(By.css('button')).click()
await delay(300)
})
- it('should show terms of use', async () => {
- await delay(300)
+ it('show terms of use', async () => {
const terms = await driver.findElement(By.css('.terms-header')).getText()
assert.equal(terms, 'TERMS OF USE', 'shows terms of use')
- await delay(300)
+ delay(300)
})
- it('should be unable to continue without scolling throught the terms of use', async () => {
+ it('checks if the TOU button is disabled', async () => {
const button = await driver.findElement(By.css('button')).isEnabled()
assert.equal(button, false, 'disabled continue button')
- const element = driver.findElement(By.linkText(
- 'Attributions'
- ))
+ const element = await driver.findElement(By.linkText('Attributions'))
await driver.executeScript('arguments[0].scrollIntoView(true)', element)
- await delay(300)
+ await delay(700)
})
- it('should be able to continue when scrolled to the bottom of terms of use', async () => {
- const button = await driver.findElement(By.css('button'))
- const buttonEnabled = await button.isEnabled()
- await delay(500)
- assert.equal(buttonEnabled, true, 'enabled continue button')
+ it('allows the button to be clicked when scrolled to the bottom of TOU', async () => {
+ const button = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > div.flex-column.flex-center.flex-grow > button'))
await button.click()
- await delay(300)
})
- it('should accept password with length of eight', async () => {
+ it('accepts password with length of eight', async () => {
const passwordBox = await driver.findElement(By.id('password-box'))
const passwordBoxConfirm = await driver.findElement(By.id('password-box-confirm'))
- const button = driver.findElement(By.css('button'))
+ const button = await driver.findElements(By.css('button'))
- passwordBox.sendKeys('123456789')
- passwordBoxConfirm.sendKeys('123456789')
+ await passwordBox.sendKeys('123456789')
+ await passwordBoxConfirm.sendKeys('123456789')
+ await button[0].click()
await delay(500)
- await button.click()
})
- it('should show value was created and seed phrase', async () => {
- await delay(700)
- this.seedPhase = await driver.findElement(By.css('.twelve-word-phrase')).getText()
- const continueAfterSeedPhrase = await driver.findElement(By.css('button'))
+ it('shows value was created and seed phrase', async () => {
+ await delay(300)
+ const seedPhrase = await driver.findElement(By.css('.twelve-word-phrase')).getText()
+ assert.equal(seedPhrase.split(' ').length, 12)
+ const continueAfterSeedPhrase = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > button:nth-child(4)'))
+ assert.equal(await continueAfterSeedPhrase.getText(), `I'VE COPIED IT SOMEWHERE SAFE`)
await continueAfterSeedPhrase.click()
await delay(300)
})
- it('should show lock account', async () => {
+ it('adds a second account', async function () {
+ await driver.findElement(By.css('#app-content > div > div.full-width > div > div:nth-child(2) > span > div')).click()
+ await delay(300)
+ await driver.findElement(By.css('#app-content > div > div.full-width > div > div:nth-child(2) > span > div > div > span > div > li:nth-child(3) > span')).click()
+ })
+
+ it('shows account address', async function () {
+ await delay(300)
+ accountAddress = await driver.findElement(By.css('#app-content > div > div.app-primary.from-left > div > div > div:nth-child(1) > flex-column > div.flex-row > div')).getText()
+ })
+
+ it('logs out of the vault', async () => {
await driver.findElement(By.css('.sandwich-expando')).click()
await delay(500)
- await driver.findElement(By.css('#app-content > div > div:nth-child(3) > span > div > li:nth-child(3)')).click()
+ const logoutButton = await driver.findElement(By.css('#app-content > div > div:nth-child(3) > span > div > li:nth-child(3)'))
+ assert.equal(await logoutButton.getText(), 'Log Out')
+ await logoutButton.click()
})
- it('should accept account password after lock', async () => {
+ it('accepts account password after lock', async () => {
await delay(500)
await driver.findElement(By.id('password-box')).sendKeys('123456789')
- await driver.findElement(By.css('button')).click()
+ await driver.findElement(By.id('password-box')).sendKeys(Key.ENTER)
await delay(500)
})
- it('should show QR code option', async () => {
+ it('shows QR code option', async () => {
await delay(300)
await driver.findElement(By.css('.fa-ellipsis-h')).click()
await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > div > div:nth-child(1) > flex-column > div.name-label > div > span > i > div > div > li:nth-child(3)')).click()
await delay(300)
})
- it('should show the account address', async () => {
- this.accountAddress = await driver.findElement(By.css('.ellip-address')).getText()
+ it('checks QR code address is the same as account details address', async () => {
+ const QRaccountAddress = await driver.findElement(By.css('.ellip-address')).getText()
+ assert.equal(accountAddress.toLowerCase(), QRaccountAddress)
await driver.findElement(By.css('.fa-arrow-left')).click()
await delay(500)
})
})
- async function setProviderType(type) {
+ describe('Import Ganache seed phrase', function () {
+
+ it('logs out', async function () {
+ await driver.findElement(By.css('.sandwich-expando')).click()
+ await delay(200)
+ const logOut = await driver.findElement(By.css('#app-content > div > div:nth-child(3) > span > div > li:nth-child(3)'))
+ assert.equal(await logOut.getText(), 'Log Out')
+ await logOut.click()
+ await delay(300)
+ })
+
+ it('restores from seed phrase', async function () {
+ const restoreSeedLink = await driver.findElement(By.css('#app-content > div > div.app-primary.from-left > div > div.flex-row.flex-center.flex-grow > p'))
+ assert.equal(await restoreSeedLink.getText(), 'Restore from seed phrase')
+ await restoreSeedLink.click()
+ await delay(100)
+ })
+
+ it('adds seed phrase', async function () {
+ const testSeedPhrase = 'phrase upgrade clock rough situate wedding elder clever doctor stamp excess tent'
+ const seedTextArea = await driver.findElement(By.css('#app-content > div > div.app-primary.from-left > div > textarea'))
+ await seedTextArea.sendKeys(testSeedPhrase)
+
+ await driver.findElement(By.id('password-box')).sendKeys('123456789')
+ await driver.findElement(By.id('password-box-confirm')).sendKeys('123456789')
+ await driver.findElement(By.css('#app-content > div > div.app-primary.from-left > div > div > button:nth-child(2)')).click()
+ await delay(500)
+ })
+
+ it('balance renders', async function () {
+ await delay(200)
+ const balance = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > div > div.flex-row > div.ether-balance.ether-balance-amount > div > div > div:nth-child(1) > div:nth-child(1)'))
+ assert.equal(await balance.getText(), '100.000')
+ await delay(200)
+ })
+
+ it('sends transaction', async function () {
+ const sendButton = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > div > div.flex-row > button:nth-child(4)'))
+ assert.equal(await sendButton.getText(), 'SEND')
+ await sendButton.click()
+ await delay(200)
+ })
+
+ it('adds recipient address and amount', async function () {
+ const sendTranscationScreen = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > h3:nth-child(2)')).getText()
+ assert.equal(sendTranscationScreen, 'SEND TRANSACTION')
+ const inputAddress = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > section:nth-child(3) > div > input'))
+ const inputAmmount = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > section:nth-child(4) > input'))
+ await inputAddress.sendKeys('0x2f318C334780961FB129D2a6c30D0763d9a5C970')
+ await inputAmmount.sendKeys('10')
+ await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > section:nth-child(4) > button')).click()
+ await delay(300)
+ })
+
+ it('confirms transaction', async function () {
+ await delay(300)
+ await driver.findElement(By.css('#pending-tx-form > div.flex-row.flex-space-around.conf-buttons > input')).click()
+ await delay(500)
+ })
+
+ it('finds the transaction in the transactions list', async function () {
+ const tranasactionAmount = await driver.findElement(By.css('#app-content > div > div.app-primary.from-left > div > section > section > div > div > div > div.ether-balance.ether-balance-amount > div > div > div > div:nth-child(1)'))
+ assert.equal(await tranasactionAmount.getText(), '10.0')
+ })
+ })
+
+ describe('Token Factory', function () {
+
+ it('navigates to token factory', async function () {
+ await driver.get('http://tokenfactory.surge.sh/')
+ })
+
+ it('navigates to create token contract link', async function () {
+ const createToken = await driver.findElement(By.css('#bs-example-navbar-collapse-1 > ul > li:nth-child(3) > a'))
+ await createToken.click()
+ })
+
+ it('adds input for token', async function () {
+ const totalSupply = await driver.findElement(By.css('#main > div > div > div > div:nth-child(2) > div > div:nth-child(5) > input'))
+ const tokenName = await driver.findElement(By.css('#main > div > div > div > div:nth-child(2) > div > div:nth-child(6) > input'))
+ const tokenDecimal = await driver.findElement(By.css('#main > div > div > div > div:nth-child(2) > div > div:nth-child(7) > input'))
+ const tokenSymbol = await driver.findElement(By.css('#main > div > div > div > div:nth-child(2) > div > div:nth-child(8) > input'))
+ const createToken = await driver.findElement(By.css('#main > div > div > div > div:nth-child(2) > div > button'))
+
+ await totalSupply.sendKeys('100')
+ await tokenName.sendKeys('Test')
+ await tokenDecimal.sendKeys('0')
+ await tokenSymbol.sendKeys('TST')
+ await createToken.click()
+ await delay(1000)
+ })
+
+ // There is an issue with blank confirmation window in Firefox, but the button is still there and the driver is able to clicked (?.?)
+ it('confirms transaction in MetaMask popup', async function () {
+ const windowHandles = await driver.getAllWindowHandles()
+ await driver.switchTo().window(windowHandles[windowHandles.length - 1])
+ const metamaskSubmit = await driver.findElement(By.css('#pending-tx-form > div.flex-row.flex-space-around.conf-buttons > input'))
+ await metamaskSubmit.click()
+ await delay(1000)
+ })
+
+ it('switches back to Token Factory to grab the token contract address', async function () {
+ const windowHandles = await driver.getAllWindowHandles()
+ await driver.switchTo().window(windowHandles[0])
+ const tokenContactAddress = await driver.findElement(By.css('#main > div > div > div > div:nth-child(2) > span:nth-child(3)'))
+ tokenAddress = await tokenContactAddress.getText()
+ await delay(500)
+ })
+
+ it('navigates back to MetaMask popup in the tab', async function () {
+ if (process.env.SELENIUM_BROWSER === 'chrome') {
+ await driver.get(`chrome-extension://${extensionId}/popup.html`)
+ } else if (process.env.SELENIUM_BROWSER === 'firefox') {
+ await driver.get(`moz-extension://${extensionId}/popup.html`)
+ }
+ await delay(700)
+ })
+ })
+
+ describe('Add Token', function () {
+
+ it('switches to the add token screen', async function () {
+ const tokensTab = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > section > div > div.inactiveForm.pointer'))
+ assert.equal(await tokensTab.getText(), 'TOKENS')
+ await tokensTab.click()
+ await delay(300)
+ })
+
+ it('navigates to the add token screen', async function () {
+ const addTokenButton = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > section > div.full-flex-height > div > button'))
+ assert.equal(await addTokenButton.getText(), 'ADD TOKEN')
+ await addTokenButton.click()
+ })
+
+ it('checks add token screen rendered', async function () {
+ const addTokenScreen = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > div.section-title.flex-row.flex-center > h2'))
+ assert.equal(await addTokenScreen.getText(), 'ADD TOKEN')
+ })
+
+ it('adds token parameters', async function () {
+ const tokenContractAddress = await driver.findElement(By.css('#token-address'))
+ await tokenContractAddress.sendKeys(tokenAddress)
+ await delay(300)
+ await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > div.flex-column.flex-justify-center.flex-grow.select-none > div > button')).click()
+ await delay(200)
+ })
+
+ it('checks the token balance', async function () {
+ const tokenBalance = await driver.findElement(By.css('#app-content > div > div.app-primary.from-left > div > section > div.full-flex-height > ol > li:nth-child(2) > h3'))
+ assert.equal(await tokenBalance.getText(), '100 TST')
+ })
+ })
+
+ async function setProviderType (type) {
await driver.executeScript('window.metamask.setProviderType(arguments[0])', type)
}
- async function verboseReportOnFailure(test) {
- const artifactDir = `./test-artifacts/${test.title}`
+ async function checkBrowserForConsoleErrors() {
+ const ignoredLogTypes = ['WARNING']
+ const ignoredErrorMessages = [
+ // React throws error warnings on "dataset", but still sets the data-* properties correctly
+ 'Warning: Unknown prop `dataset` on ',
+ // Third-party Favicon 404s show up as errors
+ 'favicon.ico - Failed to load resource: the server responded with a status of 404 (Not Found)',
+ // React Development build - known issue blocked by test build sys
+ 'Warning: It looks like you\'re using a minified copy of the development build of React.',
+ // Redux Development build - known issue blocked by test build sys
+ 'This means that you are running a slower development build of Redux.',
+ ]
+ const browserLogs = await driver.manage().logs().get('browser')
+ const errorEntries = browserLogs.filter(entry => !ignoredLogTypes.includes(entry.level.toString()))
+ const errorObjects = errorEntries.map(entry => entry.toJSON())
+ // ignore all errors that contain a message in `ignoredErrorMessages`
+ const matchedErrorObjects = errorObjects.filter(entry => !ignoredErrorMessages.some(message => entry.message.includes(message)))
+ return matchedErrorObjects
+ }
+
+ async function verboseReportOnFailure (test) {
+ let artifactDir
+ if (process.env.SELENIUM_BROWSER === 'chrome') {
+ artifactDir = `./test-artifacts/chrome/${test.title}`
+ } else if (process.env.SELENIUM_BROWSER === 'firefox') {
+ artifactDir = `./test-artifacts/firefox/${test.title}`
+ }
const filepathBase = `${artifactDir}/test-failure`
await pify(mkdirp)(artifactDir)
// capture screenshot