@marimo-team/islands 0.23.7-dev55 → 0.23.7-dev57
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/{chat-ui-DCyW3OUK.js → chat-ui-D3XBept8.js} +3 -3
- package/dist/{code-visibility-CJ7U5FE0.js → code-visibility-PjV7HUDZ.js} +10624 -1451
- package/dist/{formats-CpgZM9BM.js → formats-Dsy9kkZu.js} +1 -1
- package/dist/{html-to-image-40ZXSWP-.js → html-to-image-CpggM7u1.js} +1 -1
- package/dist/main.js +1353 -9989
- package/dist/{process-output-CCeeXIBd.js → process-output-X8TR20AK.js} +1 -1
- package/dist/{reveal-component-Bopa1DsA.js → reveal-component-Phd-LTXq.js} +3 -3
- package/dist/{toDate-CJWlVNGD.js → toDate-CIpC_34u.js} +30 -17
- package/dist/{vega-component-BtvQ-Kc4.js → vega-component-cSdqoAxe.js} +2 -2
- package/package.json +1 -1
- package/src/components/data-table/__tests__/column-header.test.tsx +106 -1
- package/src/components/data-table/__tests__/filter-pill-editor.test.tsx +88 -2
- package/src/components/data-table/__tests__/filters.test.ts +84 -13
- package/src/components/data-table/column-header.tsx +152 -26
- package/src/components/data-table/date-filter-inputs.tsx +325 -0
- package/src/components/data-table/filter-pill-editor.tsx +139 -30
- package/src/components/data-table/filter-pills.tsx +31 -57
- package/src/components/data-table/filters.ts +88 -66
- package/src/components/editor/chrome/wrapper/footer-items/backend-status.tsx +1 -1
- package/src/core/runtime/__tests__/runtime.test.ts +38 -17
- package/src/core/runtime/runtime.ts +57 -34
- package/src/core/websocket/__tests__/useMarimoKernelConnection.hook.test.tsx +5 -4
- package/src/core/websocket/__tests__/useMarimoKernelConnection.test.ts +18 -54
- package/src/core/websocket/transports/__tests__/ws.test.ts +125 -0
- package/src/core/websocket/transports/basic.ts +1 -3
- package/src/core/websocket/transports/transport.ts +0 -1
- package/src/core/websocket/transports/ws.ts +96 -0
- package/src/core/websocket/useMarimoKernelConnection.tsx +30 -26
- package/src/core/websocket/useWebSocket.tsx +3 -18
|
@@ -4,50 +4,22 @@ import { afterEach, describe, expect, it, vi } from "vitest";
|
|
|
4
4
|
import { Logger } from "@/utils/Logger";
|
|
5
5
|
import { WebSocketClosedReason, WebSocketState } from "../types";
|
|
6
6
|
import { classifyCloseEvent } from "../useMarimoKernelConnection";
|
|
7
|
-
import { MAX_RETRIES } from "../useWebSocket";
|
|
8
7
|
|
|
9
|
-
function classify(
|
|
10
|
-
|
|
11
|
-
retryCount = 0,
|
|
12
|
-
maxRetries = MAX_RETRIES,
|
|
13
|
-
) {
|
|
14
|
-
return classifyCloseEvent({ reason }, { retryCount, maxRetries });
|
|
8
|
+
function classify(reason: string | undefined) {
|
|
9
|
+
return classifyCloseEvent({ reason });
|
|
15
10
|
}
|
|
16
11
|
|
|
17
12
|
describe("classifyCloseEvent", () => {
|
|
18
13
|
describe("transient closes (default branch)", () => {
|
|
19
|
-
it("retries
|
|
20
|
-
const decision = classify(undefined
|
|
14
|
+
it("retries on empty/undefined reason", () => {
|
|
15
|
+
const decision = classify(undefined);
|
|
21
16
|
expect(decision.kind).toBe("retry");
|
|
22
17
|
expect(decision.status).toEqual({ state: WebSocketState.CONNECTING });
|
|
23
18
|
});
|
|
24
19
|
|
|
25
|
-
it("retries on each intermediate close event during a retry storm", () => {
|
|
26
|
-
for (let n = 0; n < MAX_RETRIES; n++) {
|
|
27
|
-
const decision = classify(undefined, n);
|
|
28
|
-
expect(decision.kind).toBe("retry");
|
|
29
|
-
expect(decision.status).toEqual({ state: WebSocketState.CONNECTING });
|
|
30
|
-
}
|
|
31
|
-
});
|
|
32
|
-
|
|
33
|
-
it("transitions to CLOSED when retryCount reaches maxRetries", () => {
|
|
34
|
-
const decision = classify(undefined, MAX_RETRIES);
|
|
35
|
-
expect(decision.kind).toBe("gave-up");
|
|
36
|
-
expect(decision.status).toEqual({
|
|
37
|
-
state: WebSocketState.CLOSED,
|
|
38
|
-
code: WebSocketClosedReason.KERNEL_DISCONNECTED,
|
|
39
|
-
reason: "kernel not found",
|
|
40
|
-
});
|
|
41
|
-
});
|
|
42
|
-
|
|
43
|
-
it("transitions to CLOSED when retryCount exceeds maxRetries", () => {
|
|
44
|
-
const decision = classify(undefined, MAX_RETRIES + 5);
|
|
45
|
-
expect(decision.kind).toBe("gave-up");
|
|
46
|
-
});
|
|
47
|
-
|
|
48
20
|
it("treats unknown reason strings as transient and logs a warning", () => {
|
|
49
21
|
const logger = vi.spyOn(Logger, "warn").mockImplementation(() => {});
|
|
50
|
-
const decision = classify("something-else"
|
|
22
|
+
const decision = classify("something-else");
|
|
51
23
|
expect(decision.kind).toBe("retry");
|
|
52
24
|
expect(logger).toHaveBeenCalled();
|
|
53
25
|
logger.mockRestore();
|
|
@@ -60,7 +32,7 @@ describe("classifyCloseEvent", () => {
|
|
|
60
32
|
|
|
61
33
|
describe("terminal closes (server-initiated)", () => {
|
|
62
34
|
it("MARIMO_ALREADY_CONNECTED → terminal + closeTransport, with takeover", () => {
|
|
63
|
-
const decision = classify("MARIMO_ALREADY_CONNECTED"
|
|
35
|
+
const decision = classify("MARIMO_ALREADY_CONNECTED");
|
|
64
36
|
expect(decision.kind).toBe("terminal");
|
|
65
37
|
expect(decision.status).toMatchObject({
|
|
66
38
|
state: WebSocketState.CLOSED,
|
|
@@ -79,7 +51,7 @@ describe("classifyCloseEvent", () => {
|
|
|
79
51
|
"MARIMO_NO_SESSION",
|
|
80
52
|
"MARIMO_SHUTDOWN",
|
|
81
53
|
])("%s → terminal with KERNEL_DISCONNECTED, closes transport", (reason) => {
|
|
82
|
-
const decision = classify(reason
|
|
54
|
+
const decision = classify(reason);
|
|
83
55
|
expect(decision.kind).toBe("terminal");
|
|
84
56
|
expect(decision.status).toMatchObject({
|
|
85
57
|
state: WebSocketState.CLOSED,
|
|
@@ -91,7 +63,7 @@ describe("classifyCloseEvent", () => {
|
|
|
91
63
|
});
|
|
92
64
|
|
|
93
65
|
it("MARIMO_MALFORMED_QUERY → terminal but does NOT close transport", () => {
|
|
94
|
-
const decision = classify("MARIMO_MALFORMED_QUERY"
|
|
66
|
+
const decision = classify("MARIMO_MALFORMED_QUERY");
|
|
95
67
|
expect(decision.kind).toBe("terminal");
|
|
96
68
|
expect(decision.status).toMatchObject({
|
|
97
69
|
state: WebSocketState.CLOSED,
|
|
@@ -103,7 +75,7 @@ describe("classifyCloseEvent", () => {
|
|
|
103
75
|
});
|
|
104
76
|
|
|
105
77
|
it("MARIMO_KERNEL_STARTUP_ERROR → terminal + closeTransport", () => {
|
|
106
|
-
const decision = classify("MARIMO_KERNEL_STARTUP_ERROR"
|
|
78
|
+
const decision = classify("MARIMO_KERNEL_STARTUP_ERROR");
|
|
107
79
|
expect(decision.kind).toBe("terminal");
|
|
108
80
|
expect(decision.status).toMatchObject({
|
|
109
81
|
state: WebSocketState.CLOSED,
|
|
@@ -113,25 +85,17 @@ describe("classifyCloseEvent", () => {
|
|
|
113
85
|
expect(decision.closeTransport).toBe(true);
|
|
114
86
|
}
|
|
115
87
|
});
|
|
116
|
-
|
|
117
|
-
it("terminal closes ignore retryCount entirely", () => {
|
|
118
|
-
const decision = classify("MARIMO_SHUTDOWN", 99);
|
|
119
|
-
expect(decision.kind).toBe("terminal");
|
|
120
|
-
});
|
|
121
88
|
});
|
|
122
89
|
|
|
123
|
-
describe("
|
|
124
|
-
it("
|
|
125
|
-
const
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
...Array.from({ length: MAX_RETRIES }, () => "retry"),
|
|
133
|
-
"gave-up",
|
|
134
|
-
]);
|
|
90
|
+
describe("transport exhaustion", () => {
|
|
91
|
+
it("MARIMO_TRANSPORT_EXHAUSTED → gave-up with KERNEL_DISCONNECTED", () => {
|
|
92
|
+
const decision = classify("MARIMO_TRANSPORT_EXHAUSTED");
|
|
93
|
+
expect(decision.kind).toBe("gave-up");
|
|
94
|
+
expect(decision.status).toEqual({
|
|
95
|
+
state: WebSocketState.CLOSED,
|
|
96
|
+
code: WebSocketClosedReason.KERNEL_DISCONNECTED,
|
|
97
|
+
reason: "kernel not found",
|
|
98
|
+
});
|
|
135
99
|
});
|
|
136
100
|
});
|
|
137
101
|
});
|
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
/* Copyright 2026 Marimo. All rights reserved. */
|
|
2
|
+
|
|
3
|
+
import { beforeEach, describe, expect, it, vi } from "vitest";
|
|
4
|
+
import { MAX_RETRIES, WsTransport, TRANSPORT_EXHAUSTED_REASON } from "../ws";
|
|
5
|
+
|
|
6
|
+
let innerListeners: Record<string, ((e: unknown) => void)[]>;
|
|
7
|
+
|
|
8
|
+
vi.mock("partysocket/ws", () => {
|
|
9
|
+
class FakeReconnectingWebSocket {
|
|
10
|
+
retryCount = 0;
|
|
11
|
+
readyState = WebSocket.CONNECTING;
|
|
12
|
+
constructor() {
|
|
13
|
+
innerListeners = { open: [], close: [], message: [], error: [] };
|
|
14
|
+
}
|
|
15
|
+
addEventListener(event: string, cb: (e: unknown) => void) {
|
|
16
|
+
innerListeners[event].push(cb);
|
|
17
|
+
}
|
|
18
|
+
removeEventListener(event: string, cb: (e: unknown) => void) {
|
|
19
|
+
innerListeners[event] = innerListeners[event].filter((c) => c !== cb);
|
|
20
|
+
}
|
|
21
|
+
reconnect() {}
|
|
22
|
+
close() {}
|
|
23
|
+
send() {}
|
|
24
|
+
}
|
|
25
|
+
return { default: FakeReconnectingWebSocket };
|
|
26
|
+
});
|
|
27
|
+
|
|
28
|
+
interface FakeReconnectingWebSocket {
|
|
29
|
+
retryCount: number;
|
|
30
|
+
readyState: number;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
function dispatchClose(reason = "") {
|
|
34
|
+
const evt = new CloseEvent("close", { reason, code: 1006 });
|
|
35
|
+
for (const cb of innerListeners.close) {
|
|
36
|
+
cb(evt);
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
describe("WsTransport", () => {
|
|
41
|
+
let transport: WsTransport;
|
|
42
|
+
let inner: FakeReconnectingWebSocket;
|
|
43
|
+
|
|
44
|
+
beforeEach(() => {
|
|
45
|
+
transport = new WsTransport(() => "ws://example.invalid/ws");
|
|
46
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
47
|
+
inner = (transport as any).inner;
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
describe("close event reason", () => {
|
|
51
|
+
it("forwards the original reason before retry budget is exhausted", () => {
|
|
52
|
+
const seen: CloseEvent[] = [];
|
|
53
|
+
transport.addEventListener("close", (e) => seen.push(e));
|
|
54
|
+
|
|
55
|
+
inner.retryCount = MAX_RETRIES - 1;
|
|
56
|
+
dispatchClose("");
|
|
57
|
+
|
|
58
|
+
expect(seen).toHaveLength(1);
|
|
59
|
+
expect(seen[0].reason).toBe("");
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
it("rewrites reason to MARIMO_TRANSPORT_EXHAUSTED on exhaustion", () => {
|
|
63
|
+
const seen: CloseEvent[] = [];
|
|
64
|
+
transport.addEventListener("close", (e) => seen.push(e));
|
|
65
|
+
|
|
66
|
+
inner.retryCount = MAX_RETRIES;
|
|
67
|
+
dispatchClose("");
|
|
68
|
+
|
|
69
|
+
expect(seen).toHaveLength(1);
|
|
70
|
+
expect(seen[0].reason).toBe(TRANSPORT_EXHAUSTED_REASON);
|
|
71
|
+
});
|
|
72
|
+
|
|
73
|
+
it("does not rewrite a server-sent reason on exhaustion", () => {
|
|
74
|
+
// If partysocket happens to deliver a real MARIMO_* reason at the same
|
|
75
|
+
// moment the retry budget exhausts, the exhausted-state rewrite still
|
|
76
|
+
// wins because the wrapper keys off retryCount, not the original reason.
|
|
77
|
+
// Document the behavior so a future change is deliberate.
|
|
78
|
+
const seen: CloseEvent[] = [];
|
|
79
|
+
transport.addEventListener("close", (e) => seen.push(e));
|
|
80
|
+
|
|
81
|
+
inner.retryCount = MAX_RETRIES;
|
|
82
|
+
dispatchClose("MARIMO_SHUTDOWN");
|
|
83
|
+
|
|
84
|
+
expect(seen[0].reason).toBe(TRANSPORT_EXHAUSTED_REASON);
|
|
85
|
+
});
|
|
86
|
+
});
|
|
87
|
+
|
|
88
|
+
describe("addEventListener dedupe", () => {
|
|
89
|
+
it("does not double-fire when the same close listener is added twice", () => {
|
|
90
|
+
const cb = vi.fn();
|
|
91
|
+
transport.addEventListener("close", cb);
|
|
92
|
+
transport.addEventListener("close", cb);
|
|
93
|
+
|
|
94
|
+
dispatchClose("");
|
|
95
|
+
|
|
96
|
+
expect(cb).toHaveBeenCalledTimes(1);
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
it("a single removeEventListener fully unregisters a duplicated add", () => {
|
|
100
|
+
const cb = vi.fn();
|
|
101
|
+
transport.addEventListener("close", cb);
|
|
102
|
+
transport.addEventListener("close", cb);
|
|
103
|
+
transport.removeEventListener("close", cb);
|
|
104
|
+
|
|
105
|
+
dispatchClose("");
|
|
106
|
+
|
|
107
|
+
expect(cb).not.toHaveBeenCalled();
|
|
108
|
+
// Inner socket has no orphaned wrappers left.
|
|
109
|
+
expect(innerListeners.close).toHaveLength(0);
|
|
110
|
+
});
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
describe("removeEventListener", () => {
|
|
114
|
+
it("unregisters the right wrapper for close listeners", () => {
|
|
115
|
+
const cb = vi.fn();
|
|
116
|
+
transport.addEventListener("close", cb);
|
|
117
|
+
transport.removeEventListener("close", cb);
|
|
118
|
+
|
|
119
|
+
inner.retryCount = MAX_RETRIES;
|
|
120
|
+
dispatchClose("");
|
|
121
|
+
|
|
122
|
+
expect(cb).not.toHaveBeenCalled();
|
|
123
|
+
});
|
|
124
|
+
});
|
|
125
|
+
});
|
|
@@ -43,9 +43,7 @@ export class BasicTransport implements IConnectionTransport {
|
|
|
43
43
|
return WebSocket.OPEN;
|
|
44
44
|
}
|
|
45
45
|
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
reconnect(code?: number | undefined, reason?: string | undefined): void {
|
|
46
|
+
reconnect(_code?: number | undefined, _reason?: string | undefined): void {
|
|
49
47
|
this.close();
|
|
50
48
|
this.connect();
|
|
51
49
|
return;
|
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
/* Copyright 2026 Marimo. All rights reserved. */
|
|
2
|
+
|
|
3
|
+
import ReconnectingWebSocket from "partysocket/ws";
|
|
4
|
+
import type {
|
|
5
|
+
ConnectionEvent,
|
|
6
|
+
ConnectionTransportCallback,
|
|
7
|
+
IConnectionTransport,
|
|
8
|
+
} from "./transport";
|
|
9
|
+
|
|
10
|
+
// Per-`reconnect()` retry budget. After exhaustion, partysocket stops silently;
|
|
11
|
+
// the wrapper rewrites the close-event reason to surface the give-up.
|
|
12
|
+
export const MAX_RETRIES = 10;
|
|
13
|
+
|
|
14
|
+
export const TRANSPORT_EXHAUSTED_REASON = "MARIMO_TRANSPORT_EXHAUSTED";
|
|
15
|
+
|
|
16
|
+
export class WsTransport implements IConnectionTransport {
|
|
17
|
+
private inner: ReconnectingWebSocket;
|
|
18
|
+
private closeWrappers = new WeakMap<
|
|
19
|
+
ConnectionTransportCallback<"close">,
|
|
20
|
+
ConnectionTransportCallback<"close">
|
|
21
|
+
>();
|
|
22
|
+
|
|
23
|
+
constructor(urlProvider: () => string) {
|
|
24
|
+
this.inner = new ReconnectingWebSocket(urlProvider, undefined, {
|
|
25
|
+
maxRetries: MAX_RETRIES,
|
|
26
|
+
debug: false,
|
|
27
|
+
startClosed: true,
|
|
28
|
+
// long timeout — the server can become slow when many notebooks are open.
|
|
29
|
+
connectionTimeout: 10_000,
|
|
30
|
+
});
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
get readyState(): WebSocket["readyState"] {
|
|
34
|
+
return this.inner.readyState as WebSocket["readyState"];
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
reconnect(code?: number, reason?: string): void {
|
|
38
|
+
this.inner.reconnect(code, reason);
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
close(): void {
|
|
42
|
+
this.inner.close();
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
send(data: string | ArrayBuffer | Blob | ArrayBufferView): void {
|
|
46
|
+
this.inner.send(data);
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
addEventListener<T extends ConnectionEvent>(
|
|
50
|
+
event: T,
|
|
51
|
+
callback: ConnectionTransportCallback<T>,
|
|
52
|
+
): void {
|
|
53
|
+
if (event === "close") {
|
|
54
|
+
const userCb = callback as ConnectionTransportCallback<"close">;
|
|
55
|
+
// Match native EventTarget dedupe: a second addEventListener with the
|
|
56
|
+
// same listener is a no-op. Without this, repeated adds leak wrappers
|
|
57
|
+
// on the inner socket and double-fire on close.
|
|
58
|
+
if (this.closeWrappers.has(userCb)) {
|
|
59
|
+
return;
|
|
60
|
+
}
|
|
61
|
+
const wrapper: ConnectionTransportCallback<"close"> = (e) => {
|
|
62
|
+
if (this.inner.retryCount >= MAX_RETRIES) {
|
|
63
|
+
userCb(
|
|
64
|
+
new CloseEvent("close", {
|
|
65
|
+
code: e.code,
|
|
66
|
+
reason: TRANSPORT_EXHAUSTED_REASON,
|
|
67
|
+
wasClean: e.wasClean,
|
|
68
|
+
}),
|
|
69
|
+
);
|
|
70
|
+
} else {
|
|
71
|
+
userCb(e);
|
|
72
|
+
}
|
|
73
|
+
};
|
|
74
|
+
this.closeWrappers.set(userCb, wrapper);
|
|
75
|
+
this.inner.addEventListener("close", wrapper);
|
|
76
|
+
return;
|
|
77
|
+
}
|
|
78
|
+
this.inner.addEventListener(event, callback as never);
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
removeEventListener<T extends ConnectionEvent>(
|
|
82
|
+
event: T,
|
|
83
|
+
callback: ConnectionTransportCallback<T>,
|
|
84
|
+
): void {
|
|
85
|
+
if (event === "close") {
|
|
86
|
+
const userCb = callback as ConnectionTransportCallback<"close">;
|
|
87
|
+
const wrapper = this.closeWrappers.get(userCb);
|
|
88
|
+
if (wrapper) {
|
|
89
|
+
this.closeWrappers.delete(userCb);
|
|
90
|
+
this.inner.removeEventListener("close", wrapper);
|
|
91
|
+
}
|
|
92
|
+
return;
|
|
93
|
+
}
|
|
94
|
+
this.inner.removeEventListener(event, callback as never);
|
|
95
|
+
}
|
|
96
|
+
}
|
|
@@ -11,10 +11,8 @@ import type {
|
|
|
11
11
|
NotificationMessageData,
|
|
12
12
|
NotificationPayload,
|
|
13
13
|
} from "@/core/kernel/messages";
|
|
14
|
-
import {
|
|
15
|
-
|
|
16
|
-
useConnectionTransport,
|
|
17
|
-
} from "@/core/websocket/useWebSocket";
|
|
14
|
+
import { TRANSPORT_EXHAUSTED_REASON } from "@/core/websocket/transports/ws";
|
|
15
|
+
import { useConnectionTransport } from "@/core/websocket/useWebSocket";
|
|
18
16
|
import { renderHTML } from "@/plugins/core/RenderHTML";
|
|
19
17
|
import {
|
|
20
18
|
handleWidgetMessage,
|
|
@@ -80,16 +78,27 @@ import {
|
|
|
80
78
|
|
|
81
79
|
const SUPPORTS_LAZY_KERNELS = true;
|
|
82
80
|
|
|
81
|
+
// All MARIMO_* reasons except TRANSPORT_EXHAUSTED are emitted by the backend
|
|
82
|
+
// (marimo/_server/api/endpoints/ws_endpoint.py and ws/*.py). Keep in sync with
|
|
83
|
+
// the backend literals.
|
|
84
|
+
export type CloseReason =
|
|
85
|
+
| "MARIMO_ALREADY_CONNECTED"
|
|
86
|
+
| "MARIMO_WRONG_KERNEL_ID"
|
|
87
|
+
| "MARIMO_NO_FILE_KEY"
|
|
88
|
+
| "MARIMO_NO_SESSION_ID"
|
|
89
|
+
| "MARIMO_NO_SESSION"
|
|
90
|
+
| "MARIMO_SHUTDOWN"
|
|
91
|
+
| "MARIMO_MALFORMED_QUERY"
|
|
92
|
+
| "MARIMO_KERNEL_STARTUP_ERROR"
|
|
93
|
+
| typeof TRANSPORT_EXHAUSTED_REASON;
|
|
94
|
+
|
|
83
95
|
export type CloseDecision =
|
|
84
96
|
| { kind: "terminal"; status: ConnectionStatus; closeTransport: boolean }
|
|
85
97
|
| { kind: "gave-up"; status: ConnectionStatus }
|
|
86
98
|
| { kind: "retry"; status: ConnectionStatus };
|
|
87
99
|
|
|
88
|
-
export function classifyCloseEvent(
|
|
89
|
-
event
|
|
90
|
-
context: { retryCount: number; maxRetries: number },
|
|
91
|
-
): CloseDecision {
|
|
92
|
-
switch (event.reason) {
|
|
100
|
+
export function classifyCloseEvent(event: { reason?: string }): CloseDecision {
|
|
101
|
+
switch (event.reason as CloseReason | undefined) {
|
|
93
102
|
case "MARIMO_ALREADY_CONNECTED":
|
|
94
103
|
return {
|
|
95
104
|
kind: "terminal",
|
|
@@ -101,6 +110,15 @@ export function classifyCloseEvent(
|
|
|
101
110
|
},
|
|
102
111
|
closeTransport: true,
|
|
103
112
|
};
|
|
113
|
+
case TRANSPORT_EXHAUSTED_REASON:
|
|
114
|
+
return {
|
|
115
|
+
kind: "gave-up",
|
|
116
|
+
status: {
|
|
117
|
+
state: WebSocketState.CLOSED,
|
|
118
|
+
code: WebSocketClosedReason.KERNEL_DISCONNECTED,
|
|
119
|
+
reason: "kernel not found",
|
|
120
|
+
},
|
|
121
|
+
};
|
|
104
122
|
case "MARIMO_WRONG_KERNEL_ID":
|
|
105
123
|
case "MARIMO_NO_FILE_KEY":
|
|
106
124
|
case "MARIMO_NO_SESSION_ID":
|
|
@@ -144,18 +162,7 @@ export function classifyCloseEvent(
|
|
|
144
162
|
logNever(event.reason as never);
|
|
145
163
|
}
|
|
146
164
|
}
|
|
147
|
-
|
|
148
|
-
// CLOSED so callers can detect the give-up.
|
|
149
|
-
if (context.retryCount >= context.maxRetries) {
|
|
150
|
-
return {
|
|
151
|
-
kind: "gave-up",
|
|
152
|
-
status: {
|
|
153
|
-
state: WebSocketState.CLOSED,
|
|
154
|
-
code: WebSocketClosedReason.KERNEL_DISCONNECTED,
|
|
155
|
-
reason: "kernel not found",
|
|
156
|
-
},
|
|
157
|
-
};
|
|
158
|
-
}
|
|
165
|
+
|
|
159
166
|
return {
|
|
160
167
|
kind: "retry",
|
|
161
168
|
status: { state: WebSocketState.CONNECTING },
|
|
@@ -440,7 +447,7 @@ export function useMarimoKernelConnection(opts: {
|
|
|
440
447
|
}
|
|
441
448
|
shouldTryReconnecting.current = true;
|
|
442
449
|
setConnection({ state: WebSocketState.CONNECTING });
|
|
443
|
-
const healthy = await runtimeManager.
|
|
450
|
+
const healthy = await runtimeManager.reconcileFromHealth();
|
|
444
451
|
if (!healthy) {
|
|
445
452
|
shouldTryReconnecting.current = false;
|
|
446
453
|
setConnection({
|
|
@@ -512,10 +519,7 @@ export function useMarimoKernelConnection(opts: {
|
|
|
512
519
|
*/
|
|
513
520
|
onClose: (e) => {
|
|
514
521
|
Logger.warn("WebSocket closed", e.code, e.reason);
|
|
515
|
-
const decision = classifyCloseEvent(e
|
|
516
|
-
retryCount: ws.retryCount,
|
|
517
|
-
maxRetries: MAX_RETRIES,
|
|
518
|
-
});
|
|
522
|
+
const decision = classifyCloseEvent(e);
|
|
519
523
|
setConnection(decision.status);
|
|
520
524
|
if (decision.kind === "terminal" && decision.closeTransport) {
|
|
521
525
|
ws.close(); // close to prevent reconnecting
|
|
@@ -1,11 +1,11 @@
|
|
|
1
1
|
/* Copyright 2026 Marimo. All rights reserved. */
|
|
2
2
|
|
|
3
|
-
import ReconnectingWebSocket from "partysocket/ws";
|
|
4
3
|
import { useEffect, useState } from "react";
|
|
5
4
|
import { Logger } from "@/utils/Logger";
|
|
6
5
|
import { createPyodideConnection } from "../wasm/bridge";
|
|
7
6
|
import { isWasm } from "../wasm/utils";
|
|
8
7
|
import { BasicTransport } from "./transports/basic";
|
|
8
|
+
import { WsTransport } from "./transports/ws";
|
|
9
9
|
import type { IConnectionTransport } from "./transports/transport";
|
|
10
10
|
|
|
11
11
|
interface UseConnectionTransportOptions {
|
|
@@ -18,10 +18,6 @@ interface UseConnectionTransportOptions {
|
|
|
18
18
|
onError: (event: WebSocketEventMap["error"]) => void;
|
|
19
19
|
}
|
|
20
20
|
|
|
21
|
-
// Per-`reconnect()` retry budget for partysocket. After exhaustion, partysocket
|
|
22
|
-
// stops silently; treat `retryCount >= MAX_RETRIES` as the give-up signal.
|
|
23
|
-
export const MAX_RETRIES = 10;
|
|
24
|
-
|
|
25
21
|
function createConnectionTransport(
|
|
26
22
|
options: Pick<UseConnectionTransportOptions, "url" | "static">,
|
|
27
23
|
): IConnectionTransport {
|
|
@@ -31,19 +27,8 @@ function createConnectionTransport(
|
|
|
31
27
|
if (isWasm()) {
|
|
32
28
|
return createPyodideConnection();
|
|
33
29
|
}
|
|
34
|
-
//
|
|
35
|
-
|
|
36
|
-
const urlProvider = options.url; // We don't call the URL provider now since it may change (i.e. if the runtime redirects)
|
|
37
|
-
// Cast needed: ReconnectingWebSocket types readyState as `number`
|
|
38
|
-
// but IConnectionTransport expects `0 | 1 | 2 | 3`
|
|
39
|
-
return new ReconnectingWebSocket(urlProvider, undefined, {
|
|
40
|
-
maxRetries: MAX_RETRIES,
|
|
41
|
-
debug: false,
|
|
42
|
-
startClosed: true,
|
|
43
|
-
// long timeout -- the server can become slow when many notebooks
|
|
44
|
-
// are open.
|
|
45
|
-
connectionTimeout: 10_000,
|
|
46
|
-
}) as unknown as IConnectionTransport;
|
|
30
|
+
// urlProvider is passed lazily; it may change after a runtime redirect.
|
|
31
|
+
return new WsTransport(options.url);
|
|
47
32
|
}
|
|
48
33
|
|
|
49
34
|
/**
|