Skip to content

Commit 7baa2b4

Browse files
committed
Add browser to integration test suite
1 parent bc84223 commit 7baa2b4

File tree

7 files changed

+96
-31
lines changed

7 files changed

+96
-31
lines changed

.github/workflows/ci.yml

+27-3
Original file line numberDiff line numberDiff line change
@@ -64,9 +64,6 @@ jobs:
6464
# See supported Node.js release schedule at https://nodejs.org/en/about/previous-releases
6565
node-version: [18.x, 20.x]
6666
suite: [commonjs, esm, typescript]
67-
exclude:
68-
- suite: cloudflare-worker
69-
node-version: 18.x # Only test Cloudflare suite with the latest Node version
7067
fail-fast: false
7168

7269
steps:
@@ -84,6 +81,33 @@ jobs:
8481
npm --prefix integration/${{ matrix.suite }} install "./${{ needs.build.outputs.tarball-name }}"
8582
npm --prefix integration/${{ matrix.suite }} test
8683
84+
integration-browser:
85+
needs: [test, build]
86+
runs-on: ubuntu-latest
87+
88+
env:
89+
REPLICATE_API_TOKEN: ${{ secrets.REPLICATE_API_TOKEN }}
90+
91+
strategy:
92+
matrix:
93+
browser: ["chromium", "firefox", "webkit"]
94+
suite: ["browser"]
95+
fail-fast: false
96+
97+
steps:
98+
- uses: actions/checkout@v3
99+
- uses: actions/download-artifact@v3
100+
with:
101+
name: package-tarball
102+
- name: Use Node.js ${{ matrix.node-version }}
103+
uses: actions/setup-node@v3
104+
with:
105+
node-version: ${{ matrix.node-version }}
106+
cache: "npm"
107+
- run: |
108+
npm --prefix integration/${{ matrix.suite }} install
109+
npm --prefix integration/${{ matrix.suite }} install "./${{ needs.build.outputs.tarball-name }}"
110+
npm --prefix integration/${{ matrix.suite }} npm exec -- playwright test --browser ${{ matrix.browser }}
87111
88112
integration-edge:
89113
needs: [test, build]

index.js

+2-12
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ const {
55
withAutomaticRetries,
66
validateWebhook,
77
parseProgressFromLogs,
8+
streamAsyncIterator,
89
} = require("./lib/util");
910

1011
const accounts = require("./lib/accounts");
@@ -297,18 +298,7 @@ class Replicate {
297298
options: { signal },
298299
});
299300

300-
const reader = stream.getReader();
301-
302-
while (true) {
303-
const { done, value } = await reader.read();
304-
305-
if (done) {
306-
break;
307-
}
308-
309-
yield value;
310-
}
311-
reader.releaseLock();
301+
yield* streamAsyncIterator(stream);
312302
} else {
313303
throw new Error("Prediction does not support streaming");
314304
}

integration/browser/README.md

+39
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,39 @@
1+
# Browser integration tests
2+
3+
Uses [`playwright`](https://playwright.dev/docs) to run a basic integration test against the three most common browser engines, Firefox, Chromium and WebKit.
4+
5+
It uses the `replicate/canary` model for the moment, which requires a Replicate API token available in the environment under `REPLICATE_API_TOKEN`.
6+
7+
The entire suite is a single `main()` function that calls a single model exercising the streaming API.
8+
9+
The test uses `esbuild` within the test generate a browser friendly version of the `index.js` file which is loaded into the given browser and calls the `main()` function asserting the response content.
10+
11+
## CORS
12+
13+
The Replicate API doesn't support Cross Origin Resource Sharing at this time. We work around this in Playwright by intercepting the request in a `page.route` handler. We don't modify the request/response, but this seems to work around the restriction.
14+
15+
## Setup
16+
17+
npm install
18+
19+
## Local
20+
21+
The following command will run the tests across all browsers.
22+
23+
npm test
24+
25+
To run against the default browser (chromium) run:
26+
27+
npm exec playwright test
28+
29+
Or, specify a browser with:
30+
31+
npm exec playwright test --browser firefox
32+
33+
## Debugging
34+
35+
Running `playwright test` with the `--debug` flag opens a browser window with a debugging interface, and a breakpoint set at the start of the test. It can also be connected directly to VSCode.
36+
37+
npm exec playwright test --debug
38+
39+
The browser.js file is injected into the page via a script tag, to be able to set breakpoints in this file you'll need to use a `debugger` statement and open the devtools in the spawned browser window before continuing the test suite.

integration/browser/package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@
66
"main": "index.js",
77
"type": "module",
88
"scripts": {
9-
"test": "node index.test.js"
9+
"test": "playwright test --browser all"
1010
},
1111
"license": "ISC",
1212
"dependencies": {

lib/stream.js

+2-10
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
// Attempt to use readable-stream if available, attempt to use the built-in stream module.
22

33
const ApiError = require("./error");
4+
const { streamAsyncIterator } = require("./util");
45
const {
56
EventSourceParserStream,
67
} = require("../vendor/eventsource-parser/stream");
@@ -73,15 +74,7 @@ function createReadableStream({ url, fetch, options = {} }) {
7374
.pipeThrough(new TextDecoderStream())
7475
.pipeThrough(new EventSourceParserStream());
7576

76-
const reader = stream.getReader();
77-
78-
while (true) {
79-
const { done, value: event } = await reader.read();
80-
81-
if (done) {
82-
break;
83-
}
84-
77+
for await (const event of streamAsyncIterator(stream)) {
8578
if (event.event === "error") {
8679
controller.error(new Error(event.data));
8780
break;
@@ -96,7 +89,6 @@ function createReadableStream({ url, fetch, options = {} }) {
9689
}
9790
}
9891

99-
reader.releaseLock();
10092
controller.close();
10193
},
10294
});

lib/util.js

+24
Original file line numberDiff line numberDiff line change
@@ -354,9 +354,33 @@ function parseProgressFromLogs(input) {
354354
return null;
355355
}
356356

357+
/**
358+
* Helper to make any `ReadableStream` iterable, this is supported
359+
* by most server runtimes but browsers still haven't implemented
360+
* it yet.
361+
* See: https://developer.mozilla.org/en-US/docs/Web/API/ReadableStream#browser_compatibility
362+
*
363+
* @template T
364+
* @param {ReadableStream<T>} stream an instance of a `ReadableStream`
365+
* @yields {T} a chunk/event from the stream
366+
*/
367+
async function* streamAsyncIterator(stream) {
368+
const reader = stream.getReader();
369+
try {
370+
while (true) {
371+
const { done, value } = await reader.read();
372+
if (done) return;
373+
yield value;
374+
}
375+
} finally {
376+
reader.releaseLock();
377+
}
378+
}
379+
357380
module.exports = {
358381
transformFileInputs,
359382
validateWebhook,
360383
withAutomaticRetries,
361384
parseProgressFromLogs,
385+
streamAsyncIterator,
362386
};

tsconfig.json

+1-5
Original file line numberDiff line numberDiff line change
@@ -5,9 +5,5 @@
55
"strict": true,
66
"allowJs": true
77
},
8-
"exclude": [
9-
"**/node_modules",
10-
"integration/**"
11-
]
8+
"exclude": ["integration/**", "**/node_modules"]
129
}
13-

0 commit comments

Comments
 (0)