Skip to content

Commit 804f01d

Browse files
committed
chore(serdes): rename serialize(d=>r), new api
1 parent 761240e commit 804f01d

15 files changed

+345
-173
lines changed

.changeset/nasty-planes-jam.md

+2-2
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,8 @@
22
'@qwik.dev/core': minor
33
---
44

5-
FEAT: `useSerialized$(fn)` and `createSerialized$(fn)` allow serializing custom objects. You must provide a
5+
FEAT: `useSerializer$(fn)` and `createSerializer$(fn)` allow serializing custom objects. You must provide a
66
function that converts the custom object to a serializable one via the `[SerializerSymbol]`
7-
property, and then provide `use|createSerialized$(fn)` with the function that creates the custom object
7+
property, and then provide `use|createSerializer$(fn)` with the function that creates the custom object
88
from the serialized data. This will lazily create the value when needed. Note that the serializer
99
function may return a Promise, which will be awaited. The deserializer must not return a Promise.

packages/docs/src/routes/api/qwik/api.json

+12-12
Original file line numberDiff line numberDiff line change
@@ -222,18 +222,18 @@
222222
"mdFile": "core.createcontextid.md"
223223
},
224224
{
225-
"name": "createSerialized$",
226-
"id": "createserialized_",
225+
"name": "createSerializer$",
226+
"id": "createserializer_",
227227
"hierarchy": [
228228
{
229-
"name": "createSerialized$",
230-
"id": "createserialized_"
229+
"name": "createSerializer$",
230+
"id": "createserializer_"
231231
}
232232
],
233233
"kind": "Function",
234-
"content": "Create a signal that holds a custom serializable value. See [useSerialized$](#useserialized_) for more details.\n\n\n```typescript\ncreateSerialized$: <T extends CustomSerializable<any, S>, S = T extends {\n [SerializerSymbol]: (obj: any) => infer U;\n} ? U : unknown>(qrl: (data: S | undefined) => T) => T extends Promise<any> ? never : SerializedSignal<T>\n```\n\n\n<table><thead><tr><th>\n\nParameter\n\n\n</th><th>\n\nType\n\n\n</th><th>\n\nDescription\n\n\n</th></tr></thead>\n<tbody><tr><td>\n\nqrl\n\n\n</td><td>\n\n(data: S \\| undefined) =&gt; T\n\n\n</td><td>\n\n\n</td></tr>\n</tbody></table>\n**Returns:**\n\nT extends Promise&lt;any&gt; ? never : SerializedSignal&lt;T&gt;",
234+
"content": "Create a signal that holds a custom serializable value. See [useSerializer$](#useserializer_) for more details.\n\n\n```typescript\ncreateSerializer$: <T, S>(arg: SerializerArg<T, S>) => T extends Promise<any> ? never : SerializerSignal<T>\n```\n\n\n<table><thead><tr><th>\n\nParameter\n\n\n</th><th>\n\nType\n\n\n</th><th>\n\nDescription\n\n\n</th></tr></thead>\n<tbody><tr><td>\n\narg\n\n\n</td><td>\n\nSerializerArg&lt;T, S&gt;\n\n\n</td><td>\n\n\n</td></tr>\n</tbody></table>\n**Returns:**\n\nT extends Promise&lt;any&gt; ? never : SerializerSignal&lt;T&gt;",
235235
"editUrl": "https://github.com/QwikDev/qwik/tree/main/packages/qwik/src/core/signal/signal.public.ts",
236-
"mdFile": "core.createserialized_.md"
236+
"mdFile": "core.createserializer_.md"
237237
},
238238
{
239239
"name": "createSignal",
@@ -2046,18 +2046,18 @@
20462046
"mdFile": "core.useresource_.md"
20472047
},
20482048
{
2049-
"name": "useSerialized$",
2050-
"id": "useserialized_",
2049+
"name": "useSerializer$",
2050+
"id": "useserializer_",
20512051
"hierarchy": [
20522052
{
2053-
"name": "useSerialized$",
2054-
"id": "useserialized_"
2053+
"name": "useSerializer$",
2054+
"id": "useserializer_"
20552055
}
20562056
],
20572057
"kind": "Variable",
2058-
"content": "Creates a signal which holds a custom serializable value. It requires that the value implements the `CustomSerializable` type, which means having a function under the `[SerializeSymbol]` property that returns a serializable value when called.\n\nThe `fn` you pass is called with the result of the serialization (in the browser, only when the value is needed), or `undefined` when not yet initialized. If you refer to other signals, `fn` will be called when those change just like computed signals, and then the argument will be the previous output, not the serialized result.\n\nThis is useful when using third party libraries that use custom objects that are not serializable.\n\nNote that the `fn` is called lazily, so it won't impact container resume.\n\n\n```typescript\nuseSerialized$: typeof createSerialized$\n```\n\n\n\n```tsx\nclass MyCustomSerializable {\n constructor(public n: number) {}\n inc() {\n this.n++;\n }\n [SerializeSymbol]() {\n return this.n;\n }\n}\nconst Cmp = component$(() => {\n const custom = useSerialized$<MyCustomSerializable, number>(\n (prev) =>\n new MyCustomSerializable(prev instanceof MyCustomSerializable ? prev : (prev ?? 3))\n );\n return <div onClick$={() => custom.value.inc()}>{custom.value.n}</div>;\n});\n```",
2058+
"content": "Creates a signal which holds a custom serializable value. It requires that the value implements the `CustomSerializable` type, which means having a function under the `[SerializeSymbol]` property that returns a serializable value when called.\n\nThe `fn` you pass is called with the result of the serialization (in the browser, only when the value is needed), or `undefined` when not yet initialized. If you refer to other signals, `fn` will be called when those change just like computed signals, and then the argument will be the previous output, not the serialized result.\n\nThis is useful when using third party libraries that use custom objects that are not serializable.\n\nNote that the `fn` is called lazily, so it won't impact container resume.\n\n\n```typescript\nuseSerializer$: typeof createSerializer$\n```\n\n\n\n```tsx\nclass MyCustomSerializable {\n constructor(public n: number) {}\n inc() {\n this.n++;\n }\n}\nconst Cmp = component$(() => {\n const custom = useSerializer$({\n deserialize: (data) => new MyCustomSerializable(data),\n serialize: (data) => data.n,\n initial: 2,\n });\n return <div onClick$={() => custom.value.inc()}>{custom.value.n}</div>;\n});\n```\n\n\nWhen using a Signal as the data to create the object, you may not need `serialize`<!-- -->. Furthermore, when the signal is updated, the serializer will be updated as well, and the current object will be passed as the second argument.\n\n```tsx\nconst Cmp = component$(() => {\n const n = useSignal(2);\n const custom = useSerializer$((_data, current) => {\n if (current) {\n current.n = n.value;\n return current;\n }\n return new MyCustomSerializable(n.value);\n});\n return <div onClick$={() => n.value++}>{custom.value.n}</div>;\n});\n```\n(note that in this example, the `{custom.value.n}` is not reactive, so the div text will not update)",
20592059
"editUrl": "https://github.com/QwikDev/qwik/tree/main/packages/qwik/src/core/use/use-serialized.ts",
2060-
"mdFile": "core.useserialized_.md"
2060+
"mdFile": "core.useserializer_.md"
20612061
},
20622062
{
20632063
"name": "useServerData",

packages/docs/src/routes/api/qwik/index.md

+31-19
Original file line numberDiff line numberDiff line change
@@ -723,14 +723,12 @@ The name of the context.
723723

724724
[Edit this section](https://github.com/QwikDev/qwik/tree/main/packages/qwik/src/core/use/use-context.ts)
725725

726-
## createSerialized$
726+
## createSerializer$
727727

728-
Create a signal that holds a custom serializable value. See [useSerialized$](#useserialized_) for more details.
728+
Create a signal that holds a custom serializable value. See [useSerializer$](#useserializer_) for more details.
729729

730730
```typescript
731-
createSerialized$: <T extends CustomSerializable<any, S>, S = T extends {
732-
[SerializerSymbol]: (obj: any) => infer U;
733-
} ? U : unknown>(qrl: (data: S | undefined) => T) => T extends Promise<any> ? never : SerializedSignal<T>
731+
createSerializer$: <T, S>(arg: SerializerArg<T, S>) => T extends Promise<any> ? never : SerializerSignal<T>
734732
```
735733

736734
<table><thead><tr><th>
@@ -748,19 +746,19 @@ Description
748746
</th></tr></thead>
749747
<tbody><tr><td>
750748

751-
qrl
749+
arg
752750

753751
</td><td>
754752

755-
(data: S \| undefined) =&gt; T
753+
SerializerArg&lt;T, S&gt;
756754

757755
</td><td>
758756

759757
</td></tr>
760758
</tbody></table>
761759
**Returns:**
762760

763-
T extends Promise&lt;any&gt; ? never : SerializedSignal&lt;T&gt;
761+
T extends Promise&lt;any&gt; ? never : SerializerSignal&lt;T&gt;
764762

765763
[Edit this section](https://github.com/QwikDev/qwik/tree/main/packages/qwik/src/core/signal/signal.public.ts)
766764

@@ -4946,7 +4944,7 @@ _(Optional)_
49464944
49474945
[Edit this section](https://github.com/QwikDev/qwik/tree/main/packages/qwik/src/core/use/use-resource-dollar.ts)
49484946
4949-
## useSerialized$
4947+
## useSerializer$
49504948
49514949
Creates a signal which holds a custom serializable value. It requires that the value implements the `CustomSerializable` type, which means having a function under the `[SerializeSymbol]` property that returns a serializable value when called.
49524950
@@ -4957,7 +4955,7 @@ This is useful when using third party libraries that use custom objects that are
49574955
Note that the `fn` is called lazily, so it won't impact container resume.
49584956
49594957
```typescript
4960-
useSerialized$: typeof createSerialized$;
4958+
useSerializer$: typeof createSerializer$;
49614959
```
49624960
49634961
```tsx
@@ -4966,21 +4964,35 @@ class MyCustomSerializable {
49664964
inc() {
49674965
this.n++;
49684966
}
4969-
[SerializeSymbol]() {
4970-
return this.n;
4971-
}
49724967
}
49734968
const Cmp = component$(() => {
4974-
const custom = useSerialized$<MyCustomSerializable, number>(
4975-
(prev) =>
4976-
new MyCustomSerializable(
4977-
prev instanceof MyCustomSerializable ? prev : (prev ?? 3),
4978-
),
4979-
);
4969+
const custom = useSerializer$({
4970+
deserialize: (data) => new MyCustomSerializable(data),
4971+
serialize: (data) => data.n,
4972+
initial: 2,
4973+
});
49804974
return <div onClick$={() => custom.value.inc()}>{custom.value.n}</div>;
49814975
});
49824976
```
49834977
4978+
When using a Signal as the data to create the object, you may not need `serialize`. Furthermore, when the signal is updated, the serializer will be updated as well, and the current object will be passed as the second argument.
4979+
4980+
```tsx
4981+
const Cmp = component$(() => {
4982+
const n = useSignal(2);
4983+
const custom = useSerializer$((_data, current) => {
4984+
if (current) {
4985+
current.n = n.value;
4986+
return current;
4987+
}
4988+
return new MyCustomSerializable(n.value);
4989+
});
4990+
return <div onClick$={() => n.value++}>{custom.value.n}</div>;
4991+
});
4992+
```
4993+
4994+
(note that in this example, the `{custom.value.n}` is not reactive, so the div text will not update)
4995+
49844996
[Edit this section](https://github.com/QwikDev/qwik/tree/main/packages/qwik/src/core/use/use-serialized.ts)
49854997
49864998
## useServerData

packages/qwik/public.d.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ export {
66
ComputedSignal,
77
ContextId,
88
createComputed$,
9-
createSerialized$,
9+
createSerializer$,
1010
createContextId,
1111
createSignal,
1212
CSSProperties,
@@ -61,7 +61,7 @@ export {
6161
useOnDocument,
6262
useOnWindow,
6363
useResource$,
64-
useSerialized$,
64+
useSerializer$,
6565
useServerData,
6666
useSignal,
6767
useStore,

packages/qwik/src/core/api.md

+13-12
Original file line numberDiff line numberDiff line change
@@ -120,20 +120,21 @@ export const createComputedQrl: <T>(qrl: QRL<() => T>) => ComputedSignal_2<T>;
120120
// @public
121121
export const createContextId: <STATE = unknown>(name: string) => ContextId<STATE>;
122122

123-
// Warning: (ae-forgotten-export) The symbol "CustomSerializable" needs to be exported by the entry point index.d.ts
124-
// Warning: (ae-forgotten-export) The symbol "SerializedSignal" needs to be exported by the entry point index.d.ts
123+
// Warning: (ae-forgotten-export) The symbol "SerializerArg" needs to be exported by the entry point index.d.ts
124+
// Warning: (ae-forgotten-export) The symbol "SerializerSignal" needs to be exported by the entry point index.d.ts
125125
//
126126
// @public
127-
export const createSerialized$: <T extends CustomSerializable<any, S>, S = T extends {
128-
[SerializerSymbol]: (obj: any) => infer U;
129-
} ? U : unknown>(qrl: (data: S | undefined) => T) => T extends Promise<any> ? never : SerializedSignal<T>;
127+
export const createSerializer$: <T, S>(arg: SerializerArg<T, S>) => T extends Promise<any> ? never : SerializerSignal<T>;
130128

131-
// Warning: (ae-forgotten-export) The symbol "ConstructorFn" needs to be exported by the entry point index.d.ts
132-
// Warning: (ae-forgotten-export) The symbol "SerializedSignal_2" needs to be exported by the entry point index.d.ts
133-
// Warning: (ae-internal-missing-underscore) The name "createSerializedQrl" should be prefixed with an underscore because the declaration is marked as @internal
129+
// Warning: (ae-forgotten-export) The symbol "SerializerSignal_2" needs to be exported by the entry point index.d.ts
130+
// Warning: (ae-internal-missing-underscore) The name "createSerializerQrl" should be prefixed with an underscore because the declaration is marked as @internal
134131
//
135132
// @internal (undocumented)
136-
export const createSerializedQrl: <T extends CustomSerializable<T, S>, S>(qrl: QRL<ConstructorFn<T, S>>) => SerializedSignal_2<T>;
133+
export const createSerializerQrl: <T, S>(arg: QRL<{
134+
serialize: (data: S | undefined) => T;
135+
deserialize: (data: T) => S;
136+
initial?: S;
137+
}>) => SerializerSignal_2<T, S>;
137138

138139
// @public
139140
export const createSignal: {
@@ -1109,12 +1110,12 @@ export const useResource$: <T>(generatorFn: ResourceFn<T>, opts?: ResourceOption
11091110
export const useResourceQrl: <T>(qrl: QRL<ResourceFn<T>>, opts?: ResourceOptions) => ResourceReturn<T>;
11101111

11111112
// @public
1112-
export const useSerialized$: typeof createSerialized$;
1113+
export const useSerializer$: typeof createSerializer$;
11131114

1114-
// Warning: (ae-internal-missing-underscore) The name "useSerializedQrl" should be prefixed with an underscore because the declaration is marked as @internal
1115+
// Warning: (ae-internal-missing-underscore) The name "useSerializerQrl" should be prefixed with an underscore because the declaration is marked as @internal
11151116
//
11161117
// @internal (undocumented)
1117-
export const useSerializedQrl: <F extends ConstructorFn<any, any>>(qrl: QRL<F>) => ReadonlySignal<unknown>;
1118+
export const useSerializerQrl: <T, S>(qrl: QRL<SerializerArg<T, S>>) => ReadonlySignal<unknown>;
11181119

11191120
// @public (undocumented)
11201121
export function useServerData<T>(key: string): T | undefined;

packages/qwik/src/core/index.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -110,7 +110,7 @@ export type { ContextId } from './use/use-context';
110110
export type { UseStoreOptions } from './use/use-store.public';
111111
export type { ComputedFn } from './use/use-computed';
112112
export { useComputedQrl } from './use/use-computed';
113-
export { useSerializedQrl, useSerialized$ } from './use/use-serialized';
113+
export { useSerializerQrl, useSerializer$ } from './use/use-serialized';
114114
export type { OnVisibleTaskOptions, VisibleTaskStrategy } from './use/use-visible-task';
115115
export { useVisibleTaskQrl } from './use/use-visible-task';
116116
export type { EagernessOptions, TaskCtx, TaskFn, Tracker, UseTaskOptions } from './use/use-task';
@@ -138,8 +138,8 @@ export {
138138
createSignal,
139139
createComputedQrl,
140140
createComputed$,
141-
createSerializedQrl,
142-
createSerialized$,
141+
createSerializerQrl,
142+
createSerializer$,
143143
} from './signal/signal.public';
144144
export { EffectPropData as _EffectData } from './signal/signal';
145145

packages/qwik/src/core/shared/serialization.md

+14
Original file line numberDiff line numberDiff line change
@@ -37,3 +37,17 @@ To avoid blocking the main thread on wake, we lazily restore the roots, with cac
3737
The serialized text is first parsed to get an array of encoded root data.
3838

3939
Then, a proxy gets the raw data and returns an array that deserializes properties on demand and caches them. Objects are also lazily restored.
40+
41+
## Out-of-order streaming
42+
43+
when we allow out-of-order streaming but we want interactivity while the page is still streaming, we can't scan the state once for cycles/Promises/references, because later state might refer to earlier state.
44+
45+
Therefore we must make the serializer single-pass. To do this, we could store each object as a root, but that will make references take more bytes. Instead, we could implement sub-paths for references, and when we encounter a reference, we output that instead.
46+
47+
For later references it would be best to have a root reference, so we'd probably add a root for the sub-reference and later we can refer to that.
48+
49+
We'll still need to keep track of Promises. We can do this by waiting, but that halts streaming. Instead, we could write a forward reference id, and when the promise is resolved, we store the result as the next root item. At the end of the stream we can emit a mapping from forward references to root index.
50+
51+
Then later, when we send out-of-order state, it will append to the existing state on the client.
52+
53+
This single-pass approach might just be better in general, because several bugs were found due to differences between the first and second pass code, and it allows streaming the data while the promises are being resolved.

0 commit comments

Comments
 (0)