@bquery/bquery 1.7.0 → 1.8.1
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 +760 -716
- package/dist/{a11y-C5QOVvRn.js → a11y-DVBCy09c.js} +3 -3
- package/dist/a11y-DVBCy09c.js.map +1 -0
- package/dist/a11y.es.mjs +1 -1
- package/dist/component/library.d.ts.map +1 -1
- package/dist/{component-CuuTijA6.js → component-L3-JfOFz.js} +5 -5
- package/dist/component-L3-JfOFz.js.map +1 -0
- package/dist/component.es.mjs +1 -1
- package/dist/{config-BW35FKuA.js → config-DhT9auRm.js} +1 -1
- package/dist/{config-BW35FKuA.js.map → config-DhT9auRm.js.map} +1 -1
- package/dist/{constraints-3lV9yyBw.js → constraints-D5RHQLmP.js} +1 -1
- package/dist/constraints-D5RHQLmP.js.map +1 -0
- package/dist/core/collection.d.ts +86 -0
- package/dist/core/collection.d.ts.map +1 -1
- package/dist/core/element.d.ts +28 -0
- package/dist/core/element.d.ts.map +1 -1
- package/dist/core/shared.d.ts +6 -0
- package/dist/core/shared.d.ts.map +1 -1
- package/dist/core-DdtZHzsS.js +168 -0
- package/dist/core-DdtZHzsS.js.map +1 -0
- package/dist/{core-Cjl7GUu8.js → core-EMYSLzaT.js} +289 -259
- package/dist/core-EMYSLzaT.js.map +1 -0
- package/dist/core.es.mjs +48 -47
- package/dist/{custom-directives-7wAShnnd.js → custom-directives-Dr4C5lVV.js} +1 -1
- package/dist/custom-directives-Dr4C5lVV.js.map +1 -0
- package/dist/{devtools-D2fQLhDN.js → devtools-BhB2iDPT.js} +2 -2
- package/dist/devtools-BhB2iDPT.js.map +1 -0
- package/dist/devtools.es.mjs +1 -1
- package/dist/{dnd-B8EgyzaI.js → dnd-NwZBYh4l.js} +1 -1
- package/dist/dnd-NwZBYh4l.js.map +1 -0
- package/dist/dnd.es.mjs +1 -1
- package/dist/{env-NeVmr4Gf.js → env-CTdvLaH2.js} +1 -1
- package/dist/env-CTdvLaH2.js.map +1 -0
- package/dist/forms/create-form.d.ts.map +1 -1
- package/dist/forms/index.d.ts +3 -2
- package/dist/forms/index.d.ts.map +1 -1
- package/dist/forms/types.d.ts +46 -0
- package/dist/forms/types.d.ts.map +1 -1
- package/dist/forms/use-field.d.ts +34 -0
- package/dist/forms/use-field.d.ts.map +1 -0
- package/dist/forms/validators.d.ts +25 -0
- package/dist/forms/validators.d.ts.map +1 -1
- package/dist/forms-UcRHsYxC.js +227 -0
- package/dist/forms-UcRHsYxC.js.map +1 -0
- package/dist/forms.es.mjs +14 -12
- package/dist/full.d.ts +17 -26
- package/dist/full.d.ts.map +1 -1
- package/dist/full.es.mjs +206 -181
- package/dist/full.iife.js +33 -33
- package/dist/full.iife.js.map +1 -1
- package/dist/full.umd.js +33 -33
- package/dist/full.umd.js.map +1 -1
- package/dist/function-Cybd57JV.js +33 -0
- package/dist/function-Cybd57JV.js.map +1 -0
- package/dist/{i18n-BnnhTFOS.js → i18n-kuF6Ekj6.js} +3 -3
- package/dist/i18n-kuF6Ekj6.js.map +1 -0
- package/dist/i18n.es.mjs +1 -1
- package/dist/index.es.mjs +251 -228
- package/dist/media/breakpoints.d.ts.map +1 -1
- package/dist/media/types.d.ts +2 -2
- package/dist/media/types.d.ts.map +1 -1
- package/dist/{media-Di2Ta22s.js → media-i-fB5WxI.js} +3 -3
- package/dist/media-i-fB5WxI.js.map +1 -0
- package/dist/media.es.mjs +1 -1
- package/dist/{motion-qPj_TYGv.js → motion-BJsAuULb.js} +2 -2
- package/dist/motion-BJsAuULb.js.map +1 -0
- package/dist/motion.es.mjs +1 -1
- package/dist/{mount-SM07RUa6.js → mount-B4Y8bk8Z.js} +5 -5
- package/dist/mount-B4Y8bk8Z.js.map +1 -0
- package/dist/{platform-CPbCprb6.js → platform-Dw2gE3zI.js} +3 -3
- package/dist/{platform-CPbCprb6.js.map → platform-Dw2gE3zI.js.map} +1 -1
- package/dist/platform.es.mjs +2 -2
- package/dist/plugin/registry.d.ts.map +1 -1
- package/dist/{plugin-cPoOHFLY.js → plugin-C2WuC8SF.js} +20 -18
- package/dist/plugin-C2WuC8SF.js.map +1 -0
- package/dist/plugin.es.mjs +1 -1
- package/dist/reactive/async-data.d.ts +28 -3
- package/dist/reactive/async-data.d.ts.map +1 -1
- package/dist/reactive/computed.d.ts +3 -0
- package/dist/reactive/computed.d.ts.map +1 -1
- package/dist/reactive/effect.d.ts +3 -0
- package/dist/reactive/effect.d.ts.map +1 -1
- package/dist/reactive/http.d.ts +194 -0
- package/dist/reactive/http.d.ts.map +1 -0
- package/dist/reactive/index.d.ts +2 -2
- package/dist/reactive/index.d.ts.map +1 -1
- package/dist/reactive/pagination.d.ts +126 -0
- package/dist/reactive/pagination.d.ts.map +1 -0
- package/dist/reactive/polling.d.ts +55 -0
- package/dist/reactive/polling.d.ts.map +1 -0
- package/dist/reactive/readonly.d.ts +20 -1
- package/dist/reactive/readonly.d.ts.map +1 -1
- package/dist/reactive/rest.d.ts +293 -0
- package/dist/reactive/rest.d.ts.map +1 -0
- package/dist/reactive/scope.d.ts +140 -0
- package/dist/reactive/scope.d.ts.map +1 -0
- package/dist/reactive/signal.d.ts +16 -2
- package/dist/reactive/signal.d.ts.map +1 -1
- package/dist/reactive/to-value.d.ts +57 -0
- package/dist/reactive/to-value.d.ts.map +1 -0
- package/dist/reactive/websocket.d.ts +285 -0
- package/dist/reactive/websocket.d.ts.map +1 -0
- package/dist/reactive-DwkhUJfP.js +1148 -0
- package/dist/reactive-DwkhUJfP.js.map +1 -0
- package/dist/reactive.es.mjs +38 -19
- package/dist/{registry-CWf368tT.js → registry-B08iilIh.js} +1 -1
- package/dist/{registry-CWf368tT.js.map → registry-B08iilIh.js.map} +1 -1
- package/dist/router/constraints.d.ts.map +1 -1
- package/dist/router/index.d.ts +1 -1
- package/dist/router/index.d.ts.map +1 -1
- package/dist/router/router.d.ts.map +1 -1
- package/dist/router/state.d.ts +25 -2
- package/dist/router/state.d.ts.map +1 -1
- package/dist/router-CQikC9Ed.js +492 -0
- package/dist/router-CQikC9Ed.js.map +1 -0
- package/dist/router.es.mjs +9 -8
- package/dist/ssr/hydrate.d.ts.map +1 -1
- package/dist/{ssr-B2qd_WBB.js → ssr-_dAcGdzu.js} +4 -4
- package/dist/ssr-_dAcGdzu.js.map +1 -0
- package/dist/ssr.es.mjs +1 -1
- package/dist/store/persisted.d.ts.map +1 -1
- package/dist/{store-DWpyH6p5.js → store-Cb3gPRve.js} +7 -7
- package/dist/store-Cb3gPRve.js.map +1 -0
- package/dist/store.es.mjs +2 -2
- package/dist/storybook.es.mjs.map +1 -1
- package/dist/{testing-CsqjNUyy.js → testing-C5Sjfsna.js} +8 -8
- package/dist/testing-C5Sjfsna.js.map +1 -0
- package/dist/testing.es.mjs +1 -1
- package/dist/{type-guards-Do9DWgNp.js → type-guards-BMX2c0LP.js} +1 -1
- package/dist/{type-guards-Do9DWgNp.js.map → type-guards-BMX2c0LP.js.map} +1 -1
- package/dist/untrack-D0fnO5k2.js +36 -0
- package/dist/untrack-D0fnO5k2.js.map +1 -0
- package/dist/view/custom-directives.d.ts.map +1 -1
- package/dist/view.es.mjs +4 -4
- package/package.json +177 -177
- package/src/a11y/announce.ts +131 -131
- package/src/a11y/audit.ts +314 -314
- package/src/a11y/index.ts +68 -68
- package/src/a11y/media-preferences.ts +255 -255
- package/src/a11y/roving-tab-index.ts +164 -164
- package/src/a11y/skip-link.ts +255 -255
- package/src/a11y/trap-focus.ts +184 -184
- package/src/a11y/types.ts +183 -183
- package/src/component/component.ts +599 -599
- package/src/component/html.ts +153 -153
- package/src/component/index.ts +52 -52
- package/src/component/library.ts +540 -542
- package/src/component/scope.ts +212 -212
- package/src/component/types.ts +310 -310
- package/src/core/collection.ts +876 -707
- package/src/core/element.ts +1015 -981
- package/src/core/env.ts +60 -60
- package/src/core/index.ts +49 -49
- package/src/core/shared.ts +77 -62
- package/src/core/utils/index.ts +148 -148
- package/src/devtools/devtools.ts +410 -410
- package/src/devtools/index.ts +48 -48
- package/src/devtools/types.ts +104 -104
- package/src/dnd/draggable.ts +296 -296
- package/src/dnd/droppable.ts +228 -228
- package/src/dnd/index.ts +62 -62
- package/src/dnd/sortable.ts +307 -307
- package/src/dnd/types.ts +293 -293
- package/src/forms/create-form.ts +320 -278
- package/src/forms/index.ts +70 -65
- package/src/forms/types.ts +203 -154
- package/src/forms/use-field.ts +231 -0
- package/src/forms/validators.ts +294 -265
- package/src/full.ts +554 -480
- package/src/i18n/formatting.ts +67 -67
- package/src/i18n/i18n.ts +200 -200
- package/src/i18n/index.ts +67 -67
- package/src/i18n/translate.ts +182 -182
- package/src/i18n/types.ts +171 -171
- package/src/index.ts +108 -108
- package/src/media/battery.ts +116 -116
- package/src/media/breakpoints.ts +129 -131
- package/src/media/clipboard.ts +80 -80
- package/src/media/device-sensors.ts +158 -158
- package/src/media/geolocation.ts +119 -119
- package/src/media/index.ts +76 -76
- package/src/media/media-query.ts +92 -92
- package/src/media/network.ts +115 -115
- package/src/media/types.ts +177 -177
- package/src/media/viewport.ts +84 -84
- package/src/motion/index.ts +57 -57
- package/src/motion/morph.ts +151 -151
- package/src/motion/parallax.ts +120 -120
- package/src/motion/reduced-motion.ts +66 -66
- package/src/motion/types.ts +271 -271
- package/src/motion/typewriter.ts +164 -164
- package/src/plugin/index.ts +37 -37
- package/src/plugin/registry.ts +284 -269
- package/src/plugin/types.ts +137 -137
- package/src/reactive/async-data.ts +250 -29
- package/src/reactive/computed.ts +144 -130
- package/src/reactive/effect.ts +29 -6
- package/src/reactive/http.ts +790 -0
- package/src/reactive/index.ts +60 -0
- package/src/reactive/pagination.ts +317 -0
- package/src/reactive/polling.ts +179 -0
- package/src/reactive/readonly.ts +52 -8
- package/src/reactive/rest.ts +859 -0
- package/src/reactive/scope.ts +276 -0
- package/src/reactive/signal.ts +61 -1
- package/src/reactive/to-value.ts +71 -0
- package/src/reactive/websocket.ts +849 -0
- package/src/router/bq-link.ts +279 -279
- package/src/router/constraints.ts +204 -201
- package/src/router/index.ts +49 -49
- package/src/router/match.ts +312 -312
- package/src/router/path-pattern.ts +52 -52
- package/src/router/query.ts +38 -38
- package/src/router/router.ts +421 -402
- package/src/router/state.ts +51 -3
- package/src/router/types.ts +139 -139
- package/src/router/use-route.ts +68 -68
- package/src/router/utils.ts +157 -157
- package/src/security/index.ts +12 -12
- package/src/ssr/hydrate.ts +84 -82
- package/src/ssr/index.ts +70 -70
- package/src/ssr/render.ts +508 -508
- package/src/ssr/serialize.ts +296 -296
- package/src/ssr/types.ts +81 -81
- package/src/store/create-store.ts +467 -467
- package/src/store/index.ts +27 -27
- package/src/store/persisted.ts +245 -249
- package/src/store/types.ts +247 -247
- package/src/store/utils.ts +135 -135
- package/src/storybook/index.ts +480 -480
- package/src/testing/index.ts +42 -42
- package/src/testing/testing.ts +593 -593
- package/src/testing/types.ts +170 -170
- package/src/view/custom-directives.ts +28 -30
- package/src/view/evaluate.ts +292 -292
- package/src/view/process.ts +108 -108
- package/dist/a11y-C5QOVvRn.js.map +0 -1
- package/dist/component-CuuTijA6.js.map +0 -1
- package/dist/constraints-3lV9yyBw.js.map +0 -1
- package/dist/core-Cjl7GUu8.js.map +0 -1
- package/dist/core-DnlyjbF2.js +0 -112
- package/dist/core-DnlyjbF2.js.map +0 -1
- package/dist/custom-directives-7wAShnnd.js.map +0 -1
- package/dist/devtools-D2fQLhDN.js.map +0 -1
- package/dist/dnd-B8EgyzaI.js.map +0 -1
- package/dist/env-NeVmr4Gf.js.map +0 -1
- package/dist/forms-C3yovgH9.js +0 -141
- package/dist/forms-C3yovgH9.js.map +0 -1
- package/dist/i18n-BnnhTFOS.js.map +0 -1
- package/dist/media-Di2Ta22s.js.map +0 -1
- package/dist/motion-qPj_TYGv.js.map +0 -1
- package/dist/mount-SM07RUa6.js.map +0 -1
- package/dist/plugin-cPoOHFLY.js.map +0 -1
- package/dist/reactive-Cfv0RK6x.js +0 -233
- package/dist/reactive-Cfv0RK6x.js.map +0 -1
- package/dist/router-BrthaP_z.js +0 -473
- package/dist/router-BrthaP_z.js.map +0 -1
- package/dist/ssr-B2qd_WBB.js.map +0 -1
- package/dist/store-DWpyH6p5.js.map +0 -1
- package/dist/testing-CsqjNUyy.js.map +0 -1
- package/dist/untrack-DJVQQ2WM.js +0 -33
- package/dist/untrack-DJVQQ2WM.js.map +0 -1
package/src/ssr/serialize.ts
CHANGED
|
@@ -1,296 +1,296 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Store state serialization for SSR.
|
|
3
|
-
*
|
|
4
|
-
* Provides utilities to serialize store state into a `<script>` tag
|
|
5
|
-
* for client-side hydration, and to deserialize state on the client.
|
|
6
|
-
*
|
|
7
|
-
* @module bquery/ssr
|
|
8
|
-
*/
|
|
9
|
-
|
|
10
|
-
import { getStore, listStores } from '../store/index';
|
|
11
|
-
import { isPrototypePollutionKey } from '../core/utils/object';
|
|
12
|
-
import type { DeserializedStoreState, SerializeOptions } from './types';
|
|
13
|
-
|
|
14
|
-
const isStoreStateObject = (value: unknown): value is Record<string, unknown> =>
|
|
15
|
-
typeof value === 'object' && value !== null && !Array.isArray(value);
|
|
16
|
-
|
|
17
|
-
const sanitizeHydrationState = (value: Record<string, unknown>): Record<string, unknown> => {
|
|
18
|
-
const sanitized: Record<string, unknown> = {};
|
|
19
|
-
for (const [key, entryValue] of Object.entries(value)) {
|
|
20
|
-
if (isPrototypePollutionKey(key)) continue;
|
|
21
|
-
sanitized[key] = entryValue;
|
|
22
|
-
}
|
|
23
|
-
return sanitized;
|
|
24
|
-
};
|
|
25
|
-
|
|
26
|
-
/**
|
|
27
|
-
* Result of store state serialization.
|
|
28
|
-
*/
|
|
29
|
-
export type SerializeResult = {
|
|
30
|
-
/** JSON string of the state map */
|
|
31
|
-
stateJson: string;
|
|
32
|
-
/** Complete `<script>` tag ready to embed in HTML */
|
|
33
|
-
scriptTag: string;
|
|
34
|
-
};
|
|
35
|
-
|
|
36
|
-
/**
|
|
37
|
-
* Escapes a string for safe embedding in a `<script>` tag.
|
|
38
|
-
* Prevents XSS via `</script>` injection and HTML entities.
|
|
39
|
-
*
|
|
40
|
-
* @internal
|
|
41
|
-
*/
|
|
42
|
-
const escapeForScript = (str: string): string => {
|
|
43
|
-
return str
|
|
44
|
-
.replace(/</g, '\\u003c')
|
|
45
|
-
.replace(/>/g, '\\u003e')
|
|
46
|
-
.replace(/\//g, '\\u002f')
|
|
47
|
-
.replace(/\u2028/g, '\\u2028')
|
|
48
|
-
.replace(/\u2029/g, '\\u2029');
|
|
49
|
-
};
|
|
50
|
-
|
|
51
|
-
/**
|
|
52
|
-
* Escapes a string for safe embedding in an HTML attribute value.
|
|
53
|
-
* @internal
|
|
54
|
-
*/
|
|
55
|
-
const escapeForHtmlAttribute = (str: string): string => {
|
|
56
|
-
return str
|
|
57
|
-
.replace(/&/g, '&')
|
|
58
|
-
.replace(/"/g, '"')
|
|
59
|
-
.replace(/</g, '<')
|
|
60
|
-
.replace(/>/g, '>');
|
|
61
|
-
};
|
|
62
|
-
|
|
63
|
-
/**
|
|
64
|
-
* Serializes the state of registered stores into a JSON string and
|
|
65
|
-
* a `<script>` tag suitable for embedding in server-rendered HTML.
|
|
66
|
-
*
|
|
67
|
-
* The serialized state can be picked up on the client using
|
|
68
|
-
* `deserializeStoreState()` to restore stores to their server-side values.
|
|
69
|
-
*
|
|
70
|
-
* @param options - Serialization options
|
|
71
|
-
* @returns Object with JSON string and ready-to-use script tag
|
|
72
|
-
*
|
|
73
|
-
* @example
|
|
74
|
-
* ```ts
|
|
75
|
-
* import { serializeStoreState } from '@bquery/bquery/ssr';
|
|
76
|
-
* import { createStore } from '@bquery/bquery/store';
|
|
77
|
-
*
|
|
78
|
-
* const store = createStore({
|
|
79
|
-
* id: 'counter',
|
|
80
|
-
* state: () => ({ count: 42 }),
|
|
81
|
-
* });
|
|
82
|
-
*
|
|
83
|
-
* const { scriptTag } = serializeStoreState();
|
|
84
|
-
* // '<script id="__BQUERY_STORE_STATE__">window.__BQUERY_INITIAL_STATE__={"counter":{"count":42}}</script>'
|
|
85
|
-
* ```
|
|
86
|
-
*
|
|
87
|
-
* @example
|
|
88
|
-
* ```ts
|
|
89
|
-
* // Serialize only specific stores
|
|
90
|
-
* const { scriptTag } = serializeStoreState({ storeIds: ['counter'] });
|
|
91
|
-
* ```
|
|
92
|
-
*/
|
|
93
|
-
export const serializeStoreState = (options: SerializeOptions = {}): SerializeResult => {
|
|
94
|
-
const {
|
|
95
|
-
scriptId = '__BQUERY_STORE_STATE__',
|
|
96
|
-
globalKey = '__BQUERY_INITIAL_STATE__',
|
|
97
|
-
storeIds,
|
|
98
|
-
serialize = JSON.stringify,
|
|
99
|
-
} = options;
|
|
100
|
-
|
|
101
|
-
if (isPrototypePollutionKey(globalKey)) {
|
|
102
|
-
throw new Error(
|
|
103
|
-
`serializeStoreState: invalid globalKey "${globalKey}" - prototype-pollution keys are not allowed.`
|
|
104
|
-
);
|
|
105
|
-
}
|
|
106
|
-
|
|
107
|
-
if (isPrototypePollutionKey(scriptId)) {
|
|
108
|
-
throw new Error(
|
|
109
|
-
`serializeStoreState: invalid scriptId "${scriptId}" - prototype-pollution keys are not allowed.`
|
|
110
|
-
);
|
|
111
|
-
}
|
|
112
|
-
|
|
113
|
-
const ids = storeIds ?? listStores();
|
|
114
|
-
const stateMap = Object.create(null) as Record<string, Record<string, unknown>>;
|
|
115
|
-
|
|
116
|
-
for (const id of ids) {
|
|
117
|
-
if (isPrototypePollutionKey(id)) {
|
|
118
|
-
continue;
|
|
119
|
-
}
|
|
120
|
-
|
|
121
|
-
const store = getStore<{ $state: Record<string, unknown> }>(id);
|
|
122
|
-
if (store) {
|
|
123
|
-
stateMap[id] = sanitizeHydrationState(store.$state);
|
|
124
|
-
}
|
|
125
|
-
}
|
|
126
|
-
|
|
127
|
-
const stateJson = serialize(stateMap);
|
|
128
|
-
if (typeof stateJson !== 'string') {
|
|
129
|
-
throw new Error('serializeStoreState: custom serialize function must return a string.');
|
|
130
|
-
}
|
|
131
|
-
|
|
132
|
-
if (serialize !== JSON.stringify) {
|
|
133
|
-
let parsedStateJson: unknown;
|
|
134
|
-
try {
|
|
135
|
-
parsedStateJson = JSON.parse(stateJson);
|
|
136
|
-
} catch {
|
|
137
|
-
throw new Error('serializeStoreState: custom serialize function returned invalid JSON.');
|
|
138
|
-
}
|
|
139
|
-
|
|
140
|
-
if (!isStoreStateObject(parsedStateJson)) {
|
|
141
|
-
throw new Error(
|
|
142
|
-
'serializeStoreState: custom serialize function must return a JSON object string.'
|
|
143
|
-
);
|
|
144
|
-
}
|
|
145
|
-
}
|
|
146
|
-
|
|
147
|
-
const escapedJson = escapeForScript(stateJson);
|
|
148
|
-
const escapedGlobalKey = escapeForScript(JSON.stringify(globalKey));
|
|
149
|
-
const escapedScriptId = escapeForHtmlAttribute(scriptId);
|
|
150
|
-
const scriptTag = `<script id="${escapedScriptId}">window[${escapedGlobalKey}]=${escapedJson}</script>`;
|
|
151
|
-
|
|
152
|
-
return { stateJson, scriptTag };
|
|
153
|
-
};
|
|
154
|
-
|
|
155
|
-
/**
|
|
156
|
-
* Deserializes store state from the global variable set by the SSR script tag.
|
|
157
|
-
*
|
|
158
|
-
* Call this on the client before creating stores to pre-populate them with
|
|
159
|
-
* server-rendered state. After deserialization, the script tag and global
|
|
160
|
-
* variable are cleaned up automatically.
|
|
161
|
-
*
|
|
162
|
-
* @param globalKey - The global variable name where state was serialized
|
|
163
|
-
* @param scriptId - The ID of the SSR script tag to remove after hydration
|
|
164
|
-
* @returns The deserialized state map, or an empty object if not found
|
|
165
|
-
*
|
|
166
|
-
* @example
|
|
167
|
-
* ```ts
|
|
168
|
-
* import { deserializeStoreState } from '@bquery/bquery/ssr';
|
|
169
|
-
*
|
|
170
|
-
* // Call before creating stores
|
|
171
|
-
* const state = deserializeStoreState();
|
|
172
|
-
* // state = { counter: { count: 42 } }
|
|
173
|
-
* ```
|
|
174
|
-
*/
|
|
175
|
-
export const deserializeStoreState = (
|
|
176
|
-
globalKey = '__BQUERY_INITIAL_STATE__',
|
|
177
|
-
scriptId = '__BQUERY_STORE_STATE__'
|
|
178
|
-
): DeserializedStoreState => {
|
|
179
|
-
if (isPrototypePollutionKey(globalKey)) {
|
|
180
|
-
throw new Error(
|
|
181
|
-
`deserializeStoreState: invalid globalKey "${globalKey}" - prototype-pollution keys are not allowed.`
|
|
182
|
-
);
|
|
183
|
-
}
|
|
184
|
-
|
|
185
|
-
if (isPrototypePollutionKey(scriptId)) {
|
|
186
|
-
throw new Error(
|
|
187
|
-
`deserializeStoreState: invalid scriptId "${scriptId}" - prototype-pollution keys are not allowed.`
|
|
188
|
-
);
|
|
189
|
-
}
|
|
190
|
-
|
|
191
|
-
if (typeof window === 'undefined') {
|
|
192
|
-
return {};
|
|
193
|
-
}
|
|
194
|
-
|
|
195
|
-
const state = (window as unknown as Record<string, unknown>)[globalKey];
|
|
196
|
-
if (!state) {
|
|
197
|
-
return {};
|
|
198
|
-
}
|
|
199
|
-
|
|
200
|
-
// Clean up global variable
|
|
201
|
-
try {
|
|
202
|
-
delete (window as unknown as Record<string, unknown>)[globalKey];
|
|
203
|
-
} catch {
|
|
204
|
-
// In strict mode on some environments, delete may fail
|
|
205
|
-
(window as unknown as Record<string, unknown>)[globalKey] = undefined;
|
|
206
|
-
}
|
|
207
|
-
|
|
208
|
-
// Clean up script tag
|
|
209
|
-
if (typeof document !== 'undefined' && typeof document.getElementById === 'function') {
|
|
210
|
-
const scriptEl = document.getElementById(scriptId);
|
|
211
|
-
if (scriptEl) {
|
|
212
|
-
scriptEl.remove();
|
|
213
|
-
}
|
|
214
|
-
}
|
|
215
|
-
|
|
216
|
-
if (!isStoreStateObject(state)) {
|
|
217
|
-
return {};
|
|
218
|
-
}
|
|
219
|
-
|
|
220
|
-
for (const value of Object.values(state)) {
|
|
221
|
-
if (!isStoreStateObject(value)) {
|
|
222
|
-
return {};
|
|
223
|
-
}
|
|
224
|
-
}
|
|
225
|
-
|
|
226
|
-
const sanitizedStateMap = Object.create(null) as DeserializedStoreState;
|
|
227
|
-
|
|
228
|
-
for (const [storeId, storeState] of Object.entries(state)) {
|
|
229
|
-
if (isPrototypePollutionKey(storeId) || !isStoreStateObject(storeState)) {
|
|
230
|
-
continue;
|
|
231
|
-
}
|
|
232
|
-
|
|
233
|
-
sanitizedStateMap[storeId] = sanitizeHydrationState(storeState);
|
|
234
|
-
}
|
|
235
|
-
|
|
236
|
-
return sanitizedStateMap;
|
|
237
|
-
};
|
|
238
|
-
|
|
239
|
-
/**
|
|
240
|
-
* Hydrates a store with pre-serialized state from SSR.
|
|
241
|
-
*
|
|
242
|
-
* If the store exists and has a `$patch` method, this applies the
|
|
243
|
-
* deserialized state as a patch. Otherwise, the state is ignored.
|
|
244
|
-
*
|
|
245
|
-
* @param storeId - The store ID to hydrate
|
|
246
|
-
* @param state - The plain state object to apply
|
|
247
|
-
*
|
|
248
|
-
* @example
|
|
249
|
-
* ```ts
|
|
250
|
-
* import { hydrateStore, deserializeStoreState } from '@bquery/bquery/ssr';
|
|
251
|
-
* import { createStore } from '@bquery/bquery/store';
|
|
252
|
-
*
|
|
253
|
-
* // 1. Deserialize state from SSR script tag
|
|
254
|
-
* const ssrState = deserializeStoreState();
|
|
255
|
-
*
|
|
256
|
-
* // 2. Create store (gets initial values from factory)
|
|
257
|
-
* const store = createStore({
|
|
258
|
-
* id: 'counter',
|
|
259
|
-
* state: () => ({ count: 0 }),
|
|
260
|
-
* });
|
|
261
|
-
*
|
|
262
|
-
* // 3. Apply SSR state
|
|
263
|
-
* if (ssrState.counter) {
|
|
264
|
-
* hydrateStore('counter', ssrState.counter);
|
|
265
|
-
* }
|
|
266
|
-
* // store.count is now 42 (from SSR)
|
|
267
|
-
* ```
|
|
268
|
-
*/
|
|
269
|
-
export const hydrateStore = (storeId: string, state: Record<string, unknown>): void => {
|
|
270
|
-
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
271
|
-
const store = getStore<{ $patch?: (partial: any) => void }>(storeId);
|
|
272
|
-
if (store && typeof store.$patch === 'function') {
|
|
273
|
-
store.$patch(sanitizeHydrationState(state));
|
|
274
|
-
}
|
|
275
|
-
};
|
|
276
|
-
|
|
277
|
-
/**
|
|
278
|
-
* Hydrates all stores at once from a deserialized state map.
|
|
279
|
-
*
|
|
280
|
-
* Convenience wrapper that calls `hydrateStore` for each entry in the state map.
|
|
281
|
-
*
|
|
282
|
-
* @param stateMap - Map of store IDs to their state objects
|
|
283
|
-
*
|
|
284
|
-
* @example
|
|
285
|
-
* ```ts
|
|
286
|
-
* import { hydrateStores, deserializeStoreState } from '@bquery/bquery/ssr';
|
|
287
|
-
*
|
|
288
|
-
* const ssrState = deserializeStoreState();
|
|
289
|
-
* hydrateStores(ssrState);
|
|
290
|
-
* ```
|
|
291
|
-
*/
|
|
292
|
-
export const hydrateStores = (stateMap: DeserializedStoreState): void => {
|
|
293
|
-
for (const [storeId, state] of Object.entries(stateMap)) {
|
|
294
|
-
hydrateStore(storeId, state);
|
|
295
|
-
}
|
|
296
|
-
};
|
|
1
|
+
/**
|
|
2
|
+
* Store state serialization for SSR.
|
|
3
|
+
*
|
|
4
|
+
* Provides utilities to serialize store state into a `<script>` tag
|
|
5
|
+
* for client-side hydration, and to deserialize state on the client.
|
|
6
|
+
*
|
|
7
|
+
* @module bquery/ssr
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
import { getStore, listStores } from '../store/index';
|
|
11
|
+
import { isPrototypePollutionKey } from '../core/utils/object';
|
|
12
|
+
import type { DeserializedStoreState, SerializeOptions } from './types';
|
|
13
|
+
|
|
14
|
+
const isStoreStateObject = (value: unknown): value is Record<string, unknown> =>
|
|
15
|
+
typeof value === 'object' && value !== null && !Array.isArray(value);
|
|
16
|
+
|
|
17
|
+
const sanitizeHydrationState = (value: Record<string, unknown>): Record<string, unknown> => {
|
|
18
|
+
const sanitized: Record<string, unknown> = {};
|
|
19
|
+
for (const [key, entryValue] of Object.entries(value)) {
|
|
20
|
+
if (isPrototypePollutionKey(key)) continue;
|
|
21
|
+
sanitized[key] = entryValue;
|
|
22
|
+
}
|
|
23
|
+
return sanitized;
|
|
24
|
+
};
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* Result of store state serialization.
|
|
28
|
+
*/
|
|
29
|
+
export type SerializeResult = {
|
|
30
|
+
/** JSON string of the state map */
|
|
31
|
+
stateJson: string;
|
|
32
|
+
/** Complete `<script>` tag ready to embed in HTML */
|
|
33
|
+
scriptTag: string;
|
|
34
|
+
};
|
|
35
|
+
|
|
36
|
+
/**
|
|
37
|
+
* Escapes a string for safe embedding in a `<script>` tag.
|
|
38
|
+
* Prevents XSS via `</script>` injection and HTML entities.
|
|
39
|
+
*
|
|
40
|
+
* @internal
|
|
41
|
+
*/
|
|
42
|
+
const escapeForScript = (str: string): string => {
|
|
43
|
+
return str
|
|
44
|
+
.replace(/</g, '\\u003c')
|
|
45
|
+
.replace(/>/g, '\\u003e')
|
|
46
|
+
.replace(/\//g, '\\u002f')
|
|
47
|
+
.replace(/\u2028/g, '\\u2028')
|
|
48
|
+
.replace(/\u2029/g, '\\u2029');
|
|
49
|
+
};
|
|
50
|
+
|
|
51
|
+
/**
|
|
52
|
+
* Escapes a string for safe embedding in an HTML attribute value.
|
|
53
|
+
* @internal
|
|
54
|
+
*/
|
|
55
|
+
const escapeForHtmlAttribute = (str: string): string => {
|
|
56
|
+
return str
|
|
57
|
+
.replace(/&/g, '&')
|
|
58
|
+
.replace(/"/g, '"')
|
|
59
|
+
.replace(/</g, '<')
|
|
60
|
+
.replace(/>/g, '>');
|
|
61
|
+
};
|
|
62
|
+
|
|
63
|
+
/**
|
|
64
|
+
* Serializes the state of registered stores into a JSON string and
|
|
65
|
+
* a `<script>` tag suitable for embedding in server-rendered HTML.
|
|
66
|
+
*
|
|
67
|
+
* The serialized state can be picked up on the client using
|
|
68
|
+
* `deserializeStoreState()` to restore stores to their server-side values.
|
|
69
|
+
*
|
|
70
|
+
* @param options - Serialization options
|
|
71
|
+
* @returns Object with JSON string and ready-to-use script tag
|
|
72
|
+
*
|
|
73
|
+
* @example
|
|
74
|
+
* ```ts
|
|
75
|
+
* import { serializeStoreState } from '@bquery/bquery/ssr';
|
|
76
|
+
* import { createStore } from '@bquery/bquery/store';
|
|
77
|
+
*
|
|
78
|
+
* const store = createStore({
|
|
79
|
+
* id: 'counter',
|
|
80
|
+
* state: () => ({ count: 42 }),
|
|
81
|
+
* });
|
|
82
|
+
*
|
|
83
|
+
* const { scriptTag } = serializeStoreState();
|
|
84
|
+
* // '<script id="__BQUERY_STORE_STATE__">window.__BQUERY_INITIAL_STATE__={"counter":{"count":42}}</script>'
|
|
85
|
+
* ```
|
|
86
|
+
*
|
|
87
|
+
* @example
|
|
88
|
+
* ```ts
|
|
89
|
+
* // Serialize only specific stores
|
|
90
|
+
* const { scriptTag } = serializeStoreState({ storeIds: ['counter'] });
|
|
91
|
+
* ```
|
|
92
|
+
*/
|
|
93
|
+
export const serializeStoreState = (options: SerializeOptions = {}): SerializeResult => {
|
|
94
|
+
const {
|
|
95
|
+
scriptId = '__BQUERY_STORE_STATE__',
|
|
96
|
+
globalKey = '__BQUERY_INITIAL_STATE__',
|
|
97
|
+
storeIds,
|
|
98
|
+
serialize = JSON.stringify,
|
|
99
|
+
} = options;
|
|
100
|
+
|
|
101
|
+
if (isPrototypePollutionKey(globalKey)) {
|
|
102
|
+
throw new Error(
|
|
103
|
+
`serializeStoreState: invalid globalKey "${globalKey}" - prototype-pollution keys are not allowed.`
|
|
104
|
+
);
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
if (isPrototypePollutionKey(scriptId)) {
|
|
108
|
+
throw new Error(
|
|
109
|
+
`serializeStoreState: invalid scriptId "${scriptId}" - prototype-pollution keys are not allowed.`
|
|
110
|
+
);
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
const ids = storeIds ?? listStores();
|
|
114
|
+
const stateMap = Object.create(null) as Record<string, Record<string, unknown>>;
|
|
115
|
+
|
|
116
|
+
for (const id of ids) {
|
|
117
|
+
if (isPrototypePollutionKey(id)) {
|
|
118
|
+
continue;
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
const store = getStore<{ $state: Record<string, unknown> }>(id);
|
|
122
|
+
if (store) {
|
|
123
|
+
stateMap[id] = sanitizeHydrationState(store.$state);
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
const stateJson = serialize(stateMap);
|
|
128
|
+
if (typeof stateJson !== 'string') {
|
|
129
|
+
throw new Error('serializeStoreState: custom serialize function must return a string.');
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
if (serialize !== JSON.stringify) {
|
|
133
|
+
let parsedStateJson: unknown;
|
|
134
|
+
try {
|
|
135
|
+
parsedStateJson = JSON.parse(stateJson);
|
|
136
|
+
} catch {
|
|
137
|
+
throw new Error('serializeStoreState: custom serialize function returned invalid JSON.');
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
if (!isStoreStateObject(parsedStateJson)) {
|
|
141
|
+
throw new Error(
|
|
142
|
+
'serializeStoreState: custom serialize function must return a JSON object string.'
|
|
143
|
+
);
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
const escapedJson = escapeForScript(stateJson);
|
|
148
|
+
const escapedGlobalKey = escapeForScript(JSON.stringify(globalKey));
|
|
149
|
+
const escapedScriptId = escapeForHtmlAttribute(scriptId);
|
|
150
|
+
const scriptTag = `<script id="${escapedScriptId}">window[${escapedGlobalKey}]=${escapedJson}</script>`;
|
|
151
|
+
|
|
152
|
+
return { stateJson, scriptTag };
|
|
153
|
+
};
|
|
154
|
+
|
|
155
|
+
/**
|
|
156
|
+
* Deserializes store state from the global variable set by the SSR script tag.
|
|
157
|
+
*
|
|
158
|
+
* Call this on the client before creating stores to pre-populate them with
|
|
159
|
+
* server-rendered state. After deserialization, the script tag and global
|
|
160
|
+
* variable are cleaned up automatically.
|
|
161
|
+
*
|
|
162
|
+
* @param globalKey - The global variable name where state was serialized
|
|
163
|
+
* @param scriptId - The ID of the SSR script tag to remove after hydration
|
|
164
|
+
* @returns The deserialized state map, or an empty object if not found
|
|
165
|
+
*
|
|
166
|
+
* @example
|
|
167
|
+
* ```ts
|
|
168
|
+
* import { deserializeStoreState } from '@bquery/bquery/ssr';
|
|
169
|
+
*
|
|
170
|
+
* // Call before creating stores
|
|
171
|
+
* const state = deserializeStoreState();
|
|
172
|
+
* // state = { counter: { count: 42 } }
|
|
173
|
+
* ```
|
|
174
|
+
*/
|
|
175
|
+
export const deserializeStoreState = (
|
|
176
|
+
globalKey = '__BQUERY_INITIAL_STATE__',
|
|
177
|
+
scriptId = '__BQUERY_STORE_STATE__'
|
|
178
|
+
): DeserializedStoreState => {
|
|
179
|
+
if (isPrototypePollutionKey(globalKey)) {
|
|
180
|
+
throw new Error(
|
|
181
|
+
`deserializeStoreState: invalid globalKey "${globalKey}" - prototype-pollution keys are not allowed.`
|
|
182
|
+
);
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
if (isPrototypePollutionKey(scriptId)) {
|
|
186
|
+
throw new Error(
|
|
187
|
+
`deserializeStoreState: invalid scriptId "${scriptId}" - prototype-pollution keys are not allowed.`
|
|
188
|
+
);
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
if (typeof window === 'undefined') {
|
|
192
|
+
return {};
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
const state = (window as unknown as Record<string, unknown>)[globalKey];
|
|
196
|
+
if (!state) {
|
|
197
|
+
return {};
|
|
198
|
+
}
|
|
199
|
+
|
|
200
|
+
// Clean up global variable
|
|
201
|
+
try {
|
|
202
|
+
delete (window as unknown as Record<string, unknown>)[globalKey];
|
|
203
|
+
} catch {
|
|
204
|
+
// In strict mode on some environments, delete may fail
|
|
205
|
+
(window as unknown as Record<string, unknown>)[globalKey] = undefined;
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
// Clean up script tag
|
|
209
|
+
if (typeof document !== 'undefined' && typeof document.getElementById === 'function') {
|
|
210
|
+
const scriptEl = document.getElementById(scriptId);
|
|
211
|
+
if (scriptEl) {
|
|
212
|
+
scriptEl.remove();
|
|
213
|
+
}
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
if (!isStoreStateObject(state)) {
|
|
217
|
+
return {};
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
for (const value of Object.values(state)) {
|
|
221
|
+
if (!isStoreStateObject(value)) {
|
|
222
|
+
return {};
|
|
223
|
+
}
|
|
224
|
+
}
|
|
225
|
+
|
|
226
|
+
const sanitizedStateMap = Object.create(null) as DeserializedStoreState;
|
|
227
|
+
|
|
228
|
+
for (const [storeId, storeState] of Object.entries(state)) {
|
|
229
|
+
if (isPrototypePollutionKey(storeId) || !isStoreStateObject(storeState)) {
|
|
230
|
+
continue;
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
sanitizedStateMap[storeId] = sanitizeHydrationState(storeState);
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
return sanitizedStateMap;
|
|
237
|
+
};
|
|
238
|
+
|
|
239
|
+
/**
|
|
240
|
+
* Hydrates a store with pre-serialized state from SSR.
|
|
241
|
+
*
|
|
242
|
+
* If the store exists and has a `$patch` method, this applies the
|
|
243
|
+
* deserialized state as a patch. Otherwise, the state is ignored.
|
|
244
|
+
*
|
|
245
|
+
* @param storeId - The store ID to hydrate
|
|
246
|
+
* @param state - The plain state object to apply
|
|
247
|
+
*
|
|
248
|
+
* @example
|
|
249
|
+
* ```ts
|
|
250
|
+
* import { hydrateStore, deserializeStoreState } from '@bquery/bquery/ssr';
|
|
251
|
+
* import { createStore } from '@bquery/bquery/store';
|
|
252
|
+
*
|
|
253
|
+
* // 1. Deserialize state from SSR script tag
|
|
254
|
+
* const ssrState = deserializeStoreState();
|
|
255
|
+
*
|
|
256
|
+
* // 2. Create store (gets initial values from factory)
|
|
257
|
+
* const store = createStore({
|
|
258
|
+
* id: 'counter',
|
|
259
|
+
* state: () => ({ count: 0 }),
|
|
260
|
+
* });
|
|
261
|
+
*
|
|
262
|
+
* // 3. Apply SSR state
|
|
263
|
+
* if (ssrState.counter) {
|
|
264
|
+
* hydrateStore('counter', ssrState.counter);
|
|
265
|
+
* }
|
|
266
|
+
* // store.count is now 42 (from SSR)
|
|
267
|
+
* ```
|
|
268
|
+
*/
|
|
269
|
+
export const hydrateStore = (storeId: string, state: Record<string, unknown>): void => {
|
|
270
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
271
|
+
const store = getStore<{ $patch?: (partial: any) => void }>(storeId);
|
|
272
|
+
if (store && typeof store.$patch === 'function') {
|
|
273
|
+
store.$patch(sanitizeHydrationState(state));
|
|
274
|
+
}
|
|
275
|
+
};
|
|
276
|
+
|
|
277
|
+
/**
|
|
278
|
+
* Hydrates all stores at once from a deserialized state map.
|
|
279
|
+
*
|
|
280
|
+
* Convenience wrapper that calls `hydrateStore` for each entry in the state map.
|
|
281
|
+
*
|
|
282
|
+
* @param stateMap - Map of store IDs to their state objects
|
|
283
|
+
*
|
|
284
|
+
* @example
|
|
285
|
+
* ```ts
|
|
286
|
+
* import { hydrateStores, deserializeStoreState } from '@bquery/bquery/ssr';
|
|
287
|
+
*
|
|
288
|
+
* const ssrState = deserializeStoreState();
|
|
289
|
+
* hydrateStores(ssrState);
|
|
290
|
+
* ```
|
|
291
|
+
*/
|
|
292
|
+
export const hydrateStores = (stateMap: DeserializedStoreState): void => {
|
|
293
|
+
for (const [storeId, state] of Object.entries(stateMap)) {
|
|
294
|
+
hydrateStore(storeId, state);
|
|
295
|
+
}
|
|
296
|
+
};
|