Server
This page is the PostgreSQL/server runtime overview. It explains what the server owns, which sync flows it serves, and which contracts host applications must satisfy. Detailed guidance for server-originated writes lives on the separate Server-Originated Writes page.
What The Server Owns
The server treats registered PostgreSQL business tables as authoritative state.
oversync.SyncService is the low-level PostgreSQL sync runtime. It owns:
- schema bootstrap and validation for the supported registered-table envelope
- first-connect lifecycle state
- staged push-session creation, chunk upload, and commit
- committed bundle capture from business-table transactions
- pull and snapshot serving
- source sequencing, retirement, and retained-history enforcement
Runtime Tables
Layout ownership and registered-table catalog:
sync.meta: one-row layout marker; startup fails closed if the layout name does not match the running buildsync.table_catalog: deterministic registered-table catalog with compacttable_id, visible sync-key column, and key kind
Authoritative replication state:
sync.user_state: externaluser_id, internaluser_pk, per-user bundle sequencing, and retained-floor trackingsync.source_state: durable per-source committedsource_bundle_idwatermark and source retirement/replacement statesync.row_state: authoritative current row version and tombstone state keyed by(user_pk, table_id, key_bytes)sync.bundle_log: one row per retained committed sync bundle, with the unique(user_pk, source_id, source_bundle_id)replay/idempotency keysync.bundle_rows: retained committed row effects using compacttable_id,key_bytes, andop_code
Transport and lifecycle state:
sync.scope_state: durable first-connect authority state keyed byuser_pksync.push_sessions: active staging-only push sessionssync.push_session_rows: compact staged rows for each push sessionsync.snapshot_sessions: active frozen snapshot session metadatasync.snapshot_session_rows: materialized rows for each snapshot session, ordered by compact table/key identity
Row-bearing server tables store compact internal identifiers (user_pk, table_id, key_bytes,
and op_code). Wire responses reconstruct visible schema, table, structured key, operation
strings, and payloads from sync.table_catalog and the stored row data.
Main Flows
First Connect
Clients call POST /sync/connect to resolve first authority for one scope.
Possible outcomes:
remote_authoritative: the scope was already initialized, even if it is currently emptyinitialize_local: this source won the initialization lease and may seed server state from local pending rowsinitialize_empty: the scope was uninitialized and the server established authoritative empty stateretry_later: another initializer currently owns the lease, or the server is applying a bounded empty-first deferral optimization
retry_later is a normal lifecycle result, not an auth failure.
Push
The normal client write flow is:
POST /sync/push-sessionsPOST /sync/push-sessions/{push_id}/chunksPOST /sync/push-sessions/{push_id}/commit
Accepted-push recovery fetches authoritative bundle rows through
GET /sync/committed-bundles/{bundle_seq}/rows.
Retained duplicate replay is resolved through sync.bundle_log, not through push-session state.
If the exact source tuple is older than retained bundle history but sync.source_state proves that
the source bundle id was already committed, the server returns history_pruned instead of
accepting it again. If a source sends a future bundle id, the server returns
source_sequence_out_of_order; commit revalidation can return source_sequence_changed.
Pull And Snapshot
GET /sync/pullreturns complete committed bundles onlyPOST /sync/snapshot-sessionsmaterializes one frozen current after-image inside PostgreSQLGET /sync/snapshot-sessions/{snapshot_id}returns deterministic chunks from that frozen snapshot- if a client checkpoint falls behind the retained bundle floor, the server returns
history_pruned
Pull and committed-bundle replay are only guaranteed above retained_bundle_floor. Rows at or
below that floor are outside the retained-history contract even if physical pruning has not deleted
them yet. Snapshot creation reads live business tables plus sync.row_state from one PostgreSQL
transaction snapshot and fails closed if they disagree.
Server-Originated Writes
If your application writes registered PostgreSQL tables outside client push handling, use:
ScopeManager.ExecWrite(...)in the common caseWithinSyncBundle(...)only when your application already manages exact(user_id, source_id, source_bundle_id)tuples directly
That topic has enough runtime detail to deserve its own page. See Server-Originated Writes.
Registered Table Requirements
Registered PostgreSQL tables must satisfy these rules before bootstrap:
- exactly one visible sync key column per registered table
- visible sync key type must be
uuidortext - every registered table must define
_sync_scope_id TEXT NOT NULL (_sync_scope_id, sync_key)must be unique- every unique constraint or unique index on a registered table must include
_sync_scope_id - registered table sets must be FK-closed
- registered-to-registered foreign keys must be scope-inclusive and
DEFERRABLE - supported
ON DELETE/ON UPDATEactions areNO ACTION,RESTRICT,CASCADE,SET NULL, andSET DEFAULT - supported
MATCHoptions are empty,NONE, orSIMPLE DEFERRABLE INITIALLY DEFERREDis recommendedDEFERRABLE INITIALLY IMMEDIATEis accepted- partial, predicate, and expression unique indexes are unsupported on registered tables
Bootstrap fails closed with UnsupportedSchemaError when the registered schema is outside the
supported envelope.
Auth Contract
The handlers expect the host application to authenticate first and place
oversync.Actor{UserID, SourceID} into request context.
The built-in transport helper is oversync.ActorMiddleware(...), which reads
Oversync-Source-ID after the host auth layer has already established trusted user_id in
request context.
_sync_scope_id is derived from Actor.UserID, enforced only on the authoritative PostgreSQL
side, and excluded from client-visible payloads, conflicts, pulls, and snapshots.