April 3rd, 2025

Show HN: Offline JavaScript PubSub between browser tabs

TabSub is a JavaScript library for offline PubSub communication between browser tabs using local storage. It allows message sharing on the same domain and is open-source under the MIT license.

Read original articleLink Icon
CuriosityConcernEnthusiasm
Show HN: Offline JavaScript PubSub between browser tabs

TabSub is a JavaScript library that facilitates offline PubSub communication between browser tabs using local storage, eliminating the need for a server. It allows messages to be shared across tabs on the same domain, leveraging the browser's built-in local storage. Users can test its functionality through examples, such as synchronizing audio playback between tabs. While the library appears to handle multiple concurrent writes effectively, there is no guarantee of performance under heavy load, and users are advised to proceed at their own risk. TabSub is open-source and available on GitHub under the MIT license. It is noted that the BroadcastChannel API offers similar functionality. The library provides functions for publishing messages, subscribing to topics, and retrieving the current state of a topic.

- TabSub enables offline PubSub communication between browser tabs using local storage.

- It is designed for use on the same domain due to browser security measures.

- The library is open-source and licensed under MIT, available on GitHub.

- Users can publish messages, subscribe to topics, and check the current state of topics.

- The BroadcastChannel API offers similar capabilities for inter-tab communication.

AI: What people are saying
The comments on the TabSub article reflect a variety of perspectives on the library and its implications for web development.
  • Some users compare TabSub to the `BroadcastChannel` API, suggesting it serves a similar purpose.
  • Concerns are raised about potential security vulnerabilities related to local storage access control.
  • Users share personal experiences with implementing similar PubSub systems, highlighting challenges and solutions.
  • There is interest in the simplicity of the code and its potential applications in various projects.
  • Some comments reference related technologies and past implementations, indicating a broader context of discussion around PubSub mechanisms.
Link Icon 13 comments
By @sisk - 1 day
If I understand this correctly, the `BroadcastChannel` API solves a similar purpose.

https://developer.mozilla.org/en-US/docs/Web/API/Broadcast_C...

By @sltkr - 1 day
The subscribe() implementation seems suboptimal in the case where there are many different topics:

    subscribe(topic, callback) {
        if (this.listeners[topic] == undefined) {
            // Not yet any listener for this topic
            this.listeners[topic] = [];

            window.addEventListener('storage', (e) => {
                const dataKey = topic + "_data";
                if (e.key === dataKey) {
                    const data =  JSON.parse(e.newValue)[0];
                    this.listeners[topic].forEach((v, k) => {
                        v(data);
                    });
                }
            }, false);
        }
        this.listeners[topic].push(callback)
    }
This installs a handler for every single topic, and every time a message is published, the handlers for all topics are called, even though at most one is interested in the change. A more efficient implementation would install a single handler, e.g. (untested):

    window.addEventListener('storage', (e) => {
        if (e.key.endsWith('_data')) {
            const topic = e.key.substring(0, e.key.length - 5);
            const data = JSON.parse(e.newValue)[0];
            this.listeners[topic]?.forEach(v => v(data));
       }
    }, false);
By @Lerc - 1 day
I built a ComfyUi node that let you run a little paint program in another tab. (Sorry about the lack of updates if you use it. I intend to get back to it)

Negotiating the communication between tabs was by far the hardest part. In the end I ended up using local storage for signaling to establish a dedicated messsageport channel.

It was such a fight to make something that re-established the connection when either page reloaded.

There are still some connection issues that I haven't been able to resolve. It seems some browsers on some systems reject messages between tabs if they were loaded hours apart.

It might be worth benchmarking a pure local storage fallback, but I'm guessing it would suffer with high traffic.

A generalised implementation that allowed switching multiple paint programs and multiple ComfyUi pages would be ideal. A PubSub might be the way to go.

There's also the issue of other parts of the app also using local storage. Need to not step on toes.

By @hombre_fatal - 1 day
It's nice that the 'storage' event also gives you event.newValue to spare you the race condition of reading from localStorage.
By @Minor49er - 1 day
The audio demo makes me think of Bandcamp which will pause music that you're playing if another Bandcamp tab starts playing a song separately. Must be a similar mechanism under the hood
By @kvemkon - 1 day
Related recently discussed:

Show HN: JavaScript PubSub in 163 Bytes (31.03.2025)

https://news.ycombinator.com/item?id=43529774

By @nottorp - 1 day
Isn't this a security hole waiting to be exploited?

How does the browser handle access control to the local storage, especially offline when they aren't loaded from the same site?

[Yes, I really don't know. Yes, I'm asking. Not everyone is a web dev.]

By @qudat - 1 day
Very cool, I love seeing pubsubs in the wild. We built an ssh based pubsub that we have been using in production for awhile now: https://pipe.pico.sh/
By @edweis - 1 day
By @amazingamazing - about 24 hours
I wonder how one does this on nodejs across instances gracefully.
By @EGreg - 1 day
I remember implementing the same thing in our framework like a decade ago, but eventually taking it out. We wanted to use it for caching data between tabs, so as not to hit the server again per tab:

https://github.com/Qbix/Platform/blob/main/platform/plugins/...

It had the concept of a “main frame” that they would route all requests to the server through.

I remember now — I did it not so much for tabs as for our solution of having tons of little iframes on a page: https://qbix.com/ecosystem