Skip to content

Commit 0ee8fe2

Browse files
committed
Playwright tests
1 parent 204feb5 commit 0ee8fe2

11 files changed

+216
-13
lines changed

.github/workflows/test.yml

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
name: test
2+
on: push
3+
jobs:
4+
build:
5+
runs-on: ubuntu-latest
6+
timeout-minutes: 1
7+
strategy:
8+
matrix:
9+
node-version: [18.x]
10+
steps:
11+
- name: Checkout
12+
uses: actions/checkout@v3
13+
- name: Use Node.js ${{ matrix.node-version }}
14+
uses: actions/setup-node@v3
15+
with:
16+
node-version: ${{ matrix.node-version }}
17+
- name: Cache node_modules
18+
id: cache-node-modules
19+
uses: actions/cache@v3
20+
with:
21+
key: ${{ runner.os }}-${{ matrix.node-version }}-${{ hashFiles('**/package-lock.json') }}
22+
path: node_modules
23+
- name: Install dependencies
24+
if: steps.cache-node-modules.outputs.cache-hit != 'true'
25+
run: npm ci
26+
- name: Install Playwright browsers
27+
run: npx playwright install --with-deps
28+
- name: Run Playwright tests
29+
run: npm run test-playwright

.gitignore

+2-1
Original file line numberDiff line numberDiff line change
@@ -1 +1,2 @@
1-
dist
1+
dist
2+
tests/results

build.js

+4-4
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,14 @@
11
const esbuild = require('esbuild')
2-
const fs = require('fs')
3-
const fsp = require('fs').promises
4-
const path = require('path')
2+
const fs = require('node:fs')
3+
const fsp = require('node:fs').promises
4+
const path = require('node:path')
5+
const httpdir = require('httpdir')
56

67
const srcPath = path.join(__dirname, 'src')
78
const distPath = path.join(__dirname, 'dist')
89

910
build()
1011
if (process.argv.includes('--watch')) {
11-
const httpdir = require('/usr/local/lib/node_modules/httpdir')
1212
const server = httpdir.createServer({ basePath: distPath, httpPort: 9697 })
1313
server.onStart(({ urls }) => {
1414
console.log(urls.join('\n'))

package-lock.json

+75
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

+6-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,8 @@
55
"author": "Johan Satgé",
66
"private": true,
77
"scripts": {
8-
"build": "node build.js"
8+
"build": "node build.js",
9+
"test-playwright": "npm run build && playwright test"
910
},
1011
"repository": {
1112
"type": "git",
@@ -25,5 +26,9 @@
2526
"htm": "^3.1.1",
2627
"preact": "^10.23.1",
2728
"prismjs": "^1.29.0"
29+
},
30+
"devDependencies": {
31+
"@playwright/test": "^1.50.1",
32+
"httpdir": "^2.1.0"
2833
}
2934
}

playwright.config.js

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
const { defineConfig } = require('@playwright/test')
2+
const httpdir = require('httpdir')
3+
4+
export default defineConfig({
5+
use: {
6+
baseURL: 'http://localhost:9698',
7+
headless: true,
8+
acceptDownloads: true,
9+
},
10+
testMatch: 'tests/*.spec.js',
11+
outputDir: 'tests/results',
12+
globalSetup: 'playwright.globalsetup.js',
13+
})

playwright.globalsetup.js

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
const httpdir = require('httpdir')
2+
3+
module.exports = () => {
4+
return new Promise((resolve, reject) => {
5+
const server = httpdir.createServer({
6+
basePath: 'dist',
7+
httpPort: 9698,
8+
})
9+
server.onStart(resolve)
10+
server.onError(reject)
11+
server.start()
12+
})
13+
}

src/bundler.js

-1
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,6 @@ function wasmJsResolver() {
3131
// Function sends back the code, sample usage code and stats
3232
export async function buildBundle(requestedImports, format) {
3333
const { bundleSource, bundleComments, usage } = getBundleSource(requestedImports, format)
34-
console.log('BUNDLE SOURCE', bundleSource)
3534
await esbuildInitPromise
3635
const params = {
3736
stdin: {

src/ui.css

+10
Original file line numberDiff line numberDiff line change
@@ -172,6 +172,11 @@ body {
172172
border-radius: 4px;
173173
}
174174

175+
.check-button:disabled {
176+
opacity: 0.5;
177+
cursor: default;
178+
}
179+
175180
.columns {
176181
display: grid;
177182
grid-template-columns: repeat(4, 1fr);
@@ -278,6 +283,11 @@ input[type="radio"]:checked::after {
278283
border-radius: 50%;
279284
}
280285

286+
input[type="radio"]:disabled {
287+
opacity: 0.5;
288+
cursor: default;
289+
}
290+
281291
.loader {
282292
position: absolute;
283293
box-sizing: border-box;

src/ui.js

+18-6
Original file line numberDiff line numberDiff line change
@@ -159,18 +159,25 @@ function App({ defaultImports }) {
159159
Feel free to fine-tune the imports you need to get a fully customized version of Preact though!
160160
</div>
161161
`}
162-
<button class="check-button" onClick=${onCheckAllImports}>Check all</button>
163-
<button class="check-button" onClick=${onCheckNoImports}>Check none</button>
162+
<button class="check-button" onClick=${onCheckAllImports} disabled=${isLoadingBundle}>
163+
Check all
164+
</button>
165+
<button class="check-button" onClick=${onCheckNoImports} disabled=${isLoadingBundle} data-testid="checkNone">
166+
Check none
167+
</button>
164168
<div class="columns">
165169
${Object.keys(window.preactEcosystem.imports).map((pkg) => html`
166170
<div class="column" key=${pkg}>
167171
<h3 class="column-title">
168172
${pkg}
169-
<span class="version">${window.preactEcosystem.versions[pkg]}</span>
173+
<span class="version" data-testid=${`${pkg}Version`}>
174+
${window.preactEcosystem.versions[pkg]}
175+
</span>
170176
</h3>
171177
<${ImportsList}
172178
pkg="${pkg}" imports=${window.preactEcosystem.imports[pkg]}
173179
selectedImports=${selectedImports} onImportChange=${onImportChange}
180+
isLoadingBundle=${isLoadingBundle}
174181
/>
175182
</div>
176183
`)}
@@ -187,6 +194,7 @@ function App({ defaultImports }) {
187194
type="radio" name="format" value="esm"
188195
onChange=${onFormatChange}
189196
checked=${format === 'esm'}
197+
disabled=${isLoadingBundle}
190198
/>
191199
ESM
192200
</label>
@@ -195,6 +203,7 @@ function App({ defaultImports }) {
195203
type="radio" name="format" value="iife"
196204
onChange=${onFormatChange}
197205
checked=${format === 'iife'}
206+
disabled=${isLoadingBundle}
198207
/>
199208
IIFE
200209
</label>
@@ -209,7 +218,9 @@ function App({ defaultImports }) {
209218
<button class="action ${hasCopied ? 'copied' : ''}" onClick=${onCopyToClipboard} disabled=${!bundle.filename}>
210219
Copy to clipboard
211220
</button>
212-
<button class="action" onClick=${onDownload} disabled=${!bundle.filename}>Download file</button>
221+
<button class="action" onClick=${onDownload} disabled=${!bundle.filename} data-testid="downloadBundle">
222+
Download file
223+
</button>
213224
<span class="size">
214225
Size: ${bundle.sizeKb || 0}Kb (${bundle.sizeGzippedKb || 0}Kb gzipped)
215226
</span>
@@ -221,6 +232,7 @@ function App({ defaultImports }) {
221232
<code
222233
class="code"
223234
dangerouslySetInnerHTML=${{__html: highlightedBundleUsage}}
235+
data-testid="htmlExample"
224236
></code>
225237
</pre>
226238
</section>
@@ -230,13 +242,13 @@ function App({ defaultImports }) {
230242
`
231243
}
232244

233-
function ImportsList({ pkg, imports, selectedImports, onImportChange }) {
245+
function ImportsList({ pkg, imports, selectedImports, onImportChange, isLoadingBundle }) {
234246
return imports.map((imp) => html`
235247
<label class="input-label" key=${pkg + imp}>
236248
<input
237249
type="checkbox" autocomplete="off"
238250
data-pkg=${pkg} data-imp=${imp}
239-
disabled=${mandatoryImports[pkg] && mandatoryImports[pkg].includes(imp)}
251+
disabled=${isLoadingBundle || (mandatoryImports[pkg] && mandatoryImports[pkg].includes(imp))}
240252
checked=${typeof selectedImports[pkg] === 'object' && selectedImports[pkg].includes(imp)}
241253
value="1"
242254
onChange=${onImportChange}

tests/basic.spec.js

+46
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
const { test, expect } = require('@playwright/test')
2+
const path = require('node:path')
3+
const fsp = require('node:fs').promises
4+
5+
async function downloadPreactBundle({ page }) {
6+
const downloadPromise = page.waitForEvent('download')
7+
await page.getByTestId('downloadBundle').click()
8+
const download = await downloadPromise
9+
const downloadName = download.suggestedFilename()
10+
const downloadPath = path.join(__dirname, '../dist', downloadName)
11+
await download.saveAs(downloadPath)
12+
return {
13+
preactBundleFilename: downloadName,
14+
preactBundle: await fsp.readFile(downloadPath, 'utf8')
15+
}
16+
}
17+
18+
async function downloadHtmlExample({ page, preactBundleFilename }) {
19+
const htmlExample = await page.getByTestId('htmlExample').innerText()
20+
const htmlFilename = preactBundleFilename.replace('.js', '.html')
21+
await fsp.writeFile(path.join(__dirname, '../dist', htmlFilename), htmlExample)
22+
return htmlFilename
23+
}
24+
25+
test('Generate a minimal Preact bundle', async({ page }) => {
26+
await page.goto('index.html')
27+
// Check the less possible dependencies
28+
await page.getByTestId('checkNone').click()
29+
// Download & test Preact bundle contents
30+
const { preactBundleFilename, preactBundle } = await downloadPreactBundle({ page })
31+
const preactVersion = await page.getByTestId('preactVersion').innerText()
32+
expect(preactBundle).toContain(`// preact@${preactVersion}`)
33+
expect(preactBundle).not.toContain('// preact/hooks')
34+
expect(preactBundle).not.toContain('// @preact/signals')
35+
expect(preactBundle).not.toContain('// htm')
36+
// Download & test HTML example
37+
const htmlExampleFilemane = await downloadHtmlExample({ page, preactBundleFilename })
38+
await page.goto(htmlExampleFilemane)
39+
await expect(page.getByText('Hello World!')).toBeVisible()
40+
})
41+
42+
// @todo more tests
43+
// - with htm
44+
// - with htm & hooks
45+
// - with htm & signals
46+
// - with IIFE & ESM

0 commit comments

Comments
 (0)