@livestore/sync-cf 0.4.0-dev.2 → 0.4.0-dev.5
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.
- package/README.md +60 -0
- package/dist/.tsbuildinfo +1 -1
- package/dist/cf-worker/do/durable-object.d.ts +45 -0
- package/dist/cf-worker/do/durable-object.d.ts.map +1 -0
- package/dist/cf-worker/do/durable-object.js +154 -0
- package/dist/cf-worker/do/durable-object.js.map +1 -0
- package/dist/cf-worker/do/layer.d.ts +34 -0
- package/dist/cf-worker/do/layer.d.ts.map +1 -0
- package/dist/cf-worker/do/layer.js +68 -0
- package/dist/cf-worker/do/layer.js.map +1 -0
- package/dist/cf-worker/do/pull.d.ts +6 -0
- package/dist/cf-worker/do/pull.d.ts.map +1 -0
- package/dist/cf-worker/do/pull.js +39 -0
- package/dist/cf-worker/do/pull.js.map +1 -0
- package/dist/cf-worker/do/push.d.ts +14 -0
- package/dist/cf-worker/do/push.d.ts.map +1 -0
- package/dist/cf-worker/do/push.js +99 -0
- package/dist/cf-worker/do/push.js.map +1 -0
- package/dist/cf-worker/do/sqlite.d.ts +196 -0
- package/dist/cf-worker/do/sqlite.d.ts.map +1 -0
- package/dist/cf-worker/do/sqlite.js +27 -0
- package/dist/cf-worker/do/sqlite.js.map +1 -0
- package/dist/cf-worker/do/sync-storage.d.ts +17 -0
- package/dist/cf-worker/do/sync-storage.d.ts.map +1 -0
- package/dist/cf-worker/do/sync-storage.js +73 -0
- package/dist/cf-worker/do/sync-storage.js.map +1 -0
- package/dist/cf-worker/do/transport/do-rpc-server.d.ts +8 -0
- package/dist/cf-worker/do/transport/do-rpc-server.d.ts.map +1 -0
- package/dist/cf-worker/do/transport/do-rpc-server.js +45 -0
- package/dist/cf-worker/do/transport/do-rpc-server.js.map +1 -0
- package/dist/cf-worker/do/transport/http-rpc-server.d.ts +7 -0
- package/dist/cf-worker/do/transport/http-rpc-server.d.ts.map +1 -0
- package/dist/cf-worker/do/transport/http-rpc-server.js +24 -0
- package/dist/cf-worker/do/transport/http-rpc-server.js.map +1 -0
- package/dist/cf-worker/do/transport/ws-rpc-server.d.ts +4 -0
- package/dist/cf-worker/do/transport/ws-rpc-server.d.ts.map +1 -0
- package/dist/cf-worker/do/transport/ws-rpc-server.js +21 -0
- package/dist/cf-worker/do/transport/ws-rpc-server.js.map +1 -0
- package/dist/cf-worker/mod.d.ts +4 -2
- package/dist/cf-worker/mod.d.ts.map +1 -1
- package/dist/cf-worker/mod.js +3 -2
- package/dist/cf-worker/mod.js.map +1 -1
- package/dist/cf-worker/shared.d.ts +127 -0
- package/dist/cf-worker/shared.d.ts.map +1 -0
- package/dist/cf-worker/shared.js +26 -0
- package/dist/cf-worker/shared.js.map +1 -0
- package/dist/cf-worker/worker.d.ts +36 -21
- package/dist/cf-worker/worker.d.ts.map +1 -1
- package/dist/cf-worker/worker.js +39 -32
- package/dist/cf-worker/worker.js.map +1 -1
- package/dist/client/mod.d.ts +4 -0
- package/dist/client/mod.d.ts.map +1 -0
- package/dist/client/mod.js +4 -0
- package/dist/client/mod.js.map +1 -0
- package/dist/client/transport/do-rpc-client.d.ts +40 -0
- package/dist/client/transport/do-rpc-client.d.ts.map +1 -0
- package/dist/client/transport/do-rpc-client.js +102 -0
- package/dist/client/transport/do-rpc-client.js.map +1 -0
- package/dist/client/transport/http-rpc-client.d.ts +43 -0
- package/dist/client/transport/http-rpc-client.d.ts.map +1 -0
- package/dist/client/transport/http-rpc-client.js +87 -0
- package/dist/client/transport/http-rpc-client.js.map +1 -0
- package/dist/client/transport/ws-rpc-client.d.ts +45 -0
- package/dist/client/transport/ws-rpc-client.d.ts.map +1 -0
- package/dist/client/transport/ws-rpc-client.js +94 -0
- package/dist/client/transport/ws-rpc-client.js.map +1 -0
- package/dist/common/do-rpc-schema.d.ts +76 -0
- package/dist/common/do-rpc-schema.d.ts.map +1 -0
- package/dist/common/do-rpc-schema.js +48 -0
- package/dist/common/do-rpc-schema.js.map +1 -0
- package/dist/common/http-rpc-schema.d.ts +58 -0
- package/dist/common/http-rpc-schema.d.ts.map +1 -0
- package/dist/common/http-rpc-schema.js +37 -0
- package/dist/common/http-rpc-schema.js.map +1 -0
- package/dist/common/mod.d.ts +5 -1
- package/dist/common/mod.d.ts.map +1 -1
- package/dist/common/mod.js +4 -1
- package/dist/common/mod.js.map +1 -1
- package/dist/common/sync-message-types.d.ts +236 -0
- package/dist/common/sync-message-types.d.ts.map +1 -0
- package/dist/common/sync-message-types.js +60 -0
- package/dist/common/sync-message-types.js.map +1 -0
- package/dist/common/ws-rpc-schema.d.ts +55 -0
- package/dist/common/ws-rpc-schema.d.ts.map +1 -0
- package/dist/common/ws-rpc-schema.js +32 -0
- package/dist/common/ws-rpc-schema.js.map +1 -0
- package/package.json +7 -8
- package/src/cf-worker/do/durable-object.ts +241 -0
- package/src/cf-worker/do/layer.ts +107 -0
- package/src/cf-worker/do/pull.ts +64 -0
- package/src/cf-worker/do/push.ts +162 -0
- package/src/cf-worker/do/sqlite.ts +28 -0
- package/src/cf-worker/do/sync-storage.ts +126 -0
- package/src/cf-worker/do/transport/do-rpc-server.ts +82 -0
- package/src/cf-worker/do/transport/http-rpc-server.ts +37 -0
- package/src/cf-worker/do/transport/ws-rpc-server.ts +34 -0
- package/src/cf-worker/mod.ts +4 -2
- package/src/cf-worker/shared.ts +95 -0
- package/src/cf-worker/worker.ts +72 -63
- package/src/client/mod.ts +3 -0
- package/src/client/transport/do-rpc-client.ts +171 -0
- package/src/client/transport/http-rpc-client.ts +205 -0
- package/src/client/transport/ws-rpc-client.ts +182 -0
- package/src/common/do-rpc-schema.ts +54 -0
- package/src/common/http-rpc-schema.ts +40 -0
- package/src/common/mod.ts +8 -1
- package/src/common/sync-message-types.ts +117 -0
- package/src/common/ws-rpc-schema.ts +36 -0
- package/src/cf-worker/cf-types.ts +0 -12
- package/src/cf-worker/durable-object.ts +0 -478
- package/src/common/ws-message-types.ts +0 -114
- package/src/sync-impl/mod.ts +0 -1
- package/src/sync-impl/ws-impl.ts +0 -274
|
@@ -0,0 +1,196 @@
|
|
|
1
|
+
import { State } from '@livestore/common/schema';
|
|
2
|
+
import { Schema } from '@livestore/utils/effect';
|
|
3
|
+
export declare const eventlogTable: State.SQLite.TableDef<State.SQLite.SqliteTableDefForInput<"eventlog_7_$storeId", {
|
|
4
|
+
readonly seqNum: {
|
|
5
|
+
columnType: "integer";
|
|
6
|
+
schema: Schema.Schema<number & import("effect/Brand").Brand<"GlobalEventSequenceNumber">, number, never>;
|
|
7
|
+
default: import("effect/Option").None<never>;
|
|
8
|
+
nullable: false;
|
|
9
|
+
primaryKey: true;
|
|
10
|
+
autoIncrement: false;
|
|
11
|
+
};
|
|
12
|
+
readonly parentSeqNum: {
|
|
13
|
+
columnType: "integer";
|
|
14
|
+
schema: Schema.Schema<number & import("effect/Brand").Brand<"GlobalEventSequenceNumber">, number, never>;
|
|
15
|
+
default: import("effect/Option").None<never>;
|
|
16
|
+
nullable: false;
|
|
17
|
+
primaryKey: false;
|
|
18
|
+
autoIncrement: false;
|
|
19
|
+
};
|
|
20
|
+
readonly name: {
|
|
21
|
+
columnType: "text";
|
|
22
|
+
schema: Schema.Schema<string, string, never>;
|
|
23
|
+
default: import("effect/Option").None<never>;
|
|
24
|
+
nullable: false;
|
|
25
|
+
primaryKey: false;
|
|
26
|
+
autoIncrement: false;
|
|
27
|
+
};
|
|
28
|
+
readonly args: {
|
|
29
|
+
columnType: "text";
|
|
30
|
+
schema: Schema.Schema<any, string | null, never>;
|
|
31
|
+
default: import("effect/Option").None<never>;
|
|
32
|
+
nullable: true;
|
|
33
|
+
primaryKey: false;
|
|
34
|
+
autoIncrement: false;
|
|
35
|
+
};
|
|
36
|
+
/** ISO date format. Currently only used for debugging purposes. */
|
|
37
|
+
readonly createdAt: {
|
|
38
|
+
columnType: "text";
|
|
39
|
+
schema: Schema.Schema<string, string, never>;
|
|
40
|
+
default: import("effect/Option").None<never>;
|
|
41
|
+
nullable: false;
|
|
42
|
+
primaryKey: false;
|
|
43
|
+
autoIncrement: false;
|
|
44
|
+
};
|
|
45
|
+
readonly clientId: {
|
|
46
|
+
columnType: "text";
|
|
47
|
+
schema: Schema.Schema<string, string, never>;
|
|
48
|
+
default: import("effect/Option").None<never>;
|
|
49
|
+
nullable: false;
|
|
50
|
+
primaryKey: false;
|
|
51
|
+
autoIncrement: false;
|
|
52
|
+
};
|
|
53
|
+
readonly sessionId: {
|
|
54
|
+
columnType: "text";
|
|
55
|
+
schema: Schema.Schema<string, string, never>;
|
|
56
|
+
default: import("effect/Option").None<never>;
|
|
57
|
+
nullable: false;
|
|
58
|
+
primaryKey: false;
|
|
59
|
+
autoIncrement: false;
|
|
60
|
+
};
|
|
61
|
+
}>, State.SQLite.WithDefaults<{
|
|
62
|
+
readonly seqNum: {
|
|
63
|
+
columnType: "integer";
|
|
64
|
+
schema: Schema.Schema<number & import("effect/Brand").Brand<"GlobalEventSequenceNumber">, number, never>;
|
|
65
|
+
default: import("effect/Option").None<never>;
|
|
66
|
+
nullable: false;
|
|
67
|
+
primaryKey: true;
|
|
68
|
+
autoIncrement: false;
|
|
69
|
+
};
|
|
70
|
+
readonly parentSeqNum: {
|
|
71
|
+
columnType: "integer";
|
|
72
|
+
schema: Schema.Schema<number & import("effect/Brand").Brand<"GlobalEventSequenceNumber">, number, never>;
|
|
73
|
+
default: import("effect/Option").None<never>;
|
|
74
|
+
nullable: false;
|
|
75
|
+
primaryKey: false;
|
|
76
|
+
autoIncrement: false;
|
|
77
|
+
};
|
|
78
|
+
readonly name: {
|
|
79
|
+
columnType: "text";
|
|
80
|
+
schema: Schema.Schema<string, string, never>;
|
|
81
|
+
default: import("effect/Option").None<never>;
|
|
82
|
+
nullable: false;
|
|
83
|
+
primaryKey: false;
|
|
84
|
+
autoIncrement: false;
|
|
85
|
+
};
|
|
86
|
+
readonly args: {
|
|
87
|
+
columnType: "text";
|
|
88
|
+
schema: Schema.Schema<any, string | null, never>;
|
|
89
|
+
default: import("effect/Option").None<never>;
|
|
90
|
+
nullable: true;
|
|
91
|
+
primaryKey: false;
|
|
92
|
+
autoIncrement: false;
|
|
93
|
+
};
|
|
94
|
+
/** ISO date format. Currently only used for debugging purposes. */
|
|
95
|
+
readonly createdAt: {
|
|
96
|
+
columnType: "text";
|
|
97
|
+
schema: Schema.Schema<string, string, never>;
|
|
98
|
+
default: import("effect/Option").None<never>;
|
|
99
|
+
nullable: false;
|
|
100
|
+
primaryKey: false;
|
|
101
|
+
autoIncrement: false;
|
|
102
|
+
};
|
|
103
|
+
readonly clientId: {
|
|
104
|
+
columnType: "text";
|
|
105
|
+
schema: Schema.Schema<string, string, never>;
|
|
106
|
+
default: import("effect/Option").None<never>;
|
|
107
|
+
nullable: false;
|
|
108
|
+
primaryKey: false;
|
|
109
|
+
autoIncrement: false;
|
|
110
|
+
};
|
|
111
|
+
readonly sessionId: {
|
|
112
|
+
columnType: "text";
|
|
113
|
+
schema: Schema.Schema<string, string, never>;
|
|
114
|
+
default: import("effect/Option").None<never>;
|
|
115
|
+
nullable: false;
|
|
116
|
+
primaryKey: false;
|
|
117
|
+
autoIncrement: false;
|
|
118
|
+
};
|
|
119
|
+
}>, Schema.Schema<{
|
|
120
|
+
readonly seqNum: number & import("effect/Brand").Brand<"GlobalEventSequenceNumber">;
|
|
121
|
+
readonly parentSeqNum: number & import("effect/Brand").Brand<"GlobalEventSequenceNumber">;
|
|
122
|
+
readonly name: string;
|
|
123
|
+
readonly args: any;
|
|
124
|
+
readonly createdAt: string;
|
|
125
|
+
readonly clientId: string;
|
|
126
|
+
readonly sessionId: string;
|
|
127
|
+
}, {
|
|
128
|
+
readonly seqNum: number;
|
|
129
|
+
readonly parentSeqNum: number;
|
|
130
|
+
readonly name: string;
|
|
131
|
+
readonly args: string | null;
|
|
132
|
+
readonly createdAt: string;
|
|
133
|
+
readonly clientId: string;
|
|
134
|
+
readonly sessionId: string;
|
|
135
|
+
}, never>>;
|
|
136
|
+
/** Will only ever have one row per durable object. */
|
|
137
|
+
export declare const contextTable: State.SQLite.TableDef<State.SQLite.SqliteTableDefForInput<"context_7", {
|
|
138
|
+
readonly storeId: {
|
|
139
|
+
columnType: "text";
|
|
140
|
+
schema: Schema.Schema<string, string, never>;
|
|
141
|
+
default: import("effect/Option").None<never>;
|
|
142
|
+
nullable: false;
|
|
143
|
+
primaryKey: true;
|
|
144
|
+
autoIncrement: false;
|
|
145
|
+
};
|
|
146
|
+
readonly currentHead: {
|
|
147
|
+
columnType: "integer";
|
|
148
|
+
schema: Schema.Schema<number & import("effect/Brand").Brand<"GlobalEventSequenceNumber">, number, never>;
|
|
149
|
+
default: import("effect/Option").None<never>;
|
|
150
|
+
nullable: false;
|
|
151
|
+
primaryKey: false;
|
|
152
|
+
autoIncrement: false;
|
|
153
|
+
};
|
|
154
|
+
readonly backendId: {
|
|
155
|
+
columnType: "text";
|
|
156
|
+
schema: Schema.Schema<string, string, never>;
|
|
157
|
+
default: import("effect/Option").None<never>;
|
|
158
|
+
nullable: false;
|
|
159
|
+
primaryKey: false;
|
|
160
|
+
autoIncrement: false;
|
|
161
|
+
};
|
|
162
|
+
}>, State.SQLite.WithDefaults<{
|
|
163
|
+
readonly storeId: {
|
|
164
|
+
columnType: "text";
|
|
165
|
+
schema: Schema.Schema<string, string, never>;
|
|
166
|
+
default: import("effect/Option").None<never>;
|
|
167
|
+
nullable: false;
|
|
168
|
+
primaryKey: true;
|
|
169
|
+
autoIncrement: false;
|
|
170
|
+
};
|
|
171
|
+
readonly currentHead: {
|
|
172
|
+
columnType: "integer";
|
|
173
|
+
schema: Schema.Schema<number & import("effect/Brand").Brand<"GlobalEventSequenceNumber">, number, never>;
|
|
174
|
+
default: import("effect/Option").None<never>;
|
|
175
|
+
nullable: false;
|
|
176
|
+
primaryKey: false;
|
|
177
|
+
autoIncrement: false;
|
|
178
|
+
};
|
|
179
|
+
readonly backendId: {
|
|
180
|
+
columnType: "text";
|
|
181
|
+
schema: Schema.Schema<string, string, never>;
|
|
182
|
+
default: import("effect/Option").None<never>;
|
|
183
|
+
nullable: false;
|
|
184
|
+
primaryKey: false;
|
|
185
|
+
autoIncrement: false;
|
|
186
|
+
};
|
|
187
|
+
}>, Schema.Schema<{
|
|
188
|
+
readonly storeId: string;
|
|
189
|
+
readonly currentHead: number & import("effect/Brand").Brand<"GlobalEventSequenceNumber">;
|
|
190
|
+
readonly backendId: string;
|
|
191
|
+
}, {
|
|
192
|
+
readonly storeId: string;
|
|
193
|
+
readonly currentHead: number;
|
|
194
|
+
readonly backendId: string;
|
|
195
|
+
}, never>>;
|
|
196
|
+
//# sourceMappingURL=sqlite.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sqlite.d.ts","sourceRoot":"","sources":["../../../src/cf-worker/do/sqlite.ts"],"names":[],"mappings":"AAAA,OAAO,EAAuB,KAAK,EAAE,MAAM,0BAA0B,CAAA;AACrE,OAAO,EAAE,MAAM,EAAE,MAAM,yBAAyB,CAAA;AAGhD,eAAO,MAAM,aAAa;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;IAQtB,mEAAmE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;IAAnE,mEAAmE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;UAKrE,CAAA;AAEF,sDAAsD;AACtD,eAAO,MAAM,YAAY;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;UAOvB,CAAA"}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { EventSequenceNumber, State } from '@livestore/common/schema';
|
|
2
|
+
import { Schema } from '@livestore/utils/effect';
|
|
3
|
+
import { PERSISTENCE_FORMAT_VERSION } from "../shared.js";
|
|
4
|
+
export const eventlogTable = State.SQLite.table({
|
|
5
|
+
// NOTE actual table name is determined at runtime to use proper storeId
|
|
6
|
+
name: `eventlog_${PERSISTENCE_FORMAT_VERSION}_$storeId`,
|
|
7
|
+
columns: {
|
|
8
|
+
seqNum: State.SQLite.integer({ primaryKey: true, schema: EventSequenceNumber.GlobalEventSequenceNumber }),
|
|
9
|
+
parentSeqNum: State.SQLite.integer({ schema: EventSequenceNumber.GlobalEventSequenceNumber }),
|
|
10
|
+
name: State.SQLite.text({}),
|
|
11
|
+
args: State.SQLite.text({ schema: Schema.parseJson(Schema.Any), nullable: true }),
|
|
12
|
+
/** ISO date format. Currently only used for debugging purposes. */
|
|
13
|
+
createdAt: State.SQLite.text({}),
|
|
14
|
+
clientId: State.SQLite.text({}),
|
|
15
|
+
sessionId: State.SQLite.text({}),
|
|
16
|
+
},
|
|
17
|
+
});
|
|
18
|
+
/** Will only ever have one row per durable object. */
|
|
19
|
+
export const contextTable = State.SQLite.table({
|
|
20
|
+
name: `context_${PERSISTENCE_FORMAT_VERSION}`,
|
|
21
|
+
columns: {
|
|
22
|
+
storeId: State.SQLite.text({ primaryKey: true }),
|
|
23
|
+
currentHead: State.SQLite.integer({ schema: EventSequenceNumber.GlobalEventSequenceNumber }),
|
|
24
|
+
backendId: State.SQLite.text({}),
|
|
25
|
+
},
|
|
26
|
+
});
|
|
27
|
+
//# sourceMappingURL=sqlite.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sqlite.js","sourceRoot":"","sources":["../../../src/cf-worker/do/sqlite.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,mBAAmB,EAAE,KAAK,EAAE,MAAM,0BAA0B,CAAA;AACrE,OAAO,EAAE,MAAM,EAAE,MAAM,yBAAyB,CAAA;AAChD,OAAO,EAAE,0BAA0B,EAAE,MAAM,cAAc,CAAA;AAEzD,MAAM,CAAC,MAAM,aAAa,GAAG,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC;IAC9C,wEAAwE;IACxE,IAAI,EAAE,YAAY,0BAA0B,WAAW;IACvD,OAAO,EAAE;QACP,MAAM,EAAE,KAAK,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,UAAU,EAAE,IAAI,EAAE,MAAM,EAAE,mBAAmB,CAAC,yBAAyB,EAAE,CAAC;QACzG,YAAY,EAAE,KAAK,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,MAAM,EAAE,mBAAmB,CAAC,yBAAyB,EAAE,CAAC;QAC7F,IAAI,EAAE,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC;QAC3B,IAAI,EAAE,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,MAAM,CAAC,SAAS,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC;QACjF,mEAAmE;QACnE,SAAS,EAAE,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC;QAChC,QAAQ,EAAE,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC;QAC/B,SAAS,EAAE,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC;KACjC;CACF,CAAC,CAAA;AAEF,sDAAsD;AACtD,MAAM,CAAC,MAAM,YAAY,GAAG,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC;IAC7C,IAAI,EAAE,WAAW,0BAA0B,EAAE;IAC7C,OAAO,EAAE;QACP,OAAO,EAAE,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,UAAU,EAAE,IAAI,EAAE,CAAC;QAChD,WAAW,EAAE,KAAK,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,MAAM,EAAE,mBAAmB,CAAC,yBAAyB,EAAE,CAAC;QAC5F,SAAS,EAAE,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC;KACjC;CACF,CAAC,CAAA"}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import { UnexpectedError } from '@livestore/common';
|
|
2
|
+
import type { LiveStoreEvent } from '@livestore/common/schema';
|
|
3
|
+
import type { CfTypes } from '@livestore/common-cf';
|
|
4
|
+
import { Effect, Option } from '@livestore/utils/effect';
|
|
5
|
+
import { SyncMetadata } from '../../common/sync-message-types.ts';
|
|
6
|
+
import { type Env, type StoreId } from '../shared.ts';
|
|
7
|
+
export type SyncStorage = {
|
|
8
|
+
dbName: string;
|
|
9
|
+
getEvents: (cursor: number | undefined) => Effect.Effect<ReadonlyArray<{
|
|
10
|
+
eventEncoded: LiveStoreEvent.AnyEncodedGlobal;
|
|
11
|
+
metadata: Option.Option<SyncMetadata>;
|
|
12
|
+
}>, UnexpectedError>;
|
|
13
|
+
appendEvents: (batch: ReadonlyArray<LiveStoreEvent.AnyEncodedGlobal>, createdAt: string) => Effect.Effect<void, UnexpectedError>;
|
|
14
|
+
resetStore: Effect.Effect<void, UnexpectedError>;
|
|
15
|
+
};
|
|
16
|
+
export declare const makeStorage: (ctx: CfTypes.DurableObjectState, env: Env, storeId: StoreId) => SyncStorage;
|
|
17
|
+
//# sourceMappingURL=sync-storage.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sync-storage.d.ts","sourceRoot":"","sources":["../../../src/cf-worker/do/sync-storage.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAA;AACnD,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,0BAA0B,CAAA;AAC9D,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,sBAAsB,CAAA;AACnD,OAAO,EAAE,MAAM,EAAE,MAAM,EAAU,MAAM,yBAAyB,CAAA;AAChE,OAAO,EAAE,YAAY,EAAE,MAAM,oCAAoC,CAAA;AACjE,OAAO,EAAE,KAAK,GAAG,EAA8B,KAAK,OAAO,EAAE,MAAM,cAAc,CAAA;AAGjF,MAAM,MAAM,WAAW,GAAG;IACxB,MAAM,EAAE,MAAM,CAAA;IAEd,SAAS,EAAE,CACT,MAAM,EAAE,MAAM,GAAG,SAAS,KACvB,MAAM,CAAC,MAAM,CAChB,aAAa,CAAC;QAAE,YAAY,EAAE,cAAc,CAAC,gBAAgB,CAAC;QAAC,QAAQ,EAAE,MAAM,CAAC,MAAM,CAAC,YAAY,CAAC,CAAA;KAAE,CAAC,EACvG,eAAe,CAChB,CAAA;IACD,YAAY,EAAE,CACZ,KAAK,EAAE,aAAa,CAAC,cAAc,CAAC,gBAAgB,CAAC,EACrD,SAAS,EAAE,MAAM,KACd,MAAM,CAAC,MAAM,CAAC,IAAI,EAAE,eAAe,CAAC,CAAA;IACzC,UAAU,EAAE,MAAM,CAAC,MAAM,CAAC,IAAI,EAAE,eAAe,CAAC,CAAA;CACjD,CAAA;AAED,eAAO,MAAM,WAAW,GAAI,KAAK,OAAO,CAAC,kBAAkB,EAAE,KAAK,GAAG,EAAE,SAAS,OAAO,KAAG,WAmGzF,CAAA"}
|
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
import { UnexpectedError } from '@livestore/common';
|
|
2
|
+
import { Effect, Option, Schema } from '@livestore/utils/effect';
|
|
3
|
+
import { SyncMetadata } from "../../common/sync-message-types.js";
|
|
4
|
+
import { PERSISTENCE_FORMAT_VERSION } from "../shared.js";
|
|
5
|
+
import { eventlogTable } from "./sqlite.js";
|
|
6
|
+
export const makeStorage = (ctx, env, storeId) => {
|
|
7
|
+
const dbName = `eventlog_${PERSISTENCE_FORMAT_VERSION}_${toValidTableName(storeId)}`;
|
|
8
|
+
const execDb = (cb) => Effect.tryPromise({
|
|
9
|
+
try: () => cb(env.DB),
|
|
10
|
+
catch: (error) => new UnexpectedError({ cause: error, payload: { dbName } }),
|
|
11
|
+
}).pipe(Effect.map((_) => _.results), Effect.withSpan('@livestore/sync-cf:durable-object:execDb'));
|
|
12
|
+
// const getHead: Effect.Effect<EventSequenceNumber.GlobalEventSequenceNumber, UnexpectedError> = Effect.gen(
|
|
13
|
+
// function* () {
|
|
14
|
+
// const result = yield* execDb<{ seqNum: EventSequenceNumber.GlobalEventSequenceNumber }>((db) =>
|
|
15
|
+
// db.prepare(`SELECT seqNum FROM ${dbName} ORDER BY seqNum DESC LIMIT 1`).all(),
|
|
16
|
+
// )
|
|
17
|
+
// return result[0]?.seqNum ?? EventSequenceNumber.ROOT.global
|
|
18
|
+
// },
|
|
19
|
+
// ).pipe(UnexpectedError.mapToUnexpectedError)
|
|
20
|
+
// TODO support streaming
|
|
21
|
+
const getEvents = (cursor) => Effect.gen(function* () {
|
|
22
|
+
const whereClause = cursor === undefined ? '' : `WHERE seqNum > ${cursor}`;
|
|
23
|
+
const sql = `SELECT * FROM ${dbName} ${whereClause} ORDER BY seqNum ASC`;
|
|
24
|
+
// TODO handle case where `cursor` was not found
|
|
25
|
+
const rawEvents = yield* execDb((db) => db.prepare(sql).all());
|
|
26
|
+
const events = Schema.decodeUnknownSync(Schema.Array(eventlogTable.rowSchema))(rawEvents).map(({ createdAt, ...eventEncoded }) => ({
|
|
27
|
+
eventEncoded,
|
|
28
|
+
metadata: Option.some(SyncMetadata.make({ createdAt })),
|
|
29
|
+
}));
|
|
30
|
+
return events;
|
|
31
|
+
}).pipe(UnexpectedError.mapToUnexpectedError, Effect.withSpan('@livestore/sync-cf:durable-object:getEvents', { attributes: { dbName, cursor } }));
|
|
32
|
+
const appendEvents = (batch, createdAt) => Effect.gen(function* () {
|
|
33
|
+
// If there are no events, do nothing.
|
|
34
|
+
if (batch.length === 0)
|
|
35
|
+
return;
|
|
36
|
+
// CF D1 limits:
|
|
37
|
+
// Maximum bound parameters per query 100, Maximum arguments per SQL function 32
|
|
38
|
+
// Thus we need to split the batch into chunks of max (100/7=)14 events each.
|
|
39
|
+
const CHUNK_SIZE = 14;
|
|
40
|
+
for (let i = 0; i < batch.length; i += CHUNK_SIZE) {
|
|
41
|
+
const chunk = batch.slice(i, i + CHUNK_SIZE);
|
|
42
|
+
// Create a list of placeholders ("(?, ?, ?, ?, ?, ?, ?)"), corresponding to each event.
|
|
43
|
+
const valuesPlaceholders = chunk.map(() => '(?, ?, ?, ?, ?, ?, ?)').join(', ');
|
|
44
|
+
const sql = `INSERT INTO ${dbName} (seqNum, parentSeqNum, args, name, createdAt, clientId, sessionId) VALUES ${valuesPlaceholders}`;
|
|
45
|
+
// Flatten the event properties into a parameters array.
|
|
46
|
+
const params = chunk.flatMap((event) => [
|
|
47
|
+
event.seqNum,
|
|
48
|
+
event.parentSeqNum,
|
|
49
|
+
event.args === undefined ? null : JSON.stringify(event.args),
|
|
50
|
+
event.name,
|
|
51
|
+
createdAt,
|
|
52
|
+
event.clientId,
|
|
53
|
+
event.sessionId,
|
|
54
|
+
]);
|
|
55
|
+
yield* execDb((db) => db
|
|
56
|
+
.prepare(sql)
|
|
57
|
+
.bind(...params)
|
|
58
|
+
.run());
|
|
59
|
+
}
|
|
60
|
+
}).pipe(UnexpectedError.mapToUnexpectedError, Effect.withSpan('@livestore/sync-cf:durable-object:appendEvents', {
|
|
61
|
+
attributes: { dbName, batchLength: batch.length },
|
|
62
|
+
}));
|
|
63
|
+
const resetStore = Effect.promise(() => ctx.storage.deleteAll()).pipe(UnexpectedError.mapToUnexpectedError, Effect.withSpan('@livestore/sync-cf:durable-object:resetStore'));
|
|
64
|
+
return {
|
|
65
|
+
dbName,
|
|
66
|
+
// getHead,
|
|
67
|
+
getEvents,
|
|
68
|
+
appendEvents,
|
|
69
|
+
resetStore,
|
|
70
|
+
};
|
|
71
|
+
};
|
|
72
|
+
const toValidTableName = (str) => str.replaceAll(/[^a-zA-Z0-9]/g, '_');
|
|
73
|
+
//# sourceMappingURL=sync-storage.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sync-storage.js","sourceRoot":"","sources":["../../../src/cf-worker/do/sync-storage.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
import { Effect } from '@livestore/utils/effect';
|
|
2
|
+
import { type DoCtxInput } from '../layer.ts';
|
|
3
|
+
export interface DoRpcHandlerOptions {
|
|
4
|
+
payload: Uint8Array<ArrayBuffer>;
|
|
5
|
+
input: Omit<DoCtxInput, 'from'>;
|
|
6
|
+
}
|
|
7
|
+
export declare const createDoRpcHandler: (options: DoRpcHandlerOptions) => Effect.Effect<import("@cloudflare/workers-types").ReadableStream<any> | Uint8Array<ArrayBuffer>, never, never>;
|
|
8
|
+
//# sourceMappingURL=do-rpc-server.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"do-rpc-server.d.ts","sourceRoot":"","sources":["../../../../src/cf-worker/do/transport/do-rpc-server.ts"],"names":[],"mappings":"AAEA,OAAO,EACL,MAAM,EASP,MAAM,yBAAyB,CAAA;AAGhC,OAAO,EAAS,KAAK,UAAU,EAAE,MAAM,aAAa,CAAA;AAIpD,MAAM,WAAW,mBAAmB;IAClC,OAAO,EAAE,UAAU,CAAC,WAAW,CAAC,CAAA;IAChC,KAAK,EAAE,IAAI,CAAC,UAAU,EAAE,MAAM,CAAC,CAAA;CAChC;AAED,eAAO,MAAM,kBAAkB,GAAI,SAAS,mBAAmB,mHAyDf,CAAA"}
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import { InvalidPullError, InvalidPushError } from '@livestore/common';
|
|
2
|
+
import { toDurableObjectHandler } from '@livestore/common-cf';
|
|
3
|
+
import { Effect, Headers, HttpServer, Layer, Logger, LogLevel, Option, RpcSerialization, Stream, } from '@livestore/utils/effect';
|
|
4
|
+
import { SyncDoRpc } from "../../../common/do-rpc-schema.js";
|
|
5
|
+
import { SyncMessage } from "../../../common/mod.js";
|
|
6
|
+
import { DoCtx } from "../layer.js";
|
|
7
|
+
import { makeEndingPullStream } from "../pull.js";
|
|
8
|
+
import { makePush } from "../push.js";
|
|
9
|
+
export const createDoRpcHandler = (options) => Effect.gen(this, function* () {
|
|
10
|
+
const { payload, input } = options;
|
|
11
|
+
// const { rpcSubscriptions, backendId, doOptions, ctx, env } = yield* DoCtx
|
|
12
|
+
// TODO add admin RPCs
|
|
13
|
+
const RpcLive = SyncDoRpc.toLayer({
|
|
14
|
+
'SyncDoRpc.Ping': (_req) => {
|
|
15
|
+
return Effect.succeed(SyncMessage.Pong.make({}));
|
|
16
|
+
},
|
|
17
|
+
'SyncDoRpc.Pull': (req, { headers }) => Effect.gen(this, function* () {
|
|
18
|
+
const { rpcSubscriptions } = yield* DoCtx;
|
|
19
|
+
// TODO rename `req.rpcContext` to something more appropriate
|
|
20
|
+
if (req.rpcContext) {
|
|
21
|
+
rpcSubscriptions.set(req.storeId, {
|
|
22
|
+
storeId: req.storeId,
|
|
23
|
+
payload: req.payload,
|
|
24
|
+
subscribedAt: Date.now(),
|
|
25
|
+
requestId: Headers.get(headers, 'x-rpc-request-id').pipe(Option.getOrThrow),
|
|
26
|
+
callerContext: req.rpcContext.callerContext,
|
|
27
|
+
});
|
|
28
|
+
}
|
|
29
|
+
return makeEndingPullStream(req, req.payload);
|
|
30
|
+
}).pipe(Stream.unwrap, Stream.map((res) => ({
|
|
31
|
+
...res,
|
|
32
|
+
rpcRequestId: Headers.get(headers, 'x-rpc-request-id').pipe(Option.getOrThrow),
|
|
33
|
+
})), Stream.provideLayer(DoCtx.Default({ ...input, from: { storeId: req.storeId } })), Stream.mapError((cause) => (cause._tag === 'InvalidPullError' ? cause : InvalidPullError.make({ cause }))), Stream.tapErrorCause(Effect.log)),
|
|
34
|
+
'SyncDoRpc.Push': (req) => Effect.gen(this, function* () {
|
|
35
|
+
const { doOptions, ctx, env, storeId } = yield* DoCtx;
|
|
36
|
+
const push = makePush({ storeId, payload: req.payload, options: doOptions, ctx, env });
|
|
37
|
+
return yield* push(req);
|
|
38
|
+
}).pipe(Effect.provide(DoCtx.Default({ ...input, from: { storeId: req.storeId } })), Effect.mapError((cause) => (cause._tag === 'InvalidPushError' ? cause : InvalidPushError.make({ cause }))), Effect.tapCauseLogPretty),
|
|
39
|
+
});
|
|
40
|
+
const handler = toDurableObjectHandler(SyncDoRpc, {
|
|
41
|
+
layer: Layer.mergeAll(RpcLive, RpcSerialization.layerJson, HttpServer.layerContext).pipe(Layer.provide(Logger.consoleWithThread('SyncDo')), Layer.provide(Logger.minimumLogLevel(LogLevel.Debug))),
|
|
42
|
+
});
|
|
43
|
+
return yield* handler(payload);
|
|
44
|
+
}).pipe(Effect.withSpan('createDoRpcHandler'));
|
|
45
|
+
//# sourceMappingURL=do-rpc-server.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"do-rpc-server.js","sourceRoot":"","sources":["../../../../src/cf-worker/do/transport/do-rpc-server.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import type { CfTypes } from '@livestore/common-cf';
|
|
2
|
+
import { Effect } from '@livestore/utils/effect';
|
|
3
|
+
import { DoCtx } from '../layer.ts';
|
|
4
|
+
export declare const createHttpRpcHandler: ({ request }: {
|
|
5
|
+
request: CfTypes.Request;
|
|
6
|
+
}) => Effect.Effect<CfTypes.Response, import("effect/Cause").TimeoutException, DoCtx | import("effect/Scope").Scope>;
|
|
7
|
+
//# sourceMappingURL=http-rpc-server.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"http-rpc-server.d.ts","sourceRoot":"","sources":["../../../../src/cf-worker/do/transport/http-rpc-server.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,sBAAsB,CAAA;AACnD,OAAO,EAAE,MAAM,EAA+C,MAAM,yBAAyB,CAAA;AAG7F,OAAO,EAAE,KAAK,EAAE,MAAM,aAAa,CAAA;AAInC,eAAO,MAAM,oBAAoB,GAAI,aAAa;IAAE,OAAO,EAAE,OAAO,CAAC,OAAO,CAAA;CAAE,mHAS5B,CAAA"}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { Effect, HttpApp, Layer, RpcSerialization, RpcServer } from '@livestore/utils/effect';
|
|
2
|
+
import { SyncHttpRpc } from "../../../common/http-rpc-schema.js";
|
|
3
|
+
import * as SyncMessage from "../../../common/sync-message-types.js";
|
|
4
|
+
import { DoCtx } from "../layer.js";
|
|
5
|
+
import { makeEndingPullStream } from "../pull.js";
|
|
6
|
+
import { makePush } from "../push.js";
|
|
7
|
+
export const createHttpRpcHandler = ({ request }) => Effect.gen(function* () {
|
|
8
|
+
const handlerLayer = createHttpRpcLayer;
|
|
9
|
+
const httpApp = RpcServer.toHttpApp(SyncHttpRpc).pipe(Effect.provide(handlerLayer));
|
|
10
|
+
const webHandler = yield* httpApp.pipe(Effect.map(HttpApp.toWebHandler));
|
|
11
|
+
return yield* Effect.promise(() => webHandler(request)).pipe(Effect.timeout(10000));
|
|
12
|
+
}).pipe(Effect.withSpan('createHttpRpcHandler'));
|
|
13
|
+
const createHttpRpcLayer =
|
|
14
|
+
// TODO implement admin requests
|
|
15
|
+
SyncHttpRpc.toLayer({
|
|
16
|
+
'SyncHttpRpc.Pull': (req) => makeEndingPullStream(req, req.payload),
|
|
17
|
+
'SyncHttpRpc.Push': (req) => Effect.gen(function* () {
|
|
18
|
+
const { ctx, env, doOptions, storeId } = yield* DoCtx;
|
|
19
|
+
const push = makePush({ payload: undefined, options: doOptions, storeId, ctx, env });
|
|
20
|
+
return yield* push(req);
|
|
21
|
+
}),
|
|
22
|
+
'SyncHttpRpc.Ping': () => Effect.succeed(SyncMessage.Pong.make({})),
|
|
23
|
+
}).pipe(Layer.provideMerge(RpcServer.layerProtocolHttp({ path: '/http-rpc' })), Layer.provideMerge(RpcSerialization.layerJson));
|
|
24
|
+
//# sourceMappingURL=http-rpc-server.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"http-rpc-server.js","sourceRoot":"","sources":["../../../../src/cf-worker/do/transport/http-rpc-server.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,MAAM,EAAE,OAAO,EAAE,KAAK,EAAE,gBAAgB,EAAE,SAAS,EAAE,MAAM,yBAAyB,CAAA;AAC7F,OAAO,EAAE,WAAW,EAAE,MAAM,oCAAoC,CAAA;AAChE,OAAO,KAAK,WAAW,MAAM,uCAAuC,CAAA;AACpE,OAAO,EAAE,KAAK,EAAE,MAAM,aAAa,CAAA;AACnC,OAAO,EAAE,oBAAoB,EAAE,MAAM,YAAY,CAAA;AACjD,OAAO,EAAE,QAAQ,EAAE,MAAM,YAAY,CAAA;AAErC,MAAM,CAAC,MAAM,oBAAoB,GAAG,CAAC,EAAE,OAAO,EAAgC,EAAE,EAAE,CAChF,MAAM,CAAC,GAAG,CAAC,QAAQ,CAAC;IAClB,MAAM,YAAY,GAAG,kBAAkB,CAAA;IACvC,MAAM,OAAO,GAAG,SAAS,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC,CAAA;IACnF,MAAM,UAAU,GAAG,KAAK,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC,CAAA;IAExE,OAAO,KAAK,CAAC,CAAC,MAAM,CAAC,OAAO,CAC1B,GAAG,EAAE,CAAC,UAAU,CAAC,OAA0B,CAAsC,CAClF,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAA;AAC/B,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,sBAAsB,CAAC,CAAC,CAAA;AAElD,MAAM,kBAAkB;AACtB,gCAAgC;AAChC,WAAW,CAAC,OAAO,CAAC;IAClB,kBAAkB,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,oBAAoB,CAAC,GAAG,EAAE,GAAG,CAAC,OAAO,CAAC;IAEnE,kBAAkB,EAAE,CAAC,GAAG,EAAE,EAAE,CAC1B,MAAM,CAAC,GAAG,CAAC,QAAQ,CAAC;QAClB,MAAM,EAAE,GAAG,EAAE,GAAG,EAAE,SAAS,EAAE,OAAO,EAAE,GAAG,KAAK,CAAC,CAAC,KAAK,CAAA;QACrD,MAAM,IAAI,GAAG,QAAQ,CAAC,EAAE,OAAO,EAAE,SAAS,EAAE,OAAO,EAAE,SAAS,EAAE,OAAO,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,CAAA;QAEpF,OAAO,KAAK,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;IACzB,CAAC,CAAC;IAEJ,kBAAkB,EAAE,GAAG,EAAE,CAAC,MAAM,CAAC,OAAO,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;CACpE,CAAC,CAAC,IAAI,CACL,KAAK,CAAC,YAAY,CAAC,SAAS,CAAC,iBAAiB,CAAC,EAAE,IAAI,EAAE,WAAW,EAAE,CAAC,CAAC,EACtE,KAAK,CAAC,YAAY,CAAC,gBAAgB,CAAC,SAAS,CAAC,CAC/C,CAAA"}
|
|
@@ -0,0 +1,4 @@
|
|
|
1
|
+
import { Layer, RpcServer } from '@livestore/utils/effect';
|
|
2
|
+
import { type DoCtxInput } from '../layer.ts';
|
|
3
|
+
export declare const makeRpcServer: ({ doSelf, doOptions }: Omit<DoCtxInput, "from">) => Layer.Layer<never, never, RpcServer.Protocol>;
|
|
4
|
+
//# sourceMappingURL=ws-rpc-server.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ws-rpc-server.d.ts","sourceRoot":"","sources":["../../../../src/cf-worker/do/transport/ws-rpc-server.ts"],"names":[],"mappings":"AACA,OAAO,EAAoB,KAAK,EAAE,SAAS,EAAU,MAAM,yBAAyB,CAAA;AAEpF,OAAO,EAAS,KAAK,UAAU,EAAE,MAAM,aAAa,CAAA;AAIpD,eAAO,MAAM,aAAa,GAAI,uBAAuB,IAAI,CAAC,UAAU,EAAE,MAAM,CAAC,kDA0B5E,CAAA"}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import { InvalidPullError, InvalidPushError } from '@livestore/common';
|
|
2
|
+
import { Effect, identity, Layer, RpcServer, Stream } from '@livestore/utils/effect';
|
|
3
|
+
import { SyncWsRpc } from "../../../common/ws-rpc-schema.js";
|
|
4
|
+
import { DoCtx } from "../layer.js";
|
|
5
|
+
import { makeEndingPullStream } from "../pull.js";
|
|
6
|
+
import { makePush } from "../push.js";
|
|
7
|
+
export const makeRpcServer = ({ doSelf, doOptions }) => {
|
|
8
|
+
// TODO implement admin requests
|
|
9
|
+
const handlersLayer = SyncWsRpc.toLayer({
|
|
10
|
+
'SyncWsRpc.Pull': (req) => makeEndingPullStream(req, req.payload).pipe(
|
|
11
|
+
// Needed to keep the stream alive on the client side for phase 2 (i.e. not send the `Exit` stream RPC message)
|
|
12
|
+
req.live ? Stream.concat(Stream.never) : identity, Stream.provideLayer(DoCtx.Default({ doSelf, doOptions, from: { storeId: req.storeId } })), Stream.mapError((cause) => (cause._tag === 'InvalidPullError' ? cause : InvalidPullError.make({ cause })))),
|
|
13
|
+
'SyncWsRpc.Push': (req) => Effect.gen(function* () {
|
|
14
|
+
const { doOptions, storeId, ctx, env } = yield* DoCtx;
|
|
15
|
+
const push = makePush({ options: doOptions, storeId, payload: req.payload, ctx, env });
|
|
16
|
+
return yield* push(req);
|
|
17
|
+
}).pipe(Effect.provide(DoCtx.Default({ doSelf, doOptions, from: { storeId: req.storeId } })), Effect.mapError((cause) => (cause._tag === 'InvalidPushError' ? cause : InvalidPushError.make({ cause }))), Effect.tapCauseLogPretty),
|
|
18
|
+
});
|
|
19
|
+
return RpcServer.layer(SyncWsRpc).pipe(Layer.provide(handlersLayer));
|
|
20
|
+
};
|
|
21
|
+
//# sourceMappingURL=ws-rpc-server.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ws-rpc-server.js","sourceRoot":"","sources":["../../../../src/cf-worker/do/transport/ws-rpc-server.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,gBAAgB,EAAE,gBAAgB,EAAE,MAAM,mBAAmB,CAAA;AACtE,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,KAAK,EAAE,SAAS,EAAE,MAAM,EAAE,MAAM,yBAAyB,CAAA;AACpF,OAAO,EAAE,SAAS,EAAE,MAAM,kCAAkC,CAAA;AAC5D,OAAO,EAAE,KAAK,EAAmB,MAAM,aAAa,CAAA;AACpD,OAAO,EAAE,oBAAoB,EAAE,MAAM,YAAY,CAAA;AACjD,OAAO,EAAE,QAAQ,EAAE,MAAM,YAAY,CAAA;AAErC,MAAM,CAAC,MAAM,aAAa,GAAG,CAAC,EAAE,MAAM,EAAE,SAAS,EAA4B,EAAE,EAAE;IAC/E,gCAAgC;IAChC,MAAM,aAAa,GAAG,SAAS,CAAC,OAAO,CAAC;QACtC,gBAAgB,EAAE,CAAC,GAAG,EAAE,EAAE,CACxB,oBAAoB,CAAC,GAAG,EAAE,GAAG,CAAC,OAAO,CAAC,CAAC,IAAI;QACzC,+GAA+G;QAC/G,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,QAAQ,EACjD,MAAM,CAAC,YAAY,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,MAAM,EAAE,SAAS,EAAE,IAAI,EAAE,EAAE,OAAO,EAAE,GAAG,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,EACzF,MAAM,CAAC,QAAQ,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,KAAK,CAAC,IAAI,KAAK,kBAAkB,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,gBAAgB,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,CAE3G;QACH,gBAAgB,EAAE,CAAC,GAAG,EAAE,EAAE,CACxB,MAAM,CAAC,GAAG,CAAC,QAAQ,CAAC;YAClB,MAAM,EAAE,SAAS,EAAE,OAAO,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,KAAK,CAAC,CAAC,KAAK,CAAA;YAErD,MAAM,IAAI,GAAG,QAAQ,CAAC,EAAE,OAAO,EAAE,SAAS,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,CAAC,OAAO,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,CAAA;YAEtF,OAAO,KAAK,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;QACzB,CAAC,CAAC,CAAC,IAAI,CACL,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,MAAM,EAAE,SAAS,EAAE,IAAI,EAAE,EAAE,OAAO,EAAE,GAAG,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,EACpF,MAAM,CAAC,QAAQ,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,KAAK,CAAC,IAAI,KAAK,kBAAkB,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,gBAAgB,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,EAC1G,MAAM,CAAC,iBAAiB,CACzB;KACJ,CAAC,CAAA;IAEF,OAAO,SAAS,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC,CAAA;AACtE,CAAC,CAAA"}
|
package/dist/cf-worker/mod.d.ts
CHANGED
|
@@ -1,4 +1,6 @@
|
|
|
1
|
-
export
|
|
2
|
-
export
|
|
1
|
+
export type { CfTypes } from '@livestore/common-cf';
|
|
2
|
+
export { CfDeclare } from '@livestore/common-cf/declare';
|
|
3
|
+
export * from './do/durable-object.ts';
|
|
4
|
+
export * from './shared.ts';
|
|
3
5
|
export * from './worker.ts';
|
|
4
6
|
//# sourceMappingURL=mod.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"mod.d.ts","sourceRoot":"","sources":["../../src/cf-worker/mod.ts"],"names":[],"mappings":"AAAA,cAAc,
|
|
1
|
+
{"version":3,"file":"mod.d.ts","sourceRoot":"","sources":["../../src/cf-worker/mod.ts"],"names":[],"mappings":"AAAA,YAAY,EAAE,OAAO,EAAE,MAAM,sBAAsB,CAAA;AACnD,OAAO,EAAE,SAAS,EAAE,MAAM,8BAA8B,CAAA;AACxD,cAAc,wBAAwB,CAAA;AACtC,cAAc,aAAa,CAAA;AAC3B,cAAc,aAAa,CAAA"}
|
package/dist/cf-worker/mod.js
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"mod.js","sourceRoot":"","sources":["../../src/cf-worker/mod.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"mod.js","sourceRoot":"","sources":["../../src/cf-worker/mod.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,SAAS,EAAE,MAAM,8BAA8B,CAAA;AACxD,cAAc,wBAAwB,CAAA;AACtC,cAAc,aAAa,CAAA;AAC3B,cAAc,aAAa,CAAA"}
|