Skip to content

Commit

Permalink
Revert "Upgrade webR to v0.2.2 and load R Shiny package from filesyst…
Browse files Browse the repository at this point in the history
…em image (#80)"

This reverts commit 7cdeabe.
  • Loading branch information
wch committed Nov 27, 2023
1 parent 0aae729 commit 1cd8f03
Show file tree
Hide file tree
Showing 6 changed files with 1,876 additions and 1,206 deletions.
5 changes: 1 addition & 4 deletions Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,6 @@ SHINYLIVE_VERSION = $(shell node -p "require('./package.json').version")

PYODIDE_VERSION = 0.22.1
PYODIDE_DIST_FILENAME = pyodide-$(PYODIDE_VERSION).tar.bz2
R_SHINY_VERSION = 1.8.0.9000-webr
BUILD_DIR = ./build
PACKAGE_DIR = ./packages
DIST_DIR = ./dist
Expand Down Expand Up @@ -128,15 +127,13 @@ $(BUILD_DIR)/shinylive/style-resets.css: src/style-resets.css
$(BUILD_DIR)/shinylive/pyodide:
mkdir -p $(BUILD_DIR)/shinylive/pyodide
cd $(BUILD_DIR)/shinylive && \
curl --fail -L https://github.com/pyodide/pyodide/releases/download/$(PYODIDE_VERSION)/$(PYODIDE_DIST_FILENAME) \
curl -L https://github.com/pyodide/pyodide/releases/download/$(PYODIDE_VERSION)/$(PYODIDE_DIST_FILENAME) \
| tar --exclude "*test*.tar" --exclude "node_modules" -xvj

$(BUILD_DIR)/shinylive/webr: webr
webr:
mkdir -p $(BUILD_DIR)/shinylive/webr
cp -r node_modules/webr/dist/. $(BUILD_DIR)/shinylive/webr
curl --fail -L https://github.com/r-wasm/shiny/releases/download/v$(R_SHINY_VERSION)/library.data -o $(BUILD_DIR)/shinylive/webr/library.data
curl --fail -L https://github.com/r-wasm/shiny/releases/download/v$(R_SHINY_VERSION)/library.js.metadata -o $(BUILD_DIR)/shinylive/webr/library.js.metadata

# Copy pyodide.js and .d.ts to src/pyodide/. This is a little weird in that in
# `make all`, it comes after downloading pyodide. In the future we may be able
Expand Down
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -72,7 +72,7 @@
"tsx": "^3.12.7",
"typescript": "^5.1.3",
"vscode-languageserver-protocol": "^3.17.3",
"webr": "^0.2.2",
"webr": "^0.2.1",
"xterm": "^5.2.1",
"xterm-addon-fit": "^0.7.0",
"xterm-readline": "^1.1.1"
Expand Down
18 changes: 5 additions & 13 deletions src/hooks/useWebR.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -42,13 +42,12 @@ export async function initWebR({
channelType,
},
stdout,
stderr,
stderr
);

let initError = false;
try {
const libraryUrl = utils.currentScriptDir() + "/webr/library.data";
await webRProxy.runRAsync(`webr::mount("/shiny", "${libraryUrl}")`);
await webRProxy.runRAsync('webr::install("codetools")');
await webRProxy.runRAsync(load_r_pre);
} catch (e) {
initError = true;
Expand Down Expand Up @@ -88,6 +87,8 @@ export async function initRShiny({
throw new Error("webRProxyHandle is not ready");
}

await webRProxyHandle.webRProxy.runRAsync('webr::install("renv")');
await webRProxyHandle.webRProxy.runRAsync('webr::install("shiny")');
await webRProxyHandle.webRProxy.runRAsync("library(shiny)");
// Increase webR expressions limit for deep call stack required for Shiny
await webRProxyHandle.webRProxy.runRAsync("options(expressions=1000)");
Expand All @@ -109,7 +110,7 @@ export function useWebR({
ready: false,
shinyReady: false,
initError: false,
},
}
);

useEffect(() => {
Expand Down Expand Up @@ -143,15 +144,6 @@ function ensureOpenChannelListener(webRProxy: WebRProxy): void {
}

const load_r_pre = `
# Force internal tar - silence renv warning
Sys.setenv(TAR = "internal")
# Use mounted shiny R package library
.libPaths(c(.libPaths(), "/shiny"))
# Shim R functions with webR versions (e.g. install.packages())
webr::shim_install()
.shiny_app_registry <- new.env()
# Create a httpuv app from a Shiny app directory
Expand Down
41 changes: 19 additions & 22 deletions src/messageporthttp.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,8 +13,8 @@ export async function fetchASGI(
resource: RequestInfo,
init?: RequestInit,
filter: (bodyChunk: Uint8Array, response: Response) => Uint8Array = (
bodyChunk,
) => bodyChunk,
bodyChunk
) => bodyChunk
): Promise<Response> {
if (typeof resource === "string" || typeof init !== "undefined") {
resource = new Request(resource, init);
Expand All @@ -27,7 +27,7 @@ export async function fetchASGI(
type: "makeRequest",
scope: reqToASGI(resource),
},
[channel.port2],
[channel.port2]
);

const blob = await resource.blob();
Expand Down Expand Up @@ -98,21 +98,21 @@ export async function makeRequest(
scope: ASGIHTTPRequestScope,
appName: string,
clientPort: MessagePort,
pyodide: Pyodide,
pyodide: Pyodide
) {
// We could _almost_ use app(), but unfortunately pyodide's implicit proxying
// behavior isn't compatible with ASGI (which wants dict, not JsProxy); we
// need to explicitly convert stuff first, which is what call_pyodide does.
const asgiFunc = pyodide.runPython(
`_shiny_app_registry["${appName}"].app.call_pyodide`,
`_shiny_app_registry["${appName}"].app.call_pyodide`
) as PyProxyCallable;
await connect(scope, clientPort, asgiFunc);
}

async function connect(
scope: ASGIHTTPRequestScope,
clientPort: MessagePort,
asgiFunc: PyProxyCallable,
asgiFunc: PyProxyCallable
) {
const fromClientQueue = new AwaitableQueue<Record<string, any>>();

Expand Down Expand Up @@ -230,7 +230,7 @@ export async function makeHttpuvRequest(
scope: ASGIHTTPRequestScope,
appName: string,
clientPort: MessagePort,
webRProxy: WebRProxy,
webRProxy: WebRProxy
) {
const fromClientQueue = new AwaitableQueue<Record<string, any>>();

Expand Down Expand Up @@ -265,8 +265,8 @@ export async function makeHttpuvRequest(
Object.fromEntries(
[...Array(event.headers.names.length).keys()].map((i) => {
return [event.headers.names[i], event.headers.values[i].values[0]];
}),
),
})
)
);

clientPort.postMessage({
Expand All @@ -289,7 +289,7 @@ async function handleHttpuvRequests(
appName: string,
webRProxy: WebRProxy,
fromClient: () => Promise<Record<string, any>>,
toClient: (event: Record<string, any>) => Promise<void>,
toClient: (event: Record<string, any>) => Promise<void>
) {
let body = new Uint8Array(0);
const shelter = await new webRProxy.webR.Shelter();
Expand Down Expand Up @@ -318,29 +318,26 @@ async function handleHttpuvRequests(
tryCatch(
{
app <- get(appName, env = .shiny_app_registry)
if (!is.null(app)) {
app$call(
list(
PATH_INFO = "${scope.path}",
REQUEST_METHOD = "${scope.method}",
QUERY_STRING = "${scope.query_string}",
rook.input = reader
)
app$call(
list(
PATH_INFO = "${scope.path}",
REQUEST_METHOD = "${scope.method}",
QUERY_STRING = "${scope.query_string}",
rook.input = reader
)
}
)
},
finally = {
reader$destroy()
}
)
`,
{ env, captureConditions: false, captureStreams: false },
{ env, captureConditions: false, captureStreams: false }
);

if (!isRList(httpuvResp)) {
const type = await httpuvResp.type();
throw new Error(
`Unexpected response type: "${type}", expected "list".`,
`Unexpected response type: "${httpuvResp.type()}", expected "list".`
);
}

Expand Down
13 changes: 6 additions & 7 deletions src/webr-proxy.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,13 +16,13 @@ export interface WebRProxy {
openChannel(
path: string,
appName: string,
clientPort: MessagePort,
clientPort: MessagePort
): Promise<void>;

makeRequest(
scope: ASGIHTTPRequestScope,
appName: string,
clientPort: MessagePort,
clientPort: MessagePort
): Promise<void>;
}

Expand All @@ -38,7 +38,7 @@ class WebRWorkerProxy implements WebRProxy {
constructor(
config: WebROptions,
private stdoutCallback: (text: string) => void,
private stderrCallback: (text: string) => void,
private stderrCallback: (text: string) => void
) {
this.webR = new WebR(config);
}
Expand Down Expand Up @@ -67,7 +67,6 @@ class WebRWorkerProxy implements WebRProxy {
return await this.shelter.evalR(code, options);
} catch (e) {
this.stderrCallback((e as Error).message);
throw e;
} finally {
await this.shelter.purge();
}
Expand Down Expand Up @@ -112,15 +111,15 @@ class WebRWorkerProxy implements WebRProxy {
async openChannel(
path: string,
appName: string,
clientPort: MessagePort,
clientPort: MessagePort
): Promise<void> {
await openChannelHttpuv(path, appName, clientPort, this);
}

async makeRequest(
scope: ASGIHTTPRequestScope,
appName: string,
clientPort: MessagePort,
clientPort: MessagePort
): Promise<void> {
await makeHttpuvRequest(scope, appName, clientPort, this);
}
Expand All @@ -129,7 +128,7 @@ class WebRWorkerProxy implements WebRProxy {
export async function loadWebRProxy(
config: WebROptions,
stdoutCallback: (text: string) => void = console.log,
stderrCallback: (text: string) => void = console.error,
stderrCallback: (text: string) => void = console.error
): Promise<WebRProxy> {
const webRProxy = new WebRWorkerProxy(config, stdoutCallback, stderrCallback);
await webRProxy.webR.init();
Expand Down
Loading

0 comments on commit 1cd8f03

Please sign in to comment.