Skip to main content

Documentation Index

Fetch the complete documentation index at: https://docs.deep.space/llms.txt

Use this file to discover all available pages before exploring further.

The records API is the primary surface for working with collections. Every hook and provider on this page is imported from deepspace.
import {
  RecordProvider, RecordScope, ScopeRegistryProvider,
  useQuery, useMutations, useUsers, useUserLookup, useRecordContext,
} from 'deepspace'
For schemas and column types, see the worker schemas reference. For RBAC rules, see permissions.

Providers

<RecordProvider>

Initializes the WebSocket and in-memory record store. Required ancestor of every records hook.
PropTypeDescription
roomIdstring (optional)Scope ID for the default room (usually app:<APP_NAME>). Omit for multi-scope mode and use <RecordScope> to mount scopes instead.
schemasCollectionSchema[] (optional)All collections this provider tree may query.
wsUrlstring (optional)Override the WebSocket URL. Defaults to current origin.
fetchUser() => Promise<UserProfile | null> (optional)Custom user-profile fetcher. Defaults to using the Better Auth session.
allowAnonymousboolean (optional)Connect without a JWT (default false). Required for public pages. See authentication.
getAuthToken() => Promise<string | null> (optional)Custom token fetcher. Defaults to the SDK’s.
<RecordProvider allowAnonymous>
  <App />
</RecordProvider>

<RecordScope>

Mounts a specific scope (Durable Object instance). Nest for additional scopes.
PropTypeDescription
roomIdstringScope ID (e.g. app:my-app, conv:abc123).
schemasCollectionSchema[]Collections in this scope.
appIdstringApp identifier - used for cross-app routing.
sharedScopesArray<{ roomId, schemas }>Cross-app scopes to mount alongside the primary. See cross-app shared scopes.
wsUrlstring (optional)Override WebSocket URL.
wsPathPrefixstring (optional)Override path prefix (default /ws).
isolatedbooleanIf true, this scope’s store is independent of parent stores.

<ScopeRegistryProvider>

Required once near the root if your app uses shared scopes via sharedScopes. Coordinates routing between cross-app and per-app DOs.

useQuery<T>(collection, options?)

Subscribes to a collection. Returns a reactive array of envelopes.
function useQuery<T>(
  collection: string,
  options?: {
    where?: Partial<T>
    orderBy?: string
    orderDir?: 'asc' | 'desc'
    limit?: number
  },
): {
  records: Envelope<T>[]
  status: 'loading' | 'ready' | 'error'
  error?: string
}
Subscribe to every record in a collection. The hook re-renders whenever any user mutates a record visible to the caller’s permissions.
type Note = { title: string; body: string }

function Notes() {
  const { records, status } = useQuery<Note>('notes')

  if (status === 'loading') return <Skeleton />
  return records.map((r) => <li key={r.recordId}>{r.data.title}</li>)
}
Envelope shape:
type Envelope<T> = {
  recordId: string
  data: T
  createdBy: string
  createdAt: string
  updatedAt: string
}
User fields live under .data. r.title returns undefined - always reach for r.data.title. TypeScript catches this if you pass a row type to useQuery<T>.

useMutations<T>(collection)

Returns mutation functions for the given collection. Each mutation applies optimistically - the local store updates before the server confirms.
function useMutations<T>(collection: string): {
  create:          (data: T)                       => Promise<string>
  put:             (id: string, patch: Partial<T>) => Promise<void>
  remove:          (id: string)                    => Promise<void>
  createConfirmed: (data: T)                       => Promise<string>
  putConfirmed:    (id: string, patch: Partial<T>) => Promise<void>
  removeConfirmed: (id: string)                    => Promise<void>
}
create takes the full row shape and returns the new recordId. The ID is generated on the client (timestamp + random suffix) before the write is sent, so the promise resolves with the ID immediately while the server processes the mutation in the background.
const { create } = useMutations<Note>('notes')

const id = await create({
  title: 'Untitled',
  body: '',
  pinned: false,
})
If you need to confirm the row was actually persisted (e.g., before navigating away), use createConfirmed instead - it awaits server acknowledgment:
const id = await createConfirmed({ title: 'New', body: '', pinned: false })
navigate(`/notes/${id}`)
MethodSemanticsReturns
createOptimisticPromise<string> (client-generated recordId)
putOptimistic, mergePromise<void>
removeOptimisticPromise<void>
createConfirmedWaits for DO ackPromise<string>
putConfirmedWaits for DO ackPromise<void>
removeConfirmedWaits for DO ackPromise<void>

useUsers()

Returns the users collection with role-management helpers.
type RoomUser = {
  id: string
  email: string
  name: string
  imageUrl?: string
  role: string
  createdAt: string
  lastSeenAt: string
}

function useUsers(): {
  users: RoomUser[]
  usersLoaded: boolean
  setRole: (userId: string, role: string) => void
  refresh: () => void
}
setRole accepts a free-form role string (e.g. 'admin', 'intern', or any value your schema understands) and dispatches the change without waiting for an ack. The Durable Object enforces who is allowed to call it. See permissions for how roles map onto collection RBAC rules.

useUserLookup()

O(1) wrapper around useUsers() for resolving userIds to display fields.
type UserInfo = {
  id: string
  email: string
  name: string
  imageUrl?: string
  role: string
}

function useUserLookup(): {
  users: RoomUser[]
  usersLoaded: boolean
  userMap: Map<string, UserInfo>
  getUser:  (userId: string) => UserInfo | null
  getEmail: (userId: string) => string | null
  getName:  (userId: string) => string | null
}
const { getName } = useUserLookup()
<p>By {getName(message.authorId) ?? 'unknown'}</p>
There is no getRole or getImageUrl - read those off getUser(id)?.role or getUser(id)?.imageUrl.

useRecordContext()

Low-level access to the record-store context (WebSocket send/receive primitives, ready state, user profile, etc.). Useful for building custom hooks or imperative reads outside React’s render cycle. Most apps never need this.

See also