mirror of
https://github.com/gradio-app/gradio.git
synced 2025-03-31 12:20:26 +08:00
Make the HTTP requests for the Wasm worker wait for the initial run_code()
or run_file()
to finish (#5958)
* Make the HTTP requests for the Wasm worker wait for the initial `run_code()` or `run_file()` to finish * add changeset * Support top-level await with `run_file()` --------- Co-authored-by: gradio-pr-bot <gradio-pr-bot@users.noreply.github.com> Co-authored-by: Abubakar Abid <abubakar@huggingface.co>
This commit is contained in:
parent
f769876e0f
commit
6780d660bb
5
.changeset/crazy-dancers-allow.md
Normal file
5
.changeset/crazy-dancers-allow.md
Normal file
@ -0,0 +1,5 @@
|
||||
---
|
||||
"@gradio/wasm": minor
|
||||
---
|
||||
|
||||
feat:Make the HTTP requests for the Wasm worker wait for the initial `run_code()` or `run_file()` to finish
|
26
js/wasm/src/promise-delegate.ts
Normal file
26
js/wasm/src/promise-delegate.ts
Normal file
@ -0,0 +1,26 @@
|
||||
type PromiseImplFn<T> = ConstructorParameters<typeof Promise<T>>[0];
|
||||
|
||||
export class PromiseDelegate<T> {
|
||||
private promiseInternal: Promise<T>;
|
||||
private resolveInternal!: Parameters<PromiseImplFn<T>>[0];
|
||||
private rejectInternal!: Parameters<PromiseImplFn<T>>[1];
|
||||
|
||||
constructor() {
|
||||
this.promiseInternal = new Promise((resolve, reject) => {
|
||||
this.resolveInternal = resolve;
|
||||
this.rejectInternal = reject;
|
||||
});
|
||||
}
|
||||
|
||||
get promise(): Promise<T> {
|
||||
return this.promiseInternal;
|
||||
}
|
||||
|
||||
public resolve(value: T): void {
|
||||
this.resolveInternal(value);
|
||||
}
|
||||
|
||||
public reject(reason: unknown): void {
|
||||
this.rejectInternal(reason);
|
||||
}
|
||||
}
|
@ -26,7 +26,7 @@ let call_asgi_app_from_js: (
|
||||
receive: () => Promise<unknown>,
|
||||
send: (event: any) => Promise<void>
|
||||
) => Promise<void>;
|
||||
let run_script: (path: string) => void;
|
||||
let run_script: (path: string) => Promise<void>;
|
||||
let unload_local_modules: (target_dir_path?: string) => void;
|
||||
|
||||
async function loadPyodideAndPackages(
|
||||
@ -218,7 +218,7 @@ self.onmessage = async (event: MessageEvent<InMessage>): Promise<void> => {
|
||||
case "run-python-file": {
|
||||
unload_local_modules();
|
||||
|
||||
run_script(msg.data.path);
|
||||
await run_script(msg.data.path);
|
||||
|
||||
const replyMessage: ReplyMessageSuccess = {
|
||||
type: "reply:success",
|
||||
|
@ -1,6 +1,8 @@
|
||||
import ast
|
||||
import tokenize
|
||||
import types
|
||||
import sys
|
||||
from inspect import CO_COROUTINE
|
||||
|
||||
# BSD 3-Clause License
|
||||
#
|
||||
@ -63,6 +65,7 @@ class modified_sys_path:
|
||||
|
||||
|
||||
# Copyright (c) Streamlit Inc. (2018-2022) Snowflake Inc. (2022)
|
||||
# Copyright (c) Yuichiro Tachibana (2023)
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||
# you may not use this file except in compliance with the License.
|
||||
@ -80,9 +83,10 @@ def _new_module(name: str) -> types.ModuleType:
|
||||
return types.ModuleType(name)
|
||||
|
||||
|
||||
def _run_script(script_path: str) -> None:
|
||||
async def _run_script(script_path: str) -> None:
|
||||
# This function is based on the following code from Streamlit:
|
||||
# https://github.com/streamlit/streamlit/blob/1.24.0/lib/streamlit/runtime/scriptrunner/script_runner.py#L519-L554
|
||||
# with modifications to support top-level await.
|
||||
|
||||
with tokenize.open(script_path) as f:
|
||||
filebody = f.read()
|
||||
@ -98,7 +102,7 @@ def _run_script(script_path: str) -> None:
|
||||
# mode (as opposed to "eval" or "single").
|
||||
mode="exec",
|
||||
# Don't inherit any flags or "future" statements.
|
||||
flags=0,
|
||||
flags=ast.PyCF_ALLOW_TOP_LEVEL_AWAIT, # Allow top-level await. Ref: https://github.com/whitphx/streamlit/commit/277dc580efb315a3e9296c9a0078c602a0904384
|
||||
dont_inherit=1,
|
||||
# Use the default optimization options.
|
||||
optimize=-1,
|
||||
@ -117,4 +121,9 @@ def _run_script(script_path: str) -> None:
|
||||
module.__dict__["__file__"] = script_path
|
||||
|
||||
with modified_sys_path(script_path):
|
||||
exec(bytecode, module.__dict__)
|
||||
# Allow top-level await. Ref: https://github.com/whitphx/streamlit/commit/277dc580efb315a3e9296c9a0078c602a0904384
|
||||
if bytecode.co_flags & CO_COROUTINE:
|
||||
# The source code includes top-level awaits, so the compiled code object is a coroutine.
|
||||
await eval(bytecode, module.__dict__)
|
||||
else:
|
||||
exec(bytecode, module.__dict__)
|
||||
|
@ -9,6 +9,7 @@ import type {
|
||||
ReplyMessage
|
||||
} from "./message-types";
|
||||
import { MessagePortWebSocket } from "./messageportwebsocket";
|
||||
import { PromiseDelegate } from "./promise-delegate";
|
||||
|
||||
export interface WorkerProxyOptions {
|
||||
gradioWheelUrl: string;
|
||||
@ -20,6 +21,8 @@ export interface WorkerProxyOptions {
|
||||
export class WorkerProxy {
|
||||
private worker: globalThis.Worker;
|
||||
|
||||
private firstRunPromiseDelegate = new PromiseDelegate<void>();
|
||||
|
||||
constructor(options: WorkerProxyOptions) {
|
||||
console.debug("WorkerProxy.constructor(): Create a new worker.");
|
||||
// Loading a worker here relies on Vite's support for WebWorkers (https://vitejs.dev/guide/features.html#web-workers),
|
||||
@ -49,6 +52,7 @@ export class WorkerProxy {
|
||||
code
|
||||
}
|
||||
});
|
||||
this.firstRunPromiseDelegate.resolve();
|
||||
}
|
||||
|
||||
public async runPythonFile(path: string): Promise<void> {
|
||||
@ -58,6 +62,7 @@ export class WorkerProxy {
|
||||
path
|
||||
}
|
||||
});
|
||||
this.firstRunPromiseDelegate.resolve();
|
||||
}
|
||||
|
||||
// A wrapper for this.worker.postMessage(). Unlike that function, which
|
||||
@ -84,6 +89,12 @@ export class WorkerProxy {
|
||||
}
|
||||
|
||||
public async httpRequest(request: HttpRequest): Promise<HttpResponse> {
|
||||
// Wait for the first run to be done
|
||||
// to avoid the "Gradio app has not been launched." error
|
||||
// in case running the code takes long time.
|
||||
// Ref: https://github.com/gradio-app/gradio/issues/5957
|
||||
await this.firstRunPromiseDelegate.promise;
|
||||
|
||||
console.debug("WorkerProxy.httpRequest()", request);
|
||||
const result = await this.postMessageAsync({
|
||||
type: "http-request",
|
||||
|
Loading…
x
Reference in New Issue
Block a user