npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

@ennuicastr/mprpc

v0.1.0

Published

RPC over MessagePorts and/or Workers

Downloads

71

Readme

This is a simple system for remote procedure calls over MessagePorts and Workers.

It is divided into a receiver module and a target module. In this context, a receiver is the object that actually handles and runs RPC procedures, and a target is an object that dispatches RPC calls over message ports.

Types are exported for TypeScript to make it relatively straightforward to implement both sides based on a single interface.

We will use the following simple interface as an example:

interface Calculator {
    readNumber(): Promise<number>;
    sum(l: Float64Array): number;
}

Receiver

Import @ennuicastr/mprpc/receiver. In sample code, it will be imported as rpcReceiver.

Any object with methods can be an RPC receiver. To assign an RPC receiver object to a MessagePort, use rpcReceiver.rpcReceiver:

rpcReceiver.rpcReceiver(new Handler(), messagePort);

As it's common to make a worker that primarily acts as a receiver for RPC messages, a helper function is provided that's simply equivalent to calling rpcReceiver.rpcReceiver with globalThis as the message port:

rpcReceiver.rpcWorkerMain(new Handler());

RPC methods can return either the value directly, or an object of the form {return: value, transfer: [...]}. In the latter case, the transfer array is passed as the transfer argument to postMessage.

In addition, the type constructor rpcReceiver.RPCReceiver<T> is provided. This modifies a type by allowing the return type of methods to have transfers. rpcReceiver.RPCReceiver<Calculator> is:

interface {
    readNumber(): Promise<number> | Promise<{return: number, transfer: Transferable[]}>;
    sum(l: Float64Array): number | {return: number, transfer: Transferable[]};
}

Target

Import @ennuicastr/mprpc/target. In sample code, it will be imported as rpcTarget.

The rpcTarget.RPCTarget class class provides an RPC target for a MessagePort you've created yourself. It provides two RPC methods: rpc and rpcv. The difference is simply that the latter does not wait for a return, which can be more efficient as it saves a message. rpc and rpcv take two arguments plus an optional third: the method to call on the receiver, the arguments as an array, and the transfer list for postMessage. rpc returns a Promise which resolves (or rejects) based on the result of the method on the receiver.

Additionally, rpcTarget.RPCTarget provides a tee method, which creates an additional MessagePort for communicating with the same receiver, which can then be, for instance, transferred to another worker.

For the common case of a worker servering as the RPC receiver, the class rpcTarget.RPCWorker is provided. It takes a Worker URL as its argument, just like the native Worker, and connects it as an RPC target.

The type constructor rpcTarget.Async is provided to help create wrapper classes for rpcTarget.RPCTarget. It simply adds Promise to all method return types. rpcTarget.Async<Calculator> is:

interface {
    readNumber(): Promise<number>;
    sum(l: Float64Array): Promise<number>;
}

The general design intent of rpcTarget.RPCTarget and rpcTarget.RPCWorker is to build classes that inherit from them, but they're also suitable for keeping as private fields within an exposed class. Here are examples of both designs, for Calculator:

class CalculatorWorkerA
    extends rpcTarget.RPCWorker
    implements rpcTarget.Async<Calculator>
{
    constructor() { super(calculatorWorkerURL); }

    readNumber(): Promise<number> {
        return this.rpc("readNumber", []);
    }

    sum(l: Float64Array): Promise<number> {
        return this.rpc("sum", [l], [l.buffer]);
    }
}

class CalculatorWorkerB
    implements rpcTarget.Async<Calculator>
{
    constructor() {
        this._worker = new rpcTarget.RPCWorker(calculatorWorkerURL);
    }

    readNumber(): Promise<number> {
        return this._worker.rpc("readNumber", []);
    }

    sum(l: Float64Array): Promise<number> {
        return this._worker.rpc("sum", [l], [l.buffer]);
    }
}