forked from vuejs/vue-cli
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathserveWithPuppeteer.js
108 lines (93 loc) · 2.76 KB
/
serveWithPuppeteer.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
const stripAnsi = require('strip-ansi')
const launchPuppeteer = require('./launchPuppeteer')
module.exports = async function serveWithPuppeteer (serve, test, noPuppeteer) {
let activeBrowser
let activeChild
let notifyUpdate
const nextUpdate = () => {
return new Promise(resolve => {
notifyUpdate = resolve
})
}
await new Promise((resolve, reject) => {
const child = activeChild = serve()
const exit = async (err) => {
if (activeBrowser) {
await activeBrowser.close()
activeBrowser = null
}
if (activeChild) {
activeChild.stdin.write('close')
activeBrowser = null
}
reject(err)
}
let isFirstMatch = true
child.stdout.on('data', async (data) => {
data = data.toString()
try {
const urlMatch = data.match(/http:\/\/[^/]+\//)
if (urlMatch && isFirstMatch) {
isFirstMatch = false
let url = urlMatch[0]
// fix "Protocol error (Page.navigate): Cannot navigate to invalid URL undefined" error
// when running test in vscode terminal(zsh)
url = stripAnsi(url)
if (noPuppeteer) {
await test({ url })
} else {
// start browser
const { page, browser, requestUrls } = await launchPuppeteer(url)
activeBrowser = browser
const helpers = createHelpers(page)
await test({
browser,
page,
url,
nextUpdate,
helpers,
requestUrls
})
await browser.close()
activeBrowser = null
}
// on appveyor, the spawned server process doesn't exit
// and causes the build to hang.
child.stdin.write('close')
activeChild = null
resolve()
} else if (data.match(/App updated/)) {
if (notifyUpdate) {
notifyUpdate(data)
}
} else if (data.match(/Failed to compile/)) {
exit(data)
}
} catch (err) {
exit(err)
}
})
child.on('exit', code => {
activeChild = null
if (code !== 0) {
exit(`serve exited with code ${code}`)
}
})
})
}
/* eslint-disable no-shadow */
function createHelpers (page) {
return {
getText: selector => page.evaluate(selector => {
return document.querySelector(selector).textContent
}, selector),
hasElement: selector => page.evaluate(selector => {
return !!document.querySelector(selector)
}, selector),
hasClass: (selector, cls) => page.evaluate((selector, cls) => {
const el = document.querySelector(selector)
return el && el.classList.contains(cls)
}, selector, cls)
}
}
/* eslint-enable no-shadow */