-
Notifications
You must be signed in to change notification settings - Fork 5
Expand file tree
/
Copy pathserializer.tsx
More file actions
128 lines (122 loc) · 3.78 KB
/
serializer.tsx
File metadata and controls
128 lines (122 loc) · 3.78 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
import { enablePatches, produce as immerProduce, Patch } from "immer";
import { createPlugin, fromJSON, SerovalJSON, toJSON } from "seroval";
import { Accessor, createMemo } from "solid-js";
import {
createSeriazliedMemo,
createSeriazliedProjection,
createSeriazliedRef,
SerializedMemo,
SerializedMemoClass,
SerializedProjection,
SerializedProjectionClass,
SerializedRef,
SerializedRefClass,
} from "./shared";
enablePatches();
export function serializeReactivePayload(scope: string, input: any) {
const refs = new Map<string, Function>();
const signals = new Map<string, Function>();
const value = toJSON(input, {
plugins: [
createPlugin<SerializedRefClass, SerializedRef>({
tag: "seroval-plugins/socket/ref",
test: (value) => value instanceof SerializedRefClass,
parse: {
sync(value) {
const id = crypto.randomUUID();
refs.set(id, value.handler);
return createSeriazliedRef({ scope, id });
},
},
serialize: () => ``,
deserialize: () => ({} as any),
}),
createPlugin<SerializedMemoClass, SerializedMemo>({
tag: "seroval-plugins/socket/memo",
test: (value: any) => value instanceof SerializedMemoClass,
parse: {
sync(value) {
const id = crypto.randomUUID();
signals.set(id, value.signal);
return createSeriazliedMemo({ scope, id });
},
},
serialize: () => ``,
deserialize: () => ({} as any),
}),
createPlugin<SerializedProjectionClass, SerializedProjection>({
tag: "seroval-plugins/socket/projection",
test: (value: any) => value instanceof SerializedProjectionClass,
parse: {
sync(store) {
const id = crypto.randomUUID();
const projection = createMemo(
({ state, changes: _c }) => {
let changes = [] as Patch[];
const s = immerProduce(
state,
store.mutation as any,
(patches) => {
changes.push(...patches);
}
);
return { state: s, changes };
},
{ state: store.init, changes: [] as Patch[] }
);
signals.set(id, () => projection().changes);
return createSeriazliedProjection({
scope,
id,
initial: store.init,
});
},
},
serialize: () => ``,
deserialize: () => ({} as any),
}),
],
});
return { value, refs, signals };
}
export function deserializeReactivePayload(
value: SerovalJSON,
plugins: {
createSocketRefConsumer<I extends any[], O>(
ref: SerializedRef
): (...payload: I) => Promise<O>;
createSocketMemoConsumer<O>(
ref: SerializedMemo<O>
): Accessor<O | undefined>;
createSocketProjectionConsumer<O extends object>(
ref: SerializedProjection<O>
): O;
}
) {
console.log({ value });
return fromJSON<any>(value, {
plugins: [
createPlugin<Function, SerializedRef>({
tag: "seroval-plugins/socket/ref",
test: () => true,
parse: {},
serialize: () => ``,
deserialize: plugins.createSocketRefConsumer,
}),
createPlugin<Function, SerializedMemo>({
tag: "seroval-plugins/socket/memo",
test: () => true,
parse: {},
serialize: () => ``,
deserialize: plugins.createSocketMemoConsumer,
}),
createPlugin<object, SerializedProjection>({
tag: "seroval-plugins/socket/projection",
test: () => true,
parse: {},
serialize: () => ``,
deserialize: plugins.createSocketProjectionConsumer,
}),
],
});
}