Security News
Fluent Assertions Faces Backlash After Abandoning Open Source Licensing
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
@replit/river
Advanced tools
It's like tRPC but... with JSON Schema Support, duplex streaming and support for service multiplexing. Transport agnostic!
River allows multiple clients to connect to and make remote procedure calls to a remote server as if they were local procedures.
River provides a framework for long-lived streaming Remote Procedure Calls (RPCs) in modern web applications, featuring advanced error handling and customizable retry policies to ensure seamless communication between clients and servers.
River provides a framework similar to tRPC and gRPC but with additional features:
See PROTOCOL.md for more information on the protocol.
Before proceeding, ensure you have TypeScript 5 installed and configured appropriately:
Ensure your tsconfig.json
is configured correctly:
You must verify that:
compilerOptions.moduleResolution
is set to "bundler"
compilerOptions.strictFunctionTypes
is set to true
compilerOptions.strictNullChecks
is set to true
or, preferably, that:
compilerOptions.moduleResolution
is set to "bundler"
compilerOptions.strict
is set to true
Like so:
{
"compilerOptions": {
"moduleResolution": "bundler",
"strict": true
// Other compiler options...
}
}
If these options already exist in your tsconfig.json
and don't match what is shown above, modify them. River is designed for "strict": true
, but technically only strictFunctionTypes
and strictNullChecks
being set to true
is required. Failing to set these will cause unresolvable type errors when defining services.
Install River and Dependencies:
To use River, install the required packages using npm:
npm i @replit/river @sinclair/typebox
If you plan on using WebSocket for the underlying transport, also install
npm i ws isomorphic-ws
rpc
whose handler has a signature of Input -> Result<Output, Error>
.upload
whose handler has a signature of AsyncIterableIterator<Input> -> Result<Output, Error>
.subscription
whose handler has a signature of Input -> Pushable<Result<Output, Error>>
.stream
whose handler has a signature of AsyncIterableIterator<Input> -> Pushable<Result<Output, Error>>
.Transport
to work.
First, we create a service using ServiceSchema
:
import { ServicaSchema, Procedure, Ok } from '@replit/river';
import { Type } from '@sinclair/typebox';
export const ExampleService = ServiceSchema.define(
// configuration
{
// initializer for shared state
initializeState: () => ({ count: 0 }),
},
// procedures
{
add: Procedure.rpc({
input: Type.Object({ n: Type.Number() }),
output: Type.Object({ result: Type.Number() }),
errors: Type.Never(),
// note that a handler is unique per user RPC
async handler(ctx, { n }) {
// access and mutate shared state
ctx.state.count += n;
return Ok({ result: ctx.state.count });
},
}),
},
);
Then, we create the server:
import http from 'http';
import { WebSocketServer } from 'ws';
import { WebSocketServerTransport } from '@replit/river/transport/ws/server';
import { createServer } from '@replit/river';
// start websocket server on port 3000
const httpServer = http.createServer();
const port = 3000;
const wss = new WebSocketServer({ server: httpServer });
const transport = new WebSocketServerTransport(wss, 'SERVER');
export const server = createServer(transport, {
example: ExampleService,
});
export type ServiceSurface = typeof server;
httpServer.listen(port);
In another file for the client (to create a separate entrypoint),
import WebSocket from 'isomorphic-ws';
import { WebSocketClientTransport } from '@replit/river/transport/ws/client';
import { createClient } from '@replit/river';
import type ServiceSurface from './server';
const websocketUrl = `ws://localhost:3000`;
const transport = new WebSocketClientTransport(
async () => new WebSocket(websocketUrl),
'my-client-id',
);
const client = createClient<ServiceSurface>(
transport,
'SERVER', // transport id of the server in the previous step
true, // whether to eagerly connect to the server on creation (optional argument)
);
// we get full type safety on `client`
// client.<service name>.<procedure name>.<procedure type>()
// e.g.
const result = await client.example.add.rpc({ n: 3 });
if (result.ok) {
const msg = result.payload;
console.log(msg.result); // 0 + 3 = 3
}
To add logging,
import { bindLogger, setLevel } from '@replit/river/logging';
bindLogger(console.log);
setLevel('info');
River defines two types of reconnects:
You can listen for transparent reconnects via the connectionStatus
events, but realistically, no applications should need to listen for this unless it is for debugging purposes. Hard reconnects are signaled via sessionStatus
events.
If your application is stateful on either the server or the client, the service consumer should wrap all the client-side setup with transport.addEventListener('sessionStatus', (evt) => ...)
to do appropriate setup and teardown.
transport.addEventListener('connectionStatus', (evt) => {
if (evt.status === 'connect') {
// do something
} else if (evt.status === 'disconnect') {
// do something else
}
});
transport.addEventListener('sessionStatus', (evt) => {
if (evt.status === 'connect') {
// do something
} else if (evt.status === 'disconnect') {
// do something else
}
});
We've also provided an end-to-end testing environment using Next.js
, and a simple backend connected with the WebSocket transport that you can play with on Replit.
You can find more service examples in the E2E test fixtures
npm i
-- install dependenciesnpm run check
-- lintnpm run format
-- formatnpm run test
-- run testsnpm run publish
-- cut a new release (should bump version in package.json first)FAQs
It's like tRPC but... with JSON Schema Support, duplex streaming and support for service multiplexing. Transport agnostic!
The npm package @replit/river receives a total of 9,430 weekly downloads. As such, @replit/river popularity was classified as popular.
We found that @replit/river demonstrated a healthy version release cadence and project activity because the last version was released less than a year ago. It has 25 open source maintainers collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
Research
Security News
Socket researchers uncover the risks of a malicious Python package targeting Discord developers.
Security News
The UK is proposing a bold ban on ransomware payments by public entities to disrupt cybercrime, protect critical services, and lead global cybersecurity efforts.