You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
I wanted to provide some more insight on how I accomplished broadcasting events asynchronously using this excellent package, props to @apteryxxyz for this beautiful implementation.
For my example, these are the package versions used at the moment of writing this.
next: 14.1.0
next-ws: 1.0.1
ws: 8.16.0
Exposing WebSocketServer to global
ws exposes a clients list on the WebSocket server. Normally, using next-ws one would only access the WebSocket server inside the exported SOCKET route.
To expose the WebSocket server to different parts of your Next application, export it as follows:
Using Node's global accesor, and thanks to the patch done by next-ws before-hand, this WebSocket server object persists itself statelessly on the entire application. This is good even for Next applications where the output is standalone as it is most common when dockerizing said app.
Using this method, one maintains Typescript support and avoids utilizing a custom Next server.
I'm finding I have a somewhat similar use-case. I'm building an app that needs an external source to be able to send messages to the WebSocket server, but I'm hitting a wall in getting that working. I tried the above solution, and was able to get it working locally, but since my project is built in `standalone` mode (for Docker) it wasn't super clear how to use the custom server alongside the `server.js` that gets generated by Next. Also didn't really love losing TypeScript support to have a custom server.
WebSocketServer provides a clients list. If you want, for example, wait for an incoming webhook message in order to broadcast something to all connected clients in the WebSocket, simply access the globalized object and stream it out to said clients.
// app/api/webhook/route.tsimport{NextRequest,NextResponse}from"next/server";import{WebSocket}from"ws";import{WebSocketServer}from"ws";exportasyncfunctionPOST(req: NextRequest){try{// Fetch the WebSocket server.constwebSocket=(globalasany)?.["wsServer"]asWebSocketServer;if(!webSocket)returnNextResponse.json({status: false},{status: 500,});// Parse the client set into an array.constclients=Array.from(webSocket.clients);// Emit the message to the client.for(constclientofclients)client.send('Hello from the webhook!');returnNextResponse.json({status: true});}catch(e){console.error(e);throwe;}}
You can broadcast any message you wish via this method, fully async and never losing connectivity. ws allows any Buffer derived data to go through, so as long as your data falls into this category you should be good to go (i.e: strings, files)
As next-ws uses ws for connectivity, one can send query parameters through the connection string for the exported WebSocket server route. This is great to identify clients via an ID, or off-load some data to their client connection to later check when streaming messages asynchronously on the server-side.
If you're using Typescript: ws must be augmented to allow you to set more properties on the WebSocket client, an example for this would be allowing clients to have an ID. As such:
Now you can set IDs for incoming WebSocket connection clients which are fully readable on the WebSocket server clients list. This is an example on how to accomplish a sort-of basic selective broadcast.
Client-side
Note: This is not a secure way to send parameters to the WebSocket server. This is just as an example implementation.
/app/components/SomeComponent.tsx"use client";import{WebSocketProvider}from"next-ws/client";// As an example, userId would be 12345exportdefaultfunctionSomeComponent({ children, userId }: {children: React.Node,userId: string}){return(<WebSocketProviderurl={`ws://localhost:3000/api/socket?userId=${userId}`}>{...children}</WebSocketProvider>);}
Server-side
Load the provided query parameters from the request object. You can validate stuff here and then assign the client the provided ID.
import{WebSocket,WebSocketServer}from"ws";import{IncomingMessage}from"http";exportfunctionSOCKET(client: WebSocket,request: IncomingMessage,server: WebSocketServer){// Obtain query parameters.constquery=newURLSearchParams(request.url!.split("?")[1]);// Assign the provided userId to the client.client["id"]=query.get("userId")??"Unknown";(globalasany)["wsServer"]=server;}
Whenever you wish to broadcast something specific, access the client.id property when looping through the WebSocket server clients list.
// app/api/webhook/route.tsimport{NextRequest,NextResponse}from"next/server";import{WebSocket}from"ws";import{WebSocketServer}from"ws";exportasyncfunctionPOST(req: NextRequest){try{// Fetch the WebSocket server.constwebSocket=(globalasany)?.["wsServer"]asWebSocketServer;if(!webSocket)returnNextResponse.json({status: false},{status: 500,});// Parse the client set into an array.// Typescript type augmentation done to allow for the use of the `id` property.constclients: Array<WebSocket>=Array.from(webSocket.clientsasSet<WebSocket>);// Emit the message to the client.for(constclientofclients){console.log(client.id);// 12345client.send('Hello from the webhook!');}returnNextResponse.json({status: true});}catch(e){console.error(e);throwe;}}
The text was updated successfully, but these errors were encountered:
punteroo
changed the title
[Documentation] Broadcast events to connected clients
[Documentation] Exposing WebSocketServer for server-side access
Feb 21, 2024
Hello!
I wanted to provide some more insight on how I accomplished broadcasting events asynchronously using this excellent package, props to @apteryxxyz for this beautiful implementation.
For my example, these are the package versions used at the moment of writing this.
next: 14.1.0
next-ws: 1.0.1
ws: 8.16.0
Exposing
WebSocketServer
toglobal
ws
exposes a clients list on the WebSocket server. Normally, usingnext-ws
one would only access the WebSocket server inside the exportedSOCKET
route.To expose the WebSocket server to different parts of your Next application, export it as follows:
Using Node's
global
accesor, and thanks to the patch done bynext-ws
before-hand, this WebSocket server object persists itself statelessly on the entire application. This is good even for Next applications where the output isstandalone
as it is most common when dockerizing said app.Using this method, one maintains Typescript support and avoids utilizing a custom Next server.
Originally posted by @Alex-Mastin in #8 (comment)
Broadcast to Everyone
WebSocketServer
provides a clients list. If you want, for example, wait for an incoming webhook message in order to broadcast something to all connected clients in the WebSocket, simply access the globalized object and stream it out to said clients.You can broadcast any message you wish via this method, fully async and never losing connectivity.
ws
allows anyBuffer
derived data to go through, so as long as your data falls into this category you should be good to go (i.e: strings, files)https://github.com/DefinitelyTyped/DefinitelyTyped/blob/master/types/ws/index.d.ts#L20-L36
Broadcasting to Specific Clients
As
next-ws
usesws
for connectivity, one can send query parameters through the connection string for the exported WebSocket server route. This is great to identify clients via an ID, or off-load some data to their client connection to later check when streaming messages asynchronously on the server-side.If you're using Typescript:
ws
must be augmented to allow you to set more properties on the WebSocket client, an example for this would be allowing clients to have an ID. As such:Now you can set IDs for incoming WebSocket connection clients which are fully readable on the WebSocket server clients list. This is an example on how to accomplish a sort-of basic selective broadcast.
Client-side
Note: This is not a secure way to send parameters to the WebSocket server. This is just as an example implementation.
Server-side
Load the provided query parameters from the
request
object. You can validate stuff here and then assign the client the provided ID.Whenever you wish to broadcast something specific, access the
client.id
property when looping through the WebSocket server clients list.The text was updated successfully, but these errors were encountered: