@better-state/server 0.1.0 → 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,102 @@
1
+ /**
2
+ * StorageAdapter — the contract any persistence backend must implement.
3
+ *
4
+ * SQLite ships with the MVP. To add Postgres, Redis, or any other backend,
5
+ * implement this interface and pass it to the server at boot.
6
+ *
7
+ * All methods are synchronous today (SQLite is sync). When a Postgres adapter
8
+ * is added, this interface can be migrated to async — the surface is small
9
+ * enough that the change is mechanical.
10
+ */
11
+ export interface NamespaceRecord {
12
+ id: string;
13
+ name: string;
14
+ api_key: string;
15
+ created_at: number;
16
+ }
17
+ export interface StateRecord {
18
+ id: string;
19
+ namespace: string;
20
+ key: string;
21
+ initial: string;
22
+ snapshot: string;
23
+ version: number;
24
+ created_at: number;
25
+ updated_at: number;
26
+ }
27
+ export interface EventRecord {
28
+ id: string;
29
+ state_id: string;
30
+ client_id: string;
31
+ client_ts: number;
32
+ server_ts: number;
33
+ seq: number;
34
+ mutation: string;
35
+ meta: string | null;
36
+ created_at: number;
37
+ }
38
+ export interface EventInput {
39
+ id: string;
40
+ stateId: string;
41
+ clientId: string;
42
+ clientTs: number;
43
+ serverTs: number;
44
+ seq: number;
45
+ mutation: string;
46
+ meta: string | null;
47
+ }
48
+ export interface StorageAdapter {
49
+ /** Run migrations / create tables. Called once at boot. */
50
+ init(): void;
51
+ /** Clean shutdown (close connections, etc.) */
52
+ close(): void;
53
+ /** List all namespaces (api_key excluded from result for safety). */
54
+ listNamespaces(): Pick<NamespaceRecord, "id" | "name" | "created_at">[];
55
+ /** Find a namespace by its hashed API key. Returns just the id, or null. */
56
+ findNamespaceByApiKey(apiKeyHash: string): {
57
+ id: string;
58
+ } | null;
59
+ /** Find a namespace by name. Returns the full record, or null. */
60
+ findNamespaceByName(name: string): NamespaceRecord | null;
61
+ /** Insert a new namespace. */
62
+ createNamespace(ns: NamespaceRecord): void;
63
+ /** Find a state by (namespace, key). */
64
+ findState(namespace: string, key: string): StateRecord | null;
65
+ /** Find a state by primary key. */
66
+ findStateById(id: string): StateRecord | null;
67
+ /** List states for a namespace (snapshot included). */
68
+ listStates(namespace: string): StateRecord[];
69
+ /** Insert a new state row. */
70
+ createState(state: StateRecord): void;
71
+ /** Update the snapshot, version, and updated_at for a state. */
72
+ updateSnapshot(stateId: string, snapshot: string, version: number, updatedAt: number): void;
73
+ /** Get the current max sequence number for a state. null if no events. */
74
+ getMaxSeq(stateId: string): number | null;
75
+ /** Insert a batch of events atomically (single transaction). */
76
+ insertEvents(events: EventInput[]): void;
77
+ /** Get all events for replay: just mutation + meta, ordered by (server_ts, seq). */
78
+ getEventsForReplay(stateId: string): {
79
+ mutation: string;
80
+ meta: string | null;
81
+ }[];
82
+ /** Get paginated event history (full records). */
83
+ getEventHistory(stateId: string, limit: number, cursor?: string): {
84
+ events: EventRecord[];
85
+ hasMore: boolean;
86
+ };
87
+ /** Counts for the studio overview. */
88
+ getStats(): {
89
+ namespaces: number;
90
+ states: number;
91
+ events: number;
92
+ };
93
+ /** All states with their namespace name, ordered by last updated. */
94
+ listAllStatesWithNamespace(): (StateRecord & {
95
+ namespace_name: string;
96
+ })[];
97
+ /** Recent events across all states, with state key attached. */
98
+ getRecentEvents(limit: number): (EventRecord & {
99
+ state_key: string;
100
+ })[];
101
+ }
102
+ //# sourceMappingURL=adapter.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"adapter.d.ts","sourceRoot":"","sources":["../../src/storage/adapter.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,12 @@
1
+ /**
2
+ * StorageAdapter — the contract any persistence backend must implement.
3
+ *
4
+ * SQLite ships with the MVP. To add Postgres, Redis, or any other backend,
5
+ * implement this interface and pass it to the server at boot.
6
+ *
7
+ * All methods are synchronous today (SQLite is sync). When a Postgres adapter
8
+ * is added, this interface can be migrated to async — the surface is small
9
+ * enough that the change is mechanical.
10
+ */
11
+ export {};
12
+ //# sourceMappingURL=adapter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"adapter.js","sourceRoot":"","sources":["../../src/storage/adapter.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG"}
@@ -0,0 +1,4 @@
1
+ export type { StorageAdapter, NamespaceRecord, StateRecord, EventRecord, EventInput, } from "./adapter.js";
2
+ export { SqliteAdapter } from "./sqlite.js";
3
+ export type { SqliteOptions } from "./sqlite.js";
4
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/storage/index.ts"],"names":[],"mappings":"AAAA,YAAY,EACV,cAAc,EACd,eAAe,EACf,WAAW,EACX,WAAW,EACX,UAAU,GACX,MAAM,cAAc,CAAC;AAEtB,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAC5C,YAAY,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC"}
@@ -0,0 +1,2 @@
1
+ export { SqliteAdapter } from "./sqlite.js";
2
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/storage/index.ts"],"names":[],"mappings":"AAQA,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC"}
@@ -0,0 +1,49 @@
1
+ import Database from "better-sqlite3";
2
+ import type { StorageAdapter, NamespaceRecord, StateRecord, EventRecord, EventInput } from "./adapter.js";
3
+ export interface SqliteOptions {
4
+ /** Path to the .db file. Defaults to process.env.DATABASE_PATH or ./data/state.db */
5
+ dbPath?: string;
6
+ /** If true, uses an in-memory database (useful for tests). */
7
+ memory?: boolean;
8
+ }
9
+ export declare class SqliteAdapter implements StorageAdapter {
10
+ private db;
11
+ constructor(opts?: SqliteOptions);
12
+ /** Expose the raw database for edge cases (tests, legacy code). */
13
+ get raw(): Database.Database;
14
+ init(): void;
15
+ close(): void;
16
+ listNamespaces(): Pick<NamespaceRecord, "id" | "name" | "created_at">[];
17
+ findNamespaceByApiKey(apiKeyHash: string): {
18
+ id: string;
19
+ } | null;
20
+ findNamespaceByName(name: string): NamespaceRecord | null;
21
+ createNamespace(ns: NamespaceRecord): void;
22
+ findState(namespace: string, key: string): StateRecord | null;
23
+ findStateById(id: string): StateRecord | null;
24
+ listStates(namespace: string): StateRecord[];
25
+ createState(state: StateRecord): void;
26
+ updateSnapshot(stateId: string, snapshot: string, version: number, updatedAt: number): void;
27
+ getMaxSeq(stateId: string): number | null;
28
+ insertEvents(events: EventInput[]): void;
29
+ getEventsForReplay(stateId: string): {
30
+ mutation: string;
31
+ meta: string | null;
32
+ }[];
33
+ getEventHistory(stateId: string, limit: number, cursor?: string): {
34
+ events: EventRecord[];
35
+ hasMore: boolean;
36
+ };
37
+ getStats(): {
38
+ namespaces: number;
39
+ states: number;
40
+ events: number;
41
+ };
42
+ listAllStatesWithNamespace(): (StateRecord & {
43
+ namespace_name: string;
44
+ })[];
45
+ getRecentEvents(limit: number): (EventRecord & {
46
+ state_key: string;
47
+ })[];
48
+ }
49
+ //# sourceMappingURL=sqlite.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sqlite.d.ts","sourceRoot":"","sources":["../../src/storage/sqlite.ts"],"names":[],"mappings":"AAAA,OAAO,QAAQ,MAAM,gBAAgB,CAAC;AAGtC,OAAO,KAAK,EACV,cAAc,EACd,eAAe,EACf,WAAW,EACX,WAAW,EACX,UAAU,EACX,MAAM,cAAc,CAAC;AAEtB,MAAM,WAAW,aAAa;IAC5B,qFAAqF;IACrF,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,8DAA8D;IAC9D,MAAM,CAAC,EAAE,OAAO,CAAC;CAClB;AAED,qBAAa,aAAc,YAAW,cAAc;IAClD,OAAO,CAAC,EAAE,CAAoB;gBAElB,IAAI,GAAE,aAAkB;IAepC,mEAAmE;IACnE,IAAI,GAAG,IAAI,QAAQ,CAAC,QAAQ,CAE3B;IAID,IAAI,IAAI,IAAI;IAyCZ,KAAK,IAAI,IAAI;IAMb,cAAc,IAAI,IAAI,CAAC,eAAe,EAAE,IAAI,GAAG,MAAM,GAAG,YAAY,CAAC,EAAE;IAMvE,qBAAqB,CAAC,UAAU,EAAE,MAAM,GAAG;QAAE,EAAE,EAAE,MAAM,CAAA;KAAE,GAAG,IAAI;IAQhE,mBAAmB,CAAC,IAAI,EAAE,MAAM,GAAG,eAAe,GAAG,IAAI;IAQzD,eAAe,CAAC,EAAE,EAAE,eAAe,GAAG,IAAI;IAU1C,SAAS,CAAC,SAAS,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,GAAG,WAAW,GAAG,IAAI;IAQ7D,aAAa,CAAC,EAAE,EAAE,MAAM,GAAG,WAAW,GAAG,IAAI;IAQ7C,UAAU,CAAC,SAAS,EAAE,MAAM,GAAG,WAAW,EAAE;IAQ5C,WAAW,CAAC,KAAK,EAAE,WAAW,GAAG,IAAI;IAkBrC,cAAc,CACZ,OAAO,EAAE,MAAM,EACf,QAAQ,EAAE,MAAM,EAChB,OAAO,EAAE,MAAM,EACf,SAAS,EAAE,MAAM,GAChB,IAAI;IAUP,SAAS,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI;IAOzC,YAAY,CAAC,MAAM,EAAE,UAAU,EAAE,GAAG,IAAI;IAyBxC,kBAAkB,CAChB,OAAO,EAAE,MAAM,GACd;QAAE,QAAQ,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,GAAG,IAAI,CAAA;KAAE,EAAE;IAQ9C,eAAe,CACb,OAAO,EAAE,MAAM,EACf,KAAK,EAAE,MAAM,EACb,MAAM,CAAC,EAAE,MAAM,GACd;QAAE,MAAM,EAAE,WAAW,EAAE,CAAC;QAAC,OAAO,EAAE,OAAO,CAAA;KAAE;IA8B9C,QAAQ,IAAI;QAAE,UAAU,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAA;KAAE;IAalE,0BAA0B,IAAI,CAAC,WAAW,GAAG;QAAE,cAAc,EAAE,MAAM,CAAA;KAAE,CAAC,EAAE;IAW1E,eAAe,CACb,KAAK,EAAE,MAAM,GACZ,CAAC,WAAW,GAAG;QAAE,SAAS,EAAE,MAAM,CAAA;KAAE,CAAC,EAAE;CAY3C"}
@@ -0,0 +1,192 @@
1
+ import Database from "better-sqlite3";
2
+ import { mkdirSync } from "node:fs";
3
+ import path from "node:path";
4
+ export class SqliteAdapter {
5
+ db;
6
+ constructor(opts = {}) {
7
+ if (opts.memory) {
8
+ this.db = new Database(":memory:");
9
+ }
10
+ else {
11
+ const dbPath = opts.dbPath ||
12
+ process.env.DATABASE_PATH ||
13
+ path.join(process.cwd(), "data", "state.db");
14
+ mkdirSync(path.dirname(dbPath), { recursive: true });
15
+ this.db = new Database(dbPath);
16
+ }
17
+ this.db.pragma("journal_mode = WAL");
18
+ this.db.pragma("foreign_keys = ON");
19
+ }
20
+ /** Expose the raw database for edge cases (tests, legacy code). */
21
+ get raw() {
22
+ return this.db;
23
+ }
24
+ // ── Lifecycle ───────────────────────────────────────────────────────────
25
+ init() {
26
+ this.db.exec(`
27
+ CREATE TABLE IF NOT EXISTS namespaces (
28
+ id TEXT PRIMARY KEY,
29
+ name TEXT NOT NULL,
30
+ api_key TEXT NOT NULL UNIQUE,
31
+ created_at INTEGER NOT NULL
32
+ );
33
+
34
+ CREATE TABLE IF NOT EXISTS states (
35
+ id TEXT PRIMARY KEY,
36
+ namespace TEXT NOT NULL REFERENCES namespaces(id),
37
+ key TEXT NOT NULL,
38
+ initial TEXT NOT NULL,
39
+ snapshot TEXT NOT NULL,
40
+ version INTEGER NOT NULL DEFAULT 0,
41
+ created_at INTEGER NOT NULL,
42
+ updated_at INTEGER NOT NULL,
43
+ UNIQUE(namespace, key)
44
+ );
45
+
46
+ CREATE TABLE IF NOT EXISTS event_log (
47
+ id TEXT PRIMARY KEY,
48
+ state_id TEXT NOT NULL REFERENCES states(id),
49
+ client_id TEXT NOT NULL,
50
+ client_ts INTEGER NOT NULL,
51
+ server_ts INTEGER NOT NULL,
52
+ seq INTEGER NOT NULL,
53
+ mutation TEXT NOT NULL,
54
+ meta TEXT,
55
+ created_at INTEGER NOT NULL
56
+ );
57
+
58
+ CREATE INDEX IF NOT EXISTS idx_event_log_state_ts
59
+ ON event_log(state_id, server_ts, seq);
60
+
61
+ CREATE INDEX IF NOT EXISTS idx_states_namespace_key
62
+ ON states(namespace, key);
63
+ `);
64
+ }
65
+ close() {
66
+ this.db.close();
67
+ }
68
+ // ── Namespaces ──────────────────────────────────────────────────────────
69
+ listNamespaces() {
70
+ return this.db
71
+ .prepare("SELECT id, name, created_at FROM namespaces")
72
+ .all();
73
+ }
74
+ findNamespaceByApiKey(apiKeyHash) {
75
+ return (this.db
76
+ .prepare("SELECT id FROM namespaces WHERE api_key = ?")
77
+ .get(apiKeyHash) ?? null);
78
+ }
79
+ findNamespaceByName(name) {
80
+ return (this.db
81
+ .prepare("SELECT * FROM namespaces WHERE name = ?")
82
+ .get(name) ?? null);
83
+ }
84
+ createNamespace(ns) {
85
+ this.db
86
+ .prepare("INSERT INTO namespaces (id, name, api_key, created_at) VALUES (?, ?, ?, ?)")
87
+ .run(ns.id, ns.name, ns.api_key, ns.created_at);
88
+ }
89
+ // ── States ──────────────────────────────────────────────────────────────
90
+ findState(namespace, key) {
91
+ return (this.db
92
+ .prepare("SELECT * FROM states WHERE namespace = ? AND key = ?")
93
+ .get(namespace, key) ?? null);
94
+ }
95
+ findStateById(id) {
96
+ return (this.db
97
+ .prepare("SELECT * FROM states WHERE id = ?")
98
+ .get(id) ?? null);
99
+ }
100
+ listStates(namespace) {
101
+ return this.db
102
+ .prepare("SELECT id, key, snapshot, version, created_at, updated_at FROM states WHERE namespace = ?")
103
+ .all(namespace);
104
+ }
105
+ createState(state) {
106
+ this.db
107
+ .prepare(`INSERT INTO states (id, namespace, key, initial, snapshot, version, created_at, updated_at)
108
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?)`)
109
+ .run(state.id, state.namespace, state.key, state.initial, state.snapshot, state.version, state.created_at, state.updated_at);
110
+ }
111
+ updateSnapshot(stateId, snapshot, version, updatedAt) {
112
+ this.db
113
+ .prepare("UPDATE states SET snapshot = ?, version = ?, updated_at = ? WHERE id = ?")
114
+ .run(snapshot, version, updatedAt, stateId);
115
+ }
116
+ // ── Event Log ───────────────────────────────────────────────────────────
117
+ getMaxSeq(stateId) {
118
+ const row = this.db
119
+ .prepare("SELECT MAX(seq) as max_seq FROM event_log WHERE state_id = ?")
120
+ .get(stateId);
121
+ return row.max_seq;
122
+ }
123
+ insertEvents(events) {
124
+ const insert = this.db.prepare(`INSERT INTO event_log (id, state_id, client_id, client_ts, server_ts, seq, mutation, meta, created_at)
125
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)`);
126
+ const batch = this.db.transaction(() => {
127
+ for (const e of events) {
128
+ insert.run(e.id, e.stateId, e.clientId, e.clientTs, e.serverTs, e.seq, e.mutation, e.meta, e.serverTs);
129
+ }
130
+ });
131
+ batch();
132
+ }
133
+ getEventsForReplay(stateId) {
134
+ return this.db
135
+ .prepare("SELECT mutation, meta FROM event_log WHERE state_id = ? ORDER BY server_ts ASC, seq ASC")
136
+ .all(stateId);
137
+ }
138
+ getEventHistory(stateId, limit, cursor) {
139
+ let events;
140
+ if (cursor) {
141
+ events = this.db
142
+ .prepare(`SELECT * FROM event_log
143
+ WHERE state_id = ? AND (server_ts, seq) > (
144
+ SELECT server_ts, seq FROM event_log WHERE id = ?
145
+ )
146
+ ORDER BY server_ts ASC, seq ASC
147
+ LIMIT ?`)
148
+ .all(stateId, cursor, limit + 1);
149
+ }
150
+ else {
151
+ events = this.db
152
+ .prepare("SELECT * FROM event_log WHERE state_id = ? ORDER BY server_ts ASC, seq ASC LIMIT ?")
153
+ .all(stateId, limit + 1);
154
+ }
155
+ const hasMore = events.length > limit;
156
+ if (hasMore)
157
+ events.pop();
158
+ return { events, hasMore };
159
+ }
160
+ // ── Studio ──────────────────────────────────────────────────────────────
161
+ getStats() {
162
+ const ns = this.db
163
+ .prepare("SELECT COUNT(*) as count FROM namespaces")
164
+ .get();
165
+ const st = this.db
166
+ .prepare("SELECT COUNT(*) as count FROM states")
167
+ .get();
168
+ const ev = this.db
169
+ .prepare("SELECT COUNT(*) as count FROM event_log")
170
+ .get();
171
+ return { namespaces: ns.count, states: st.count, events: ev.count };
172
+ }
173
+ listAllStatesWithNamespace() {
174
+ return this.db
175
+ .prepare(`SELECT s.id, s.namespace, s.key, s.snapshot, s.version, s.created_at, s.updated_at, n.name as namespace_name
176
+ FROM states s
177
+ JOIN namespaces n ON s.namespace = n.id
178
+ ORDER BY s.updated_at DESC`)
179
+ .all();
180
+ }
181
+ getRecentEvents(limit) {
182
+ const rows = this.db
183
+ .prepare(`SELECT e.id, e.state_id, e.client_id, e.client_ts, e.server_ts, e.mutation, e.meta, s.key as state_key
184
+ FROM event_log e
185
+ JOIN states s ON e.state_id = s.id
186
+ ORDER BY e.server_ts DESC, e.seq DESC
187
+ LIMIT ?`)
188
+ .all(limit);
189
+ return rows.reverse();
190
+ }
191
+ }
192
+ //# sourceMappingURL=sqlite.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sqlite.js","sourceRoot":"","sources":["../../src/storage/sqlite.ts"],"names":[],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@better-state/server",
3
- "version": "0.1.0",
3
+ "version": "0.2.0",
4
4
  "description": "Better-State server — shared state primitive for developers",
5
5
  "type": "module",
6
6
  "bin": {