mirror of
https://github.com/kremalicious/metamask-extension.git
synced 2024-11-22 18:00:18 +01:00
tests - integration - Add Drizzle tests (#5467)
* added drizzle app for testing * working * clean up * clean up script * make build step required * add drizzle-tests to .eslintignore * clean up drizzle run script * lint * use truffle unbox * undo eslintignore changes * revert change * dont use global * dont need this steps * use the new account flow * restore package-lock.json
This commit is contained in:
parent
1eec21b945
commit
ccab4ee1a4
@ -36,6 +36,10 @@ workflows:
|
|||||||
requires:
|
requires:
|
||||||
- prep-deps-npm
|
- prep-deps-npm
|
||||||
- prep-build
|
- prep-build
|
||||||
|
- test-e2e-beta-drizzle:
|
||||||
|
requires:
|
||||||
|
- prep-deps-npm
|
||||||
|
- prep-build
|
||||||
- test-unit:
|
- test-unit:
|
||||||
requires:
|
requires:
|
||||||
- prep-deps-npm
|
- prep-deps-npm
|
||||||
@ -68,6 +72,7 @@ workflows:
|
|||||||
- test-e2e-firefox
|
- test-e2e-firefox
|
||||||
- test-e2e-beta-chrome
|
- test-e2e-beta-chrome
|
||||||
- test-e2e-beta-firefox
|
- test-e2e-beta-firefox
|
||||||
|
- test-e2e-beta-drizzle
|
||||||
- test-integration-mascara-chrome
|
- test-integration-mascara-chrome
|
||||||
- test-integration-mascara-firefox
|
- test-integration-mascara-firefox
|
||||||
- test-integration-flat-chrome
|
- test-integration-flat-chrome
|
||||||
@ -222,6 +227,19 @@ jobs:
|
|||||||
path: test-artifacts
|
path: test-artifacts
|
||||||
destination: test-artifacts
|
destination: test-artifacts
|
||||||
|
|
||||||
|
test-e2e-beta-drizzle:
|
||||||
|
docker:
|
||||||
|
- image: circleci/node:8.11.3-browsers
|
||||||
|
steps:
|
||||||
|
- checkout
|
||||||
|
- attach_workspace:
|
||||||
|
at: .
|
||||||
|
- run:
|
||||||
|
name: test:e2e:drizzle:beta
|
||||||
|
command: npm run test:e2e:drizzle:beta
|
||||||
|
- store_artifacts:
|
||||||
|
path: test-artifacts
|
||||||
|
destination: test-artifacts
|
||||||
test-e2e-beta-chrome:
|
test-e2e-beta-chrome:
|
||||||
docker:
|
docker:
|
||||||
- image: circleci/node:8.11.3-browsers
|
- image: circleci/node:8.11.3-browsers
|
||||||
|
@ -20,3 +20,4 @@ test/integration/bundle.js
|
|||||||
test/integration/jquery-3.1.0.min.js
|
test/integration/jquery-3.1.0.min.js
|
||||||
test/integration/helpers.js
|
test/integration/helpers.js
|
||||||
test/integration/lib/first-time.js
|
test/integration/lib/first-time.js
|
||||||
|
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
window.onload = function() {
|
window.onload = function () {
|
||||||
if (window.location.pathname === '/phishing.html') {
|
if (window.location.pathname === '/phishing.html') {
|
||||||
const {hostname} = parseHash()
|
const {hostname} = parseHash()
|
||||||
document.getElementById('esdbLink').innerHTML = '<b>To read more about this scam, navigate to: <a href="https://etherscamdb.info/domain/' + hostname + '"> https://etherscamdb.info/domain/' + hostname + '</a></b>'
|
document.getElementById('esdbLink').innerHTML = '<b>To read more about this scam, navigate to: <a href="https://etherscamdb.info/domain/' + hostname + '"> https://etherscamdb.info/domain/' + hostname + '</a></b>'
|
||||||
|
@ -16,6 +16,7 @@
|
|||||||
"test:integration": "npm run test:integration:build && 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:integration:build": "gulp build:scss",
|
||||||
"test:e2e:chrome": "shell-parallel -s 'npm run ganache:start' -x 'sleep 3 && npm run test:e2e:run:chrome'",
|
"test:e2e:chrome": "shell-parallel -s 'npm run ganache:start' -x 'sleep 3 && npm run test:e2e:run:chrome'",
|
||||||
|
"test:e2e:drizzle:beta": "SELENIUM_BROWSER=chrome test/e2e/beta/run-drizzle.sh",
|
||||||
"test:e2e:chrome:beta": "SELENIUM_BROWSER=chrome test/e2e/beta/run-all.sh",
|
"test:e2e:chrome:beta": "SELENIUM_BROWSER=chrome test/e2e/beta/run-all.sh",
|
||||||
"test:e2e:firefox": "shell-parallel -s 'npm run ganache:start' -x 'sleep 3 && npm run test:e2e:run:firefox'",
|
"test:e2e:firefox": "shell-parallel -s 'npm run ganache:start' -x 'sleep 3 && npm run test:e2e:run:firefox'",
|
||||||
"test:e2e:firefox:beta": "SELENIUM_BROWSER=firefox test/e2e/beta/run-all.sh",
|
"test:e2e:firefox:beta": "SELENIUM_BROWSER=firefox test/e2e/beta/run-all.sh",
|
||||||
|
286
test/e2e/beta/drizzle.spec.js
Normal file
286
test/e2e/beta/drizzle.spec.js
Normal file
@ -0,0 +1,286 @@
|
|||||||
|
const path = require('path')
|
||||||
|
const assert = require('assert')
|
||||||
|
const webdriver = require('selenium-webdriver')
|
||||||
|
const { By, until } = webdriver
|
||||||
|
const {
|
||||||
|
delay,
|
||||||
|
buildChromeWebDriver,
|
||||||
|
buildFirefoxWebdriver,
|
||||||
|
installWebExt,
|
||||||
|
getExtensionIdChrome,
|
||||||
|
getExtensionIdFirefox,
|
||||||
|
} = require('../func')
|
||||||
|
const {
|
||||||
|
checkBrowserForConsoleErrors,
|
||||||
|
closeAllWindowHandlesExcept,
|
||||||
|
findElement,
|
||||||
|
findElements,
|
||||||
|
loadExtension,
|
||||||
|
openNewPage,
|
||||||
|
verboseReportOnFailure,
|
||||||
|
waitUntilXWindowHandles,
|
||||||
|
} = require('./helpers')
|
||||||
|
|
||||||
|
describe('MetaMask', function () {
|
||||||
|
let extensionId
|
||||||
|
let driver
|
||||||
|
|
||||||
|
const tinyDelayMs = 200
|
||||||
|
const regularDelayMs = tinyDelayMs * 2
|
||||||
|
const largeDelayMs = regularDelayMs * 2
|
||||||
|
|
||||||
|
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(driver, this.currentTest)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
after(async function () {
|
||||||
|
await driver.quit()
|
||||||
|
})
|
||||||
|
|
||||||
|
|
||||||
|
describe('New UI setup', async function () {
|
||||||
|
it('switches to first tab', async function () {
|
||||||
|
await delay(tinyDelayMs)
|
||||||
|
const [firstTab] = await driver.getAllWindowHandles()
|
||||||
|
await driver.switchTo().window(firstTab)
|
||||||
|
await delay(regularDelayMs)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('selects the new UI option', async () => {
|
||||||
|
try {
|
||||||
|
const overlay = await findElement(driver, By.css('.full-flex-height'))
|
||||||
|
await driver.wait(until.stalenessOf(overlay))
|
||||||
|
} catch (e) {}
|
||||||
|
|
||||||
|
let button
|
||||||
|
try {
|
||||||
|
button = await findElement(driver, By.xpath("//button[contains(text(), 'Try it now')]"))
|
||||||
|
} catch (e) {
|
||||||
|
await loadExtension(driver, extensionId)
|
||||||
|
await delay(largeDelayMs)
|
||||||
|
button = await findElement(driver, By.xpath("//button[contains(text(), 'Try it now')]"))
|
||||||
|
}
|
||||||
|
await button.click()
|
||||||
|
await delay(regularDelayMs)
|
||||||
|
|
||||||
|
// Close all other tabs
|
||||||
|
const [tab0, tab1, tab2] = await driver.getAllWindowHandles()
|
||||||
|
await driver.switchTo().window(tab0)
|
||||||
|
await delay(tinyDelayMs)
|
||||||
|
|
||||||
|
let selectedUrl = await driver.getCurrentUrl()
|
||||||
|
await delay(tinyDelayMs)
|
||||||
|
if (tab0 && selectedUrl.match(/popup.html/)) {
|
||||||
|
await closeAllWindowHandlesExcept(driver, tab0)
|
||||||
|
} else if (tab1) {
|
||||||
|
await driver.switchTo().window(tab1)
|
||||||
|
selectedUrl = await driver.getCurrentUrl()
|
||||||
|
await delay(tinyDelayMs)
|
||||||
|
if (selectedUrl.match(/popup.html/)) {
|
||||||
|
await closeAllWindowHandlesExcept(driver, tab1)
|
||||||
|
} else if (tab2) {
|
||||||
|
await driver.switchTo().window(tab2)
|
||||||
|
selectedUrl = await driver.getCurrentUrl()
|
||||||
|
selectedUrl.match(/popup.html/) && await closeAllWindowHandlesExcept(driver, tab2)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
throw new Error('popup.html not found')
|
||||||
|
}
|
||||||
|
await delay(regularDelayMs)
|
||||||
|
const [appTab] = await driver.getAllWindowHandles()
|
||||||
|
await driver.switchTo().window(appTab)
|
||||||
|
await delay(tinyDelayMs)
|
||||||
|
|
||||||
|
await loadExtension(driver, extensionId)
|
||||||
|
await delay(regularDelayMs)
|
||||||
|
|
||||||
|
const continueBtn = await findElement(driver, 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 findElement(driver, By.css('.create-password #create-password'))
|
||||||
|
const passwordBoxConfirm = await findElement(driver, By.css('.create-password #confirm-password'))
|
||||||
|
const button = await findElement(driver, 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 findElement(driver, By.css('.unique-image button'))
|
||||||
|
await nextScreen.click()
|
||||||
|
await delay(regularDelayMs)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('clicks through the ToS', async () => {
|
||||||
|
// terms of use
|
||||||
|
const canClickThrough = await driver.findElement(By.css('.tou button')).isEnabled()
|
||||||
|
assert.equal(canClickThrough, false, 'disabled continue button')
|
||||||
|
const bottomOfTos = await findElement(driver, By.linkText('Attributions'))
|
||||||
|
await driver.executeScript('arguments[0].scrollIntoView(true)', bottomOfTos)
|
||||||
|
await delay(regularDelayMs)
|
||||||
|
const acceptTos = await findElement(driver, By.css('.tou button'))
|
||||||
|
driver.wait(until.elementIsEnabled(acceptTos))
|
||||||
|
await acceptTos.click()
|
||||||
|
await delay(regularDelayMs)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('clicks through the privacy notice', async () => {
|
||||||
|
// privacy notice
|
||||||
|
const nextScreen = await findElement(driver, By.css('.tou button'))
|
||||||
|
await nextScreen.click()
|
||||||
|
await delay(regularDelayMs)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('clicks through the phishing notice', async () => {
|
||||||
|
// phishing notice
|
||||||
|
const noticeElement = await driver.findElement(By.css('.markdown'))
|
||||||
|
await driver.executeScript('arguments[0].scrollTop = arguments[0].scrollHeight', noticeElement)
|
||||||
|
await delay(regularDelayMs)
|
||||||
|
const nextScreen = await findElement(driver, By.css('.tou button'))
|
||||||
|
await nextScreen.click()
|
||||||
|
await delay(regularDelayMs)
|
||||||
|
})
|
||||||
|
|
||||||
|
let seedPhrase
|
||||||
|
|
||||||
|
it('reveals the seed phrase', async () => {
|
||||||
|
const byRevealButton = By.css('.backup-phrase__secret-blocker .backup-phrase__reveal-button')
|
||||||
|
await driver.wait(until.elementLocated(byRevealButton, 10000))
|
||||||
|
const revealSeedPhraseButton = await findElement(driver, byRevealButton, 10000)
|
||||||
|
await revealSeedPhraseButton.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 findElement(driver, By.css('.backup-phrase button'))
|
||||||
|
await nextScreen.click()
|
||||||
|
await delay(regularDelayMs)
|
||||||
|
})
|
||||||
|
|
||||||
|
async function clickWordAndWait (word) {
|
||||||
|
const xpathClass = 'backup-phrase__confirm-seed-option backup-phrase__confirm-seed-option--unselected'
|
||||||
|
const xpath = `//button[@class='${xpathClass}' and contains(text(), '${word}')]`
|
||||||
|
const word0 = await findElement(driver, By.xpath(xpath), 10000)
|
||||||
|
|
||||||
|
await word0.click()
|
||||||
|
await delay(tinyDelayMs)
|
||||||
|
}
|
||||||
|
|
||||||
|
async function retypeSeedPhrase (words, wasReloaded, count = 0) {
|
||||||
|
try {
|
||||||
|
if (wasReloaded) {
|
||||||
|
const byRevealButton = By.css('.backup-phrase__secret-blocker .backup-phrase__reveal-button')
|
||||||
|
await driver.wait(until.elementLocated(byRevealButton, 10000))
|
||||||
|
const revealSeedPhraseButton = await findElement(driver, byRevealButton, 10000)
|
||||||
|
await revealSeedPhraseButton.click()
|
||||||
|
await delay(regularDelayMs)
|
||||||
|
|
||||||
|
const nextScreen = await findElement(driver, By.css('.backup-phrase button'))
|
||||||
|
await nextScreen.click()
|
||||||
|
await delay(regularDelayMs)
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let i = 0; i < 12; i++) {
|
||||||
|
await clickWordAndWait(words[i])
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
if (count > 2) {
|
||||||
|
throw e
|
||||||
|
} else {
|
||||||
|
await loadExtension(driver, extensionId)
|
||||||
|
await retypeSeedPhrase(words, true, count + 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
it('can retype the seed phrase', async () => {
|
||||||
|
const words = seedPhrase.split(' ')
|
||||||
|
|
||||||
|
await retypeSeedPhrase(words)
|
||||||
|
|
||||||
|
const confirm = await findElement(driver, By.xpath(`//button[contains(text(), 'Confirm')]`))
|
||||||
|
await confirm.click()
|
||||||
|
await delay(regularDelayMs)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('clicks through the deposit modal', async () => {
|
||||||
|
const byBuyModal = By.css('span .modal')
|
||||||
|
const buyModal = await driver.wait(until.elementLocated(byBuyModal))
|
||||||
|
const closeModal = await findElement(driver, By.css('.page-container__header-close'))
|
||||||
|
await closeModal.click()
|
||||||
|
await driver.wait(until.stalenessOf(buyModal))
|
||||||
|
await delay(regularDelayMs)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('switches to localhost', async () => {
|
||||||
|
const networkDropdown = await findElement(driver, By.css('.network-name'))
|
||||||
|
await networkDropdown.click()
|
||||||
|
await delay(regularDelayMs)
|
||||||
|
|
||||||
|
const [localhost] = await findElements(driver, By.xpath(`//span[contains(text(), 'Localhost')]`))
|
||||||
|
await localhost.click()
|
||||||
|
await delay(largeDelayMs * 2)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('Drizzle', () => {
|
||||||
|
it('should be able to detect our eth address', async () => {
|
||||||
|
await openNewPage(driver, 'http://127.0.0.1:3000/')
|
||||||
|
await delay(regularDelayMs)
|
||||||
|
|
||||||
|
await waitUntilXWindowHandles(driver, 2)
|
||||||
|
const windowHandles = await driver.getAllWindowHandles()
|
||||||
|
const dapp = windowHandles[1]
|
||||||
|
|
||||||
|
await driver.switchTo().window(dapp)
|
||||||
|
await delay(regularDelayMs)
|
||||||
|
|
||||||
|
|
||||||
|
const addressElement = await findElement(driver, By.css(`.pure-u-1-1 h4`))
|
||||||
|
const addressText = await addressElement.getText()
|
||||||
|
assert(addressText.match(/^0x[a-fA-F0-9]{40}$/))
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
@ -6,5 +6,5 @@ set -o pipefail
|
|||||||
|
|
||||||
export PATH="$PATH:./node_modules/.bin"
|
export PATH="$PATH:./node_modules/.bin"
|
||||||
|
|
||||||
shell-parallel -s 'npm run ganache:start -- -b 2' -x 'sleep 5 && static-server test/e2e/beta/contract-test/ --port 8080' -x 'sleep 5 && mocha test/e2e/beta/metamask-beta-ui.spec'
|
shell-parallel -s 'npm run ganache:start -- -b 2' -x 'sleep 5 && static-server test/e2e/beta/contract-test --port 8080' -x 'sleep 5 && mocha test/e2e/beta/metamask-beta-ui.spec'
|
||||||
shell-parallel -s 'npm run ganache:start -- -d -b 2' -x 'sleep 5 && static-server test/e2e/beta/contract-test/ --port 8080' -x 'sleep 5 && mocha test/e2e/beta/from-import-beta-ui.spec'
|
shell-parallel -s 'npm run ganache:start -- -d -b 2' -x 'sleep 5 && mocha test/e2e/beta/from-import-beta-ui.spec'
|
||||||
|
20
test/e2e/beta/run-drizzle.sh
Executable file
20
test/e2e/beta/run-drizzle.sh
Executable file
@ -0,0 +1,20 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
|
set -e
|
||||||
|
set -u
|
||||||
|
set -o pipefail
|
||||||
|
|
||||||
|
export PATH="$PATH:./node_modules/.bin"
|
||||||
|
|
||||||
|
npm run ganache:start -- -b 2 >> /dev/null 2>&1 &
|
||||||
|
sleep 5
|
||||||
|
cd test/e2e/beta/
|
||||||
|
rm -rf drizzle-test
|
||||||
|
mkdir drizzle-test && cd drizzle-test
|
||||||
|
npm install truffle
|
||||||
|
truffle unbox drizzle
|
||||||
|
echo "Deploying contracts for Drizzle test..."
|
||||||
|
truffle compile && truffle migrate
|
||||||
|
BROWSER=none npm start >> /dev/null 2>&1 &
|
||||||
|
cd ../../../../
|
||||||
|
mocha test/e2e/beta/drizzle.spec
|
@ -49,7 +49,7 @@ class NewAccountCreateForm extends Component {
|
|||||||
h(Button, {
|
h(Button, {
|
||||||
type: 'primary',
|
type: 'primary',
|
||||||
large: true,
|
large: true,
|
||||||
className:'new-account-create-form__button',
|
className: 'new-account-create-form__button',
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
createAccount(newAccountName || defaultAccountName)
|
createAccount(newAccountName || defaultAccountName)
|
||||||
.then(() => history.push(DEFAULT_ROUTE))
|
.then(() => history.push(DEFAULT_ROUTE))
|
||||||
|
Loading…
Reference in New Issue
Block a user