-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtest.js
137 lines (114 loc) · 4.48 KB
/
test.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
const server = require('./server')
const configuration = require('./package.json')
const puppeteer = configuration.devDependencies.puppeteer
? require('puppeteer')
: require('puppeteer-firefox');
// configuration
const rootPath = __dirname // which folder to serve over http
const port = 8888 // which port to use for http server
const mainPage = `http://localhost:${port}/`
const headless = false // false: show browser, true: hide browser
const slowMo = true // true: each browser action will take 100 milliseconds
? 100
: 0
// globals
let browser = null
let page = null
before(async function () {
this.timeout(5 * 1000) // starting browser may take more than 2 seconds
await server.start(rootPath, port)
browser = await puppeteer.launch({ headless, slowMo })
page = (await browser.pages())[0]
page.on('console', async function (msg) {
if (msg.type() === 'error' && msg.args().length) {
let args = await Promise.all(msg.args().map(arg => arg.jsonValue()))
console.error("Browser console.error:", ...args)
} else {
console.log(msg._text)
}
})
})
beforeEach(async function () {
await page.goto(mainPage)
})
after(function () {
browser.close()
server.shutdown()
})
describe('tests', function () {
this.timeout(slowMo === 0 ? 2000 : 0)
it('typing in regular input element', async function () {
await waitFor({ text: 'regular', click: true })
await page.keyboard.type('asdf')
await waitFor({ text: 'asdf' })
})
it('typing in shadowed input element', async function () {
await waitFor({ text: 'shadowed', click: true })
await page.keyboard.type('qwert')
await waitFor({ text: 'qwert' })
})
it('typing in iframed input element', async function () {
await waitFor({ text: 'iframe', click: true })
await page.keyboard.type('1234')
await waitFor({ text: '1234' })
})
it('should fail', async function () {
await waitFor({ text: 'non existent text', timeout: 100 })
})
})
/**
* Waits for a visible element containing given text, possibly clicks it.
*
* @param {object} params - What to wait for, in which selector, if it should be clicked and how long to wait.
* @param {string} params.text - What text to wait for.
* @param {string} params.selector - What selector to use, defaults to any element: `*`.
* @param {bool} params.click - Wheter to click when found.
* @param {number} params.timeout - How long to wait in milliseconds.
*/
async function waitFor({ text, selector = '*', click = false, timeout = 1000 }) {
const start = Date.now()
while ((Date.now() - start) < timeout) {
let frames = await page.frames()
let scopes = [page, ...frames]
for (let scope of scopes) {
let result
try {
result = await scope.evaluate(pageFunction, text, selector, click)
} catch (err) {
// probably lost execution context, break and get new scopes
break
}
if (result) {
return true
}
}
}
throw new Error(`'${text}' not found on page in selector '${selector}', waited ${timeout} milliseconds.`)
function pageFunction (text, selector, click) {
let match = findElement(text)
if (match) {
if (click) {
match.click()
}
return true
}
return false
function findElement(text) {
let matchingElements = Array.from(document.querySelectorAll(selector))
.filter(element => element.textContent.includes(text))
.sort((a, b) => a.textContent.length - b.textContent.length) // shortest text first, e.g. "best" search result
if (matchingElements.length > 0 && matchingElements[0].offsetParent !== null) {
return matchingElements[0]
}
let shadowedElements = Array.from(document.querySelectorAll(selector))
.filter(element => element.shadowRoot)
.flatMap(element => Array.from(element.shadowRoot.querySelectorAll(selector)))
.filter(element => element.textContent.includes(text))
.sort((a, b) => a.textContent.length - b.textContent.length)
if (shadowedElements.length > 0 && shadowedElements[0].offsetParent !== null) {
return shadowedElements[0]
}
return null
}
}
}