Typescript implementation of Arri RPC. It's built on top of H3 and uses esbuild for bundling.
Parameters and responses are defined using @arrirpc/schema for automatic validation and serialization of inputs and outputs and to generate Arri Type Definitions for client generators.
# npm
npx arri init [project-name]
cd [project-name]
npm install
npm run dev
# pnpm
pnpm dlx arri init [project-name]
cd [project-name]
pnpm install
pnpm run dev
# npm
npm install -D arri
npm install @arrirpc/server @arrirpc/schema
# pnpm
pnpm install -D arri
pnpm install @arrirpc/server @arrirpc/schema
A basic Arri app directory looks something like this:
|-- project-dir
|-- .arri // temp files go here
|-- .output // final bundle goes here
|-- src
|-- procedures // .rpc.ts files go here
|-- app.ts
|-- arri.config.ts
|-- package.json
|-- tsconfig.json
|
Both .arri
and .output
should be added to your .gitignore
file
.arri
.output
node_modules
Create an arri.config.ts
in the project directory
// arri.config.ts
import { defineConfig, servers, generators } from 'arri';
export default defineConfig({
server: servers.tsServer(),
generators: [
generators.typescriptClient({
// options
}),
generators.dartClient({
// options
}),
],
});
Create an app entry file in your src directory. The name of the file must match whatever you put as the entry
in your arri.config.ts
.
// ./src/app.ts
import { ArriApp } from 'arri';
const app = new ArriApp();
export default app;
Setup your npm scripts:
{
"name": "my-arri-app",
"type": "module",
"scripts": {
"dev": "arri dev",
"build": "arri build"
},
"dependencies": {
...
},
"devDependencies": {
...
}
}
Arri RPC comes with an optional file based router that will automatically register functions in the ./procedures
directory that end with the .rpc.ts
file extension.
|-- src
|-- procedures
|-- sayHello.rpc.ts // becomes sayHello()
|-- users
|-- getUser.rpc.ts // becomes users.getUser()
|-- updateUser.rpc.ts // becomes users.updateUser()
Example .rpc.ts
file
// ./src/users/getUser.rpc.ts
import { defineRpc } from '@arrirpc/server';
import { a } from '@arrirpc/schema';
export default defineRpc({
params: a.object({
userId: a.string(),
}),
response: a.object({
id: a.string(),
name: a.string(),
createdAt: a.timestamp(),
}),
handler({ params }) {
// function body
},
});
export default defineConfig({
servers: servers.tsServer({
procedureDir: 'procedures', // change which directory to look for procedures (This is relative to the srcDir)
procedureGlobPatterns: ['**/*.rpc.ts'], // change the file name glob pattern for finding rpcs
}),
// rest of config
});
For those that want to opt out of the file-based routing system you can manually register procedures like so.
// using the app instance
const app = new ArriApp()
app.rpc('sayHello',
defineRpc({...})
);
// defining a service
const app = new ArriApp();
const usersService = defineService("users");
usersService.rpc("getUser", defineRpc({...}));
usersService.rpc("createUser", defineRpc({...}));
// register the service on the app instance
app.use(usersService);
There's also a shorthand for initializing services with procedures
// this is equivalent to what we showed above
const usersService = defineService("users", {
getUser: defineRpc({..}),
createUser: defineRpc({..}),
});
Event stream procedures make use of Server Sent Events to stream events to clients.
Arri Event streams sent the following event types:
message
- A standard message with the response data serialized as JSONdone
- A message to tell clients that there will be no more eventsping
- A message periodically sent by the server to keep the connection alive.
/// message event ///
id: string | undefined;
event: 'message';
data: Response; // whatever you have specified as the response serialized to json
/// done event ///
event: 'done';
data: 'this stream has ended';
/// ping event ///
event: 'ping';
data: '';
// procedures/users/watchUser.rpc.ts
export default defineEventStreamRpc({
params: a.object({
userId: a.string(),
}),
response: a.object({
id: a.string(),
name: a.string(),
createdAt: a.timestamp(),
updatedAt: a.timestamp(),
}),
handler({ params, stream }) {
// initialize the stream and send it to the client
stream.send();
// send a message every second
const interval = setInterval(async () => {
await stream.push({
id: '1',
name: 'John Doe',
createdAt: new Date(),
updatedAt: new Date(),
});
}, 1000);
// cleanup when the client disconnects
stream.on('close', () => {
clearInterval(interval);
});
},
});
// send the stream to the client. Must be called before pushing any messages
stream.send()
// push a new message to the client
stream.push(data: Data, eventId?: string)
// close the stream and tell the client that there will be no more messages
stream.close()
// register a callback that will fire after the stream has been close by the server or the connection has been dropped
stream.onClosed(cb: () => any)
Very experimental. Do not use in production.
// Websocket procedures work really well with discriminated unions
const IncomingMsg = a.discriminator('type', {
FOO: a.object({
message: a.string(),
}),
PING: a.object({
message: a.string(),
})
});
const OutgoingMsg = a.discriminator('type', {
BAR: a.object({
message: a.string(),
}),
PONG: a.object({
message: a.string()
})
});
export default defineWebsocketRpc(
params: IncomingMsg,
response: OutgoingMsg,
handler: {
onOpen: (peer) => {},
onMessage: (peer, message) => {
switch(message.type) {
case "FOO":
peer.send({
type: "BAR",
message: "You sent a FOO message"
});
break;
case "PING":
peer.send({
type: "PONG",
message: "You sent a PING message"
});
break;
}
},
onError: (peer, error) => {}
}
)
Under the hood Websocket RPCs use crossws.
The possible payloads sent by the server will look like the following:
event: message
data: <response serialized to json>
event: error
data: {"code": <some-err-code>, "message": <some-error-msg>}
You can also add generic endpoints for instances when a message-based RPC endpoint doesn't fit.
// using the app instance
const app = new ArriApp();
app.route({
method: "get",
path: "/hello-world",
handler(event) {
return "hello world";
},
});
// using a sub-router
const app = new ArriApp();
const router = new ArriRouter();
router.route({
method: "get",
path: "/hello-world",
handler(event) {
return "hello world",
}
})
app.use(router)
// sup-routers can also specify a route prefix
const router = new ArriRouter("/v1")
router.route({
method: "get",
path: "/hello-world", // this will become /v1/hello-world
handler(event) {
return "hello world"
}
});
const app = new ArriApp();
const requestLoggerMiddleware = defineMiddleware((event) => {
console.log(`new request at ${event.path}`);
});
app.use(requestLoggerMiddleware);
Any values added to event.context
will become available in the rpc instance
const authMiddleware = defineMiddleware(async (event) => {
// assume you did something to get the user from the request
event.context.user = {
id: 1,
name: 'John Doe',
email: 'johndoe@gmail.com',
};
});
app.rpc('sayHello', {
params: undefined,
response: a.object({
message: a.string(),
}),
// user is available here
handler({ user }) {
return {
message: `Hello ${user.name}`,
};
},
});
To get type safety for these new properties create a .d.ts
file and augment the ArriEventContext
provided by @arrirpc/server
import '@arrirpc/server';
declare module '@arrirpc/server' {
interface ArriEventContext {
user?: {
id: number;
name: string;
email: string;
};
}
}
// arri.config.ts
import { defineConfig, servers, generators } from "arri";
export default defineConfig({
server: servers.tsServer(),
generators: [
generators.typescriptClient({...}),
generators.dartClient({...}),
generators.kotlinClient({...})
generators.someGenerator({...})
]
});
For info on what generators are available see here
For info on how to create your own generator see @arrirpc/codegen-utils
The server generates a __definition.json
file that acts as a schema for all of the procedures and models in the API. By default this schema is viewable from /__definition
when the server is running, but it can be modified. The endpoint is also relative to the rpcRoutePrefix
option.
It looks something like this:
{
"procedures": {
"sayHello": {
"transport": "http",
"path": "/say-hello",
"method": "post",
"params": "SayHelloParams",
"response": "SayHelloResponse"
}
// rest of procedures
},
"definitions": {
"SayHelloParams": {
"properties": {
"name": {
"type": "string"
}
}
},
"SayHelloResponse": {
"properties": {
"message": {
"type": "string"
}
}
}
// rest of models
}
}
Arri is able to use this schema file to automatically generate clients in multiple languages. In this way it works similarly to an Open API schema, but with much better code-generation support. I've made a lot of deliberate choices in designing this schema to make code-generation easier and more consistent across languages. For example, Arri schemas use a superset of JSON Type Definition for their models instead of JSON Schema.
Every procedure maps to a different url based on it's name. For example given the following file structure:
|--src
|--procedures
|--getStatus.rpc.ts
|--users
|--getUser.rpc.ts
|--updateUser.rpc.ts
We will get the following endpoints:
POST /get-status
POST /users/get-user
POST /users/update-user
(Note: these will always be relative to the `rpcRoutePrefix` option)
By default all procedures will become post requests, but you can change this when creating a procedure:
// procedures/users/getUser.rpc.ts
export default defineRpc({
method: 'get',
// rest of config
});
The supported HTTP methods are as follows:
- post
- get
- delete
- patch
- put
When using a get method the RPC params will be mapped as query parameters which will be coerced into their type using the a.coerce
method from arri-validate
. Get methods support all basic scalar types however arrays and nested objects are not supported.
Arri is built on top of H3 so many of the concepts that apply to H3 also apply to Arri.
Arri re-eports all of the H3 utilities.
import { getRequestIP, setResponseHeader } from '@arrirpc/server';
You can access H3 events from inside procedures handlers.
defineRpc({
params: undefined,
response: undefined,
handler(_, event) {
getRequestIP(event);
},
});
defineEventStreamRpc({
params: undefined,
response: undefined,
handler(_, event) {
getRequestIP(event);
},
});
Arri server is just an H3 app under the hood so you can start it the same way you would start an H3 app. Although you should note that currently the filed based router only works when using the Arri CLI.
import { createServer } from 'node:http';
import { ArriApp, toNodeListener } from '@arrirpc/server';
const app = new ArriApp();
createServer(toNodeListener(app.h3App)).listen(process.env.PORT || 3000);
# start the dev server
arri dev [flags]
# create a production build
arri build [flags]
# create a new project
arri init [dir]
# run codegen
arri codegen [path-to-definition-file]
Run nx build arri-rpc
to build the library.
Run nx test arri-rpc
to execute the unit tests via Vitest.