UNPKG

convex

Version:

Client for the Convex Cloud

329 lines (297 loc) 7.42 kB
import type { UserIdentityAttributes } from "../../server/authentication.js"; export type { UserIdentityAttributes } from "../../server/authentication.js"; import { JSONValue, Base64 } from "../../values/index.js"; import { Long } from "../long.js"; /** * Shared schema */ export function u64ToLong(encoded: EncodedU64): U64 { const integerBytes = Base64.toByteArray(encoded); return Long.fromBytesLE(Array.from(integerBytes)); } export function longToU64(raw: U64): EncodedU64 { const integerBytes = new Uint8Array(raw.toBytesLE()); return Base64.fromByteArray(integerBytes); } export function parseServerMessage( encoded: EncodedServerMessage, ): ServerMessage { switch (encoded.type) { case "FatalError": case "AuthError": case "ActionResponse": case "Ping": { return { ...encoded }; } case "MutationResponse": { if (encoded.success) { return { ...encoded, ts: u64ToLong(encoded.ts) }; } else { return { ...encoded }; } } case "Transition": { return { ...encoded, startVersion: { ...encoded.startVersion, ts: u64ToLong(encoded.startVersion.ts), }, endVersion: { ...encoded.endVersion, ts: u64ToLong(encoded.endVersion.ts), }, }; } default: { const _exhaustivenessCheck: never = encoded; } } return undefined as never; } export function encodeClientMessage( message: ClientMessage, ): EncodedClientMessage { switch (message.type) { case "Authenticate": case "ModifyQuerySet": case "Mutation": case "Action": case "Event": { return { ...message }; } case "Connect": { if (message.maxObservedTimestamp !== undefined) { return { ...message, maxObservedTimestamp: longToU64(message.maxObservedTimestamp), }; } else { return { ...message, maxObservedTimestamp: undefined }; } } default: { const _exhaustivenessCheck: never = message; } } return undefined as never; } type U64 = Long; type EncodedU64 = string; /** * Unique nonnegative integer identifying a single query. */ export type QueryId = number; // nonnegative int export type QuerySetVersion = number; // nonnegative int export type RequestId = number; // nonnegative int export type IdentityVersion = number; // nonnegative int /** * A serialized representation of decisions made during a query's execution. * * A journal is produced when a query function first executes and is re-used * when a query is re-executed. * * Currently this is used to store pagination end cursors to ensure * that pages of paginated queries will always end at the same cursor. This * enables gapless, reactive pagination. * * `null` is used to represent empty journals. * @public */ export type QueryJournal = string | null; /** * Client message schema */ type Connect = { type: "Connect"; sessionId: string; connectionCount: number; lastCloseReason: string | null; maxObservedTimestamp?: TS; }; export type AddQuery = { type: "Add"; queryId: QueryId; udfPath: string; args: JSONValue[]; journal?: QueryJournal; /** * @internal */ componentPath?: string; }; export type RemoveQuery = { type: "Remove"; queryId: QueryId; }; export type QuerySetModification = { type: "ModifyQuerySet"; baseVersion: QuerySetVersion; newVersion: QuerySetVersion; modifications: (AddQuery | RemoveQuery)[]; }; export type MutationRequest = { type: "Mutation"; requestId: RequestId; udfPath: string; args: JSONValue[]; // Execute the mutation on a specific component. // Only admin auth is allowed to run mutations on non-root components. componentPath?: string; }; export type ActionRequest = { type: "Action"; requestId: RequestId; udfPath: string; args: JSONValue[]; // Execute the action on a specific component. // Only admin auth is allowed to run actions on non-root components. componentPath?: string; }; export type AdminAuthentication = { type: "Authenticate"; tokenType: "Admin"; value: string; baseVersion: IdentityVersion; impersonating?: UserIdentityAttributes; }; export type Authenticate = | AdminAuthentication | { type: "Authenticate"; tokenType: "User"; value: string; baseVersion: IdentityVersion; } | { type: "Authenticate"; tokenType: "None"; baseVersion: IdentityVersion; }; export type Event = { type: "Event"; eventType: string; event: any; }; export type ClientMessage = | Connect | Authenticate | QuerySetModification | MutationRequest | ActionRequest | Event; type EncodedConnect = Omit<Connect, "maxObservedTimestamp"> & { maxObservedTimestamp?: EncodedTS; }; type EncodedClientMessage = | EncodedConnect | Authenticate | QuerySetModification | MutationRequest | ActionRequest | Event; /** * Server message schema */ export type TS = U64; type EncodedTS = EncodedU64; type LogLines = string[]; export type StateVersion = { querySet: QuerySetVersion; ts: TS; identity: IdentityVersion; }; type EncodedStateVersion = Omit<StateVersion, "ts"> & { ts: EncodedTS }; type StateModification = | { type: "QueryUpdated"; queryId: QueryId; value: JSONValue; logLines: LogLines; // Optional because old backend versions don't send this. journal?: QueryJournal; } | { type: "QueryFailed"; queryId: QueryId; errorMessage: string; logLines: LogLines; errorData: JSONValue; // Optional because old backend versions don't send this. journal?: QueryJournal; } | { type: "QueryRemoved"; queryId: QueryId; }; export type Transition = { type: "Transition"; startVersion: StateVersion; endVersion: StateVersion; modifications: StateModification[]; }; type MutationSuccess = { type: "MutationResponse"; requestId: RequestId; success: true; result: JSONValue; ts: TS; logLines: LogLines; }; type MutationFailed = { type: "MutationResponse"; requestId: RequestId; success: false; result: string; logLines: LogLines; errorData?: JSONValue; }; export type MutationResponse = MutationSuccess | MutationFailed; type ActionSuccess = { type: "ActionResponse"; requestId: RequestId; success: true; result: JSONValue; logLines: LogLines; }; type ActionFailed = { type: "ActionResponse"; requestId: RequestId; success: false; result: string; logLines: LogLines; errorData?: JSONValue; }; export type ActionResponse = ActionSuccess | ActionFailed; export type AuthError = { type: "AuthError"; error: string; baseVersion?: IdentityVersion; }; type FatalError = { type: "FatalError"; error: string; }; type Ping = { type: "Ping"; }; export type ServerMessage = | Transition | MutationResponse | ActionResponse | FatalError | AuthError | Ping; type EncodedTransition = Omit<Transition, "startVersion" | "endVersion"> & { startVersion: EncodedStateVersion; endVersion: EncodedStateVersion; }; type EncodedMutationSuccess = Omit<MutationSuccess, "ts"> & { ts: EncodedTS }; type EncodedMutationResponse = MutationFailed | EncodedMutationSuccess; type EncodedServerMessage = | EncodedTransition | EncodedMutationResponse | ActionResponse | FatalError | AuthError | Ping;