rivetkit

작성자: rivet-dev

AI 에이전트, 멀티플레이어 앱, 워크플로 자동화를 위한 장기 실행 상태 저장 컴퓨팅입니다. 액터는 데이터베이스 왕복 없이 요청 간 상태를 유지하는 영구적인 인메모리 프로세스로, 0에서 수백만 개의 동시 인스턴스까지 자동 확장됩니다. WebSocket 및 이벤트를 통한 내장 실시간 통신, 순서가 지정된 메시지 처리를 위한 내구성 있는 큐, 구조화된 데이터 쿼리를 위한 SQLite를 제공합니다. 상태는 Rivet Cloud 또는 자체 호스팅을 통해 재시작 및 충돌 후에도 유지됩니다...

npx skills add https://github.com/rivet-dev/skills --skill rivetkit

Rivet Actors

Build high-performance, long-lived, in-memory processes on Rivet's actor runtime. Use this skill whenever HTTP endpoints, databases, or queues no longer meet the requirements for AI agents, realtime collaboration, workflow orchestration, or geo-distributed data.

What To Do If You Get Stuck

If something is not working as intended or you are stuck, prompt the user to join the Rivet Discord or file an issue on the Rivet GitHub to report an issue and get help.

Provide the user with a pre-generated report with:

  • Symptoms
    • If this is happening in local dev, deployed, or both
    • The error you're seeing
    • Relevant source code related to this
  • What you've tried to solve it
  • Environment
    • RivetKit version
    • Runtime (Node, Bun, etc) including version
    • If applicable, provider in use (e.g. Vercel, Railway, etc)
    • If applicable, HTTP router in use (e.g. Hono, Express, Elysia)

Debugging Actors

Use the inspector HTTP API to examine running actors. These endpoints are accessible through the gateway at /gateway/{actor_id}/inspector/*. Key endpoints:

  • GET /inspector/summary - full actor snapshot (state, connections, RPCs, queue)
  • GET /inspector/state / PATCH /inspector/state - read/write actor state
  • GET /inspector/connections - active connections
  • GET /inspector/rpcs - available actions
  • POST /inspector/action/{name} - execute an action with {"args": [...]}
  • POST /inspector/database/execute - run SQL with {"sql": "...", "args": [...]} or {"sql": "...", "properties": {...}} for reads or mutations
  • GET /inspector/queue?limit=50 - queue status
  • GET /inspector/traces?startMs=0&endMs=...&limit=1000 - trace spans (OTLP JSON)
  • GET /inspector/workflow-history - workflow history and status as JSON (nameRegistry, entries, entryMetadata)
  • POST /inspector/workflow/replay - replay a workflow from a specific step or from the beginning; returns 409 actor/workflow_in_flight if the workflow is still running
  • GET /inspector/database/schema - SQLite tables and views exposed by c.db
  • GET /inspector/database/rows?table=...&limit=100&offset=0 - paged SQLite rows for a table or view

In local dev, no auth token is needed. In production, pass Authorization: Bearer <inspector-token>, where the inspector token is the actor-specific token auto-generated on first start and persisted in the actor's internal KV at key 0x03. The Rivet dashboard retrieves this token automatically; for direct API access, fetch it through the management KV endpoint. See the debugging docs for details.

Citing Sources

When providing information from Rivet documentation, cite the canonical URL so users can learn more. Each reference file includes its canonical URL in the header metadata.

How to cite:

  • Use inline links for key concepts: "Use actor keys to uniquely identify instances."
  • Add a "Learn more" link after explanations for complex topics

Finding canonical URLs:

The Reference Map below links to reference files. Each file's header contains:

> Canonical URL: https://rivet.dev/docs/actors/actions

Use that canonical URL when citing, not the reference file path.

Examples:

  • Actions → https://rivet.dev/docs/actors/actions
  • React client → https://rivet.dev/docs/clients/react
  • Self-hosting on Kubernetes → https://rivet.dev/docs/self-hosting/kubernetes

Version Check

Before starting any work, check if the user's project is on the latest version of RivetKit (latest: 2.3.2). Look at the rivetkit version in the user's package.json (check both dependencies and devDependencies). If the installed version is older than 2.3.2, inform the user and suggest upgrading:

npm install [email protected]

If the user also uses @rivetkit/react, @rivetkit/next-js, or other @rivetkit/* client packages, suggest upgrading those too. Outdated versions may have known bugs or missing features that cause issues.

First Steps

  1. Install RivetKit (latest: 2.3.2)
    npm install [email protected]
    
  2. Define a registry with setup({ use: { /* actors */ } }).
  3. Call registry.start() to start the server. For custom HTTP server integration, use registry.handler() with a router like Hono. For serverless deployments, use registry.serve(). For runner-only mode, use registry.startEnvoy().
  4. Verify /api/rivet/metadata returns 200 before deploying.
  5. Configure Rivet Cloud or self-hosted engine
    • You must configure versioning for production builds. This is not needed for local development. See Versions & Upgrades.
  6. Integrate clients (see client guides below for JavaScript, React, or Swift)
  7. Prompt the user if they want to deploy. If so, go to Deploying Rivet Backends.

For more information, read the quickstart guide relevant to the user's project.

Project Setup

.gitignore

Every RivetKit project should have a .gitignore. Include at minimum:

node_modules/
dist/
.env

.dockerignore

Every project with a Dockerfile should have a .dockerignore to keep the image small and avoid leaking secrets:

node_modules/
dist/
.env
.git/

Dockerfile

Use this as a base Dockerfile for deploying a RivetKit project. The RIVET_RUNNER_VERSION build arg is only needed when self-hosting or using a custom runner (not needed for Rivet Compute). It lets Rivet track which version of the actor is running and drain old actors on deploy. See https://rivet.dev/docs/actors/versions for details.

FROM node:24-alpine

ARG RIVET_RUNNER_VERSION
ENV RIVET_RUNNER_VERSION=$RIVET_RUNNER_VERSION

WORKDIR /app

COPY package.json package-lock.json ./
RUN npm ci

COPY . .
RUN npm run build --if-present

CMD ["node", "dist/index.js"]

Build with:

docker build --build-arg RIVET_RUNNER_VERSION=$(date +%s) .

Adjust the CMD to match the project's entry point. If the project uses a different output directory or start command, update accordingly.

Error Handling Policy

  • Prefer fail-fast behavior by default.
  • Avoid try/catch unless it is required for a real recovery path, cleanup boundary, or to add actionable context.
  • Never swallow errors. If you add a catch, you must handle the error explicitly, at minimum by logging it.
  • When you cannot recover, log context and rethrow.

State vs Vars: Persistence Rules

c.vars is ephemeral. Data in c.vars is lost on every restart, crash, upgrade, or sleep/wake cycle. Only use c.vars for non-serializable objects (e.g., physics engines, WebSocket references, event emitters, caches) or truly transient runtime data (e.g., current input direction that doesn't matter after disconnect).

Persistent storage options. Any data that must survive restarts belongs in one of these, NOT in c.vars:

  • c.state — CBOR-serializable data for small, bounded datasets. Ideal for configuration, counters, small player lists, phase flags, etc. Keep under 128 KB. Do not store unbounded or growing data here (e.g., chat logs, event histories, spawned entity lists that grow without limit). State is read/written as a single blob on every persistence cycle.
  • c.kv — Key-value store for unbounded data. This is what c.state uses under the hood. Supports binary values. Use for larger or variable-size data like user inventories, world chunks, file blobs, or any collection that may grow over time. Keys are scoped to the actor instance.
  • c.db — SQLite database for structured or complex data. Use when you need queries, indexes, joins, aggregations, or relational modeling. Ideal for leaderboards, match histories, player pools, or any data that benefits from SQL.

Common mistake: Storing meaningful game/application data in c.vars instead of persisting it. For example, if users can spawn objects in a physics simulation, the spawn definitions (position, size, type) must be persisted in c.state (or c.kv if unbounded), even though the physics engine handles (non-serializable) live in c.vars. On restart, run() should recreate the runtime objects from the persisted data.

Deploying Rivet Backends

Assume the user is deploying to Rivet Cloud, unless otherwise specified. If user is self-hosting, read the self-hosting guides below.

  1. Verify that Rivet Actors are working in local dev
  2. Prompt the user to choose a provider to deploy to (see Connect for a list of providers, such as Vercel, Railway, etc)
  3. Follow the deploy guide for that given provider. You will need to instruct the user when you need manual intervention.

API Reference

The RivetKit OpenAPI specification is available in the skill directory at openapi.json. This file documents all HTTP endpoints for managing actors.

Misc Notes

  • The Rivet domain is rivet.dev, not rivet.gg

TypeScript Caveat: Actor Client Inference

  • In multi-file TypeScript projects, bidirectional actor calls can create a circular type dependency when both actors use c.client<typeof registry>().
  • Symptoms usually include c.state becoming unknown, actor methods becoming possibly undefined, or TS2322 / TS2722 errors after the first cross-actor call.
  • If an action returns the result of another actor call, prefer an explicit return type annotation on that action instead of relying on inference through c.client<typeof registry>().
  • If explicit return types are not enough, use a narrower client or registry type for only the actors that action needs.
  • As a last resort, pass unknown for the registry type and be explicit that this gives up type safety at that call site.

Features

  • Long-Lived, Stateful Compute: Each unit of compute is like a tiny server that remembers things between requests – no need to re-fetch data from a database or worry about timeouts. Like AWS Lambda, but with memory and no timeouts.
  • Blazing-Fast Reads & Writes: State is stored on the same machine as your compute, so reads and writes are ultra-fast. No database round trips, no latency spikes. State is persisted to Rivet for long term storage, so it survives server restarts.
  • Realtime: Update state and broadcast changes in realtime with WebSockets. No external pub/sub systems, no polling – just built-in low-latency events.
  • Infinitely Scalable: Automatically scale from zero to millions of concurrent actors. Pay only for what you use with instant scaling and no cold starts.
  • Fault Tolerant: Built-in error handling and recovery. Actors automatically restart on failure while preserving state integrity and continuing operations.

When to Use Rivet Actors

  • AI agents & sandboxes: multi-step toolchains, conversation memory, sandbox orchestration.
  • Multiplayer or collaborative apps: CRDT docs, shared cursors, realtime dashboards, chat.
  • Workflow automation: background jobs, cron, rate limiters, durable queues, backpressure control.
  • Data-intensive backends: geo-distributed or per-tenant databases, in-memory caches, sharded SQL.
  • Networking workloads: WebSocket servers, custom protocols, local-first sync, edge fanout.

Minimal Project

Backend

index.ts

Client Docs

Use the client SDK that matches your app:

Actor Quick Reference

In-Memory State

Persistent data that survives restarts, crashes, and deployments. State is persisted on Rivet Cloud or Rivet self-hosted, so it survives restarts if the current process crashes or exits.

Static Initial State

Dynamic Initial State

Documentation

Keys

Keys uniquely identify actor instances. Use compound keys (arrays) for hierarchical addressing:

Don't build keys with string interpolation like "org:${userId}" when userId contains user data. Use arrays instead to prevent key injection attacks.

Documentation

Input

Pass initialization data when creating actors. Input is only available in createState and onCreate, so store it in state if you need it later.

Documentation

Temporary Variables

Temporary data that doesn't survive restarts. Use for non-serializable objects (event emitters, connections, etc).

Static Initial Vars

Dynamic Initial Vars

Documentation

Actions

Actions are the primary way clients and other actors communicate with an actor.

Documentation

Events & Broadcasts

Events enable real-time communication from actors to connected clients.

Documentation

Connections

Access the current connection via c.conn or all connected clients via c.conns. Use c.conn.id or c.conn.state to securely identify who is calling an action. c.conn is only available for actions invoked through a connected client; stateless actor-handle calls run without a connection, so guard against that. Connection state is initialized via connState or createConnState, which receives parameters passed by the client on connect.

Static Connection Initial State

Dynamic Connection Initial State

Documentation

Queues

Use queues to process durable messages in order inside a run loop.

Documentation

Workflows

Use workflows when your run logic needs durable, replayable multi-step execution.

Documentation

Actor-to-Actor Communication

Actors can call other actors using c.client().

Documentation

Scheduling

Schedule actions to run after a delay or at a specific time. Schedules persist across restarts, upgrades, and crashes.

Documentation

Destroying Actors

Permanently delete an actor and its state using c.destroy().

Documentation

Lifecycle Hooks

Actors support hooks for initialization, background processing, connections, networking, and state changes. Use run for long-lived background loops, and use c.aborted or c.abortSignal for graceful shutdown.

Documentation

Context Types

When writing helper functions outside the actor definition, use *ContextOf<typeof myActor> to extract the correct context type. Helpers like ActionContextOf, CreateContextOf, ConnContextOf, and ConnInitContextOf are exported from "rivetkit". Do not manually define your own context interface. Always derive it from the actor definition.

Documentation

Errors

Use UserError to throw errors that are safely returned to clients. Pass metadata to include structured data. Other errors are converted to generic "internal error" for security.

Actor

Client

Documentation

Low-Level HTTP & WebSocket Handlers

For custom protocols or integrating libraries that need direct access to HTTP Request/Response or WebSocket connections, use onRequest and onWebSocket.

HTTP Handler Documentation · WebSocket Handler Documentation

Icons & Names

Customize how actors appear in the UI with display names and icons. It's recommended to always provide a name and icon to actors in order to make them easier to distinguish in the dashboard.

import { actor } from "rivetkit";

const chatRoom = actor({
	options: {
		name: "Chat Room",
		icon: "💬", // or FontAwesome: "comments", "chart-line", etc.
	},
	// ...
});

Documentation

Client Documentation

Find the full client guides here:

Common Patterns

Actors scale naturally through isolated state and message-passing. Structure your applications with these patterns:

Documentation

Actor Per Entity

Create one actor per user, document, or room. Use compound keys to scope entities:

Coordinator & Data Actors

Data actors handle core logic (chat rooms, game sessions, user data). Coordinator actors track and manage collections of data actors—think of them as an index.

Run Loop

Use a run loop for continuous background work inside an actor. Process queue messages in order, run logic on intervals, stream AI responses, or coordinate long-running tasks.

Workflow Loop

Use this pattern for long-lived, durable workflows that initialize resources, process commands in a loop, then clean up.

Documentation

Actions vs Queues

  • Actions are not durable. Use them for realtime reads, ephemeral data, and low-latency communication like player input.
  • Queues are durable. Use them to serialize mutations through the run loop, avoiding race conditions with SQLite and other local state. Callers can still wait for a response from queued work.

Authentication, Security, & CORS

  • Validate credentials in onBeforeConnect or createConnState and throw an error to reject unauthorized connections.
  • Use c.conn.state to securely identify users in actions rather than trusting action parameters.
  • For cross-origin access, validate the request origin in onBeforeConnect.

Authentication Documentation · CORS Documentation

Versions & Upgrades

When deploying new code, set a version number so Rivet can route new actors to the latest runner and optionally drain old ones. Use a build timestamp, git commit count, or CI build number as the version. It is very important to configure versioning before deploying to production. Without versioning, actors can regress by running on older runner versions, and existing actors will never be forced to migrate to new runners. They will continue running indefinitely on the old runners until they exit.

Documentation

Anti-Patterns

Never build a "god" actor

Do not put all your logic in a single actor. A god actor serializes every operation through one bottleneck, kills parallelism, and makes the entire system fail as a unit. Split into focused actors per entity.

Never create an actor per request

Actors are long-lived and maintain state across requests. Creating a new actor for every incoming request throws away the core benefit of the model and wastes resources on actor creation and teardown. Use actors for persistent entities and regular functions for stateless work.

Reference Map

Actors

Cli

Clients

Cookbook

Deploy

General

Self Hosting

rivet-dev의 다른 스킬

rivetkit-actors
rivet-dev
Rivet Actors로 상태 저장형 백엔드 구축: 고성능, 장기 실행, 인메모리, 지속형 프로세스. HTTP, 데이터베이스, 큐의 한계를 넘어설 때 사용하세요.…
official
agent-browser
rivet-dev
웹 테스트, 양식 작성, 스크린샷, 데이터 추출을 위한 브라우저 상호작용을 자동화합니다. 사용자가 웹사이트를 탐색하거나 웹과 상호작용해야 할 때 사용하세요.
official
multiplayer-game
rivet-dev
멀티플레이어 게임 구축을 위한 실용적 패턴: 매치메이킹, 틱 루프, 실시간 상태, 관심 관리, 검증. 10가지 게임 유형(배틀로얄, 아레나, IO 스타일, 오픈 월드, 파티, 물리 2D/3D, 랭크, 턴제, 방치형)에 대한 시작 코드와 아키텍처 패턴, 액터 토폴로지, 생명주기 다이어그램, 넷코드 전략 포함. 고정 실시간 루프 대 액션 기반 업데이트, 물리 엔진 선택(Rapier 2D/3D), 공간... 등 서버 시뮬레이션 기본 사항 다룸.
official
rivetkit-actors
rivet-dev
Rivet Actors로 상태 저장형 백엔드 구축: 고성능, 장기 실행, 인메모리, 지속형 프로세스. HTTP, 데이터베이스, 큐의 한계를 넘어설 때 사용하세요.…
official
rivetkit-client-javascript
rivet-dev
JavaScript 클라이언트로, 무상태 또는 상태 저장 연결을 통해 Rivet Actors에 연결합니다. 브라우저, Node.js 및 Bun 환경을 지원하며, 환경 변수 또는 명시적 구성을 통한 자동 엔드포인트 감지 기능을 제공합니다. 독립적인 요청을 위한 무상태 액션 호출과 실시간 이벤트 구독이 가능한 상태 저장 연결의 두 가지 상호작용 모드를 제공합니다. onRequest 또는 onWebSocket 핸들러를 구현하는 액터를 위한 저수준 HTTP 및 WebSocket 액세스를 포함하며, 복합 배열 기반...
official
rivetkit-client-react
rivet-dev
React 클라이언트로, 훅과 실시간 상태 관리를 통해 Rivet Actors에 연결합니다. createRivetKit()으로 타입이 지정된 훅을 생성하고, useActor()로 키와 선택적 매개변수를 사용하여 액터 인스턴스에 연결합니다. useEvent()로 액터 이벤트를 구독하고, connStatus와 error 상태를 통해 연결 수명 주기를 모니터링합니다. createClient()를 사용하여 상태 비저장 단일 호출, 액터 검색 메서드(get, getOrCreate, create, getForId), 그리고 저수준 HTTP/WebSocket 액세스를 수행합니다. 복합 배열 키를 지원합니다...
official
rivetkit-client-swift
rivet-dev
Swift 클라이언트로, 무상태 및 상태 저장 연결 모드를 통해 Rivet Actors에 연결합니다. 무상태 액션 호출과 AsyncStream을 통한 실시간 이벤트 스트리밍이 가능한 지속적 WebSocket 연결을 지원합니다. 0~5개의 위치 인수를 사용한 타입화된 액션 호출과 더 많은 인수에 대한 원시 JSON 폴백을 제공합니다. 상태 변경, 오류, 열림, 닫힘 스트림을 통한 연결 수명 주기 모니터링을 포함합니다. 커스텀 onRequest 및...을 위한 저수준 HTTP 및 WebSocket 접근을 지원합니다.
official
rivetkit-client-swiftui
rivet-dev
SwiftUI 클라이언트 라이브러리로, 반응형 바인딩 및 이벤트 스트리밍을 통해 Rivet Actors에 연결합니다. 연결 수명 주기 관리를 위한 get-or-create 의미 체계의 @Actor 프로퍼티 래퍼와 환경 설정 및 이벤트 구독을 위한 뷰 모디파이어를 제공합니다. 타입화된 비동기 액션, fire-and-forget 전송, 0–5개의 타입화된 인수를 가진 AsyncStream 이벤트 구독을 지원합니다. 연결 상태 추적, ActorError를 통한 오류 처리, 계층적 액터를 위한 복합 키 지원을 포함합니다...
official