Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Can't use Bun websocket publish() or subscribe() #66

Open
dukejones opened this issue Sep 22, 2024 · 2 comments
Open

Can't use Bun websocket publish() or subscribe() #66

dukejones opened this issue Sep 22, 2024 · 2 comments

Comments

@dukejones
Copy link

The current method works well enough for a single client of a websocket. But the main use case of websockets (especially with HTTP2 solving the head-of-line blocking for many requests) is to be able to publish to all of the subscribers.

Bun includes a simple pub/sub facility in its websockets. For this to work, you call .publish() on the server instance.
https://bun.sh/docs/api/websockets#pub-sub

The way svelte-adapter-bun works, there's no way to get a server instance from the websocket handlers. It should be possible from something like .upgrade(), save that ref locally in the websocketHandler file, but the websocket handler / upgrade method has been patched to receive only the upgrade method from the server rather than the entire server.

Bun.serve({
  fetch(req, server) {
    const sessionId = await generateSessionId();
    server.upgrade(req, {
      headers: {
        "Set-Cookie": `SessionId=${sessionId}`,
      },
    });
  },
  websocket: {}, // handlers
});

^^^ instead of server, .upgrade() receives just the server.upgrade() function.

I'd suggest getting rid of the custom .upgrade() function in line with the other websocket handlers -- it's a special hack that makes it hard to debug, coming from the Bun docs into the svelte-adapter-bun codebase.

Not exactly sure how you'd go about wiring up the upgrade etc., and it'd be a big-ish refactor. So I'm afraid I'm dropping back to Node and using a slower websocket library for now.

I'll be checking back in~ wishing this project much success.

@KyleFontenot
Copy link

KyleFontenot commented Oct 3, 2024

Help me understand if this doesnt answer your question. I think you might just be mixing up the uses for subscribe, upgrade, and publish.

.upgrade() belongs to Server which is the second parameter in .fetch() so yes you can access the server instance within the fetch method. Why would you need the upgrade() method from within the websocket handler, since the handler is only for upgraded connections? You could pass in the server instance into the upgrade's data object, but that might dirty it up a little.

.upgrade() is in Bun.fetch, while .subscribe() and publish() are in the websocket handler methods.

Bun.serve({
  fetch: (req: Request, server: Server) => {
    server.upgrade(req, {
      data: {
        server
      }
    });
  },
  websocket: {
    open(ws: ServerWebSocket) {
      ws.subscribe('chatroom')
    },
    message(ws: ServerWebSocket, msg: string | Buffer) {
      console.log(msg.toString());
    },
  },
});

Bun types clarifies it a little

@dukejones
Copy link
Author

Thanks for taking the time to write this up.. i'll try to find the time soon to dig up the old code and test out what you are saying.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants