Skip to content

Commit a7bec3a

Browse files
authored
Merge pull request #604 from adroitwhiz/playwright
Playwright: the re-wright
2 parents 3f68e18 + 0ce50b6 commit a7bec3a

File tree

7 files changed

+148
-121
lines changed

7 files changed

+148
-121
lines changed

.travis.yml

Lines changed: 3 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -1,21 +1,18 @@
11
language: node_js
22
dist: trusty
3-
addons:
4-
chrome: stable
53
node_js:
6-
- 8
4+
- 10
75
- node
86
env:
97
- NODE_ENV=production
10-
before_install:
11-
- google-chrome-stable --headless --no-sandbox --remote-debugging-port=9222 &
128
install:
139
- npm --production=false install
1410
- npm --production=false update
1511
sudo: false
1612
cache:
1713
directories:
1814
- node_modules
15+
- ~/.cache/ms-playwright
1916
jobs:
2017
include:
2118
- stage: test
@@ -24,7 +21,7 @@ jobs:
2421
- npm run docs
2522
- npm run tap
2623
- stage: deploy
27-
node_js: 8
24+
node_js: 10
2825
script: npm run build
2926
before_deploy:
3027
- VPKG=$($(npm bin)/json -f package.json version)

package.json

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -29,14 +29,14 @@
2929
"babel-loader": "^7.1.4",
3030
"babel-polyfill": "^6.22.0",
3131
"babel-preset-env": "^1.6.1",
32-
"chromeless": "^1.5.1",
3332
"copy-webpack-plugin": "^4.5.1",
3433
"docdash": "^0.4.0",
3534
"eslint": "^4.6.1",
3635
"eslint-config-scratch": "^5.0.0",
3736
"gh-pages": "^1.0.0",
3837
"jsdoc": "^3.5.5",
3938
"json": "^9.0.4",
39+
"playwright-chromium": "^1.0.1",
4040
"scratch-vm": "0.2.0-prerelease.20191227164934",
4141
"tap": "^11.0.0",
4242
"travis-after-all": "^1.4.4",

test/helper/page-util.js

Lines changed: 54 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,54 @@
1+
/* global window, VirtualMachine, ScratchStorage, ScratchSVGRenderer */
2+
/* eslint-env browser */
3+
4+
// Wait for all SVG skins to be loaded.
5+
// TODO: this is extremely janky and should be removed once vm.loadProject waits for SVG skins to load
6+
// https://github.com/LLK/scratch-render/issues/563
7+
window.waitForSVGSkinLoad = renderer => new Promise(resolve => {
8+
// eslint-disable-next-line prefer-const
9+
let interval;
10+
11+
const waitInner = () => {
12+
let numSVGSkins = 0;
13+
let numLoadedSVGSkins = 0;
14+
for (const skin of renderer._allSkins) {
15+
if (skin.constructor.name !== 'SVGSkin') continue;
16+
numSVGSkins++;
17+
if (skin._svgRenderer.loaded) numLoadedSVGSkins++;
18+
}
19+
20+
if (numSVGSkins === numLoadedSVGSkins) {
21+
clearInterval(interval);
22+
resolve();
23+
}
24+
};
25+
26+
interval = setInterval(waitInner, 1);
27+
});
28+
29+
window.loadFileInputIntoVM = (fileInput, vm, render) => {
30+
const reader = new FileReader();
31+
return new Promise(resolve => {
32+
reader.onload = () => {
33+
vm.start();
34+
vm.loadProject(reader.result)
35+
.then(() => window.waitForSVGSkinLoad(render))
36+
.then(() => {
37+
resolve();
38+
});
39+
};
40+
reader.readAsArrayBuffer(fileInput.files[0]);
41+
});
42+
};
43+
44+
window.initVM = render => {
45+
const vm = new VirtualMachine();
46+
const storage = new ScratchStorage();
47+
48+
vm.attachStorage(storage);
49+
vm.attachRenderer(render);
50+
vm.attachV2SVGAdapter(new ScratchSVGRenderer.SVGRenderer());
51+
vm.attachV2BitmapAdapter(new ScratchSVGRenderer.BitmapAdapter());
52+
53+
return vm;
54+
};

test/integration/cpu-render.html

Lines changed: 12 additions & 31 deletions
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22
<script src="../../node_modules/scratch-vm/dist/web/scratch-vm.js"></script>
33
<script src="../../node_modules/scratch-storage/dist/web/scratch-storage.js"></script>
44
<script src="../../node_modules/scratch-svg-renderer/dist/web/scratch-svg-renderer.js"></script>
5+
<script src="../helper/page-util.js"></script>
56
<!-- note: this uses the BUILT version of scratch-render! make sure to npm run build -->
67
<script src="../../dist/web/scratch-render.js"></script>
78

@@ -17,38 +18,18 @@
1718
window.devicePixelRatio = 1;
1819
const gpuCanvas = document.getElementById('test');
1920
var render = new ScratchRender(gpuCanvas);
20-
var vm = new VirtualMachine();
21-
var storage = new ScratchStorage();
21+
var vm = initVM(render);
2222

23-
vm.attachStorage(storage);
24-
vm.attachRenderer(render);
25-
vm.attachV2SVGAdapter(new ScratchSVGRenderer.SVGRenderer());
26-
vm.attachV2BitmapAdapter(new ScratchSVGRenderer.BitmapAdapter());
27-
28-
document.getElementById('file').addEventListener('click', e => {
29-
document.body.removeChild(document.getElementById('loaded'));
30-
});
31-
32-
document.getElementById('file').addEventListener('change', e => {
33-
const reader = new FileReader();
34-
const thisFileInput = e.target;
35-
reader.onload = () => {
36-
vm.start();
37-
vm.loadProject(reader.result)
38-
.then(() => {
39-
// we add a `#loaded` div to our document, the integration suite
40-
// waits for that element to show up to assume the vm is ready
41-
// to play!
42-
const div = document.createElement('div');
43-
div.id='loaded';
44-
document.body.appendChild(div);
45-
vm.greenFlag();
46-
setTimeout(() => {
47-
renderCpu();
48-
}, 1000);
49-
});
50-
};
51-
reader.readAsArrayBuffer(thisFileInput.files[0]);
23+
const fileInput = document.getElementById('file');
24+
const loadFile = loadFileInputIntoVM.bind(null, fileInput, vm, render);
25+
fileInput.addEventListener('change', e => {
26+
loadFile()
27+
.then(() => {
28+
vm.greenFlag();
29+
setTimeout(() => {
30+
renderCpu();
31+
}, 1000);
32+
});
5233
});
5334

5435
const cpuCanvas = document.getElementById('cpu');

test/integration/index.html

Lines changed: 3 additions & 28 deletions
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22
<script src="../../node_modules/scratch-vm/dist/web/scratch-vm.js"></script>
33
<script src="../../node_modules/scratch-storage/dist/web/scratch-storage.js"></script>
44
<script src="../../node_modules/scratch-svg-renderer/dist/web/scratch-svg-renderer.js"></script>
5+
<script src="../helper/page-util.js"></script>
56
<!-- note: this uses the BUILT version of scratch-render! make sure to npm run build -->
67
<script src="../../dist/web/scratch-render.js"></script>
78

@@ -15,39 +16,13 @@
1516

1617
var canvas = document.getElementById('test');
1718
var render = new ScratchRender(canvas);
18-
var vm = new VirtualMachine();
19-
var storage = new ScratchStorage();
19+
var vm = initVM(render);
2020
var mockMouse = data => vm.runtime.postIOData('mouse', {
2121
canvasWidth: canvas.width,
2222
canvasHeight: canvas.height,
2323
...data,
2424
});
2525

26-
vm.attachStorage(storage);
27-
vm.attachRenderer(render);
28-
vm.attachV2SVGAdapter(new ScratchSVGRenderer.SVGRenderer());
29-
vm.attachV2BitmapAdapter(new ScratchSVGRenderer.BitmapAdapter());
30-
31-
document.getElementById('file').addEventListener('click', e => {
32-
document.body.removeChild(document.getElementById('loaded'));
33-
});
34-
35-
document.getElementById('file').addEventListener('change', e => {
36-
const reader = new FileReader();
37-
const thisFileInput = e.target;
38-
reader.onload = () => {
39-
vm.start();
40-
vm.loadProject(reader.result)
41-
.then(() => {
42-
// we add a `#loaded` div to our document, the integration suite
43-
// waits for that element to show up to assume the vm is ready
44-
// to play!
45-
const div = document.createElement('div');
46-
div.id='loaded';
47-
document.body.appendChild(div);
48-
});
49-
};
50-
reader.readAsArrayBuffer(thisFileInput.files[0]);
51-
});
26+
const loadFile = loadFileInputIntoVM.bind(null, document.getElementById('file'), vm, render);
5227
</script>
5328
</body>

test/integration/pick-tests.js

Lines changed: 23 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -1,29 +1,34 @@
11
/* global vm, render, Promise */
2-
const {Chromeless} = require('chromeless');
2+
const {chromium} = require('playwright-chromium');
33
const test = require('tap').test;
44
const path = require('path');
5-
const chromeless = new Chromeless();
65

76
const indexHTML = path.resolve(__dirname, 'index.html');
87
const testDir = (...args) => path.resolve(__dirname, 'pick-tests', ...args);
98

10-
const runFile = (file, action, script) =>
9+
const runFile = async (file, action, page, script) => {
1110
// start each test by going to the index.html, and loading the scratch file
12-
chromeless.goto(`file://${indexHTML}`)
13-
.setFileInput('#file', testDir(file))
14-
// the index.html handler for file input will add a #loaded element when it
15-
// finishes.
16-
.wait('#loaded')
17-
.evaluate(`function () {return (${script})(${action});}`)
18-
;
11+
await page.goto(`file://${indexHTML}`);
12+
const fileInput = await page.$('#file');
13+
await fileInput.setInputFiles(testDir(file));
14+
15+
await page.evaluate(() =>
16+
// `loadFile` is defined on the page itself.
17+
// eslint-disable-next-line no-undef
18+
loadFile()
19+
);
20+
return page.evaluate(`(function () {return (${script})(${action});})()`);
21+
};
1922

2023
// immediately invoked async function to let us wait for each test to finish before starting the next.
2124
(async () => {
25+
const browser = await chromium.launch();
26+
const page = await browser.newPage();
2227

2328
const testOperation = async function (name, action, expect) {
2429
await test(name, async t => {
2530

26-
const results = await runFile('test-mouse-touch.sb2', action, boundAction => {
31+
const results = await runFile('test-mouse-touch.sb2', action, page, boundAction => {
2732
vm.greenFlag();
2833
const sendResults = [];
2934

@@ -97,5 +102,10 @@ const runFile = (file, action, script) =>
97102
}
98103

99104
// close the browser window we used
100-
await chromeless.end();
101-
})();
105+
await browser.close();
106+
})().catch(err => {
107+
// Handle promise rejections by exiting with a nonzero code to ensure that tests don't erroneously pass
108+
// eslint-disable-next-line no-console
109+
console.error(err.message);
110+
process.exit(1);
111+
});

test/integration/scratch-tests.js

Lines changed: 52 additions & 42 deletions
Original file line numberDiff line numberDiff line change
@@ -1,54 +1,56 @@
11
/* global vm, Promise */
2-
const {Chromeless} = require('chromeless');
2+
const {chromium} = require('playwright-chromium');
33
const test = require('tap').test;
44
const path = require('path');
55
const fs = require('fs');
6-
const chromeless = new Chromeless();
76

87
const indexHTML = path.resolve(__dirname, 'index.html');
98
const testDir = (...args) => path.resolve(__dirname, 'scratch-tests', ...args);
109

11-
const testFile = file => test(file, async t => {
10+
const testFile = (file, page) => test(file, async t => {
1211
// start each test by going to the index.html, and loading the scratch file
13-
const says = await chromeless.goto(`file://${indexHTML}`)
14-
.setFileInput('#file', testDir(file))
15-
// the index.html handler for file input will add a #loaded element when it
16-
// finishes.
17-
.wait('#loaded')
18-
.evaluate(() => {
19-
// This function is run INSIDE the integration chrome browser via some
20-
// injection and .toString() magic. We can return some "simple data"
21-
// back across as a promise, so we will just log all the says that happen
22-
// for parsing after.
23-
24-
// this becomes the `says` in the outer scope
25-
const messages = [];
26-
const TIMEOUT = 5000;
27-
28-
vm.runtime.on('SAY', (_, __, message) => {
29-
messages.push(message);
30-
});
12+
await page.goto(`file://${indexHTML}`);
13+
const fileInput = await page.$('#file');
14+
await fileInput.setInputFiles(testDir(file));
15+
await page.evaluate(() =>
16+
// `loadFile` is defined on the page itself.
17+
// eslint-disable-next-line no-undef
18+
loadFile()
19+
);
20+
const says = await page.evaluate(() => {
21+
// This function is run INSIDE the integration chrome browser via some
22+
// injection and .toString() magic. We can return some "simple data"
23+
// back across as a promise, so we will just log all the says that happen
24+
// for parsing after.
25+
26+
// this becomes the `says` in the outer scope
27+
const messages = [];
28+
const TIMEOUT = 5000;
29+
30+
vm.runtime.on('SAY', (_, __, message) => {
31+
messages.push(message);
32+
});
3133

32-
vm.greenFlag();
33-
const startTime = Date.now();
34-
35-
return Promise.resolve()
36-
.then(async () => {
37-
// waiting for all threads to complete, then we return
38-
while (vm.runtime.threads.some(thread => vm.runtime.isActiveThread(thread))) {
39-
if ((Date.now() - startTime) >= TIMEOUT) {
40-
// if we push the message after end, the failure from tap is not very useful:
41-
// "not ok test after end() was called"
42-
messages.unshift(`fail Threads still running after ${TIMEOUT}ms`);
43-
break;
44-
}
45-
46-
await new Promise(resolve => setTimeout(resolve, 50));
34+
vm.greenFlag();
35+
const startTime = Date.now();
36+
37+
return Promise.resolve()
38+
.then(async () => {
39+
// waiting for all threads to complete, then we return
40+
while (vm.runtime.threads.some(thread => vm.runtime.isActiveThread(thread))) {
41+
if ((Date.now() - startTime) >= TIMEOUT) {
42+
// if we push the message after end, the failure from tap is not very useful:
43+
// "not ok test after end() was called"
44+
messages.unshift(`fail Threads still running after ${TIMEOUT}ms`);
45+
break;
4746
}
4847

49-
return messages;
50-
});
51-
});
48+
await new Promise(resolve => setTimeout(resolve, 50));
49+
}
50+
51+
return messages;
52+
});
53+
});
5254

5355
// Map string messages to tap reporting methods. This will be used
5456
// with events from scratch's runtime emitted on block instructions.
@@ -103,13 +105,21 @@ const testFile = file => test(file, async t => {
103105

104106
// immediately invoked async function to let us wait for each test to finish before starting the next.
105107
(async () => {
108+
const browser = await chromium.launch();
109+
const page = await browser.newPage();
110+
106111
const files = fs.readdirSync(testDir())
107112
.filter(uri => uri.endsWith('.sb2') || uri.endsWith('.sb3'));
108113

109114
for (const file of files) {
110-
await testFile(file);
115+
await testFile(file, page);
111116
}
112117

113118
// close the browser window we used
114-
await chromeless.end();
115-
})();
119+
await browser.close();
120+
})().catch(err => {
121+
// Handle promise rejections by exiting with a nonzero code to ensure that tests don't erroneously pass
122+
// eslint-disable-next-line no-console
123+
console.error(err.message);
124+
process.exit(1);
125+
});

0 commit comments

Comments
 (0)