diff --git a/.changeset/websocket-solid-2-async.md b/.changeset/websocket-solid-2-async.md new file mode 100644 index 000000000..c7a971f09 --- /dev/null +++ b/.changeset/websocket-solid-2-async.md @@ -0,0 +1,32 @@ +--- +"@solid-primitives/websocket": major +--- + +Upgrade to Solid.js 2.0 (`^2.0.0-beta.7`) and add async-reactive message primitives. + +**Breaking changes** + +- Peer dependency is now `solid-js@^2.0.0-beta.7`. All `createEffect` examples in docs now use the Solid 2.0 split form: `createEffect(compute, effect)`. + +**New: `createWSMessage`** + +Reactive `Accessor` for the most recently received WebSocket message. Cleans up its event listener on owner disposal via `onCleanup`. + +```ts +const message = createWSMessage(ws); +return

{message()}

; +``` + +> Note: uses a signal internally, so under burst conditions only the final message before a flush is tracked by effects. For every-message processing, use the planned `wsMessageIterable` / `createWSData` primitives. + +**`createWSState` signal fix** + +Internal signal now uses `{ ownedWrite: true }` to suppress the Solid 2.0 dev-mode `SIGNAL_WRITE_IN_OWNED_SCOPE` diagnostic, which would fire if `ws.close()` is called from inside a component body or reactive computation. + +**Planned for next minor: async message primitives** + +The following are designed and documented but not yet implemented, based on Solid 2.0's `createMemo(AsyncIterable)` model: + +- `wsMessageIterable` — buffered `AsyncIterable` that never drops burst messages; works with `makeReconnectingWS` +- `createWSData` — async memo over `wsMessageIterable`; suspends `` until first message; integrates with `isPending` and `latest` +- `createWSStore` — reactive store driven by WS messages as draft-mutation patches via `createStore(fn, seed)` diff --git a/packages/memo/README.md b/packages/memo/README.md index d3e5bdb6a..8d164fddf 100644 --- a/packages/memo/README.md +++ b/packages/memo/README.md @@ -156,7 +156,7 @@ The lazy memo, as it is implemented now, doesn't allow for setting a `equals` fu ### Not ownerless -Lazy memos in Solid 2.0 will be ownerless — the reactive context of the callback will depend of the place of read, not creation. +Lazy memos in Solid will be ownerless — the reactive context of the callback will depend of the place of read, not creation. This implementation will always execute it's callback with the context of owner it was created under. So ti won't work with [Suspense](https://www.solidjs.com/docs/latest/api#) the way you might expect — meaning that it won't activate any Suspense that is below place of creation. diff --git a/packages/websocket/CHANGELOG.md b/packages/websocket/CHANGELOG.md index d123f3f90..04bb848db 100644 --- a/packages/websocket/CHANGELOG.md +++ b/packages/websocket/CHANGELOG.md @@ -1,5 +1,14 @@ # @solid-primitives/websocket +## 2.0.0-beta.0 + +### Major Changes + +- Upgrade to Solid.js 2.0 (`^2.0.0-beta.7`). +- `createWSState`: signal now uses `ownedWrite: true` to suppress dev-mode warnings when `ws.close()` is called from within a component or effect. +- New primitive `createWSMessage`: reactive signal containing the latest received WebSocket message. Cleans up its event listener automatically on owner disposal. +- Updated all JSDoc examples to use the Solid 2.0 split `createEffect(compute, effect)` form. + ## 1.3.2 ### Patch Changes diff --git a/packages/websocket/README.md b/packages/websocket/README.md index 44ccd0248..5050d0739 100644 --- a/packages/websocket/README.md +++ b/packages/websocket/README.md @@ -6,148 +6,227 @@ [![stage](https://img.shields.io/endpoint?style=for-the-badge&url=https%3A%2F%2Fraw.githubusercontent.com%2Fsolidjs-community%2Fsolid-primitives%2Fmain%2Fassets%2Fbadges%2Fstage-0.json)](https://github.com/solidjs-community/solid-primitives#contribution-process) -Primitive to help establish, maintain and operate a websocket connection. +Primitives to help establish, maintain, and operate WebSocket connections in Solid. -- `makeWS` - sets up a web socket connection with a buffered send -- `createWS` - sets up a web socket connection that disconnects on cleanup -- `createWSState` - creates a reactive signal containing the readyState of a websocket -- `makeReconnectingWS` - sets up a web socket connection that reconnects if involuntarily closed -- `createReconnectingWS` - sets up a reconnecting web socket connection that disconnects on cleanup -- `makeHeartbeatWS` - wraps a reconnecting web socket to send a heart beat and reconnect if the answer fails +### Connection primitives -All of them return a WebSocket instance extended with a `message` prop containing an accessor for the last received message for convenience and the ability to receive messages to send before the connection is opened. +- [`makeWS`](#makews) — raw WebSocket with a buffered send queue (manual cleanup) +- [`createWS`](#createws) — same, but closes on owner disposal +- [`createWSState`](#createwsstate) — reactive `readyState` signal (`0`–`3`) +- [`makeReconnectingWS`](#makereconnectingws) — auto-reconnects on involuntary close (manual cleanup) +- [`createReconnectingWS`](#createreconnectingws) — same, but closes on owner disposal +- [`makeHeartbeatWS`](#makeheartbeatws) — wraps a reconnecting WS with a heartbeat/pong watchdog -## How to use it +### Message primitives + +- [`createWSMessage`](#createwsmessage) — reactive signal for the **latest** received message +- [`wsMessageIterable`](#wsmessageiterable-planned) — buffered `AsyncIterable` over WS messages *(planned)* +- [`createWSData`](#createwsdata-planned) — async memo compatible with ``, `isPending`, and `latest` *(planned)* +- [`createWSStore`](#createwsstore-planned) — reactive store driven by WS message patches *(planned)* + +--- + +## Connection primitives + +### `makeWS` + +Sets up a WebSocket with a buffered send queue. Messages sent before the connection opens are queued and flushed on `open`. Does **not** close on cleanup — use `createWS` for that. ```ts -const ws = createWS("ws://localhost:5000"); -const state = createWSState(ws); -const states = ["Connecting", "Connected", "Disconnecting", "Disconnected"]; -ws.send("it works"); -createEffect(on(ws.message, msg => console.log(msg), { defer: true })); -return

Connection: {states[state()]}

; - -const socket = makeHeartbeatWS( - makeReconnectingWS(`ws://${location.hostName}/api/ws`, undefined, { timeout: 500 }), - { message: "👍" }, +const ws = makeWS("ws://localhost:5000"); +createEffect( + () => serverMessage(), + (msg) => ws.send(msg), ); -// with the primitives starting with `make...`, one needs to manually clean up: -socket.send("this will reconnect if connection fails"); +onCleanup(() => ws.close()); ``` -### Definitions +### `createWS` + +Same as `makeWS`, but registers `ws.close()` with `onCleanup`. ```ts -/** Arguments of the primitives */ -type WSProps = [url: string, protocols?: string | string[]]; -type WSMessage = string | ArrayBufferLike | ArrayBufferView | Blob; -type WSReadyState = WebSocket.CONNECTING | WebSocket.OPEN | WebSocket.CLOSING | WebSocket.CLOSED; -type WSEventMap = { - close: CloseEvent; - error: Event; - message: MessageEvent; - open: Event; -}; -type ReconnectingWebSocket = WebSocket & { - reconnect: () => void; - // ws.send.before is meant to be used by heartbeat - send: ((msg: WSMessage) => void) & { before: () => void }; -}; -type WSHeartbeatOptions = { - /** - * Heartbeat message being sent to the server in order to validate the connection - * @default "ping" - */ - message?: WSMessage; - /** - * The time between messages being sent in milliseconds - * @default 1000 - */ - interval?: number; - /** - * The time after the heartbeat message being sent to wait for the next message in milliseconds - * @default 1500 - */ - wait?: number; -}; +const ws = createWS("ws://localhost:5000"); +createEffect( + () => serverMessage(), + (msg) => ws.send(msg), +); ``` -If you want to use the messages as a signal, have a look at the [`event-listener`](../event-listener/README.md) package: +### `createWSState` + +Returns a reactive `Accessor<0 | 1 | 2 | 3>` tracking the WebSocket's `readyState`. ```ts -import { createWS } from "@solid-primitives/websocket"; -import { createEventSignal } from "@solid-primitives/event-listener"; +const ws = createWS("ws://localhost:5000"); +const state = createWSState(ws); +const labels = ["Connecting", "Open", "Closing", "Closed"] as const; + +return

Status: {labels[state()]}

; +``` + +### `createWSMessage` + +Returns a reactive `Accessor` that holds the **most recently received** message. Starts as `undefined`. +```ts const ws = createWS("ws://localhost:5000"); -const messageEvent = createEventSignal(ws, "message"); -const message = () => messageEvent().data; +const message = createWSMessage(ws); + +return

Last message: {message()}

; ``` -Otherwise, you can simply use the message event to get message.data: +> **Note — "latest wins" semantics.** `createWSMessage` uses a signal internally. In Solid 2.0, signal writes are batched: if two messages arrive before the reactive flush, only the second is seen by effects. This is fine for "current state" displays, but if your protocol can burst messages and you need to process every one, use [`wsMessageIterable`](#wsmessageiterable-planned) or [`createWSData`](#createwsdata-planned) instead. + +### `makeReconnectingWS` + +Returns a `WebSocket`-shaped proxy that transparently opens a new underlying connection whenever the server closes it involuntarily. ```ts -import { createStore } from "solid-js/store"; -import { createReconnectingWS, WSMessage } from "@solid-primitives/websocket"; +const ws = makeReconnectingWS("ws://localhost:5000", undefined, { delay: 3000, retries: Infinity }); +createEffect( + () => serverMessage(), + (msg) => ws.send(msg), +); +onCleanup(() => ws.close()); +``` + +### `createReconnectingWS` + +Same as `makeReconnectingWS`, but closes on owner disposal. -const ws = createReconnectingWS("ws://localhost:5000"); -const [messages, setMessages] = createStore(); -ws.addEventListener("message", (ev) => setMessages(messages.length, ev.data)); +### `makeHeartbeatWS` - messages}> - {(message) => ...} - +Wraps a `ReconnectingWebSocket` to send a periodic heartbeat. If no response arrives within `wait` ms the connection is force-reconnected. + +```ts +const ws = makeHeartbeatWS( + createReconnectingWS("ws://localhost:5000"), + { message: "ping", interval: 1000, wait: 1500 }, +); ``` -## Setting up a websocket server +--- + +## Async message primitives *(planned for next minor)* + +These three primitives leverage Solid's async reactivity — `createMemo` with `AsyncIterable`, `` boundaries, `isPending`, and `latest` — to provide a more powerful and correct model for WebSocket data. + +### `wsMessageIterable` *(planned)* -While you can use this primitive with solid-start, it already provides a package for websockets that handles both the server and the client side: +The foundational building block. Returns a buffered `AsyncIterable` over a WebSocket's message stream. Cleanup (`ws.removeEventListener`) happens automatically when the iterator is returned (Solid calls `it.return()` on memo disposal). ```ts -import { createWebSocketServer } from "solid-start/websocket"; -import server$ from "solid-start/server"; - -const pingPong = createWebSocketServer( - server$(function (webSocket) { - webSocket.addEventListener("message", async msg => { - try { - // Parse the incoming message - let incomingMessage = JSON.parse(msg.data); - console.log(incomingMessage); - - switch (incomingMessage.type) { - case "ping": - webSocket.send( - JSON.stringify([ - { - type: "pong", - data: { - id: incomingMessage.data.id, - time: Date.now(), - }, - }, - ]), - ); - break; - } - } catch (err: any) { - // Report any exceptions directly back to the client. As with our handleErrors() this - // probably isn't what you'd want to do in production, but it's convenient when testing. - webSocket.send(JSON.stringify({ error: err.stack })); - } - }); - }), +import { wsMessageIterable } from "@solid-primitives/websocket"; + +// Compose freely with any Solid 2.0 async primitive: +const latestQuote = createMemo(async function* () { + for await (const raw of wsMessageIterable(ws)) { + yield JSON.parse(raw) as Quote; + } +}); +``` + +Works correctly with `makeReconnectingWS` — event listeners are re-attached to each new underlying connection, so the iterable survives reconnects transparently. + +**Why this doesn't drop messages:** Unlike `createWSMessage`, each yielded value triggers its own `flush()` inside the Solid runtime. Messages that arrive while an earlier one is being processed are buffered and drained synchronously, so no message is skipped by reactive effects. + +### `createWSData` *(planned)* + +An async memo wrapping `wsMessageIterable`. Suspends the nearest `` boundary until the first message arrives; subsequent updates work with `isPending` and `latest`. + +```tsx +const price = createWSData(ws, { transform: JSON.parse }); + +return ( + Waiting for data…

}> + {/* isPending: true while the next tick is in-flight with a stale value showing */} +

price()) ? "stale" : ""}> + Bid: {price().bid} / Ask: {price().ask} +

+
); ``` -Otherwise, in order to set up your own production-use websocket server, we recommend packages like +Comparison with `createWSMessage`: + +| | `createWSMessage` | `createWSData` | +|---|---|---| +| Drops burst messages | Yes | No | +| Works with `` | No | Yes | +| `isPending()` support | No | Yes | +| `latest()` support | No | Yes | +| Returns `undefined` before first message | Yes | No — throws (suspends) | +| Best for | Simple last-value display | State-source WS, real-time feeds | + +### `createWSStore` *(planned)* + +A reactive store driven by WebSocket messages as incremental patches. Uses Solid `createStore(fn, seed)` form — each message is applied as a draft mutation. + +```tsx +const [appState] = createWSStore(ws, { + initial: { users: [], status: "connecting" }, + patch(draft, msg) { + Object.assign(draft, JSON.parse(msg)); + }, +}); + +return

Users online: {appState.users.length}

; +``` + +--- + +## Composing with `action` (request/response pattern) + +For protocols with correlated request/response over a shared WebSocket, Solid `action` is used: + +```ts +const queryServer = action(function* (payload: RequestPayload) { + const id = crypto.randomUUID(); -- nodejs: [`ws`](https://github.com/websockets/ws) -- rust: [`websocket`](https://docs.rs/websocket/latest/websocket/) + setOptimisticState(draft => { draft.loading = true; }); -## Demo + ws.send(JSON.stringify({ ...payload, id })); -You may view a working example here: -https://primitives.solidjs.community/playground/websocket/ + const response: ResponsePayload = yield new Promise(resolve => { + const handler = (e: MessageEvent) => { + const msg = JSON.parse(e.data); + if (msg.id === id) { + ws.removeEventListener("message", handler); + resolve(msg); + } + }; + ws.addEventListener("message", handler); + }); + + refresh(() => serverData()); + return response; +}); +``` + +--- + +## Type reference + +```ts +type WSMessage = string | ArrayBufferLike | ArrayBufferView | Blob; + +type WSReconnectOptions = { + delay?: number; // ms between reconnect attempts — default: 3000 + retries?: number; // max reconnect attempts — default: Infinity +}; + +type ReconnectingWebSocket = WebSocket & { + reconnect: () => void; + send: ((msg: WSMessage) => void) & { before?: () => void }; +}; + +type WSHeartbeatOptions = { + message?: WSMessage; // default: "ping" + interval?: number; // ms between heartbeats — default: 1000 + wait?: number; // ms to wait for pong before reconnecting — default: 1500 +}; +``` ## Changelog diff --git a/packages/websocket/package.json b/packages/websocket/package.json index 7e3607734..f5eeb8547 100644 --- a/packages/websocket/package.json +++ b/packages/websocket/package.json @@ -1,6 +1,6 @@ { "name": "@solid-primitives/websocket", - "version": "1.3.2", + "version": "2.0.0-beta.0", "description": "Primitive to create a web socket connection", "author": "David Di Biase ", "contributors": [ @@ -19,6 +19,7 @@ "makeWS", "createWS", "createWSState", + "createWSMessage", "makeReconnectingWS", "createReconnectingWS", "makeHeartbeatWS" @@ -55,10 +56,10 @@ "primitives" ], "peerDependencies": { - "solid-js": "^1.6.12" + "solid-js": "^2.0.0-beta.7" }, "typesVersions": {}, "devDependencies": { - "solid-js": "^1.9.7" + "solid-js": "2.0.0-beta.7" } } diff --git a/packages/websocket/src/index.ts b/packages/websocket/src/index.ts index e8946e7b9..733c3a3d6 100644 --- a/packages/websocket/src/index.ts +++ b/packages/websocket/src/index.ts @@ -3,10 +3,13 @@ import { type Accessor, onCleanup, createSignal } from "solid-js"; export type WSMessage = string | ArrayBufferLike | ArrayBufferView | Blob; /** - * opens a web socket connection with a queued send + * Opens a web socket connection with a queued send. * ```ts * const ws = makeWS("ws://localhost:5000"); - * createEffect(() => ws.send(serverMessage())); + * createEffect( + * () => serverMessage(), + * (msg) => ws.send(msg), + * ); * onCleanup(() => ws.close()); * ``` * Will not throw if you attempt to send messages before the connection opened; instead, it will enqueue the message to be sent when the connection opens. @@ -28,10 +31,13 @@ export const makeWS = ( }; /** - * opens a web socket connection with a queued send that closes on cleanup + * Opens a web socket connection with a queued send that closes on cleanup. * ```ts - * const ws = makeWS("ws://localhost:5000"); - * createEffect(() => ws.send(serverMessage())); + * const ws = createWS("ws://localhost:5000"); + * createEffect( + * () => serverMessage(), + * (msg) => ws.send(msg), + * ); * ``` * Will not throw if you attempt to send messages before the connection opened; instead, it will enqueue the message to be sent when the connection opens. */ @@ -42,12 +48,12 @@ export const createWS = (url: string, protocols?: string | string[]): WebSocket }; /** - * Returns a reactive state signal for the web socket's readyState: + * Returns a reactive signal for the WebSocket's `readyState`: * - * WebSocket.CONNECTING = 0 - * WebSocket.OPEN = 1 - * WebSocket.CLOSING = 2 - * WebSocket.CLOSED = 3 + * - `0` — CONNECTING + * - `1` — OPEN + * - `2` — CLOSING + * - `3` — CLOSED * * ```ts * const ws = createWS('ws://localhost:5000'); @@ -57,7 +63,10 @@ export const createWS = (url: string, protocols?: string | string[]): WebSocket * ``` */ export const createWSState = (ws: WebSocket): Accessor<0 | 1 | 2 | 3> => { - const [state, setState] = createSignal(ws.readyState as 0 | 1 | 2 | 3); + // ownedWrite: true — setState may be called from ws.close(), which the user + // could invoke inside a component or effect. This suppresses the dev-mode + // owned-scope write warning for this intentionally internal signal. + const [state, setState] = createSignal(ws.readyState as 0 | 1 | 2 | 3, { ownedWrite: true }); const _close = ws.close.bind(ws); ws.addEventListener("open", () => setState(1)); ws.close = (...args) => { @@ -68,6 +77,33 @@ export const createWSState = (ws: WebSocket): Accessor<0 | 1 | 2 | 3> => { return state; }; +/** + * Returns a reactive signal containing the latest message received from the WebSocket. + * Starts as `undefined` until the first message arrives. + * + * ```ts + * const ws = createWS("ws://localhost:5000"); + * const message = createWSMessage(ws); + * return

Last message: {message()}

; + * ``` + * + * The signal updates on every incoming message. Pair it with `createEffect` to + * react to each new value: + * ```ts + * createEffect( + * () => message(), + * (msg) => msg !== undefined && console.log("received:", msg), + * ); + * ``` + */ +export const createWSMessage = (ws: WebSocket): Accessor => { + const [message, setMessage] = createSignal(undefined); + const handler = (e: MessageEvent) => setMessage(() => e.data as T); + ws.addEventListener("message", handler); + onCleanup(() => ws.removeEventListener("message", handler)); + return message; +}; + export type WSReconnectOptions = { delay?: number; retries?: number; @@ -83,7 +119,10 @@ export type ReconnectingWebSocket = WebSocket & { * Returns a WebSocket-like object that under the hood opens new connections on disconnect: * ```ts * const ws = makeReconnectingWS("ws:localhost:5000"); - * createEffect(() => ws.send(serverMessage())); + * createEffect( + * () => serverMessage(), + * (msg) => ws.send(msg), + * ); * onCleanup(() => ws.close()); * ``` * Will not throw if you attempt to send messages before the connection opened; instead, it will enqueue the message to be sent when the connection opens. @@ -148,8 +187,11 @@ export const makeReconnectingWS = ( /** * Returns a WebSocket-like object that under the hood opens new connections on disconnect and closes on cleanup: * ```ts - * const ws = makeReconnectingWS("ws:localhost:5000"); - * createEffect(() => ws.send(serverMessage())); + * const ws = createReconnectingWS("ws:localhost:5000"); + * createEffect( + * () => serverMessage(), + * (msg) => ws.send(msg), + * ); * ``` * Will not throw if you attempt to send messages before the connection opened; instead, it will enqueue the message to be sent when the connection opens. */ @@ -178,7 +220,7 @@ export type WSHeartbeatOptions = { }; /** - * Wraps a reconnecting WebSocket to send a heartbeat to check the connection + * Wraps a reconnecting WebSocket to send a heartbeat to check the connection. * ```ts * const ws = makeHeartbeatWS(createReconnectingWS('ws://localhost:5000')) * ``` diff --git a/packages/websocket/test/index.test.ts b/packages/websocket/test/index.test.ts index 0a240c2cd..5314d3eb3 100644 --- a/packages/websocket/test/index.test.ts +++ b/packages/websocket/test/index.test.ts @@ -4,6 +4,7 @@ import { createRoot } from "solid-js"; import { createWS, createWSState, + createWSMessage, createReconnectingWS, makeReconnectingWS, makeHeartbeatWS, @@ -73,6 +74,38 @@ describe("createWSState", () => { })); }); +describe("createWSMessage", () => { + it("is undefined before any messages arrive", () => + createRoot(dispose => { + const ws = createWS("ws://localhost:5000"); + const message = createWSMessage(ws); + expect(message()).toBeUndefined(); + dispose(); + })); + + it("reflects the latest received message", () => + createRoot(dispose => { + const ws = createWS("ws://localhost:5000"); + const message = createWSMessage(ws); + vi.advanceTimersByTime(20); // wait for open + ws.dispatchEvent(new MessageEvent("message", { data: "hello" })); + expect(message()).toBe("hello"); + ws.dispatchEvent(new MessageEvent("message", { data: "world" })); + expect(message()).toBe("world"); + dispose(); + })); + + it("removes the event listener on disposal", () => + createRoot(dispose => { + const ws = makeWS("ws://localhost:5000"); + const spy = vi.spyOn(ws, "removeEventListener"); + createWSMessage(ws); + dispose(); + expect(spy).toHaveBeenCalledWith("message", expect.any(Function)); + ws.close(); + })); +}); + describe("makeReconnectingWS", () => { it("reconnects after being closed by external circumstances", () => { const ws = makeReconnectingWS("ws://localhost:5000", undefined, { delay: 100 });