# birpc [![NPM version](https://img.shields.io/npm/v/birpc?color=a1b858&label=)](https://www.npmjs.com/package/birpc) Message-based two-way remote procedure call. Useful for WebSockets and Workers communication. ## Features - Intuitive - call remote functions just like locals, with Promise to get the response - TypeScript - safe function calls for arguments and returns - Protocol agonostic - WebSocket, MessageChannel, any protocols with messages communication would work! - Zero deps, ~0.5KB ## Examples ### Using WebSocket When using WebSocket, you need to pass your custom serializer and deserializer. #### Client ```ts import type { ServerFunctions } from './types' const ws = new WebSocket('ws://url') const clientFunctions: ClientFunctions = { hey(name: string) { return `Hey ${name} from client` } } const rpc = createBirpc( clientFunctions, { post: data => ws.send(data), on: data => ws.on('message', data), // these are required when using WebSocket serialize: v => JSON.stringify(v), deserialize: v => JSON.parse(v), }, ) await rpc.hi('Client') // Hi Client from server ``` #### Server ```ts import type { ClientFunctions } from './types' import { WebSocketServer } from 'ws' const serverFunctions: ServerFunctions = { hi(name: string) { return `Hi ${name} from server` } } const wss = new WebSocketServer() wss.on('connection', (ws) => { const rpc = createBirpc( serverFunctions, { post: data => ws.send(data), on: fn => ws.on('message', fn), serialize: v => JSON.stringify(v), deserialize: v => JSON.parse(v), }, ) await rpc.hey('Server') // Hey Server from client }) ``` ### Circular References As `JSON.stringify` does not supporting circular references, we recommend using [`flatted`](https://github.com/WebReflection/flatted) as the serializer when you expect to have circular references. ```ts import { parse, stringify } from 'flatted' const rpc = createBirpc( functions, { post: data => ws.send(data), on: fn => ws.on('message', fn), // use flatted as serializer serialize: v => stringify(v), deserialize: v => parse(v), }, ) ``` ### Using MessageChannel [MessageChannel](https://developer.mozilla.org/en-US/docs/Web/API/MessageChannel) will automatically serialize the message and support circular references out-of-box. ```ts export const channel = new MessageChannel() ``` #### Bob ``` ts import type { AliceFunctions } from './types' import { channel } from './channel' const Bob: BobFunctions = { hey(name: string) { return `Hey ${name}, I am Bob` } } const rpc = createBirpc( Bob, { post: data => channel.port1.postMessage(data), on: fn => channel.port1.on('message', fn), }, ) await rpc.hi('Alice') // Hi Bob, I am Alice ``` #### Alice ``` ts import type { BobFunctions } from './types' import { channel } from './channel' const Alice: AliceFunctions = { hi(name: string) { return `Hi ${name}, I am Alice` } } const rpc = createBirpc( Alice, { post: data => channel.port2.postMessage(data), on: fn => channel.port2.on('message', fn), }, ) await rpc.hey('Alice') // Hey Alice, I am Bob ``` ### One-to-multiple Communication Refer to [./test/group.test.ts](./test/group.test.ts) as an example. ## Sponsors

## License [MIT](./LICENSE) License © 2021 [Anthony Fu](https://github.com/antfu)