atom.io 0.14.1 → 0.14.3
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/data/dist/index.cjs +12 -64
- package/data/dist/index.cjs.map +1 -1
- package/data/dist/index.d.cts +2 -0
- package/data/dist/index.d.ts +459 -0
- package/data/dist/index.js +1 -1
- package/dist/{chunk-KW7RA7IM.js → chunk-C4YZZNRH.js} +4 -4
- package/dist/{chunk-LFXB7Y6M.js → chunk-CWKKQKVQ.js} +2 -2
- package/dist/{chunk-CK7GNCU5.js → chunk-GMN5KH6A.js} +2 -2
- package/dist/{chunk-5VJ77LZE.js → chunk-N7ADBQJG.js} +2 -2
- package/dist/{chunk-TE3ZSTQ6.js → chunk-PURABO5G.js} +92 -27
- package/dist/chunk-PURABO5G.js.map +1 -0
- package/dist/index.cjs +106 -20
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +4 -1
- package/dist/index.d.ts +493 -0
- package/dist/index.js +1 -1
- package/dist/metafile-cjs.json +1 -1
- package/dist/metafile-esm.json +1 -0
- package/internal/dist/index.cjs +93 -29
- package/internal/dist/index.cjs.map +1 -1
- package/internal/dist/index.d.cts +6 -4
- package/internal/dist/index.d.ts +622 -0
- package/internal/dist/index.js +1 -1
- package/internal/src/atom/create-atom.ts +3 -1
- package/internal/src/families/create-atom-family.ts +3 -4
- package/internal/src/mutable/get-update-family.ts +23 -0
- package/internal/src/mutable/index.ts +1 -4
- package/internal/src/mutable/is-mutable.ts +44 -0
- package/internal/src/mutable/tracker.ts +9 -2
- package/internal/src/subject.ts +2 -1
- package/internal/src/timeline/add-atom-to-timeline.ts +10 -1
- package/internal/src/timeline/create-timeline.ts +44 -16
- package/introspection/dist/index.cjs +12 -64
- package/introspection/dist/index.cjs.map +1 -1
- package/introspection/dist/index.d.cts +2 -0
- package/introspection/dist/index.d.ts +396 -0
- package/introspection/dist/index.js +2 -2
- package/json/dist/index.cjs +9 -64
- package/json/dist/index.cjs.map +1 -1
- package/json/dist/index.d.cts +2 -0
- package/json/dist/index.d.ts +417 -0
- package/json/dist/index.js +1 -1
- package/package.json +3 -2
- package/react/dist/index.cjs +9 -64
- package/react/dist/index.cjs.map +1 -1
- package/react/dist/index.d.cts +2 -0
- package/react/dist/index.d.ts +396 -0
- package/react/dist/index.js +2 -2
- package/react-devtools/dist/index.cjs +12 -64
- package/react-devtools/dist/index.cjs.map +1 -1
- package/react-devtools/dist/index.d.cts +2 -0
- package/react-devtools/dist/index.d.ts +467 -0
- package/react-devtools/dist/index.js +3 -3
- package/realtime-client/dist/index.cjs +9 -64
- package/realtime-client/dist/index.cjs.map +1 -1
- package/realtime-client/dist/index.d.cts +2 -0
- package/realtime-client/dist/index.d.ts +394 -0
- package/realtime-client/dist/index.js +2 -2
- package/realtime-react/dist/index.cjs +9 -64
- package/realtime-react/dist/index.cjs.map +1 -1
- package/realtime-react/dist/index.d.ts +67 -0
- package/realtime-react/dist/index.js +4 -4
- package/realtime-server/dist/index.cjs +9 -64
- package/realtime-server/dist/index.cjs.map +1 -1
- package/realtime-server/dist/index.d.cts +4 -1
- package/realtime-server/dist/index.d.ts +408 -0
- package/realtime-server/dist/index.js +1 -1
- package/realtime-testing/dist/index.cjs +104 -79
- package/realtime-testing/dist/index.cjs.map +1 -1
- package/realtime-testing/dist/index.d.cts +4 -1
- package/realtime-testing/dist/index.d.ts +535 -0
- package/realtime-testing/dist/index.js +4 -4
- package/src/atom.ts +3 -1
- package/transceivers/set-rtx/dist/index.cjs +2 -1
- package/transceivers/set-rtx/dist/index.cjs.map +1 -1
- package/transceivers/set-rtx/dist/index.d.ts +66 -0
- package/transceivers/set-rtx/dist/index.js +1 -1
- package/dist/chunk-TE3ZSTQ6.js.map +0 -1
- package/internal/src/mutable/is-atom-token-mutable.ts +0 -7
- /package/dist/{chunk-KW7RA7IM.js.map → chunk-C4YZZNRH.js.map} +0 -0
- /package/dist/{chunk-LFXB7Y6M.js.map → chunk-CWKKQKVQ.js.map} +0 -0
- /package/dist/{chunk-CK7GNCU5.js.map → chunk-GMN5KH6A.js.map} +0 -0
- /package/dist/{chunk-5VJ77LZE.js.map → chunk-N7ADBQJG.js.map} +0 -0
|
@@ -243,6 +243,7 @@ declare class Store implements Lineage {
|
|
|
243
243
|
type Atom<T> = {
|
|
244
244
|
key: string;
|
|
245
245
|
type: `atom`;
|
|
246
|
+
mutable?: boolean;
|
|
246
247
|
family?: FamilyMetadata;
|
|
247
248
|
install: (store: Store) => void;
|
|
248
249
|
subject: Subject<{
|
|
@@ -281,14 +282,16 @@ type AtomFamily<T, K extends Serializable = Serializable> = ((key: K) => AtomTok
|
|
|
281
282
|
key: string;
|
|
282
283
|
type: `atom_family`;
|
|
283
284
|
subject: Subject<AtomToken<T>>;
|
|
285
|
+
mutable?: boolean;
|
|
284
286
|
};
|
|
285
287
|
type MutableAtomFamilyOptions<T extends Transceiver<any>, J extends Serializable, K extends Serializable> = AtomFamilyOptions<T, K> & JsonInterface<T, J> & {
|
|
286
288
|
mutable: true;
|
|
287
289
|
};
|
|
288
290
|
type MutableAtomFamily<Core extends Transceiver<any>, SerializableCore extends Serializable, Key extends Serializable> = JsonInterface<Core, SerializableCore> & ((key: Key) => MutableAtomToken<Core, SerializableCore>) & {
|
|
289
|
-
key: `${string}
|
|
291
|
+
key: `${string}`;
|
|
290
292
|
type: `atom_family`;
|
|
291
293
|
subject: Subject<MutableAtomToken<Core, SerializableCore>>;
|
|
294
|
+
mutable: true;
|
|
292
295
|
};
|
|
293
296
|
declare function atomFamily<T extends Transceiver<any>, J extends Serializable, K extends Serializable>(options: MutableAtomFamilyOptions<T, J, K>): MutableAtomFamily<T, J, K>;
|
|
294
297
|
declare function atomFamily<T, K extends Serializable>(options: AtomFamilyOptions<T, K>): AtomFamily<T, K>;
|
|
@@ -0,0 +1,535 @@
|
|
|
1
|
+
import { RenderResult } from '@testing-library/react';
|
|
2
|
+
import * as React from 'react';
|
|
3
|
+
import * as SocketIO from 'socket.io';
|
|
4
|
+
|
|
5
|
+
declare class Subject<T> {
|
|
6
|
+
Subscriber: (value: T) => void;
|
|
7
|
+
subscribers: Map<string, this[`Subscriber`]>;
|
|
8
|
+
subscribe(key: string, subscriber: this[`Subscriber`]): () => void;
|
|
9
|
+
private unsubscribe;
|
|
10
|
+
next(value: T): void;
|
|
11
|
+
}
|
|
12
|
+
declare class StatefulSubject<T> extends Subject<T> {
|
|
13
|
+
state: T;
|
|
14
|
+
constructor(initialState: T);
|
|
15
|
+
next(value: T): void;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
type Selector<T> = {
|
|
19
|
+
key: string;
|
|
20
|
+
type: `selector`;
|
|
21
|
+
family?: FamilyMetadata;
|
|
22
|
+
install: (store: Store) => void;
|
|
23
|
+
subject: Subject<{
|
|
24
|
+
newValue: T;
|
|
25
|
+
oldValue: T;
|
|
26
|
+
}>;
|
|
27
|
+
get: () => T;
|
|
28
|
+
set: (newValue: T | ((oldValue: T) => T)) => void;
|
|
29
|
+
};
|
|
30
|
+
type ReadonlySelector<T> = {
|
|
31
|
+
key: string;
|
|
32
|
+
type: `readonly_selector`;
|
|
33
|
+
family?: FamilyMetadata;
|
|
34
|
+
install: (store: Store) => void;
|
|
35
|
+
subject: Subject<{
|
|
36
|
+
newValue: T;
|
|
37
|
+
oldValue: T;
|
|
38
|
+
}>;
|
|
39
|
+
get: () => T;
|
|
40
|
+
};
|
|
41
|
+
|
|
42
|
+
type Transaction<ƒ extends ƒn> = {
|
|
43
|
+
key: string;
|
|
44
|
+
type: `transaction`;
|
|
45
|
+
install: (store: Store) => void;
|
|
46
|
+
subject: Subject<TransactionUpdate<ƒ>>;
|
|
47
|
+
run: (...parameters: Parameters<ƒ>) => ReturnType<ƒ>;
|
|
48
|
+
};
|
|
49
|
+
|
|
50
|
+
type TransactionMeta<ƒ extends ƒn> = {
|
|
51
|
+
phase: `applying` | `building`;
|
|
52
|
+
time: number;
|
|
53
|
+
update: TransactionUpdate<ƒ>;
|
|
54
|
+
};
|
|
55
|
+
|
|
56
|
+
type primitive = boolean | number | string | null;
|
|
57
|
+
|
|
58
|
+
type Serializable = primitive | Readonly<{
|
|
59
|
+
[key: string]: Serializable;
|
|
60
|
+
}> | ReadonlyArray<Serializable>;
|
|
61
|
+
type Object$1<Key extends string = string, Value extends Serializable = Serializable> = Record<Key, Value>;
|
|
62
|
+
|
|
63
|
+
type Refinement<Unrefined, Refined extends Unrefined> = (value: Unrefined) => value is Refined;
|
|
64
|
+
type Cardinality = `1:1` | `1:n` | `n:n`;
|
|
65
|
+
|
|
66
|
+
interface JunctionEntries<Content extends Object$1 | null> extends Object$1 {
|
|
67
|
+
readonly relations: [string, string[]][];
|
|
68
|
+
readonly contents: [string, Content][];
|
|
69
|
+
}
|
|
70
|
+
interface JunctionSchema<ASide extends string, BSide extends string> extends Object$1 {
|
|
71
|
+
readonly between: [a: ASide, b: BSide];
|
|
72
|
+
readonly cardinality: Cardinality;
|
|
73
|
+
}
|
|
74
|
+
type BaseExternalStoreConfiguration = {
|
|
75
|
+
addRelation: (a: string, b: string) => void;
|
|
76
|
+
deleteRelation: (a: string, b: string) => void;
|
|
77
|
+
replaceRelationsSafely: (a: string, bs: string[]) => void;
|
|
78
|
+
replaceRelationsUnsafely: (a: string, bs: string[]) => void;
|
|
79
|
+
getRelatedKeys: (key: string) => Set<string> | undefined;
|
|
80
|
+
has: (a: string, b?: string) => boolean;
|
|
81
|
+
};
|
|
82
|
+
type ExternalStoreWithContentConfiguration<Content extends Object$1> = {
|
|
83
|
+
getContent: (contentKey: string) => Content | undefined;
|
|
84
|
+
setContent: (contentKey: string, content: Content) => void;
|
|
85
|
+
deleteContent: (contentKey: string) => void;
|
|
86
|
+
};
|
|
87
|
+
type Empty<Obj extends object> = {
|
|
88
|
+
[Key in keyof Obj]?: undefined;
|
|
89
|
+
};
|
|
90
|
+
type ExternalStoreConfiguration<Content extends Object$1 | null> = Content extends Object$1 ? BaseExternalStoreConfiguration & ExternalStoreWithContentConfiguration<Content> : BaseExternalStoreConfiguration & Empty<ExternalStoreWithContentConfiguration<Object$1>>;
|
|
91
|
+
type JunctionAdvancedConfiguration<Content extends Object$1 | null> = {
|
|
92
|
+
externalStore?: ExternalStoreConfiguration<Content>;
|
|
93
|
+
isContent?: Refinement<unknown, Content>;
|
|
94
|
+
makeContentKey?: (a: string, b: string) => string;
|
|
95
|
+
};
|
|
96
|
+
type JunctionJSON<ASide extends string, BSide extends string, Content extends Object$1 | null> = JunctionEntries<Content> & JunctionSchema<ASide, BSide>;
|
|
97
|
+
declare class Junction<const ASide extends string, const BSide extends string, const Content extends Object$1 | null = null> {
|
|
98
|
+
readonly a: ASide;
|
|
99
|
+
readonly b: BSide;
|
|
100
|
+
readonly cardinality: Cardinality;
|
|
101
|
+
readonly relations: Map<string, Set<string>>;
|
|
102
|
+
readonly contents: Map<string, Content>;
|
|
103
|
+
isContent: Refinement<unknown, Content> | null;
|
|
104
|
+
makeContentKey: (a: string, b: string) => string;
|
|
105
|
+
getRelatedKeys(key: string): Set<string> | undefined;
|
|
106
|
+
protected addRelation(a: string, b: string): void;
|
|
107
|
+
protected deleteRelation(a: string, b: string): void;
|
|
108
|
+
protected replaceRelationsUnsafely(a: string, bs: string[]): void;
|
|
109
|
+
protected replaceRelationsSafely(a: string, bs: string[]): void;
|
|
110
|
+
protected getContentInternal(contentKey: string): Content | undefined;
|
|
111
|
+
protected setContent(contentKey: string, content: Content): void;
|
|
112
|
+
protected deleteContent(contentKey: string): void;
|
|
113
|
+
constructor(data: JunctionSchema<ASide, BSide> & Partial<JunctionEntries<Content>>, config?: JunctionAdvancedConfiguration<Content>);
|
|
114
|
+
toJSON(): JunctionJSON<ASide, BSide, Content>;
|
|
115
|
+
set(a: string, ...rest: Content extends null ? [b: string] : [b: string, content: Content]): this;
|
|
116
|
+
set(relation: {
|
|
117
|
+
[Key in ASide | BSide]: string;
|
|
118
|
+
}, ...rest: Content extends null ? [] | [b?: undefined] : [content: Content]): this;
|
|
119
|
+
delete(a: string, b?: string): this;
|
|
120
|
+
delete(relation: Record<ASide | BSide, string> | Record<ASide, string> | Record<BSide, string>, b?: undefined): this;
|
|
121
|
+
getRelatedKey(key: string): string | undefined;
|
|
122
|
+
replaceRelations(a: string, relations: Content extends null ? string[] : Record<string, Content>, config?: {
|
|
123
|
+
reckless: boolean;
|
|
124
|
+
}): this;
|
|
125
|
+
getContent(a: string, b: string): Content | undefined;
|
|
126
|
+
getRelationEntries(input: Record<ASide, string> | Record<BSide, string>): [string, Content][];
|
|
127
|
+
has(a: string, b?: string): boolean;
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
interface Lineage {
|
|
131
|
+
parent: typeof this | null;
|
|
132
|
+
child: typeof this | null;
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
type JsonInterface<T, J extends Serializable = Serializable> = {
|
|
136
|
+
toJson: (t: T) => J;
|
|
137
|
+
fromJson: (json: J) => T;
|
|
138
|
+
};
|
|
139
|
+
|
|
140
|
+
interface Transceiver<Signal extends Serializable> {
|
|
141
|
+
do: (update: Signal) => void;
|
|
142
|
+
undo: (update: Signal) => void;
|
|
143
|
+
subscribe: (key: string, fn: (update: Signal) => void) => () => void;
|
|
144
|
+
cacheUpdateNumber: number;
|
|
145
|
+
getUpdateNumber: (update: Signal) => number;
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
/**
|
|
149
|
+
* @internal Give the tracker a transceiver state and a store, and it will
|
|
150
|
+
* subscribe to the transceiver's inner value. When the inner value changes,
|
|
151
|
+
* the tracker will update its own state to reflect the change.
|
|
152
|
+
*/
|
|
153
|
+
declare class Tracker<Mutable extends Transceiver<any>> {
|
|
154
|
+
private Update;
|
|
155
|
+
private initializeState;
|
|
156
|
+
private unsubscribeFromInnerValue;
|
|
157
|
+
private observeCore;
|
|
158
|
+
private updateCore;
|
|
159
|
+
mutableState: MutableAtomToken<Mutable, Serializable>;
|
|
160
|
+
latestUpdateState: AtomToken<typeof this.Update | null>;
|
|
161
|
+
constructor(mutableState: MutableAtomToken<Mutable, Serializable>, store: Store);
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
interface MutableAtom<T> extends Atom<T> {
|
|
165
|
+
mutable: true;
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
type OperationProgress = {
|
|
169
|
+
open: false;
|
|
170
|
+
} | {
|
|
171
|
+
open: true;
|
|
172
|
+
done: Set<string>;
|
|
173
|
+
prev: Map<string, any>;
|
|
174
|
+
time: number;
|
|
175
|
+
token: StateToken<any>;
|
|
176
|
+
};
|
|
177
|
+
|
|
178
|
+
type TimelineAtomUpdate = StateUpdate<unknown> & {
|
|
179
|
+
key: string;
|
|
180
|
+
type: `atom_update`;
|
|
181
|
+
timestamp: number;
|
|
182
|
+
family?: FamilyMetadata;
|
|
183
|
+
};
|
|
184
|
+
type TimelineSelectorUpdate = {
|
|
185
|
+
key: string;
|
|
186
|
+
type: `selector_update`;
|
|
187
|
+
timestamp: number;
|
|
188
|
+
atomUpdates: Omit<TimelineAtomUpdate, `timestamp`>[];
|
|
189
|
+
};
|
|
190
|
+
type TimelineTransactionUpdate = TransactionUpdate<ƒn> & {
|
|
191
|
+
key: string;
|
|
192
|
+
type: `transaction_update`;
|
|
193
|
+
timestamp: number;
|
|
194
|
+
};
|
|
195
|
+
type Timeline = {
|
|
196
|
+
type: `timeline`;
|
|
197
|
+
key: string;
|
|
198
|
+
at: number;
|
|
199
|
+
shouldCapture?: (update: TimelineUpdate, timeline: Timeline) => boolean;
|
|
200
|
+
timeTraveling: `into_future` | `into_past` | null;
|
|
201
|
+
history: TimelineUpdate[];
|
|
202
|
+
selectorTime: number | null;
|
|
203
|
+
transactionKey: string | null;
|
|
204
|
+
install: (store: Store) => void;
|
|
205
|
+
subject: Subject<TimelineAtomUpdate | TimelineSelectorUpdate | TimelineTransactionUpdate | `redo` | `undo`>;
|
|
206
|
+
};
|
|
207
|
+
|
|
208
|
+
declare class Store implements Lineage {
|
|
209
|
+
parent: Store | null;
|
|
210
|
+
child: Store | null;
|
|
211
|
+
valueMap: Map<string, any>;
|
|
212
|
+
atoms: Map<string, Atom<any> | MutableAtom<any>>;
|
|
213
|
+
selectors: Map<string, Selector<any>>;
|
|
214
|
+
readonlySelectors: Map<string, ReadonlySelector<any>>;
|
|
215
|
+
trackers: Map<string, Tracker<Transceiver<any>>>;
|
|
216
|
+
families: Map<string, AtomFamily<any, any> | ReadonlySelectorFamily<any, any> | SelectorFamily<any, any>>;
|
|
217
|
+
timelines: Map<string, Timeline>;
|
|
218
|
+
transactions: Map<string, Transaction<ƒn>>;
|
|
219
|
+
atomsThatAreDefault: Set<string>;
|
|
220
|
+
timelineAtoms: Junction<"timelineKey", "atomKey", null>;
|
|
221
|
+
selectorAtoms: Junction<"selectorKey", "atomKey", null>;
|
|
222
|
+
selectorGraph: Junction<"upstreamSelectorKey", "downstreamSelectorKey", {
|
|
223
|
+
source: string;
|
|
224
|
+
}>;
|
|
225
|
+
subject: {
|
|
226
|
+
atomCreation: Subject<AtomToken<unknown>>;
|
|
227
|
+
selectorCreation: Subject<ReadonlySelectorToken<unknown> | SelectorToken<unknown>>;
|
|
228
|
+
transactionCreation: Subject<TransactionToken<ƒn>>;
|
|
229
|
+
timelineCreation: Subject<TimelineToken>;
|
|
230
|
+
transactionApplying: StatefulSubject<TransactionMeta<ƒn> | null>;
|
|
231
|
+
operationStatus: Subject<OperationProgress>;
|
|
232
|
+
};
|
|
233
|
+
operation: OperationProgress;
|
|
234
|
+
transactionMeta: TransactionMeta<ƒn> | null;
|
|
235
|
+
config: {
|
|
236
|
+
name: string;
|
|
237
|
+
};
|
|
238
|
+
loggers: AtomIOLogger[];
|
|
239
|
+
logger: Logger;
|
|
240
|
+
constructor(name: string, store?: Store | null);
|
|
241
|
+
}
|
|
242
|
+
|
|
243
|
+
type Atom<T> = {
|
|
244
|
+
key: string;
|
|
245
|
+
type: `atom`;
|
|
246
|
+
mutable?: boolean;
|
|
247
|
+
family?: FamilyMetadata;
|
|
248
|
+
install: (store: Store) => void;
|
|
249
|
+
subject: Subject<{
|
|
250
|
+
newValue: T;
|
|
251
|
+
oldValue: T;
|
|
252
|
+
}>;
|
|
253
|
+
default: T | (() => T);
|
|
254
|
+
cleanup?: () => void;
|
|
255
|
+
};
|
|
256
|
+
|
|
257
|
+
type Effectors<T> = {
|
|
258
|
+
setSelf: <V extends T>(next: V | ((oldValue: T) => V)) => void;
|
|
259
|
+
onSet: (callback: (options: {
|
|
260
|
+
newValue: T;
|
|
261
|
+
oldValue: T;
|
|
262
|
+
}) => void) => void;
|
|
263
|
+
};
|
|
264
|
+
type AtomEffect<T> = (tools: Effectors<T>) => (() => void) | void;
|
|
265
|
+
type AtomOptions<T> = {
|
|
266
|
+
key: string;
|
|
267
|
+
default: T | (() => T);
|
|
268
|
+
effects?: AtomEffect<T>[];
|
|
269
|
+
};
|
|
270
|
+
type MutableAtomOptions<T extends Transceiver<any>, J extends Serializable> = JsonInterface<T, J> & Omit<AtomOptions<T>, `default`> & {
|
|
271
|
+
default: () => T;
|
|
272
|
+
mutable: true;
|
|
273
|
+
};
|
|
274
|
+
declare function atom<T extends Transceiver<any>, J extends Serializable>(options: MutableAtomOptions<T, J>): MutableAtomToken<T, J>;
|
|
275
|
+
declare function atom<T>(options: AtomOptions<T>): AtomToken<T>;
|
|
276
|
+
type AtomFamilyOptions<T, K extends Serializable> = {
|
|
277
|
+
key: string;
|
|
278
|
+
default: T | ((key: K) => T);
|
|
279
|
+
effects?: (key: K) => AtomEffect<T>[];
|
|
280
|
+
};
|
|
281
|
+
type AtomFamily<T, K extends Serializable = Serializable> = ((key: K) => AtomToken<T>) & {
|
|
282
|
+
key: string;
|
|
283
|
+
type: `atom_family`;
|
|
284
|
+
subject: Subject<AtomToken<T>>;
|
|
285
|
+
mutable?: boolean;
|
|
286
|
+
};
|
|
287
|
+
type MutableAtomFamilyOptions<T extends Transceiver<any>, J extends Serializable, K extends Serializable> = AtomFamilyOptions<T, K> & JsonInterface<T, J> & {
|
|
288
|
+
mutable: true;
|
|
289
|
+
};
|
|
290
|
+
type MutableAtomFamily<Core extends Transceiver<any>, SerializableCore extends Serializable, Key extends Serializable> = JsonInterface<Core, SerializableCore> & ((key: Key) => MutableAtomToken<Core, SerializableCore>) & {
|
|
291
|
+
key: `${string}`;
|
|
292
|
+
type: `atom_family`;
|
|
293
|
+
subject: Subject<MutableAtomToken<Core, SerializableCore>>;
|
|
294
|
+
mutable: true;
|
|
295
|
+
};
|
|
296
|
+
declare function atomFamily<T extends Transceiver<any>, J extends Serializable, K extends Serializable>(options: MutableAtomFamilyOptions<T, J, K>): MutableAtomFamily<T, J, K>;
|
|
297
|
+
declare function atomFamily<T, K extends Serializable>(options: AtomFamilyOptions<T, K>): AtomFamily<T, K>;
|
|
298
|
+
|
|
299
|
+
declare function getState<T>(token: ReadonlySelectorToken<T> | StateToken<T>, store?: Store): T;
|
|
300
|
+
|
|
301
|
+
declare const LoggerIconDictionary: {
|
|
302
|
+
readonly "\u231B": "Timeline event fully captured";
|
|
303
|
+
readonly "\u23E9": "Timeline redo";
|
|
304
|
+
readonly "\u23EA": "Timeline undo";
|
|
305
|
+
readonly "\u23ED\uFE0F": "Transaction redo";
|
|
306
|
+
readonly "\u23EE\uFE0F": "Transaction undo";
|
|
307
|
+
readonly "\u23F3": "Timeline event partially captured";
|
|
308
|
+
readonly "\u23F9\uFE0F": "Time-travel complete";
|
|
309
|
+
readonly "\uD83D\uDC81": "Notice";
|
|
310
|
+
readonly "\uD83D\uDD04": "Realtime transaction synchronized";
|
|
311
|
+
readonly "\u2705": "Realtime transaction success";
|
|
312
|
+
readonly "\u2728": "Computation complete";
|
|
313
|
+
readonly "\u274C": "Conflict prevents attempted action";
|
|
314
|
+
readonly "\u2B55": "Operation start";
|
|
315
|
+
readonly "\uD83D\uDC1E": "Possible bug in AtomIO";
|
|
316
|
+
readonly "\uD83D\uDC40": "Subscription added";
|
|
317
|
+
readonly "\uD83D\uDC6A": "Family member added";
|
|
318
|
+
readonly "\uD83D\uDCC1": "Stow update";
|
|
319
|
+
readonly "\uD83D\uDCC3": "Copy mutable";
|
|
320
|
+
readonly "\uD83D\uDCD6": "Read state";
|
|
321
|
+
readonly "\uD83D\uDCDD": "Write state";
|
|
322
|
+
readonly "\uD83D\uDCE2": "Notify subscribers";
|
|
323
|
+
readonly "\uD83D\uDD0C": "Register dependency";
|
|
324
|
+
readonly "\uD83D\uDD0D": "Discover root";
|
|
325
|
+
readonly "\uD83D\uDD25": "Delete state";
|
|
326
|
+
readonly "\uD83D\uDD27": "Create mutable atom";
|
|
327
|
+
readonly "\uD83D\uDD28": "Create immutable atom";
|
|
328
|
+
readonly "\uD83D\uDD34": "Operation complete";
|
|
329
|
+
readonly "\uD83D\uDDD1": "Evict cached value";
|
|
330
|
+
readonly "\uD83D\uDCA5": "Caught";
|
|
331
|
+
readonly "\uD83D\uDE48": "Subscription canceled";
|
|
332
|
+
readonly "\uD83D\uDEC4": "Apply transaction";
|
|
333
|
+
readonly "\uD83D\uDEE0\uFE0F": "Install atom into store";
|
|
334
|
+
readonly "\uD83D\uDEEB": "Begin transaction";
|
|
335
|
+
readonly "\uD83D\uDEEC": "Complete transaction";
|
|
336
|
+
readonly "\uD83E\uDDEE": "Computing selector";
|
|
337
|
+
readonly "\uD83E\uDDF9": "Prepare to evict";
|
|
338
|
+
readonly "\uD83E\uDE82": "Abort transaction";
|
|
339
|
+
};
|
|
340
|
+
type LoggerIcon = keyof typeof LoggerIconDictionary;
|
|
341
|
+
declare const LOG_LEVELS: readonly ["info", "warn", "error"];
|
|
342
|
+
type LogLevel = (typeof LOG_LEVELS)[number];
|
|
343
|
+
type LogFn = (icon: LoggerIcon, tokenType: `atom` | `readonly_selector` | `selector` | `state` | `timeline` | `transaction` | `unknown`, tokenKey: string, message: string, ...rest: unknown[]) => void;
|
|
344
|
+
type LogFilter = (...params: Parameters<LogFn>) => boolean;
|
|
345
|
+
type Logger = Record<LogLevel, LogFn>;
|
|
346
|
+
declare class AtomIOLogger implements Logger {
|
|
347
|
+
logLevel: `error` | `info` | `warn` | null;
|
|
348
|
+
private readonly filter?;
|
|
349
|
+
private readonly logger;
|
|
350
|
+
constructor(logLevel: `error` | `info` | `warn` | null, filter?: LogFilter | undefined, logger?: Logger);
|
|
351
|
+
error: LogFn;
|
|
352
|
+
info: LogFn;
|
|
353
|
+
warn: LogFn;
|
|
354
|
+
}
|
|
355
|
+
|
|
356
|
+
type TransactionToken<_> = {
|
|
357
|
+
key: string;
|
|
358
|
+
type: `transaction`;
|
|
359
|
+
__brand?: _;
|
|
360
|
+
};
|
|
361
|
+
type TransactionUpdate<ƒ extends ƒn> = {
|
|
362
|
+
key: string;
|
|
363
|
+
updates: (KeyedStateUpdate<unknown> | TransactionUpdate<ƒn>)[];
|
|
364
|
+
params: Parameters<ƒ>;
|
|
365
|
+
output: ReturnType<ƒ>;
|
|
366
|
+
};
|
|
367
|
+
type Transactors = {
|
|
368
|
+
get: <S>(state: ReadonlySelectorToken<S> | StateToken<S>) => S;
|
|
369
|
+
set: <S>(state: StateToken<S>, newValue: S | ((oldValue: S) => S)) => void;
|
|
370
|
+
};
|
|
371
|
+
type ReadonlyTransactors = Pick<Transactors, `get`>;
|
|
372
|
+
type Read<ƒ extends ƒn> = (transactors: ReadonlyTransactors, ...parameters: Parameters<ƒ>) => ReturnType<ƒ>;
|
|
373
|
+
type Write<ƒ extends ƒn> = (transactors: Transactors, ...parameters: Parameters<ƒ>) => ReturnType<ƒ>;
|
|
374
|
+
type Act<ƒ extends ƒn> = (transactors: Transactors & {
|
|
375
|
+
run: typeof runTransaction;
|
|
376
|
+
}, ...parameters: Parameters<ƒ>) => ReturnType<ƒ>;
|
|
377
|
+
type TransactionOptions<ƒ extends ƒn> = {
|
|
378
|
+
key: string;
|
|
379
|
+
do: Act<ƒ>;
|
|
380
|
+
};
|
|
381
|
+
declare function transaction<ƒ extends ƒn>(options: TransactionOptions<ƒ>): TransactionToken<ƒ>;
|
|
382
|
+
declare const runTransaction: <ƒ extends ƒn>(token: TransactionToken<ƒ>, store?: Store) => (...parameters: Parameters<ƒ>) => ReturnType<ƒ>;
|
|
383
|
+
|
|
384
|
+
type SelectorOptions<T> = {
|
|
385
|
+
key: string;
|
|
386
|
+
get: Read<() => T>;
|
|
387
|
+
set: Write<(newValue: T) => void>;
|
|
388
|
+
};
|
|
389
|
+
type ReadonlySelectorOptions<T> = {
|
|
390
|
+
key: string;
|
|
391
|
+
get: Read<() => T>;
|
|
392
|
+
};
|
|
393
|
+
declare function selector<T>(options: SelectorOptions<T>): SelectorToken<T>;
|
|
394
|
+
declare function selector<T>(options: ReadonlySelectorOptions<T>): ReadonlySelectorToken<T>;
|
|
395
|
+
type SelectorFamilyOptions<T, K extends Serializable> = {
|
|
396
|
+
key: string;
|
|
397
|
+
get: (key: K) => Read<() => T>;
|
|
398
|
+
set: (key: K) => Write<(newValue: T) => void>;
|
|
399
|
+
};
|
|
400
|
+
type ReadonlySelectorFamilyOptions<T, K extends Serializable> = {
|
|
401
|
+
key: string;
|
|
402
|
+
get: (key: K) => Read<() => T>;
|
|
403
|
+
};
|
|
404
|
+
type SelectorFamily<T, K extends Serializable = Serializable> = ((key: K) => SelectorToken<T>) & {
|
|
405
|
+
key: string;
|
|
406
|
+
type: `selector_family`;
|
|
407
|
+
subject: Subject<SelectorToken<T>>;
|
|
408
|
+
};
|
|
409
|
+
type ReadonlySelectorFamily<T, K extends Serializable = Serializable> = ((key: K) => ReadonlySelectorToken<T>) & {
|
|
410
|
+
key: string;
|
|
411
|
+
type: `readonly_selector_family`;
|
|
412
|
+
subject: Subject<ReadonlySelectorToken<T>>;
|
|
413
|
+
};
|
|
414
|
+
declare function selectorFamily<T, K extends Serializable>(options: SelectorFamilyOptions<T, K>): SelectorFamily<T, K>;
|
|
415
|
+
declare function selectorFamily<T, K extends Serializable>(options: ReadonlySelectorFamilyOptions<T, K>): ReadonlySelectorFamily<T, K>;
|
|
416
|
+
|
|
417
|
+
declare function setState<T, New extends T>(token: StateToken<T>, value: New | ((oldValue: T) => New), store?: Store): void;
|
|
418
|
+
|
|
419
|
+
declare class Silo {
|
|
420
|
+
store: Store;
|
|
421
|
+
atom: typeof atom;
|
|
422
|
+
atomFamily: typeof atomFamily;
|
|
423
|
+
selector: typeof selector;
|
|
424
|
+
selectorFamily: typeof selectorFamily;
|
|
425
|
+
transaction: typeof transaction;
|
|
426
|
+
timeline: typeof timeline;
|
|
427
|
+
getState: typeof getState;
|
|
428
|
+
setState: typeof setState;
|
|
429
|
+
subscribe: typeof subscribe;
|
|
430
|
+
undo: typeof undo;
|
|
431
|
+
redo: typeof redo;
|
|
432
|
+
constructor(name: string, fromStore?: Store | null);
|
|
433
|
+
}
|
|
434
|
+
|
|
435
|
+
type StateUpdate<T> = {
|
|
436
|
+
newValue: T;
|
|
437
|
+
oldValue: T;
|
|
438
|
+
};
|
|
439
|
+
type KeyedStateUpdate<T> = StateUpdate<T> & {
|
|
440
|
+
key: string;
|
|
441
|
+
family?: FamilyMetadata;
|
|
442
|
+
};
|
|
443
|
+
type UpdateHandler<T> = (update: StateUpdate<T>) => void;
|
|
444
|
+
type TransactionUpdateHandler<ƒ extends ƒn> = (data: TransactionUpdate<ƒ>) => void;
|
|
445
|
+
declare function subscribe<T>(token: ReadonlySelectorToken<T> | StateToken<T>, handleUpdate: UpdateHandler<T>, key?: string, store?: Store): () => void;
|
|
446
|
+
declare function subscribe<ƒ extends ƒn>(token: TransactionToken<ƒ>, handleUpdate: TransactionUpdateHandler<ƒ>, key?: string, store?: Store): () => void;
|
|
447
|
+
declare function subscribe(token: TimelineToken, handleUpdate: (update: TimelineUpdate | `redo` | `undo`) => void, key?: string, store?: Store): () => void;
|
|
448
|
+
|
|
449
|
+
type TimelineToken = {
|
|
450
|
+
key: string;
|
|
451
|
+
type: `timeline`;
|
|
452
|
+
};
|
|
453
|
+
type TimelineOptions = {
|
|
454
|
+
key: string;
|
|
455
|
+
atoms: (AtomFamily<any, any> | AtomToken<any>)[];
|
|
456
|
+
shouldCapture?: (update: TimelineUpdate, timeline: Timeline) => boolean;
|
|
457
|
+
};
|
|
458
|
+
type TimelineUpdate = TimelineAtomUpdate | TimelineSelectorUpdate | TimelineTransactionUpdate;
|
|
459
|
+
declare const timeline: (options: TimelineOptions) => TimelineToken;
|
|
460
|
+
declare const redo: (timeline: TimelineToken) => void;
|
|
461
|
+
declare const undo: (timeline: TimelineToken) => void;
|
|
462
|
+
|
|
463
|
+
type ƒn = (...parameters: any[]) => any;
|
|
464
|
+
type AtomToken<_> = {
|
|
465
|
+
key: string;
|
|
466
|
+
type: `atom`;
|
|
467
|
+
family?: FamilyMetadata;
|
|
468
|
+
__brand?: _;
|
|
469
|
+
};
|
|
470
|
+
interface MutableAtomToken<T extends Transceiver<any>, J extends Serializable> extends AtomToken<T> {
|
|
471
|
+
__asJSON?: J;
|
|
472
|
+
__update?: T extends Transceiver<infer Update> ? Update : never;
|
|
473
|
+
}
|
|
474
|
+
type SelectorToken<_> = {
|
|
475
|
+
key: string;
|
|
476
|
+
type: `selector`;
|
|
477
|
+
family?: FamilyMetadata;
|
|
478
|
+
__brand?: _;
|
|
479
|
+
};
|
|
480
|
+
type StateToken<T> = AtomToken<T> | SelectorToken<T>;
|
|
481
|
+
type ReadonlySelectorToken<_> = {
|
|
482
|
+
key: string;
|
|
483
|
+
type: `readonly_selector`;
|
|
484
|
+
family?: FamilyMetadata;
|
|
485
|
+
__brand?: _;
|
|
486
|
+
};
|
|
487
|
+
type FamilyMetadata = {
|
|
488
|
+
key: string;
|
|
489
|
+
subKey: string;
|
|
490
|
+
};
|
|
491
|
+
|
|
492
|
+
type TestSetupOptions = {
|
|
493
|
+
server: (tools: {
|
|
494
|
+
socket: SocketIO.Socket;
|
|
495
|
+
silo: Silo;
|
|
496
|
+
}) => void;
|
|
497
|
+
};
|
|
498
|
+
type TestSetupOptions__SingleClient = TestSetupOptions & {
|
|
499
|
+
client: React.FC;
|
|
500
|
+
};
|
|
501
|
+
type TestSetupOptions__MultiClient<ClientNames extends string> = TestSetupOptions & {
|
|
502
|
+
clients: {
|
|
503
|
+
[K in ClientNames]: React.FC;
|
|
504
|
+
};
|
|
505
|
+
};
|
|
506
|
+
type RealtimeTestTools = {
|
|
507
|
+
name: string;
|
|
508
|
+
silo: Silo;
|
|
509
|
+
dispose: () => void;
|
|
510
|
+
};
|
|
511
|
+
type RealtimeTestClient = RealtimeTestTools & {
|
|
512
|
+
renderResult: RenderResult;
|
|
513
|
+
prettyPrint: () => void;
|
|
514
|
+
reconnect: () => void;
|
|
515
|
+
disconnect: () => void;
|
|
516
|
+
};
|
|
517
|
+
type RealtimeTestServer = RealtimeTestTools & {
|
|
518
|
+
port: number;
|
|
519
|
+
};
|
|
520
|
+
type RealtimeTestAPI = {
|
|
521
|
+
server: RealtimeTestServer;
|
|
522
|
+
teardown: () => void;
|
|
523
|
+
};
|
|
524
|
+
type RealtimeTestAPI__SingleClient = RealtimeTestAPI & {
|
|
525
|
+
client: RealtimeTestClient;
|
|
526
|
+
};
|
|
527
|
+
type RealtimeTestAPI__MultiClient<ClientNames extends string> = RealtimeTestAPI & {
|
|
528
|
+
clients: Record<ClientNames, RealtimeTestClient>;
|
|
529
|
+
};
|
|
530
|
+
declare const setupRealtimeTestServer: (options: TestSetupOptions) => RealtimeTestServer;
|
|
531
|
+
declare const setupRealtimeTestClient: (options: TestSetupOptions__SingleClient, name: string, port: number) => RealtimeTestClient;
|
|
532
|
+
declare const singleClient: (options: TestSetupOptions__SingleClient) => RealtimeTestAPI__SingleClient;
|
|
533
|
+
declare const multiClient: <ClientNames extends string>(options: TestSetupOptions__MultiClient<ClientNames>) => RealtimeTestAPI__MultiClient<ClientNames>;
|
|
534
|
+
|
|
535
|
+
export { type RealtimeTestAPI, type RealtimeTestAPI__MultiClient, type RealtimeTestAPI__SingleClient, type RealtimeTestClient, type RealtimeTestServer, type RealtimeTestTools, type TestSetupOptions, type TestSetupOptions__MultiClient, type TestSetupOptions__SingleClient, multiClient, setupRealtimeTestClient, setupRealtimeTestServer, singleClient };
|
|
@@ -1,8 +1,8 @@
|
|
|
1
|
-
import { RealtimeProvider } from '../../dist/chunk-
|
|
1
|
+
import { RealtimeProvider } from '../../dist/chunk-C4YZZNRH.js';
|
|
2
2
|
import { recordToEntries } from '../../dist/chunk-NYCVSXQB.js';
|
|
3
|
-
import { StoreProvider } from '../../dist/chunk-
|
|
4
|
-
import '../../dist/chunk-
|
|
5
|
-
import { Silo, IMPLICIT, __spreadProps, __spreadValues, clearStore } from '../../dist/chunk-
|
|
3
|
+
import { StoreProvider } from '../../dist/chunk-CWKKQKVQ.js';
|
|
4
|
+
import '../../dist/chunk-N7ADBQJG.js';
|
|
5
|
+
import { Silo, IMPLICIT, __spreadProps, __spreadValues, clearStore } from '../../dist/chunk-PURABO5G.js';
|
|
6
6
|
import * as http from 'http';
|
|
7
7
|
import { render, prettyDOM } from '@testing-library/react';
|
|
8
8
|
import * as Happy from 'happy-dom';
|
package/src/atom.ts
CHANGED
|
@@ -56,6 +56,7 @@ export type AtomFamily<T, K extends Json.Serializable = Json.Serializable> = ((
|
|
|
56
56
|
key: string
|
|
57
57
|
type: `atom_family`
|
|
58
58
|
subject: Subject<AtomToken<T>>
|
|
59
|
+
mutable?: boolean
|
|
59
60
|
}
|
|
60
61
|
// biome-ignore format: intersection
|
|
61
62
|
export type MutableAtomFamilyOptions<
|
|
@@ -76,9 +77,10 @@ export type MutableAtomFamily<
|
|
|
76
77
|
& JsonInterface<Core, SerializableCore>
|
|
77
78
|
& ((key: Key) => MutableAtomToken<Core, SerializableCore>)
|
|
78
79
|
& {
|
|
79
|
-
key: `${string}
|
|
80
|
+
key: `${string}`
|
|
80
81
|
type: `atom_family`
|
|
81
82
|
subject: Subject<MutableAtomToken<Core, SerializableCore>>
|
|
83
|
+
mutable: true
|
|
82
84
|
}
|
|
83
85
|
|
|
84
86
|
export function atomFamily<
|
|
@@ -18,7 +18,8 @@ var Subject = class {
|
|
|
18
18
|
this.subscribers.delete(key);
|
|
19
19
|
}
|
|
20
20
|
next(value) {
|
|
21
|
-
|
|
21
|
+
const subscribers = this.subscribers.values();
|
|
22
|
+
for (const subscriber of subscribers) {
|
|
22
23
|
subscriber(value);
|
|
23
24
|
}
|
|
24
25
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../../../anvl/src/json/index.ts","../../../internal/src/subject.ts","../src/set-rtx.ts"],"names":["update","value"],"mappings":"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","sourcesContent":["export * from \"./json-interface\"\nimport type * as Json from \"./json\"\nexport { Json }\n\nexport const parseJson = <S extends Stringified<Json.Serializable>>(\n\tstr: S | string,\n): S extends Stringified<infer J> ? J : Json.Serializable => JSON.parse(str)\n\nexport type Stringified<J extends Json.Serializable> = string & { __json: J }\n\nexport const stringifyJson = <J extends Json.Serializable>(\n\tjson: J,\n): Stringified<J> => JSON.stringify(json) as Stringified<J>\n\nexport type Empty = Record<string, never>\n\nexport const JSON_TYPE_NAMES = [\n\t`array`,\n\t`boolean`,\n\t`null`,\n\t`number`,\n\t`object`,\n\t`string`,\n] as const\n\nexport type JsonTypeName = (typeof JSON_TYPE_NAMES)[number]\n\nexport interface JsonTypes extends Record<JsonTypeName, Json.Serializable> {\n\tarray: Json.Array\n\tboolean: boolean\n\tnull: null\n\tnumber: number\n\tobject: Json.Object\n\tstring: string\n}\n\nexport const JSON_DEFAULTS: JsonTypes = {\n\tarray: [],\n\tboolean: false,\n\tnull: null,\n\tnumber: 0,\n\tobject: {},\n\tstring: ``,\n}\n","export class Subject<T> {\n\tpublic Subscriber: (value: T) => void\n\n\tpublic subscribers: Map<string, this[`Subscriber`]> = new Map()\n\n\tpublic subscribe(key: string, subscriber: this[`Subscriber`]): () => void {\n\t\tthis.subscribers.set(key, subscriber)\n\t\tconst unsubscribe = () => this.unsubscribe(key)\n\t\treturn unsubscribe\n\t}\n\n\tprivate unsubscribe(key: string) {\n\t\tthis.subscribers.delete(key)\n\t}\n\n\tpublic next(value: T): void {\n\t\tfor (const subscriber of this.subscribers.values()) {\n\t\t\tsubscriber(value)\n\t\t}\n\t}\n}\n\nexport class StatefulSubject<T> extends Subject<T> {\n\tpublic state: T\n\n\tpublic constructor(initialState: T) {\n\t\tsuper()\n\t\tthis.state = initialState\n\t}\n\n\tpublic next(value: T): void {\n\t\tthis.state = value\n\t\tsuper.next(value)\n\t}\n}\n","import type { Lineage, Transceiver, TransceiverMode } from \"atom.io/internal\"\nimport { Subject } from \"atom.io/internal\"\nimport type { Json, Stringified, primitive } from \"atom.io/json\"\nimport { parseJson, stringifyJson } from \"atom.io/json\"\n\nexport type SetUpdate =\n\t| `add:${string}`\n\t| `clear:${string}`\n\t| `del:${string}`\n\t| `tx:${string}`\nexport type NumberedSetUpdate = `${number}=${SetUpdate}`\n\nexport interface SetRTXJson<P extends primitive> extends Json.Object {\n\tmembers: P[]\n\tcache: (NumberedSetUpdate | null)[]\n\tcacheLimit: number\n\tcacheIdx: number\n\tcacheUpdateNumber: number\n}\nexport class SetRTX<P extends primitive>\n\textends Set<P>\n\timplements Transceiver<NumberedSetUpdate>, Lineage\n{\n\tpublic mode: TransceiverMode = `record`\n\tpublic readonly subject = new Subject<SetUpdate>()\n\tpublic cacheLimit = 0\n\tpublic cache: (NumberedSetUpdate | null)[] = []\n\tpublic cacheIdx = -1\n\tpublic cacheUpdateNumber = -1\n\n\tpublic constructor(values?: Iterable<P>, cacheLimit = 0) {\n\t\tsuper(values)\n\t\tif (values instanceof SetRTX) {\n\t\t\tthis.parent = values\n\t\t\tthis.cacheUpdateNumber = values.cacheUpdateNumber\n\t\t}\n\t\tif (cacheLimit) {\n\t\t\tthis.cacheLimit = cacheLimit\n\t\t\tthis.cache = new Array(cacheLimit)\n\t\t\tthis.subscribe(`auto cache`, (update) => {\n\t\t\t\tthis.cacheIdx++\n\t\t\t\tthis.cacheIdx %= this.cacheLimit\n\t\t\t\tthis.cache[this.cacheIdx] = update\n\t\t\t})\n\t\t}\n\t}\n\n\tpublic toJSON(): SetRTXJson<P> {\n\t\treturn {\n\t\t\tmembers: [...this],\n\t\t\tcache: this.cache,\n\t\t\tcacheLimit: this.cacheLimit,\n\t\t\tcacheIdx: this.cacheIdx,\n\t\t\tcacheUpdateNumber: this.cacheUpdateNumber,\n\t\t}\n\t}\n\n\tpublic static fromJSON<P extends primitive>(json: SetRTXJson<P>): SetRTX<P> {\n\t\tconst set = new SetRTX<P>(json.members, json.cacheLimit)\n\t\tset.cache = json.cache\n\t\tset.cacheIdx = json.cacheIdx\n\t\tset.cacheUpdateNumber = json.cacheUpdateNumber\n\t\treturn set\n\t}\n\n\tpublic add(value: P): this {\n\t\tif (this.mode === `record`) {\n\t\t\tthis.cacheUpdateNumber++\n\t\t\tthis.emit(`add:${stringifyJson<P>(value)}`)\n\t\t}\n\t\treturn super.add(value)\n\t}\n\n\tpublic clear(): void {\n\t\tif (this.mode === `record`) {\n\t\t\tthis.cacheUpdateNumber++\n\t\t\tthis.emit(`clear:${JSON.stringify([...this])}`)\n\t\t}\n\t\tsuper.clear()\n\t}\n\n\tpublic delete(value: P): boolean {\n\t\tif (this.mode === `record`) {\n\t\t\tthis.cacheUpdateNumber++\n\t\t\tthis.emit(`del:${stringifyJson<P>(value)}`)\n\t\t}\n\t\treturn super.delete(value)\n\t}\n\n\tpublic readonly parent: SetRTX<P> | null\n\tpublic child: SetRTX<P> | null = null\n\tpublic transactionUpdates: SetUpdate[] | null = null\n\tpublic transaction(run: (child: SetRTX<P>) => boolean): void {\n\t\tthis.mode = `transaction`\n\t\tthis.transactionUpdates = []\n\t\tthis.child = new SetRTX(this)\n\t\tconst unsubscribe = this.child._subscribe(`transaction`, (update) => {\n\t\t\tthis.transactionUpdates?.push(update)\n\t\t})\n\t\ttry {\n\t\t\tconst shouldCommit = run(this.child)\n\t\t\tif (shouldCommit) {\n\t\t\t\tthis.cacheUpdateNumber++\n\t\t\t\tthis.emit(`tx:${this.transactionUpdates.join(`;`)}`)\n\t\t\t\tfor (const update of this.transactionUpdates) {\n\t\t\t\t\tthis.doStep(update)\n\t\t\t\t}\n\t\t\t}\n\t\t} catch (thrown) {\n\t\t\tconsole.error(`Failed to apply transaction: ${thrown}`)\n\t\t} finally {\n\t\t\tunsubscribe()\n\t\t\tthis.child = null\n\t\t\tthis.transactionUpdates = null\n\t\t\tthis.mode = `record`\n\t\t}\n\t}\n\n\tprotected _subscribe(\n\t\tkey: string,\n\t\tfn: (update: SetUpdate) => void,\n\t): () => void {\n\t\treturn this.subject.subscribe(key, fn)\n\t}\n\tpublic subscribe(\n\t\tkey: string,\n\t\tfn: (update: NumberedSetUpdate) => void,\n\t): () => void {\n\t\treturn this.subject.subscribe(key, (update) =>\n\t\t\tfn(`${this.cacheUpdateNumber}=${update}`),\n\t\t)\n\t}\n\n\tpublic emit(update: SetUpdate): void {\n\t\tthis.subject.next(update)\n\t}\n\n\tprivate doStep(update: SetUpdate): void {\n\t\tconst typeValueBreak = update.indexOf(`:`)\n\t\tconst type = update.substring(0, typeValueBreak)\n\t\tconst value = update.substring(typeValueBreak + 1)\n\t\tswitch (type) {\n\t\t\tcase `add`:\n\t\t\t\tthis.add(parseJson(value as Stringified<P>))\n\t\t\t\tbreak\n\t\t\tcase `clear`:\n\t\t\t\tthis.clear()\n\t\t\t\tbreak\n\t\t\tcase `del`:\n\t\t\t\tthis.delete(parseJson(value as Stringified<P>))\n\t\t\t\tbreak\n\t\t\tcase `tx`:\n\t\t\t\tfor (const update of value.split(`;`)) {\n\t\t\t\t\tthis.doStep(update as SetUpdate)\n\t\t\t\t}\n\t\t}\n\t}\n\n\tpublic getUpdateNumber(update: NumberedSetUpdate): number {\n\t\tconst breakpoint = update.indexOf(`=`)\n\t\treturn Number(update.substring(0, breakpoint))\n\t}\n\n\tpublic do(update: NumberedSetUpdate): number | `OUT_OF_RANGE` | null {\n\t\tconst breakpoint = update.indexOf(`=`)\n\t\tconst updateNumber = Number(update.substring(0, breakpoint))\n\t\tconst eventOffset = updateNumber - this.cacheUpdateNumber\n\t\tconst isFuture = eventOffset > 0\n\t\tif (isFuture) {\n\t\t\tif (eventOffset === 1) {\n\t\t\t\tthis.mode = `playback`\n\t\t\t\tconst innerUpdate = update.substring(breakpoint + 1) as SetUpdate\n\t\t\t\tthis.doStep(innerUpdate)\n\t\t\t\tthis.mode = `record`\n\t\t\t\tthis.cacheUpdateNumber = updateNumber\n\t\t\t\treturn null\n\t\t\t}\n\t\t\treturn this.cacheUpdateNumber + 1\n\t\t}\n\t\tif (Math.abs(eventOffset) < this.cacheLimit) {\n\t\t\tconst eventIdx = this.cacheIdx + eventOffset\n\t\t\tconst cachedUpdate = this.cache[eventIdx]\n\t\t\tif (cachedUpdate === update) {\n\t\t\t\treturn null\n\t\t\t}\n\t\t\tthis.mode = `playback`\n\t\t\tlet done = false\n\t\t\twhile (!done) {\n\t\t\t\tthis.cacheIdx %= this.cacheLimit\n\t\t\t\tconst update = this.cache[this.cacheIdx]\n\t\t\t\tthis.cacheIdx--\n\t\t\t\tif (!update) {\n\t\t\t\t\treturn `OUT_OF_RANGE`\n\t\t\t\t}\n\t\t\t\tthis.undo(update)\n\t\t\t\tdone = this.cacheIdx === eventIdx - 1\n\t\t\t}\n\t\t\tconst innerUpdate = update.substring(breakpoint + 1) as SetUpdate\n\t\t\tthis.doStep(innerUpdate)\n\t\t\tthis.mode = `record`\n\t\t\tthis.cacheUpdateNumber = updateNumber\n\t\t\treturn null\n\t\t}\n\t\treturn `OUT_OF_RANGE`\n\t}\n\n\tpublic undoStep(update: SetUpdate): void {\n\t\tconst breakpoint = update.indexOf(`:`)\n\t\tconst type = update.substring(0, breakpoint)\n\t\tconst value = update.substring(breakpoint + 1)\n\t\tswitch (type) {\n\t\t\tcase `add`:\n\t\t\t\tthis.delete(parseJson(value as Stringified<P>))\n\t\t\t\tbreak\n\t\t\tcase `del`:\n\t\t\t\tthis.add(parseJson(value as Stringified<P>))\n\t\t\t\tbreak\n\t\t\tcase `clear`: {\n\t\t\t\tconst values = JSON.parse(value) as P[]\n\t\t\t\tfor (const value of values) this.add(value)\n\t\t\t\tbreak\n\t\t\t}\n\t\t\tcase `tx`: {\n\t\t\t\tconst updates = value.split(`;`) as SetUpdate[]\n\t\t\t\tfor (let i = updates.length - 1; i >= 0; i--) {\n\t\t\t\t\tthis.undoStep(updates[i])\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\tpublic undo(update: NumberedSetUpdate): number | null {\n\t\tconst breakpoint = update.indexOf(`=`)\n\t\tconst updateNumber = Number(update.substring(0, breakpoint))\n\t\tif (updateNumber === this.cacheUpdateNumber) {\n\t\t\tthis.mode = `playback`\n\t\t\tconst innerUpdate = update.substring(breakpoint + 1) as SetUpdate\n\t\t\tthis.undoStep(innerUpdate)\n\t\t\tthis.mode = `record`\n\t\t\tthis.cacheUpdateNumber--\n\t\t\treturn null\n\t\t}\n\t\treturn this.cacheUpdateNumber\n\t}\n}\n"]}
|
|
1
|
+
{"version":3,"sources":["../../../../anvl/src/json/index.ts","../../../internal/src/subject.ts","../src/set-rtx.ts"],"names":["update","value"],"mappings":"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","sourcesContent":["export * from \"./json-interface\"\nimport type * as Json from \"./json\"\nexport { Json }\n\nexport const parseJson = <S extends Stringified<Json.Serializable>>(\n\tstr: S | string,\n): S extends Stringified<infer J> ? J : Json.Serializable => JSON.parse(str)\n\nexport type Stringified<J extends Json.Serializable> = string & { __json: J }\n\nexport const stringifyJson = <J extends Json.Serializable>(\n\tjson: J,\n): Stringified<J> => JSON.stringify(json) as Stringified<J>\n\nexport type Empty = Record<string, never>\n\nexport const JSON_TYPE_NAMES = [\n\t`array`,\n\t`boolean`,\n\t`null`,\n\t`number`,\n\t`object`,\n\t`string`,\n] as const\n\nexport type JsonTypeName = (typeof JSON_TYPE_NAMES)[number]\n\nexport interface JsonTypes extends Record<JsonTypeName, Json.Serializable> {\n\tarray: Json.Array\n\tboolean: boolean\n\tnull: null\n\tnumber: number\n\tobject: Json.Object\n\tstring: string\n}\n\nexport const JSON_DEFAULTS: JsonTypes = {\n\tarray: [],\n\tboolean: false,\n\tnull: null,\n\tnumber: 0,\n\tobject: {},\n\tstring: ``,\n}\n","export class Subject<T> {\n\tpublic Subscriber: (value: T) => void\n\n\tpublic subscribers: Map<string, this[`Subscriber`]> = new Map()\n\n\tpublic subscribe(key: string, subscriber: this[`Subscriber`]): () => void {\n\t\tthis.subscribers.set(key, subscriber)\n\t\tconst unsubscribe = () => this.unsubscribe(key)\n\t\treturn unsubscribe\n\t}\n\n\tprivate unsubscribe(key: string) {\n\t\tthis.subscribers.delete(key)\n\t}\n\n\tpublic next(value: T): void {\n\t\tconst subscribers = this.subscribers.values()\n\t\tfor (const subscriber of subscribers) {\n\t\t\tsubscriber(value)\n\t\t}\n\t}\n}\n\nexport class StatefulSubject<T> extends Subject<T> {\n\tpublic state: T\n\n\tpublic constructor(initialState: T) {\n\t\tsuper()\n\t\tthis.state = initialState\n\t}\n\n\tpublic next(value: T): void {\n\t\tthis.state = value\n\t\tsuper.next(value)\n\t}\n}\n","import type { Lineage, Transceiver, TransceiverMode } from \"atom.io/internal\"\nimport { Subject } from \"atom.io/internal\"\nimport type { Json, Stringified, primitive } from \"atom.io/json\"\nimport { parseJson, stringifyJson } from \"atom.io/json\"\n\nexport type SetUpdate =\n\t| `add:${string}`\n\t| `clear:${string}`\n\t| `del:${string}`\n\t| `tx:${string}`\nexport type NumberedSetUpdate = `${number}=${SetUpdate}`\n\nexport interface SetRTXJson<P extends primitive> extends Json.Object {\n\tmembers: P[]\n\tcache: (NumberedSetUpdate | null)[]\n\tcacheLimit: number\n\tcacheIdx: number\n\tcacheUpdateNumber: number\n}\nexport class SetRTX<P extends primitive>\n\textends Set<P>\n\timplements Transceiver<NumberedSetUpdate>, Lineage\n{\n\tpublic mode: TransceiverMode = `record`\n\tpublic readonly subject = new Subject<SetUpdate>()\n\tpublic cacheLimit = 0\n\tpublic cache: (NumberedSetUpdate | null)[] = []\n\tpublic cacheIdx = -1\n\tpublic cacheUpdateNumber = -1\n\n\tpublic constructor(values?: Iterable<P>, cacheLimit = 0) {\n\t\tsuper(values)\n\t\tif (values instanceof SetRTX) {\n\t\t\tthis.parent = values\n\t\t\tthis.cacheUpdateNumber = values.cacheUpdateNumber\n\t\t}\n\t\tif (cacheLimit) {\n\t\t\tthis.cacheLimit = cacheLimit\n\t\t\tthis.cache = new Array(cacheLimit)\n\t\t\tthis.subscribe(`auto cache`, (update) => {\n\t\t\t\tthis.cacheIdx++\n\t\t\t\tthis.cacheIdx %= this.cacheLimit\n\t\t\t\tthis.cache[this.cacheIdx] = update\n\t\t\t})\n\t\t}\n\t}\n\n\tpublic toJSON(): SetRTXJson<P> {\n\t\treturn {\n\t\t\tmembers: [...this],\n\t\t\tcache: this.cache,\n\t\t\tcacheLimit: this.cacheLimit,\n\t\t\tcacheIdx: this.cacheIdx,\n\t\t\tcacheUpdateNumber: this.cacheUpdateNumber,\n\t\t}\n\t}\n\n\tpublic static fromJSON<P extends primitive>(json: SetRTXJson<P>): SetRTX<P> {\n\t\tconst set = new SetRTX<P>(json.members, json.cacheLimit)\n\t\tset.cache = json.cache\n\t\tset.cacheIdx = json.cacheIdx\n\t\tset.cacheUpdateNumber = json.cacheUpdateNumber\n\t\treturn set\n\t}\n\n\tpublic add(value: P): this {\n\t\tif (this.mode === `record`) {\n\t\t\tthis.cacheUpdateNumber++\n\t\t\tthis.emit(`add:${stringifyJson<P>(value)}`)\n\t\t}\n\t\treturn super.add(value)\n\t}\n\n\tpublic clear(): void {\n\t\tif (this.mode === `record`) {\n\t\t\tthis.cacheUpdateNumber++\n\t\t\tthis.emit(`clear:${JSON.stringify([...this])}`)\n\t\t}\n\t\tsuper.clear()\n\t}\n\n\tpublic delete(value: P): boolean {\n\t\tif (this.mode === `record`) {\n\t\t\tthis.cacheUpdateNumber++\n\t\t\tthis.emit(`del:${stringifyJson<P>(value)}`)\n\t\t}\n\t\treturn super.delete(value)\n\t}\n\n\tpublic readonly parent: SetRTX<P> | null\n\tpublic child: SetRTX<P> | null = null\n\tpublic transactionUpdates: SetUpdate[] | null = null\n\tpublic transaction(run: (child: SetRTX<P>) => boolean): void {\n\t\tthis.mode = `transaction`\n\t\tthis.transactionUpdates = []\n\t\tthis.child = new SetRTX(this)\n\t\tconst unsubscribe = this.child._subscribe(`transaction`, (update) => {\n\t\t\tthis.transactionUpdates?.push(update)\n\t\t})\n\t\ttry {\n\t\t\tconst shouldCommit = run(this.child)\n\t\t\tif (shouldCommit) {\n\t\t\t\tthis.cacheUpdateNumber++\n\t\t\t\tthis.emit(`tx:${this.transactionUpdates.join(`;`)}`)\n\t\t\t\tfor (const update of this.transactionUpdates) {\n\t\t\t\t\tthis.doStep(update)\n\t\t\t\t}\n\t\t\t}\n\t\t} catch (thrown) {\n\t\t\tconsole.error(`Failed to apply transaction: ${thrown}`)\n\t\t} finally {\n\t\t\tunsubscribe()\n\t\t\tthis.child = null\n\t\t\tthis.transactionUpdates = null\n\t\t\tthis.mode = `record`\n\t\t}\n\t}\n\n\tprotected _subscribe(\n\t\tkey: string,\n\t\tfn: (update: SetUpdate) => void,\n\t): () => void {\n\t\treturn this.subject.subscribe(key, fn)\n\t}\n\tpublic subscribe(\n\t\tkey: string,\n\t\tfn: (update: NumberedSetUpdate) => void,\n\t): () => void {\n\t\treturn this.subject.subscribe(key, (update) =>\n\t\t\tfn(`${this.cacheUpdateNumber}=${update}`),\n\t\t)\n\t}\n\n\tpublic emit(update: SetUpdate): void {\n\t\tthis.subject.next(update)\n\t}\n\n\tprivate doStep(update: SetUpdate): void {\n\t\tconst typeValueBreak = update.indexOf(`:`)\n\t\tconst type = update.substring(0, typeValueBreak)\n\t\tconst value = update.substring(typeValueBreak + 1)\n\t\tswitch (type) {\n\t\t\tcase `add`:\n\t\t\t\tthis.add(parseJson(value as Stringified<P>))\n\t\t\t\tbreak\n\t\t\tcase `clear`:\n\t\t\t\tthis.clear()\n\t\t\t\tbreak\n\t\t\tcase `del`:\n\t\t\t\tthis.delete(parseJson(value as Stringified<P>))\n\t\t\t\tbreak\n\t\t\tcase `tx`:\n\t\t\t\tfor (const update of value.split(`;`)) {\n\t\t\t\t\tthis.doStep(update as SetUpdate)\n\t\t\t\t}\n\t\t}\n\t}\n\n\tpublic getUpdateNumber(update: NumberedSetUpdate): number {\n\t\tconst breakpoint = update.indexOf(`=`)\n\t\treturn Number(update.substring(0, breakpoint))\n\t}\n\n\tpublic do(update: NumberedSetUpdate): number | `OUT_OF_RANGE` | null {\n\t\tconst breakpoint = update.indexOf(`=`)\n\t\tconst updateNumber = Number(update.substring(0, breakpoint))\n\t\tconst eventOffset = updateNumber - this.cacheUpdateNumber\n\t\tconst isFuture = eventOffset > 0\n\t\tif (isFuture) {\n\t\t\tif (eventOffset === 1) {\n\t\t\t\tthis.mode = `playback`\n\t\t\t\tconst innerUpdate = update.substring(breakpoint + 1) as SetUpdate\n\t\t\t\tthis.doStep(innerUpdate)\n\t\t\t\tthis.mode = `record`\n\t\t\t\tthis.cacheUpdateNumber = updateNumber\n\t\t\t\treturn null\n\t\t\t}\n\t\t\treturn this.cacheUpdateNumber + 1\n\t\t}\n\t\tif (Math.abs(eventOffset) < this.cacheLimit) {\n\t\t\tconst eventIdx = this.cacheIdx + eventOffset\n\t\t\tconst cachedUpdate = this.cache[eventIdx]\n\t\t\tif (cachedUpdate === update) {\n\t\t\t\treturn null\n\t\t\t}\n\t\t\tthis.mode = `playback`\n\t\t\tlet done = false\n\t\t\twhile (!done) {\n\t\t\t\tthis.cacheIdx %= this.cacheLimit\n\t\t\t\tconst update = this.cache[this.cacheIdx]\n\t\t\t\tthis.cacheIdx--\n\t\t\t\tif (!update) {\n\t\t\t\t\treturn `OUT_OF_RANGE`\n\t\t\t\t}\n\t\t\t\tthis.undo(update)\n\t\t\t\tdone = this.cacheIdx === eventIdx - 1\n\t\t\t}\n\t\t\tconst innerUpdate = update.substring(breakpoint + 1) as SetUpdate\n\t\t\tthis.doStep(innerUpdate)\n\t\t\tthis.mode = `record`\n\t\t\tthis.cacheUpdateNumber = updateNumber\n\t\t\treturn null\n\t\t}\n\t\treturn `OUT_OF_RANGE`\n\t}\n\n\tpublic undoStep(update: SetUpdate): void {\n\t\tconst breakpoint = update.indexOf(`:`)\n\t\tconst type = update.substring(0, breakpoint)\n\t\tconst value = update.substring(breakpoint + 1)\n\t\tswitch (type) {\n\t\t\tcase `add`:\n\t\t\t\tthis.delete(parseJson(value as Stringified<P>))\n\t\t\t\tbreak\n\t\t\tcase `del`:\n\t\t\t\tthis.add(parseJson(value as Stringified<P>))\n\t\t\t\tbreak\n\t\t\tcase `clear`: {\n\t\t\t\tconst values = JSON.parse(value) as P[]\n\t\t\t\tfor (const value of values) this.add(value)\n\t\t\t\tbreak\n\t\t\t}\n\t\t\tcase `tx`: {\n\t\t\t\tconst updates = value.split(`;`) as SetUpdate[]\n\t\t\t\tfor (let i = updates.length - 1; i >= 0; i--) {\n\t\t\t\t\tthis.undoStep(updates[i])\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\tpublic undo(update: NumberedSetUpdate): number | null {\n\t\tconst breakpoint = update.indexOf(`=`)\n\t\tconst updateNumber = Number(update.substring(0, breakpoint))\n\t\tif (updateNumber === this.cacheUpdateNumber) {\n\t\t\tthis.mode = `playback`\n\t\t\tconst innerUpdate = update.substring(breakpoint + 1) as SetUpdate\n\t\t\tthis.undoStep(innerUpdate)\n\t\t\tthis.mode = `record`\n\t\t\tthis.cacheUpdateNumber--\n\t\t\treturn null\n\t\t}\n\t\treturn this.cacheUpdateNumber\n\t}\n}\n"]}
|