@squide/firefly 9.3.2 → 9.3.4

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.
Files changed (68) hide show
  1. package/CHANGELOG.md +19 -0
  2. package/dist/AppRouter.js +18 -16
  3. package/dist/AppRouter.js.map +1 -0
  4. package/dist/AppRouterContext.js +6 -4
  5. package/dist/AppRouterContext.js.map +1 -0
  6. package/dist/AppRouterReducer.js +42 -40
  7. package/dist/AppRouterReducer.js.map +1 -0
  8. package/dist/FireflyRuntime.js +13 -11
  9. package/dist/FireflyRuntime.js.map +1 -0
  10. package/dist/GlobalDataQueriesError.js +2 -0
  11. package/dist/GlobalDataQueriesError.js.map +1 -0
  12. package/dist/RootRoute.js +10 -8
  13. package/dist/RootRoute.js.map +1 -0
  14. package/dist/boostrap.js +12 -10
  15. package/dist/boostrap.js.map +1 -0
  16. package/dist/index.js +6 -4
  17. package/dist/index.js.map +1 -0
  18. package/dist/useCanFetchProtectedData.js +4 -2
  19. package/dist/useCanFetchProtectedData.js.map +1 -0
  20. package/dist/useCanFetchPublicData.js +4 -2
  21. package/dist/useCanFetchPublicData.js.map +1 -0
  22. package/dist/useCanRegisterDeferredRegistrations.js +4 -2
  23. package/dist/useCanRegisterDeferredRegistrations.js.map +1 -0
  24. package/dist/useCanUpdateDeferredRegistrations.js +4 -2
  25. package/dist/useCanUpdateDeferredRegistrations.js.map +1 -0
  26. package/dist/useDeferredRegistrations.js +13 -11
  27. package/dist/useDeferredRegistrations.js.map +1 -0
  28. package/dist/useExecuteOnce.js +2 -0
  29. package/dist/useExecuteOnce.js.map +1 -0
  30. package/dist/useIsActiveRouteProtected.js +8 -6
  31. package/dist/useIsActiveRouteProtected.js.map +1 -0
  32. package/dist/useIsBootstrapping.js +4 -2
  33. package/dist/useIsBootstrapping.js.map +1 -0
  34. package/dist/useNavigationItems.js +7 -5
  35. package/dist/useNavigationItems.js.map +1 -0
  36. package/dist/useProtectedDataQueries.js +17 -15
  37. package/dist/useProtectedDataQueries.js.map +1 -0
  38. package/dist/usePublicDataQueries.js +16 -14
  39. package/dist/usePublicDataQueries.js.map +1 -0
  40. package/dist/useRegisterDeferredRegistrations.js +5 -3
  41. package/dist/useRegisterDeferredRegistrations.js.map +1 -0
  42. package/dist/useStrictRegistrationMode.js +14 -12
  43. package/dist/useStrictRegistrationMode.js.map +1 -0
  44. package/dist/useUpdateDeferredRegistrations.js +7 -5
  45. package/dist/useUpdateDeferredRegistrations.js.map +1 -0
  46. package/package.json +18 -16
  47. package/src/AppRouter.tsx +63 -0
  48. package/src/AppRouterContext.ts +27 -0
  49. package/src/AppRouterReducer.ts +363 -0
  50. package/src/FireflyRuntime.tsx +71 -0
  51. package/src/GlobalDataQueriesError.ts +17 -0
  52. package/src/RootRoute.tsx +25 -0
  53. package/src/boostrap.ts +62 -0
  54. package/src/index.ts +28 -0
  55. package/src/useCanFetchProtectedData.ts +26 -0
  56. package/src/useCanFetchPublicData.ts +23 -0
  57. package/src/useCanRegisterDeferredRegistrations.ts +24 -0
  58. package/src/useCanUpdateDeferredRegistrations.ts +23 -0
  59. package/src/useDeferredRegistrations.ts +59 -0
  60. package/src/useExecuteOnce.ts +23 -0
  61. package/src/useIsActiveRouteProtected.ts +12 -0
  62. package/src/useIsBootstrapping.ts +45 -0
  63. package/src/useNavigationItems.ts +18 -0
  64. package/src/useProtectedDataQueries.ts +101 -0
  65. package/src/usePublicDataQueries.ts +88 -0
  66. package/src/useRegisterDeferredRegistrations.ts +9 -0
  67. package/src/useStrictRegistrationMode.ts +28 -0
  68. package/src/useUpdateDeferredRegistrations.ts +16 -0
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@squide/firefly",
3
3
  "author": "Workleap",
4
- "version": "9.3.2",
4
+ "version": "9.3.4",
5
5
  "description": "Helpers to facilitate the creation of an application with the Squide firefly technology stack.",
6
6
  "license": "Apache-2.0",
7
7
  "repository": {
@@ -22,7 +22,8 @@
22
22
  }
23
23
  },
24
24
  "files": [
25
- "/dist",
25
+ "src",
26
+ "dist",
26
27
  "CHANGELOG.md",
27
28
  "README.md"
28
29
  ],
@@ -34,29 +35,30 @@
34
35
  "react-router-dom": "*"
35
36
  },
36
37
  "dependencies": {
37
- "@types/react": "19.0.1",
38
- "@types/react-dom": "19.0.2",
39
- "@squide/core": "5.4.1",
40
- "@squide/module-federation": "6.2.1",
41
- "@squide/msw": "3.2.1",
42
- "@squide/react-router": "6.4.2"
38
+ "@types/react": "19.0.7",
39
+ "@types/react-dom": "19.0.3",
40
+ "@squide/core": "5.4.2",
41
+ "@squide/module-federation": "6.2.2",
42
+ "@squide/msw": "3.2.2",
43
+ "@squide/react-router": "6.4.4"
43
44
  },
44
45
  "devDependencies": {
45
- "@rsbuild/plugin-react": "1.1.0",
46
- "@rslib/core": "0.1.4",
47
- "@swc/core": "1.10.1",
46
+ "@rsbuild/core": "1.1.13",
47
+ "@rslib/core": "0.3.1",
48
+ "@swc/core": "1.10.7",
48
49
  "@swc/jest": "0.2.37",
49
50
  "@testing-library/jest-dom": "6.6.3",
50
- "@testing-library/react": "16.1.0",
51
+ "@testing-library/react": "16.2.0",
51
52
  "@types/jest": "29.5.14",
52
- "@typescript-eslint/parser": "8.18.0",
53
- "@workleap/eslint-plugin": "3.2.5",
54
- "@workleap/swc-configs": "2.2.3",
53
+ "@typescript-eslint/parser": "8.20.0",
54
+ "@workleap/eslint-plugin": "3.2.6",
55
+ "@workleap/rslib-configs": "1.0.2",
56
+ "@workleap/swc-configs": "2.2.4",
55
57
  "@workleap/typescript-configs": "3.0.2",
56
58
  "eslint": "8.57.0",
57
59
  "jest": "29.7.0",
58
60
  "jest-environment-jsdom": "29.7.0",
59
- "msw": "2.6.9",
61
+ "msw": "2.7.0",
60
62
  "react": "19.0.0",
61
63
  "react-dom": "19.0.0",
62
64
  "react-router-dom": "6.27.0",
@@ -0,0 +1,63 @@
1
+ import { useLogger } from "@squide/core";
2
+ import { useRoutes, type Route } from "@squide/react-router";
3
+ import { useEffect, useMemo, type ReactElement } from "react";
4
+ import type { RouterProviderProps } from "react-router-dom";
5
+ import { AppRouterDispatcherContext, AppRouterStateContext } from "./AppRouterContext.ts";
6
+ import { useAppRouterReducer } from "./AppRouterReducer.ts";
7
+ import { RootRoute } from "./RootRoute.tsx";
8
+ import { useStrictRegistrationMode } from "./useStrictRegistrationMode.ts";
9
+
10
+ export interface AppRouterRenderFunctionArgs {
11
+ routes: Route[];
12
+ }
13
+
14
+ export interface RenderRouterProviderFunctionArgs {
15
+ rootRoute: ReactElement;
16
+ registeredRoutes: Route[];
17
+ routerProviderProps: Omit<RouterProviderProps, "router">;
18
+ }
19
+
20
+ export type RenderRouterProviderFunction = (args: RenderRouterProviderFunctionArgs) => ReactElement;
21
+
22
+ export interface AppRouterProps {
23
+ waitForMsw: boolean;
24
+ waitForPublicData?: boolean;
25
+ waitForProtectedData?: boolean;
26
+ children: RenderRouterProviderFunction;
27
+ }
28
+
29
+ export function AppRouter(props: AppRouterProps) {
30
+ const {
31
+ waitForMsw,
32
+ waitForPublicData = false,
33
+ waitForProtectedData = false,
34
+ children: renderRouterProvider
35
+ } = props;
36
+
37
+ const [state, dispatch] = useAppRouterReducer(waitForMsw, waitForPublicData, waitForProtectedData);
38
+
39
+ const logger = useLogger();
40
+ const routes = useRoutes();
41
+
42
+ useStrictRegistrationMode();
43
+
44
+ useEffect(() => {
45
+ logger.debug("[squide] AppRouter state updated:", state);
46
+ }, [state, logger]);
47
+
48
+ const routerProvider = useMemo(() => {
49
+ return renderRouterProvider({
50
+ rootRoute: <RootRoute />,
51
+ registeredRoutes: routes,
52
+ routerProviderProps: {}
53
+ });
54
+ }, [routes, renderRouterProvider]);
55
+
56
+ return (
57
+ <AppRouterDispatcherContext.Provider value={dispatch}>
58
+ <AppRouterStateContext.Provider value={state}>
59
+ {routerProvider}
60
+ </AppRouterStateContext.Provider>
61
+ </AppRouterDispatcherContext.Provider>
62
+ );
63
+ }
@@ -0,0 +1,27 @@
1
+ import { isNil } from "@squide/core";
2
+ import { createContext, useContext } from "react";
3
+ import type { AppRouterDispatch, AppRouterState } from "./AppRouterReducer.ts";
4
+
5
+ export const AppRouterStateContext = createContext<AppRouterState | undefined>(undefined);
6
+
7
+ export function useAppRouterState() {
8
+ const state = useContext(AppRouterStateContext);
9
+
10
+ if (isNil(state)) {
11
+ throw new Error("[squide] The useAppRouterState hook must be called by a children of the AppRouter component.");
12
+ }
13
+
14
+ return state;
15
+ }
16
+
17
+ export const AppRouterDispatcherContext = createContext<AppRouterDispatch | undefined>(undefined);
18
+
19
+ export function useAppRouterDispatcher() {
20
+ const dispatch = useContext(AppRouterDispatcherContext);
21
+
22
+ if (isNil(dispatch)) {
23
+ throw new Error("[squide] The useAppRouterDispatcher hook must be called by a children of the AppRouter component.");
24
+ }
25
+
26
+ return dispatch;
27
+ }
@@ -0,0 +1,363 @@
1
+ import { addLocalModuleRegistrationStatusChangedListener, getLocalModuleRegistrationStatus, removeLocalModuleRegistrationStatusChangedListener, useEventBus, useLogger } from "@squide/core";
2
+ import { addRemoteModuleRegistrationStatusChangedListener, areModulesReady, areModulesRegistered, getRemoteModuleRegistrationStatus, removeRemoteModuleRegistrationStatusChangedListener } from "@squide/module-federation";
3
+ import { addMswStateChangedListener, isMswReady, removeMswStateChangedListener } from "@squide/msw";
4
+ import { useCallback, useEffect, useMemo, useReducer, type Dispatch } from "react";
5
+ import { useExecuteOnce } from "./useExecuteOnce.ts";
6
+ import { isApplicationBootstrapping } from "./useIsBootstrapping.ts";
7
+
8
+ export type ActiveRouteVisiblity = "unknown" | "public" | "protected";
9
+
10
+ export interface AppRouterState {
11
+ waitForMsw: boolean;
12
+ waitForPublicData: boolean;
13
+ waitForProtectedData: boolean;
14
+ areModulesRegistered: boolean;
15
+ areModulesReady: boolean;
16
+ isMswReady: boolean;
17
+ isPublicDataReady: boolean;
18
+ isProtectedDataReady: boolean;
19
+ publicDataUpdatedAt?: number;
20
+ protectedDataUpdatedAt?: number;
21
+ deferredRegistrationsUpdatedAt?: number;
22
+ activeRouteVisibility: ActiveRouteVisiblity;
23
+ isUnauthorized: boolean;
24
+ }
25
+
26
+ export type AppRouterActionType =
27
+ | "modules-registered"
28
+ | "modules-ready"
29
+ | "msw-ready"
30
+ | "public-data-ready"
31
+ | "protected-data-ready"
32
+ | "public-data-updated"
33
+ | "protected-data-updated"
34
+ | "deferred-registrations-updated"
35
+ | "active-route-is-public"
36
+ | "active-route-is-protected"
37
+ | "is-unauthorized";
38
+
39
+ // The followings event const are a concatenation of "squide-" with AppRouterActionType.
40
+ // They are dispatched by the useEnhancedReducerDispatch hook.
41
+ export const ModulesRegisteredEvent = "squide-modules-registered";
42
+ export const ModulesReadyEvent = "squide-modules-ready";
43
+ export const MswReadyEvent = "squide-msw-ready";
44
+ export const PublicDataReadyEvent = "squide-public-data-ready";
45
+ export const ProtectedDataReadyEvent = "squide-protected-data-ready";
46
+ export const PublicDataUpdatedEvent = "squide-public-data-updated";
47
+ export const ProtectedDataUpdatedEvent = "squide-protected-data-updated";
48
+ export const DeferredRegistrationsUpdatedEvent = "squide-deferred-registrations-updated";
49
+ export const ApplicationBoostrappedEvent = "squide-app-boostrapped";
50
+
51
+ export interface AppRouterAction {
52
+ type: AppRouterActionType;
53
+ }
54
+
55
+ export type AppRouterDispatch = Dispatch<AppRouterAction>;
56
+
57
+ function reducer(state: AppRouterState, action: AppRouterAction) {
58
+ let newState = state;
59
+
60
+ switch (action.type) {
61
+ case "modules-registered": {
62
+ newState = {
63
+ ...newState,
64
+ areModulesRegistered: true
65
+ };
66
+
67
+ break;
68
+ }
69
+ case "modules-ready": {
70
+ newState = {
71
+ ...newState,
72
+ areModulesReady: true,
73
+ // Will be set even if the app is not using deferred registrations.
74
+ deferredRegistrationsUpdatedAt: Date.now()
75
+ };
76
+
77
+ break;
78
+ }
79
+ case "msw-ready": {
80
+ newState = {
81
+ ...newState,
82
+ isMswReady: true
83
+ };
84
+
85
+ break;
86
+ }
87
+ case "public-data-ready": {
88
+ newState = {
89
+ ...newState,
90
+ isPublicDataReady: true,
91
+ publicDataUpdatedAt: Date.now()
92
+ };
93
+
94
+ break;
95
+ }
96
+ case "protected-data-ready": {
97
+ newState = {
98
+ ...newState,
99
+ isProtectedDataReady: true,
100
+ protectedDataUpdatedAt: Date.now()
101
+ };
102
+
103
+ break;
104
+ }
105
+ case "public-data-updated": {
106
+ newState = {
107
+ ...newState,
108
+ publicDataUpdatedAt: Date.now()
109
+ };
110
+
111
+ break;
112
+ }
113
+ case "protected-data-updated": {
114
+ newState = {
115
+ ...newState,
116
+ protectedDataUpdatedAt: Date.now()
117
+ };
118
+
119
+ break;
120
+ }
121
+ case "deferred-registrations-updated": {
122
+ newState = {
123
+ ...newState,
124
+ deferredRegistrationsUpdatedAt: Date.now()
125
+ };
126
+
127
+ break;
128
+ }
129
+ case "active-route-is-public": {
130
+ newState = {
131
+ ...newState,
132
+ activeRouteVisibility: "public"
133
+ };
134
+
135
+ break;
136
+ }
137
+ case "active-route-is-protected": {
138
+ newState = {
139
+ ...newState,
140
+ activeRouteVisibility: "protected"
141
+ };
142
+
143
+ break;
144
+ }
145
+ case "is-unauthorized": {
146
+ newState = {
147
+ ...newState,
148
+ isUnauthorized: true
149
+ };
150
+
151
+ break;
152
+ }
153
+ default: {
154
+ throw new Error(`[squide] The AppRouter component state reducer doesn't support action type "${action.type}".`);
155
+ }
156
+ }
157
+
158
+ return newState;
159
+ }
160
+
161
+ export function getAreModulesRegistered() {
162
+ const localModuleStatus = getLocalModuleRegistrationStatus();
163
+ const remoteModuleStatus = getRemoteModuleRegistrationStatus();
164
+
165
+ return areModulesRegistered(localModuleStatus, remoteModuleStatus);
166
+ }
167
+
168
+ export function getAreModulesReady() {
169
+ const localModuleStatus = getLocalModuleRegistrationStatus();
170
+ const remoteModuleStatus = getRemoteModuleRegistrationStatus();
171
+
172
+ return areModulesReady(localModuleStatus, remoteModuleStatus);
173
+ }
174
+
175
+ export function useModuleRegistrationStatusDispatcher(areModulesRegisteredValue: boolean, areModulesReadyValue: boolean, dispatch: AppRouterDispatch) {
176
+ const logger = useLogger();
177
+
178
+ const dispatchModulesRegistered = useExecuteOnce(useCallback(() => {
179
+ if (getAreModulesRegistered()) {
180
+ dispatch({ type: "modules-registered" });
181
+
182
+ logger.debug("[squide] %cModules are registered%c.", "color: white; background-color: green;", "");
183
+
184
+ return true;
185
+ }
186
+
187
+ return false;
188
+ }, [dispatch, logger]));
189
+
190
+ const dispatchModulesReady = useExecuteOnce(useCallback(() => {
191
+ if (getAreModulesReady()) {
192
+ dispatch({ type: "modules-ready" });
193
+
194
+ logger.debug("[squide] %cModules are ready%c.", "color: white; background-color: green;", "");
195
+
196
+ return true;
197
+ }
198
+
199
+ return false;
200
+ }, [dispatch, logger]));
201
+
202
+ return useEffect(() => {
203
+ if (!areModulesRegisteredValue) {
204
+ addLocalModuleRegistrationStatusChangedListener(dispatchModulesRegistered);
205
+ addRemoteModuleRegistrationStatusChangedListener(dispatchModulesRegistered);
206
+ }
207
+
208
+ if (!areModulesReadyValue) {
209
+ addLocalModuleRegistrationStatusChangedListener(dispatchModulesReady);
210
+ addRemoteModuleRegistrationStatusChangedListener(dispatchModulesReady);
211
+ }
212
+
213
+ return () => {
214
+ removeLocalModuleRegistrationStatusChangedListener(dispatchModulesRegistered);
215
+ removeRemoteModuleRegistrationStatusChangedListener(dispatchModulesRegistered);
216
+
217
+ removeLocalModuleRegistrationStatusChangedListener(dispatchModulesReady);
218
+ removeRemoteModuleRegistrationStatusChangedListener(dispatchModulesReady);
219
+ };
220
+ }, [areModulesRegisteredValue, areModulesReadyValue, dispatchModulesRegistered, dispatchModulesReady]);
221
+ }
222
+
223
+ export function useMswStatusDispatcher(isMswReadyValue: boolean, dispatch: AppRouterDispatch) {
224
+ const logger = useLogger();
225
+
226
+ const dispatchMswReady = useExecuteOnce(useCallback(() => {
227
+ if (isMswReady()) {
228
+ dispatch({ type: "msw-ready" });
229
+
230
+ logger.debug("[squide] %cMSW is ready%c.", "color: white; background-color: green;", "");
231
+
232
+ return true;
233
+ }
234
+
235
+ return false;
236
+ }, [dispatch, logger]));
237
+
238
+ useEffect(() => {
239
+ if (!isMswReadyValue) {
240
+ addMswStateChangedListener(dispatchMswReady);
241
+ }
242
+
243
+ return () => {
244
+ removeMswStateChangedListener(dispatchMswReady);
245
+ };
246
+ }, [isMswReadyValue, dispatchMswReady]);
247
+ }
248
+
249
+ export function useBootstrappingCompletedDispatcher(state: AppRouterState) {
250
+ const eventBus = useEventBus();
251
+
252
+ const areModulesRegisteredValue = state.areModulesRegistered;
253
+ const isBoostrapping = isApplicationBootstrapping(state);
254
+
255
+ useExecuteOnce(useCallback(() => {
256
+ if (areModulesRegisteredValue && !isBoostrapping) {
257
+ eventBus.dispatch(ApplicationBoostrappedEvent);
258
+
259
+ return true;
260
+ }
261
+
262
+ return false;
263
+ }, [areModulesRegisteredValue, isBoostrapping, eventBus]), true);
264
+ }
265
+
266
+ let dispatchProxyFactory: ((reactDispatch: AppRouterDispatch) => AppRouterDispatch) | undefined;
267
+
268
+ // This function should only be used by tests.
269
+ export function __setAppReducerDispatchProxyFactory(factory: (reactDispatch: AppRouterDispatch) => AppRouterDispatch) {
270
+ dispatchProxyFactory = factory;
271
+ }
272
+
273
+ // This function should only be used by tests.
274
+ export function __clearAppReducerDispatchProxy() {
275
+ dispatchProxyFactory = undefined;
276
+ }
277
+
278
+ function useReducerDispatchProxy(reactDispatch: AppRouterDispatch) {
279
+ return useMemo(() => {
280
+ return dispatchProxyFactory ? dispatchProxyFactory(reactDispatch) : reactDispatch;
281
+ }, [reactDispatch]);
282
+ }
283
+
284
+ function useEnhancedReducerDispatch(reducerDispatch: AppRouterDispatch) {
285
+ const logger = useLogger();
286
+ const eventBus = useEventBus();
287
+
288
+ return useCallback((action: AppRouterAction) => {
289
+ logger.debug("[squide] The following action has been dispatched to the AppRouter reducer:", action);
290
+ eventBus.dispatch(`squide-${action.type}`);
291
+
292
+ reducerDispatch(action);
293
+ }, [reducerDispatch, logger, eventBus]);
294
+ }
295
+
296
+ export function useAppRouterReducer(waitForMsw: boolean, waitForPublicData: boolean, waitForProtectedData: boolean): [AppRouterState, AppRouterDispatch] {
297
+ const areModulesInitiallyRegistered = getAreModulesRegistered();
298
+ const areModulesInitiallyReady = getAreModulesReady();
299
+ const isMswInitiallyReady = isMswReady();
300
+
301
+ const eventBus = useEventBus();
302
+
303
+ // When modules are initially registered, the reducer action will never be dispatched, therefore the event would not be dispatched as well.
304
+ // To ensure the bootstrapping events sequencing, the event is manually dispatched when the modules are initially registered.
305
+ useExecuteOnce(useCallback(() => {
306
+ if (areModulesInitiallyRegistered) {
307
+ eventBus.dispatch(ModulesRegisteredEvent);
308
+ }
309
+
310
+ return true;
311
+ }, [areModulesInitiallyRegistered, eventBus]), true);
312
+
313
+ // When modules are initially registered, the reducer action will never be dispatched, therefore the event would not be dispatched as well.
314
+ // To ensure the bootstrapping events sequencing, the event is manually dispatched when the modules are initially registered.
315
+ useExecuteOnce(useCallback(() => {
316
+ if (areModulesInitiallyReady) {
317
+ eventBus.dispatch(ModulesReadyEvent);
318
+ }
319
+
320
+ return true;
321
+ }, [areModulesInitiallyReady, eventBus]), true);
322
+
323
+ // When modules are initially registered, the reducer action will never be dispatched, therefore the event would not be dispatched as well.
324
+ // To ensure the bootstrapping events sequencing, the event is manually dispatched when the modules are initially registered.
325
+ useExecuteOnce(useCallback(() => {
326
+ if (isMswInitiallyReady) {
327
+ eventBus.dispatch(MswReadyEvent);
328
+ }
329
+
330
+ return true;
331
+ }, [isMswInitiallyReady, eventBus]), true);
332
+
333
+ const [state, reactDispatch] = useReducer(reducer, {
334
+ waitForMsw,
335
+ waitForPublicData,
336
+ waitForProtectedData,
337
+ // When the modules registration functions are awaited, the event listeners are registered after the modules are registered.
338
+ areModulesRegistered: areModulesInitiallyRegistered,
339
+ areModulesReady: areModulesInitiallyReady,
340
+ isMswReady: isMswInitiallyReady,
341
+ isPublicDataReady: false,
342
+ isProtectedDataReady: false,
343
+ activeRouteVisibility: "unknown",
344
+ isUnauthorized: false
345
+ });
346
+
347
+ const {
348
+ areModulesRegistered: areModulesRegisteredValue,
349
+ areModulesReady: areModulesReadyValue,
350
+ isMswReady: isMswReadyValue
351
+ } = state;
352
+
353
+ // The dispatch proxy is strictly an utility allowing tests to mock the useReducer dispatch function. It's easier
354
+ // than mocking the import from React.
355
+ const dispatchProxy = useReducerDispatchProxy(reactDispatch);
356
+ const dispatch = useEnhancedReducerDispatch(dispatchProxy);
357
+
358
+ useModuleRegistrationStatusDispatcher(areModulesRegisteredValue, areModulesReadyValue, dispatch);
359
+ useMswStatusDispatcher(isMswReadyValue, dispatch);
360
+ useBootstrappingCompletedDispatcher(state);
361
+
362
+ return [state, dispatch];
363
+ }
@@ -0,0 +1,71 @@
1
+ import type { RegisterRouteOptions, RuntimeOptions } from "@squide/core";
2
+ import { MswPlugin, MswPluginName } from "@squide/msw";
3
+ import { ReactRouterRuntime, type Route } from "@squide/react-router";
4
+ import type { RequestHandler } from "msw";
5
+ import { getAreModulesRegistered } from "./AppRouterReducer.ts";
6
+
7
+ export interface FireflyRuntimeOptions extends RuntimeOptions {
8
+ useMsw?: boolean;
9
+ }
10
+
11
+ export class FireflyRuntime extends ReactRouterRuntime {
12
+ readonly #useMsw: boolean;
13
+
14
+ constructor({ plugins, useMsw, ...options }: FireflyRuntimeOptions = {}) {
15
+ if (useMsw) {
16
+ super({
17
+ plugins: [
18
+ ...(plugins ?? []),
19
+ runtime => new MswPlugin(runtime)
20
+ ],
21
+ ...options
22
+ });
23
+
24
+ this.#useMsw = true;
25
+ } else {
26
+ super({
27
+ plugins,
28
+ ...options
29
+ });
30
+
31
+ this.#useMsw = false;
32
+ }
33
+ }
34
+
35
+ registerRequestHandlers(handlers: RequestHandler[]) {
36
+ const mswPlugin = this.getPlugin(MswPluginName) as MswPlugin;
37
+
38
+ if (!mswPlugin) {
39
+ throw new Error("[squide] Cannot register the provided MSW request handlers because the runtime hasn't been initialized with MSW. Did you instanciate the FireflyRuntime with the \"useMsw\" option?");
40
+ }
41
+
42
+ if (getAreModulesRegistered()) {
43
+ throw new Error("[squide] Cannot register an MSW request handlers once the modules are registered. Are you trying to register an MSW request handler in a deferred registration function? Only navigation items can be registered in a deferred registration function.");
44
+ }
45
+
46
+ mswPlugin.registerRequestHandlers(handlers);
47
+ }
48
+
49
+ // Must define a return type otherwise we get an "error TS2742: The inferred type of 'requestHandlers' cannot be named" error.
50
+ get requestHandlers(): RequestHandler[] {
51
+ const mswPlugin = this.getPlugin(MswPluginName) as MswPlugin;
52
+
53
+ if (!mswPlugin) {
54
+ throw new Error("[squide] Cannot retrieve MSW request handlers because the runtime hasn't been initialized with MSW. Did you instanciate the FireflyRuntime with the \"useMsw\" option?");
55
+ }
56
+
57
+ return mswPlugin.requestHandlers;
58
+ }
59
+
60
+ registerRoute(route: Route, options: RegisterRouteOptions = {}) {
61
+ if (getAreModulesRegistered()) {
62
+ throw new Error("[squide] Cannot register a route once the modules are registered. Are you trying to register a route in a deferred registration function? Only navigation items can be registered in a deferred registration function.");
63
+ }
64
+
65
+ super.registerRoute(route, options);
66
+ }
67
+
68
+ get isMswEnabled() {
69
+ return this.#useMsw;
70
+ }
71
+ }
@@ -0,0 +1,17 @@
1
+ export class GlobalDataQueriesError extends Error {
2
+ readonly #errors: Error[];
3
+
4
+ constructor(message: string, errors: Error[]) {
5
+ super(message);
6
+
7
+ this.#errors = errors;
8
+ }
9
+
10
+ get errors() {
11
+ return this.#errors;
12
+ }
13
+ }
14
+
15
+ export function isGlobalDataQueriesError(error?: unknown): error is GlobalDataQueriesError {
16
+ return error !== undefined && error !== null && error instanceof GlobalDataQueriesError;
17
+ }
@@ -0,0 +1,25 @@
1
+ import { useEffect } from "react";
2
+ import { Outlet } from "react-router-dom";
3
+ import { useAppRouterDispatcher, useAppRouterState } from "./AppRouterContext.ts";
4
+ import { useIsActiveRouteProtected } from "./useIsActiveRouteProtected.ts";
5
+
6
+ export function RootRoute() {
7
+ const state = useAppRouterState();
8
+ const isActiveRouteProtected = useIsActiveRouteProtected(state.areModulesReady);
9
+
10
+ const dispatch = useAppRouterDispatcher();
11
+
12
+ useEffect(() => {
13
+ // Dispatching the active route visibility must be done in a route because React Router's useLocation
14
+ // hook throws if it's not called from a child of the router component.
15
+ if (isActiveRouteProtected) {
16
+ dispatch({ type: "active-route-is-protected" });
17
+ } else {
18
+ dispatch({ type: "active-route-is-public" });
19
+ }
20
+ }, [isActiveRouteProtected, dispatch]);
21
+
22
+ return (
23
+ <Outlet />
24
+ );
25
+ }