Add eventsource polyfill for Node.js and browser environments (#8118)

* add msw setup and initialisation tests

* add changeset

* add eventsource polyfill for node and browser envs

* add changeset

* add changeset

* config tweak

* types

* update eventsource usage

* add changeset

* add walk_and_store_blobs improvements and add tests

* add changeset

* api_info tests

* add direct space URL link tests

* fix tests

* add view_api tests

* add post_message test

* tweak

* add spaces tests

* jwt and protocol tests

* add post_data tests

* test tweaks

* dynamically import eventsource

* revet eventsource imports

* add node test

* lockfile

* add client test in root pkg file

* lcokfile

* remove eventsource from js/app

* add changeset

* remove ts ignore

* move eventsource polyfill to eventsource factory

* add changeset

* tweak

---------

Co-authored-by: gradio-pr-bot <gradio-pr-bot@users.noreply.github.com>
This commit is contained in:
Hannah 2024-05-02 22:49:55 +02:00 committed by GitHub
parent 5671ff129a
commit 7aca673b38
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
16 changed files with 79 additions and 14 deletions

View File

@ -0,0 +1,7 @@
---
"@gradio/client": patch
"@gradio/upload": patch
"gradio": patch
---
fix:Add eventsource polyfill for Node.js and browser environments

View File

@ -13,7 +13,9 @@
"./package.json": "./package.json"
},
"dependencies": {
"@types/eventsource": "^1.1.15",
"bufferutil": "^4.0.7",
"eventsource": "^2.0.2",
"msw": "^2.2.1",
"semiver": "^1.1.0",
"typescript": "^5.0.0",
@ -26,7 +28,10 @@
"scripts": {
"bundle": "vite build --ssr",
"generate_types": "tsc",
"build": "pnpm bundle && pnpm generate_types"
"build": "pnpm bundle && pnpm generate_types",
"test": "pnpm test:client && pnpm test:client:node",
"test:client": "vitest run -c vite.config.js",
"test:client:node": "TEST_MODE=node vitest run -c vite.config.js"
},
"engines": {
"node": ">=18.0.0"

View File

@ -62,12 +62,19 @@ export class Client {
return fetch(input, init);
}
eventSource_factory(url: URL): EventSource {
if (typeof window !== undefined && typeof EventSource !== "undefined") {
eventSource_factory(url: URL): EventSource | null {
if (typeof window === "undefined" || typeof EventSource === "undefined") {
import("eventsource")
.then((EventSourceModule) => {
return new EventSourceModule.default(url.toString());
})
.catch((error) =>
console.error("Failed to load EventSource module:", error)
);
} else {
return new EventSource(url.toString());
}
// @ts-ignore
return null; // todo: polyfill eventsource for node envs
return null;
}
view_api: () => Promise<ApiInfo<JsApiData>>;

View File

@ -1,4 +1,5 @@
import { QUEUE_FULL_MSG, SPACE_METADATA_ERROR_MSG } from "../constants";
import { beforeAll, afterEach, afterAll, it, expect, describe } from "vitest";
import {
handle_message,
get_description,

View File

@ -1,4 +1,4 @@
import { describe, it, expect, vi } from "vitest";
import { describe, it, expect, vi, afterEach } from "vitest";
import {
update_object,
walk_and_store_blobs,

View File

@ -425,5 +425,14 @@ export const handlers: RequestHandler[] = [
"Content-Type": "application/json"
}
});
}),
// heartbeat requests
http.get(`*/heartbeat/*`, () => {
return new HttpResponse(null, {
status: 200,
headers: {
"Content-Type": "application/json"
}
});
})
];

View File

@ -4,6 +4,7 @@ import {
determine_protocol
} from "../helpers/init_helpers";
import { initialise_server } from "./server";
import { beforeAll, afterEach, afterAll, it, expect, describe } from "vitest";
const server = initialise_server();

View File

@ -3,6 +3,7 @@ import { Client } from "../client";
import { initialise_server } from "./server";
import { BROKEN_CONNECTION_MSG } from "../constants";
const server = initialise_server();
import { beforeAll, afterEach, afterAll, it, expect, describe } from "vitest";
beforeAll(() => server.listen());
afterEach(() => server.resetHandlers());

View File

@ -5,6 +5,7 @@ import {
set_space_timeout,
check_space_status
} from "../helpers/spaces";
import { beforeAll, afterEach, afterAll, it, expect, describe } from "vitest";
import { initialise_server } from "./server";
import { hardware_sleeptime_response } from "./test_data";

View File

@ -1,4 +1,4 @@
import { describe, it, expect, afterEach } from "vitest";
import { describe, it, expect, afterEach, beforeAll, afterAll } from "vitest";
import { Client } from "..";
import { initialise_server } from "./server";

View File

@ -28,7 +28,7 @@ export function open_stream(this: Client): void {
throw new Error("Cannot connect to sse endpoint: " + url.toString());
}
event_source.onmessage = async function (event) {
event_source.onmessage = async function (event: MessageEvent) {
let _data = JSON.parse(event.data);
if (_data.msg === "close_stream") {
close_stream(stream_status, event_source);
@ -53,8 +53,13 @@ export function open_stream(this: Client): void {
close_stream(stream_status, event_source);
}
}
let fn = event_callbacks[event_id];
window.setTimeout(fn, 0, _data); // need to do this to put the event on the end of the event loop, so the browser can refresh between callbacks and not freeze in case of quick generations. See https://github.com/gradio-app/gradio/pull/7055
let fn: (data: any) => void = event_callbacks[event_id];
if (typeof window !== "undefined") {
window.setTimeout(fn, 0, _data); // need to do this to put the event on the end of the event loop, so the browser can refresh between callbacks and not freeze in case of quick generations. See https://github.com/gradio-app/gradio/pull/7055
} else {
setImmediate(fn, _data);
}
} else {
if (!pending_stream_messages[event_id]) {
pending_stream_messages[event_id] = [];

View File

@ -376,7 +376,7 @@ export function submit(
);
}
event_source.onmessage = async function (event) {
event_source.onmessage = async function (event: MessageEvent) {
const _data = JSON.parse(event.data);
const { type, status, data } = handle_message(
_data,
@ -476,7 +476,11 @@ export function submit(
fn_index,
time: new Date()
});
let hostname = window.location.hostname;
let hostname = "";
if (typeof window !== "undefined") {
hostname = window?.location?.hostname;
}
let hfhubdev = "dev.spaces.huggingface.tech";
const origin = hostname.includes(".dev.")
? `https://moon-${hostname.split(".")[1]}.${hfhubdev}`

View File

@ -1,6 +1,8 @@
import { defineConfig } from "vite";
import { svelte } from "@sveltejs/vite-plugin-svelte";
const TEST_MODE = process.env.TEST_MODE || "happy-dom";
export default defineConfig({
build: {
lib: {
@ -17,6 +19,11 @@ export default defineConfig({
},
plugins: [svelte()],
mode: process.env.MODE || "development",
test: {
include: ["./src/test/*.test.*"],
environment: TEST_MODE
},
ssr: {
target: "node",
format: "esm",

View File

@ -50,7 +50,7 @@
const _data = JSON.parse(event.data);
if (!progress) progress = true;
if (_data.msg === "done") {
event_source.close();
event_source?.close();
dispatch("done");
} else {
current_file_upload = _data;

View File

@ -15,7 +15,7 @@
"ts:check": "svelte-check --tsconfig tsconfig.json --threshold error",
"test": "pnpm --filter @gradio/client build && vitest dev --config .config/vitest.config.ts",
"test:run": "pnpm --filter @gradio/client build && vitest run --config .config/vitest.config.ts --reporter=verbose",
"test:node": "TEST_MODE=node pnpm vitest run --config .config/vitest.config.ts",
"test:client": "pnpm --filter=@gradio/client test",
"test:browser": "pnpm --filter @gradio/app test:browser",
"test:browser:reload": "CUSTOM_TEST=1 pnpm --filter @gradio/app test:browser:reload",
"test:browser:full": "run-s build test:browser",

17
pnpm-lock.yaml generated
View File

@ -216,9 +216,15 @@ importers:
client/js:
dependencies:
'@types/eventsource':
specifier: ^1.1.15
version: 1.1.15
bufferutil:
specifier: ^4.0.7
version: 4.0.7
eventsource:
specifier: ^2.0.2
version: 2.0.2
msw:
specifier: ^2.2.1
version: 2.2.8(typescript@5.4.3)
@ -4283,6 +4289,9 @@ packages:
'@types/estree@1.0.5':
resolution: {integrity: sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw==}
'@types/eventsource@1.1.15':
resolution: {integrity: sha512-XQmGcbnxUNa06HR3VBVkc9+A2Vpi9ZyLJcdS5dwaQQ/4ZMWFO+5c90FnMUpbtMZwB/FChoYHwuVg8TvkECacTA==}
'@types/express-serve-static-core@4.17.41':
resolution: {integrity: sha512-OaJ7XLaelTgrvlZD8/aa0vvvxZdUmlCn6MtWeB7TkiKW70BQLc9XEPpDLPdbo52ZhXUCrznlWdCHWxJWtdyajA==}
@ -5886,6 +5895,10 @@ packages:
eventemitter3@4.0.7:
resolution: {integrity: sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw==}
eventsource@2.0.2:
resolution: {integrity: sha512-IzUmBGPR3+oUG9dUeXynyNmf91/3zUSJg1lCktzKw47OXuhco54U3r9B7O4XX+Rb1Itm9OZ2b0RkTs10bICOxA==}
engines: {node: '>=12.0.0'}
execa@5.1.1:
resolution: {integrity: sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==}
engines: {node: '>=10'}
@ -12303,6 +12316,8 @@ snapshots:
'@types/estree@1.0.5': {}
'@types/eventsource@1.1.15': {}
'@types/express-serve-static-core@4.17.41':
dependencies:
'@types/node': 20.9.0
@ -14111,6 +14126,8 @@ snapshots:
eventemitter3@4.0.7: {}
eventsource@2.0.2: {}
execa@5.1.1:
dependencies:
cross-spawn: 7.0.3