@webiny/app 6.3.0 → 6.4.0-beta.0

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 (276) hide show
  1. package/App.js +70 -77
  2. package/App.js.map +1 -1
  3. package/AppContainer.js +3 -6
  4. package/AppContainer.js.map +1 -1
  5. package/apollo-client/InMemoryCache.js +11 -13
  6. package/apollo-client/InMemoryCache.js.map +1 -1
  7. package/apollo-client/IntrospectionFragmentMatcher.js +26 -38
  8. package/apollo-client/IntrospectionFragmentMatcher.js.map +1 -1
  9. package/components/Image.js +11 -17
  10. package/components/Image.js.map +1 -1
  11. package/components/index.js +0 -2
  12. package/config/RouterConfig/Route.js +23 -28
  13. package/config/RouterConfig/Route.js.map +1 -1
  14. package/config/RouterConfig.js +11 -10
  15. package/config/RouterConfig.js.map +1 -1
  16. package/config.js +19 -20
  17. package/config.js.map +1 -1
  18. package/contexts/Ui/index.js +26 -28
  19. package/contexts/Ui/index.js.map +1 -1
  20. package/core/Plugin.js +7 -9
  21. package/core/Plugin.js.map +1 -1
  22. package/core/Plugins.js +15 -34
  23. package/core/Plugins.js.map +1 -1
  24. package/core/Provider.js +5 -13
  25. package/core/Provider.js.map +1 -1
  26. package/core/createProvider.js +3 -6
  27. package/core/createProvider.js.map +1 -1
  28. package/core/createProviderPlugin.js +8 -13
  29. package/core/createProviderPlugin.js.map +1 -1
  30. package/errors/AuthenticationErrorEvent.js +8 -5
  31. package/errors/AuthenticationErrorEvent.js.map +1 -1
  32. package/errors/NetworkErrorEvent.js +8 -5
  33. package/errors/NetworkErrorEvent.js.map +1 -1
  34. package/errors/abstractions.js +3 -2
  35. package/errors/abstractions.js.map +1 -1
  36. package/errors/index.js +1 -4
  37. package/exports/admin/env-config.js +0 -2
  38. package/exports/admin/graphql-client.js +0 -2
  39. package/exports/admin/local-storage.js +0 -2
  40. package/exports/admin/router.js +0 -2
  41. package/exports/admin/security.js +0 -2
  42. package/exports/admin.js +0 -2
  43. package/features/envConfig/EnvConfig.js +12 -13
  44. package/features/envConfig/EnvConfig.js.map +1 -1
  45. package/features/envConfig/abstractions.js +2 -1
  46. package/features/envConfig/abstractions.js.map +1 -1
  47. package/features/envConfig/feature.js +9 -8
  48. package/features/envConfig/feature.js.map +1 -1
  49. package/features/envConfig/index.js +0 -2
  50. package/features/eventPublisher/EventPublisher.js +13 -21
  51. package/features/eventPublisher/EventPublisher.js.map +1 -1
  52. package/features/eventPublisher/abstractions.js +7 -9
  53. package/features/eventPublisher/abstractions.js.map +1 -1
  54. package/features/eventPublisher/feature.js +13 -12
  55. package/features/eventPublisher/feature.js.map +1 -1
  56. package/features/eventPublisher/index.js +1 -3
  57. package/features/graphqlClient/AuthenticationErrorPublishing.js +26 -26
  58. package/features/graphqlClient/AuthenticationErrorPublishing.js.map +1 -1
  59. package/features/graphqlClient/BatchingGraphQLClient.js +104 -149
  60. package/features/graphqlClient/BatchingGraphQLClient.js.map +1 -1
  61. package/features/graphqlClient/FetchGraphQLClient.js +41 -48
  62. package/features/graphqlClient/FetchGraphQLClient.js.map +1 -1
  63. package/features/graphqlClient/NetworkErrorPublishing.js +35 -42
  64. package/features/graphqlClient/NetworkErrorPublishing.js.map +1 -1
  65. package/features/graphqlClient/RequestValue.js +42 -41
  66. package/features/graphqlClient/RequestValue.js.map +1 -1
  67. package/features/graphqlClient/RetryGraphQLClient.js +36 -45
  68. package/features/graphqlClient/RetryGraphQLClient.js.map +1 -1
  69. package/features/graphqlClient/__tests__/GraphQLClient.test.js +383 -356
  70. package/features/graphqlClient/__tests__/GraphQLClient.test.js.map +1 -1
  71. package/features/graphqlClient/abstractions.js +2 -1
  72. package/features/graphqlClient/abstractions.js.map +1 -1
  73. package/features/graphqlClient/feature.js +14 -20
  74. package/features/graphqlClient/feature.js.map +1 -1
  75. package/features/graphqlClient/index.js +0 -2
  76. package/features/graphqlClient/types.js +0 -3
  77. package/features/localStorage/BrowserLocalStorageGateway.js +56 -63
  78. package/features/localStorage/BrowserLocalStorageGateway.js.map +1 -1
  79. package/features/localStorage/LocalStorage.js +27 -24
  80. package/features/localStorage/LocalStorage.js.map +1 -1
  81. package/features/localStorage/LocalStorageRepository.js +75 -88
  82. package/features/localStorage/LocalStorageRepository.js.map +1 -1
  83. package/features/localStorage/abstractions.js +5 -16
  84. package/features/localStorage/abstractions.js.map +1 -1
  85. package/features/localStorage/feature.js +17 -28
  86. package/features/localStorage/feature.js.map +1 -1
  87. package/features/localStorage/index.js +1 -3
  88. package/features/mainGraphQLClient/MainGraphQLClient.js +18 -14
  89. package/features/mainGraphQLClient/MainGraphQLClient.js.map +1 -1
  90. package/features/mainGraphQLClient/abstractions.js +2 -1
  91. package/features/mainGraphQLClient/abstractions.js.map +1 -1
  92. package/features/mainGraphQLClient/feature.js +11 -10
  93. package/features/mainGraphQLClient/feature.js.map +1 -1
  94. package/features/mainGraphQLClient/index.js +0 -2
  95. package/features/router/HistoryRouterGateway.js +75 -105
  96. package/features/router/HistoryRouterGateway.js.map +1 -1
  97. package/features/router/HistoryRouterGateway.test.js +184 -193
  98. package/features/router/HistoryRouterGateway.test.js.map +1 -1
  99. package/features/router/Route.js +38 -54
  100. package/features/router/Route.js.map +1 -1
  101. package/features/router/RouteUrl.js +84 -217
  102. package/features/router/RouteUrl.js.map +1 -1
  103. package/features/router/Router.js +67 -124
  104. package/features/router/Router.js.map +1 -1
  105. package/features/router/RouterPresenter.js +44 -49
  106. package/features/router/RouterPresenter.js.map +1 -1
  107. package/features/router/RouterRepository.js +103 -111
  108. package/features/router/RouterRepository.js.map +1 -1
  109. package/features/router/RouterRepository.test.js +119 -150
  110. package/features/router/RouterRepository.test.js.map +1 -1
  111. package/features/router/abstractions.js +4 -14
  112. package/features/router/abstractions.js.map +1 -1
  113. package/features/router/feature.js +13 -12
  114. package/features/router/feature.js.map +1 -1
  115. package/features/router/index.js +1 -3
  116. package/helpers/InterfaceGenerator/date.js +0 -3
  117. package/helpers/InterfaceGenerator/id.js +0 -3
  118. package/helpers/InterfaceGenerator/identity.js +0 -3
  119. package/helpers/InterfaceGenerator/index.js +0 -3
  120. package/helpers/InterfaceGenerator/numeric.js +0 -3
  121. package/helpers/InterfaceGenerator/truthful.js +0 -3
  122. package/hooks/useAutocomplete/index.js +0 -2
  123. package/hooks/useAutocomplete/useAutocomplete.js +12 -15
  124. package/hooks/useAutocomplete/useAutocomplete.js.map +1 -1
  125. package/hooks/useDataList/functions/getData.js +2 -1
  126. package/hooks/useDataList/functions/getData.js.map +1 -1
  127. package/hooks/useDataList/functions/getError.js +2 -1
  128. package/hooks/useDataList/functions/getError.js.map +1 -1
  129. package/hooks/useDataList/functions/getMeta.js +2 -1
  130. package/hooks/useDataList/functions/getMeta.js.map +1 -1
  131. package/hooks/useDataList/functions/index.js +0 -2
  132. package/hooks/useDataList/functions/searchDataByKey.js +9 -14
  133. package/hooks/useDataList/functions/searchDataByKey.js.map +1 -1
  134. package/hooks/useDataList/index.js +0 -2
  135. package/hooks/useDataList/useDataList.js +131 -152
  136. package/hooks/useDataList/useDataList.js.map +1 -1
  137. package/hooks/useDataList/utils/index.js +0 -2
  138. package/hooks/useDataList/utils/prepareLoadListParams.js +25 -34
  139. package/hooks/useDataList/utils/prepareLoadListParams.js.map +1 -1
  140. package/hooks/useHandler.js +12 -16
  141. package/hooks/useHandler.js.map +1 -1
  142. package/hooks/useHandlers.js +18 -20
  143. package/hooks/useHandlers.js.map +1 -1
  144. package/hooks/useRegisterLegacyPlugin.js +7 -6
  145. package/hooks/useRegisterLegacyPlugin.js.map +1 -1
  146. package/hooks/useUi.js +2 -3
  147. package/hooks/useUi.js.map +1 -1
  148. package/i18n/i18n.js +8 -4
  149. package/i18n/i18n.js.map +1 -1
  150. package/i18n/index.js +0 -2
  151. package/index.js +6 -11
  152. package/package.json +13 -13
  153. package/plugins/AddQuerySelectionPlugin.js +36 -61
  154. package/plugins/AddQuerySelectionPlugin.js.map +1 -1
  155. package/plugins/ApolloCacheObjectIdPlugin.js +12 -11
  156. package/plugins/ApolloCacheObjectIdPlugin.js.map +1 -1
  157. package/plugins/ApolloDynamicLink.js +20 -22
  158. package/plugins/ApolloDynamicLink.js.map +1 -1
  159. package/plugins/ApolloLinkPlugin.js +16 -15
  160. package/plugins/ApolloLinkPlugin.js.map +1 -1
  161. package/plugins/ConsoleLinkPlugin.js +19 -21
  162. package/plugins/ConsoleLinkPlugin.js.map +1 -1
  163. package/plugins/NetworkErrorLinkPlugin/ErrorOverlay.js +39 -45
  164. package/plugins/NetworkErrorLinkPlugin/ErrorOverlay.js.map +1 -1
  165. package/plugins/NetworkErrorLinkPlugin/GqlErrorOverlay.js +18 -22
  166. package/plugins/NetworkErrorLinkPlugin/GqlErrorOverlay.js.map +1 -1
  167. package/plugins/NetworkErrorLinkPlugin/LocalAwsLambdaTimeoutMessage.js +21 -29
  168. package/plugins/NetworkErrorLinkPlugin/LocalAwsLambdaTimeoutMessage.js.map +1 -1
  169. package/plugins/NetworkErrorLinkPlugin/StyledComponents.js +6 -15
  170. package/plugins/NetworkErrorLinkPlugin/StyledComponents.js.map +1 -1
  171. package/plugins/NetworkErrorLinkPlugin/Typography.js +9 -14
  172. package/plugins/NetworkErrorLinkPlugin/Typography.js.map +1 -1
  173. package/plugins/NetworkErrorLinkPlugin/assets/close_24px.js +19 -0
  174. package/plugins/NetworkErrorLinkPlugin/assets/close_24px.js.map +1 -0
  175. package/plugins/NetworkErrorLinkPlugin/createErrorOverlay.js +11 -18
  176. package/plugins/NetworkErrorLinkPlugin/createErrorOverlay.js.map +1 -1
  177. package/plugins/NetworkErrorLinkPlugin.js +41 -58
  178. package/plugins/NetworkErrorLinkPlugin.js.map +1 -1
  179. package/plugins/OmitTypenameLinkPlugin.js +9 -14
  180. package/plugins/OmitTypenameLinkPlugin.js.map +1 -1
  181. package/plugins/TenantHeaderLinkPlugin.js +23 -32
  182. package/plugins/TenantHeaderLinkPlugin.js.map +1 -1
  183. package/plugins/components/Image.js +8 -15
  184. package/plugins/components/Image.js.map +1 -1
  185. package/plugins/image.js +94 -133
  186. package/plugins/image.js.map +1 -1
  187. package/plugins/index.js +35 -59
  188. package/plugins/index.js.map +1 -1
  189. package/presentation/envConfig/useEnvConfig.js +4 -8
  190. package/presentation/envConfig/useEnvConfig.js.map +1 -1
  191. package/presentation/localStorage/index.js +0 -2
  192. package/presentation/localStorage/useLocalStorage.js +10 -16
  193. package/presentation/localStorage/useLocalStorage.js.map +1 -1
  194. package/presentation/localStorage/useLocalStorageValue.js +15 -13
  195. package/presentation/localStorage/useLocalStorageValue.js.map +1 -1
  196. package/presentation/localStorage/useLocalStorageValues.js +23 -30
  197. package/presentation/localStorage/useLocalStorageValues.js.map +1 -1
  198. package/presentation/router/RouteElementRegistry.js +22 -25
  199. package/presentation/router/RouteElementRegistry.js.map +1 -1
  200. package/presentation/router/abstractions.js +2 -1
  201. package/presentation/router/abstractions.js.map +1 -1
  202. package/presentation/router/components/Redirect.js +8 -10
  203. package/presentation/router/components/Redirect.js.map +1 -1
  204. package/presentation/router/components/RouteContent.js +14 -17
  205. package/presentation/router/components/RouteContent.js.map +1 -1
  206. package/presentation/router/components/RouteLink.js +11 -16
  207. package/presentation/router/components/RouteLink.js.map +1 -1
  208. package/presentation/router/components/SimpleLink.js +18 -24
  209. package/presentation/router/components/SimpleLink.js.map +1 -1
  210. package/presentation/router/hooks/useRoute.js +15 -23
  211. package/presentation/router/hooks/useRoute.js.map +1 -1
  212. package/presentation/router/hooks/useRouter.js +22 -23
  213. package/presentation/router/hooks/useRouter.js.map +1 -1
  214. package/presentation/router/index.js +0 -2
  215. package/presentation/router/types.js +0 -3
  216. package/react-butterfiles/Files.js +148 -217
  217. package/react-butterfiles/Files.js.map +1 -1
  218. package/react-butterfiles/index.js +2 -1
  219. package/react-butterfiles/index.js.map +1 -1
  220. package/react-butterfiles/utils/generateId.js +2 -3
  221. package/react-butterfiles/utils/generateId.js.map +1 -1
  222. package/react-butterfiles/utils/readFileContent.js +9 -13
  223. package/react-butterfiles/utils/readFileContent.js.map +1 -1
  224. package/renderApp.js +7 -6
  225. package/renderApp.js.map +1 -1
  226. package/router.js +0 -3
  227. package/rslib-runtime.js +14 -0
  228. package/rslib-runtime.js.map +1 -0
  229. package/shared/di/DiContainerProvider.js +8 -12
  230. package/shared/di/DiContainerProvider.js.map +1 -1
  231. package/shared/di/createFeature.js +7 -8
  232. package/shared/di/createFeature.js.map +1 -1
  233. package/shared/di/useFeature.js +7 -3
  234. package/shared/di/useFeature.js.map +1 -1
  235. package/static/svg/close_24px.33adaadc.svg +1 -0
  236. package/types.js +0 -3
  237. package/utils/createGenericContext.js +17 -21
  238. package/utils/createGenericContext.js.map +1 -1
  239. package/utils/createHashing.js +13 -16
  240. package/utils/createHashing.js.map +1 -1
  241. package/utils/index.js +0 -2
  242. package/utils/legacyPluginToReactComponent.js +11 -10
  243. package/utils/legacyPluginToReactComponent.js.map +1 -1
  244. package/components/index.js.map +0 -1
  245. package/errors/index.js.map +0 -1
  246. package/exports/admin/env-config.js.map +0 -1
  247. package/exports/admin/graphql-client.js.map +0 -1
  248. package/exports/admin/local-storage.js.map +0 -1
  249. package/exports/admin/router.js.map +0 -1
  250. package/exports/admin/security.js.map +0 -1
  251. package/exports/admin.js.map +0 -1
  252. package/features/envConfig/index.js.map +0 -1
  253. package/features/eventPublisher/index.js.map +0 -1
  254. package/features/graphqlClient/index.js.map +0 -1
  255. package/features/graphqlClient/types.js.map +0 -1
  256. package/features/localStorage/index.js.map +0 -1
  257. package/features/mainGraphQLClient/index.js.map +0 -1
  258. package/features/router/index.js.map +0 -1
  259. package/helpers/InterfaceGenerator/date.js.map +0 -1
  260. package/helpers/InterfaceGenerator/id.js.map +0 -1
  261. package/helpers/InterfaceGenerator/identity.js.map +0 -1
  262. package/helpers/InterfaceGenerator/index.js.map +0 -1
  263. package/helpers/InterfaceGenerator/numeric.js.map +0 -1
  264. package/helpers/InterfaceGenerator/truthful.js.map +0 -1
  265. package/hooks/useAutocomplete/index.js.map +0 -1
  266. package/hooks/useDataList/functions/index.js.map +0 -1
  267. package/hooks/useDataList/index.js.map +0 -1
  268. package/hooks/useDataList/utils/index.js.map +0 -1
  269. package/i18n/index.js.map +0 -1
  270. package/index.js.map +0 -1
  271. package/presentation/localStorage/index.js.map +0 -1
  272. package/presentation/router/index.js.map +0 -1
  273. package/presentation/router/types.js.map +0 -1
  274. package/router.js.map +0 -1
  275. package/types.js.map +0 -1
  276. package/utils/index.js.map +0 -1
@@ -1,232 +1,99 @@
1
- /**
2
- * Utility class for route URL pattern matching and generation.
3
- * Handles simple route patterns with :param syntax (e.g., /cms/entries/:modelId).
4
- *
5
- * Public API uses static methods, which internally construct RouteUrl instances
6
- * to handle the logic through proper instance methods.
7
- */
8
- export class RouteUrl {
9
- static patternCache = new Map();
10
- constructor(pattern, params, baseUrl) {
11
- this.pattern = pattern;
12
- this.params = params;
13
- this.baseUrl = this.normalizeBaseUrl(baseUrl);
14
- }
15
-
16
- /**
17
- * Generate a URL from a route pattern and parameters.
18
- * Path parameters are injected into the pattern, remaining parameters become query strings.
19
- *
20
- * @param pattern - Route pattern (e.g., '/security/api-keys/:id')
21
- * @param params - Parameters object (e.g., { id: 123, new: true })
22
- * @param baseUrl - Optional base URL to prepend (e.g., '/tenant123')
23
- * @returns Generated URL string
24
- *
25
- * @example
26
- * RouteUrl.fromPattern('/security/api-keys/:id', { id: 123, new: true })
27
- * // Returns: '/security/api-keys/123?new=true'
28
- *
29
- * @example
30
- * RouteUrl.fromPattern('/file-manager', {}, '/tenant123')
31
- * // Returns: '/tenant123/file-manager'
32
- */
33
- static fromPattern(pattern, params, baseUrl) {
34
- const routeUrl = new RouteUrl(pattern, params, baseUrl);
35
- return routeUrl.generate();
36
- }
37
-
38
- /**
39
- * Match a pathname against a route pattern and extract parameters.
40
- * If baseUrl is provided, it will be stripped from the pathname before matching.
41
- *
42
- * @param pathname - The pathname to match (e.g., '/tenant123/security/api-keys/123')
43
- * @param pattern - Route pattern (e.g., '/security/api-keys/:id')
44
- * @param baseUrl - Optional base URL to strip before matching (e.g., '/tenant123')
45
- * @returns Object with extracted params if matched, null otherwise
46
- *
47
- * @example
48
- * RouteUrl.match('/security/api-keys/123', '/security/api-keys/:id')
49
- * // Returns: { params: { id: '123' } }
50
- *
51
- * @example
52
- * RouteUrl.match('/tenant123/file-manager', '/file-manager', '/tenant123')
53
- * // Returns: { params: {} }
54
- */
55
- static match(pathname, pattern, baseUrl) {
56
- const routeUrl = new RouteUrl(pattern, undefined, baseUrl);
57
- return routeUrl.matchPathname(pathname);
58
- }
59
-
60
- /**
61
- * Generate a URL from the instance's pattern, params, and baseUrl.
62
- */
63
- generate() {
64
- if (!this.params) {
65
- return this.baseUrl + this.pattern;
1
+ class RouteUrl {
2
+ static{
3
+ this.patternCache = new Map();
66
4
  }
67
-
68
- // Extract path parameter names from pattern
69
- const pathKeys = this.extractPathKeys();
70
-
71
- // Replace :param with actual values
72
- let url = this.replacePathParams();
73
-
74
- // Handle empty url
75
- if (!url || url === "") {
76
- url = "/";
5
+ constructor(pattern, params, baseUrl){
6
+ this.pattern = pattern;
7
+ this.params = params;
8
+ this.baseUrl = this.normalizeBaseUrl(baseUrl);
77
9
  }
78
-
79
- // Prepend baseUrl
80
- url = this.baseUrl + url;
81
-
82
- // Add query parameters for keys not used in the path
83
- const queryString = this.buildQueryString(pathKeys);
84
- if (queryString) {
85
- url += `?${queryString}`;
86
- }
87
- return url;
88
- }
89
-
90
- /**
91
- * Match a pathname against the instance's pattern.
92
- */
93
- matchPathname(pathname) {
94
- // Strip baseUrl from pathname if provided
95
- if (this.baseUrl && pathname.startsWith(this.baseUrl)) {
96
- pathname = pathname.slice(this.baseUrl.length) || "/";
97
- } else if (this.baseUrl) {
98
- // pathname doesn't start with baseUrl, no match
99
- return null;
10
+ static fromPattern(pattern, params, baseUrl) {
11
+ const routeUrl = new RouteUrl(pattern, params, baseUrl);
12
+ return routeUrl.generate();
100
13
  }
101
-
102
- // Handle wildcard pattern
103
- if (this.pattern === "*" || this.pattern === "(.*)") {
104
- return {
105
- params: {}
106
- };
107
- }
108
-
109
- // Get or create cached pattern
110
- let compiled = RouteUrl.patternCache.get(this.pattern);
111
- if (!compiled) {
112
- compiled = this.compilePattern();
113
- RouteUrl.patternCache.set(this.pattern, compiled);
14
+ static match(pathname, pattern, baseUrl) {
15
+ const routeUrl = new RouteUrl(pattern, void 0, baseUrl);
16
+ return routeUrl.matchPathname(pathname);
114
17
  }
115
- const match = pathname.match(compiled.regex);
116
- if (!match) {
117
- return null;
18
+ generate() {
19
+ if (!this.params) return this.baseUrl + this.pattern;
20
+ const pathKeys = this.extractPathKeys();
21
+ let url = this.replacePathParams();
22
+ if (!url || "" === url) url = "/";
23
+ url = this.baseUrl + url;
24
+ const queryString = this.buildQueryString(pathKeys);
25
+ if (queryString) url += `?${queryString}`;
26
+ return url;
118
27
  }
119
-
120
- // Extract parameters from capture groups
121
- const params = {};
122
- compiled.keys.forEach((key, index) => {
123
- const value = match[index + 1];
124
- if (value !== undefined) {
125
- params[key] = decodeURIComponent(value);
126
- }
127
- });
128
- return {
129
- params
130
- };
131
- }
132
-
133
- /**
134
- * Extract path parameter names from the pattern.
135
- */
136
- extractPathKeys() {
137
- const pathKeys = new Set();
138
- const paramRegex = /:([a-zA-Z_][a-zA-Z0-9_]*)/g;
139
- let match;
140
- while ((match = paramRegex.exec(this.pattern)) !== null) {
141
- pathKeys.add(match[1]);
28
+ matchPathname(pathname) {
29
+ if (this.baseUrl && pathname.startsWith(this.baseUrl)) pathname = pathname.slice(this.baseUrl.length) || "/";
30
+ else if (this.baseUrl) return null;
31
+ if ("*" === this.pattern || "(.*)" === this.pattern) return {
32
+ params: {}
33
+ };
34
+ let compiled = RouteUrl.patternCache.get(this.pattern);
35
+ if (!compiled) {
36
+ compiled = this.compilePattern();
37
+ RouteUrl.patternCache.set(this.pattern, compiled);
38
+ }
39
+ const match = pathname.match(compiled.regex);
40
+ if (!match) return null;
41
+ const params = {};
42
+ compiled.keys.forEach((key, index)=>{
43
+ const value = match[index + 1];
44
+ if (void 0 !== value) params[key] = decodeURIComponent(value);
45
+ });
46
+ return {
47
+ params
48
+ };
142
49
  }
143
- return pathKeys;
144
- }
145
-
146
- /**
147
- * Replace :param placeholders with actual values from params.
148
- */
149
- replacePathParams() {
150
- return this.pattern.replace(/:([a-zA-Z_][a-zA-Z0-9_]*)/g, (_, key) => {
151
- const value = this.params[key];
152
- if (value === undefined || value === null) {
153
- return "";
154
- }
155
- return encodeURIComponent(String(value));
156
- });
157
- }
158
-
159
- /**
160
- * Build query string from params that are not path parameters.
161
- */
162
- buildQueryString(pathKeys) {
163
- const queryParams = {};
164
- for (const key in this.params) {
165
- if (!pathKeys.has(key)) {
166
- queryParams[key] = this.params[key];
167
- }
50
+ extractPathKeys() {
51
+ const pathKeys = new Set();
52
+ const paramRegex = /:([a-zA-Z_][a-zA-Z0-9_]*)/g;
53
+ let match;
54
+ while(null !== (match = paramRegex.exec(this.pattern)))pathKeys.add(match[1]);
55
+ return pathKeys;
168
56
  }
169
- if (Object.keys(queryParams).length === 0) {
170
- return "";
57
+ replacePathParams() {
58
+ return this.pattern.replace(/:([a-zA-Z_][a-zA-Z0-9_]*)/g, (_, key)=>{
59
+ const value = this.params[key];
60
+ if (null == value) return "";
61
+ return encodeURIComponent(String(value));
62
+ });
171
63
  }
172
- const searchParams = new URLSearchParams();
173
- for (const key in queryParams) {
174
- const value = queryParams[key];
175
- if (value !== undefined && value !== null) {
176
- if (Array.isArray(value)) {
177
- value.forEach(v => searchParams.append(key, String(v)));
178
- } else {
179
- searchParams.append(key, String(value));
64
+ buildQueryString(pathKeys) {
65
+ const queryParams = {};
66
+ for(const key in this.params)if (!pathKeys.has(key)) queryParams[key] = this.params[key];
67
+ if (0 === Object.keys(queryParams).length) return "";
68
+ const searchParams = new URLSearchParams();
69
+ for(const key in queryParams){
70
+ const value = queryParams[key];
71
+ if (null != value) if (Array.isArray(value)) value.forEach((v)=>searchParams.append(key, String(v)));
72
+ else searchParams.append(key, String(value));
180
73
  }
181
- }
74
+ return searchParams.toString();
182
75
  }
183
- return searchParams.toString();
184
- }
185
-
186
- /**
187
- * Normalize baseUrl by ensuring it starts with / and doesn't end with /
188
- */
189
- normalizeBaseUrl(baseUrl) {
190
- if (!baseUrl || baseUrl === "/") {
191
- return "";
76
+ normalizeBaseUrl(baseUrl) {
77
+ if (!baseUrl || "/" === baseUrl) return "";
78
+ let normalized = baseUrl;
79
+ if (!normalized.startsWith("/")) normalized = "/" + normalized;
80
+ if (normalized.endsWith("/")) normalized = normalized.slice(0, -1);
81
+ return normalized;
192
82
  }
193
- let normalized = baseUrl;
194
-
195
- // Ensure it starts with /
196
- if (!normalized.startsWith("/")) {
197
- normalized = "/" + normalized;
83
+ compilePattern() {
84
+ const keys = [];
85
+ let regexPattern = this.pattern.replace(/[.+?^${}()|[\]\\]/g, "\\$&").replace(/\\\*/g, ".*");
86
+ regexPattern = regexPattern.replace(/:([a-zA-Z_][a-zA-Z0-9_]*)/g, (_, key)=>{
87
+ keys.push(key);
88
+ return "([^/]+)";
89
+ });
90
+ const regex = new RegExp(`^${regexPattern}$`);
91
+ return {
92
+ regex,
93
+ keys
94
+ };
198
95
  }
199
-
200
- // Remove trailing /
201
- if (normalized.endsWith("/")) {
202
- normalized = normalized.slice(0, -1);
203
- }
204
- return normalized;
205
- }
206
-
207
- /**
208
- * Compile the instance's pattern into a regex and extract parameter names.
209
- */
210
- compilePattern() {
211
- const keys = [];
212
-
213
- // Escape special regex characters except for :param syntax
214
- let regexPattern = this.pattern.replace(/[.+?^${}()|[\]\\]/g, "\\$&") // Escape special chars
215
- .replace(/\\\*/g, ".*"); // Handle * wildcard (already escaped by above)
216
-
217
- // Replace :param with capture groups
218
- regexPattern = regexPattern.replace(/:([a-zA-Z_][a-zA-Z0-9_]*)/g, (_, key) => {
219
- keys.push(key);
220
- return "([^/]+)"; // Match any non-slash characters
221
- });
222
-
223
- // Create regex that matches the entire pathname
224
- const regex = new RegExp(`^${regexPattern}$`);
225
- return {
226
- regex,
227
- keys
228
- };
229
- }
230
96
  }
97
+ export { RouteUrl };
231
98
 
232
99
  //# sourceMappingURL=RouteUrl.js.map
@@ -1 +1 @@
1
- {"version":3,"names":["RouteUrl","patternCache","Map","constructor","pattern","params","baseUrl","normalizeBaseUrl","fromPattern","routeUrl","generate","match","pathname","undefined","matchPathname","pathKeys","extractPathKeys","url","replacePathParams","queryString","buildQueryString","startsWith","slice","length","compiled","get","compilePattern","set","regex","keys","forEach","key","index","value","decodeURIComponent","Set","paramRegex","exec","add","replace","_","encodeURIComponent","String","queryParams","has","Object","searchParams","URLSearchParams","Array","isArray","v","append","toString","normalized","endsWith","regexPattern","push","RegExp"],"sources":["RouteUrl.ts"],"sourcesContent":["interface Params {\n [key: string]: any;\n}\n\ninterface MatchResult {\n params: Record<string, string>;\n}\n\n/**\n * Utility class for route URL pattern matching and generation.\n * Handles simple route patterns with :param syntax (e.g., /cms/entries/:modelId).\n *\n * Public API uses static methods, which internally construct RouteUrl instances\n * to handle the logic through proper instance methods.\n */\nexport class RouteUrl {\n private static patternCache = new Map<string, { regex: RegExp; keys: string[] }>();\n\n private readonly pattern: string;\n private readonly params: Params | undefined;\n private readonly baseUrl: string;\n\n private constructor(pattern: string, params?: Params, baseUrl?: string) {\n this.pattern = pattern;\n this.params = params;\n this.baseUrl = this.normalizeBaseUrl(baseUrl);\n }\n\n /**\n * Generate a URL from a route pattern and parameters.\n * Path parameters are injected into the pattern, remaining parameters become query strings.\n *\n * @param pattern - Route pattern (e.g., '/security/api-keys/:id')\n * @param params - Parameters object (e.g., { id: 123, new: true })\n * @param baseUrl - Optional base URL to prepend (e.g., '/tenant123')\n * @returns Generated URL string\n *\n * @example\n * RouteUrl.fromPattern('/security/api-keys/:id', { id: 123, new: true })\n * // Returns: '/security/api-keys/123?new=true'\n *\n * @example\n * RouteUrl.fromPattern('/file-manager', {}, '/tenant123')\n * // Returns: '/tenant123/file-manager'\n */\n static fromPattern(pattern: string, params?: Params, baseUrl?: string): string {\n const routeUrl = new RouteUrl(pattern, params, baseUrl);\n return routeUrl.generate();\n }\n\n /**\n * Match a pathname against a route pattern and extract parameters.\n * If baseUrl is provided, it will be stripped from the pathname before matching.\n *\n * @param pathname - The pathname to match (e.g., '/tenant123/security/api-keys/123')\n * @param pattern - Route pattern (e.g., '/security/api-keys/:id')\n * @param baseUrl - Optional base URL to strip before matching (e.g., '/tenant123')\n * @returns Object with extracted params if matched, null otherwise\n *\n * @example\n * RouteUrl.match('/security/api-keys/123', '/security/api-keys/:id')\n * // Returns: { params: { id: '123' } }\n *\n * @example\n * RouteUrl.match('/tenant123/file-manager', '/file-manager', '/tenant123')\n * // Returns: { params: {} }\n */\n static match(pathname: string, pattern: string, baseUrl?: string): MatchResult | null {\n const routeUrl = new RouteUrl(pattern, undefined, baseUrl);\n return routeUrl.matchPathname(pathname);\n }\n\n /**\n * Generate a URL from the instance's pattern, params, and baseUrl.\n */\n private generate(): string {\n if (!this.params) {\n return this.baseUrl + this.pattern;\n }\n\n // Extract path parameter names from pattern\n const pathKeys = this.extractPathKeys();\n\n // Replace :param with actual values\n let url = this.replacePathParams();\n\n // Handle empty url\n if (!url || url === \"\") {\n url = \"/\";\n }\n\n // Prepend baseUrl\n url = this.baseUrl + url;\n\n // Add query parameters for keys not used in the path\n const queryString = this.buildQueryString(pathKeys);\n if (queryString) {\n url += `?${queryString}`;\n }\n\n return url;\n }\n\n /**\n * Match a pathname against the instance's pattern.\n */\n private matchPathname(pathname: string): MatchResult | null {\n // Strip baseUrl from pathname if provided\n if (this.baseUrl && pathname.startsWith(this.baseUrl)) {\n pathname = pathname.slice(this.baseUrl.length) || \"/\";\n } else if (this.baseUrl) {\n // pathname doesn't start with baseUrl, no match\n return null;\n }\n\n // Handle wildcard pattern\n if (this.pattern === \"*\" || this.pattern === \"(.*)\") {\n return { params: {} };\n }\n\n // Get or create cached pattern\n let compiled = RouteUrl.patternCache.get(this.pattern);\n if (!compiled) {\n compiled = this.compilePattern();\n RouteUrl.patternCache.set(this.pattern, compiled);\n }\n\n const match = pathname.match(compiled.regex);\n if (!match) {\n return null;\n }\n\n // Extract parameters from capture groups\n const params: Record<string, string> = {};\n compiled.keys.forEach((key, index) => {\n const value = match[index + 1];\n if (value !== undefined) {\n params[key] = decodeURIComponent(value);\n }\n });\n\n return { params };\n }\n\n /**\n * Extract path parameter names from the pattern.\n */\n private extractPathKeys(): Set<string> {\n const pathKeys = new Set<string>();\n const paramRegex = /:([a-zA-Z_][a-zA-Z0-9_]*)/g;\n let match;\n while ((match = paramRegex.exec(this.pattern)) !== null) {\n pathKeys.add(match[1]);\n }\n return pathKeys;\n }\n\n /**\n * Replace :param placeholders with actual values from params.\n */\n private replacePathParams(): string {\n return this.pattern.replace(/:([a-zA-Z_][a-zA-Z0-9_]*)/g, (_, key) => {\n const value = this.params![key];\n if (value === undefined || value === null) {\n return \"\";\n }\n return encodeURIComponent(String(value));\n });\n }\n\n /**\n * Build query string from params that are not path parameters.\n */\n private buildQueryString(pathKeys: Set<string>): string {\n const queryParams: Record<string, any> = {};\n\n for (const key in this.params) {\n if (!pathKeys.has(key)) {\n queryParams[key] = this.params[key];\n }\n }\n\n if (Object.keys(queryParams).length === 0) {\n return \"\";\n }\n\n const searchParams = new URLSearchParams();\n\n for (const key in queryParams) {\n const value = queryParams[key];\n if (value !== undefined && value !== null) {\n if (Array.isArray(value)) {\n value.forEach(v => searchParams.append(key, String(v)));\n } else {\n searchParams.append(key, String(value));\n }\n }\n }\n\n return searchParams.toString();\n }\n\n /**\n * Normalize baseUrl by ensuring it starts with / and doesn't end with /\n */\n private normalizeBaseUrl(baseUrl?: string): string {\n if (!baseUrl || baseUrl === \"/\") {\n return \"\";\n }\n\n let normalized = baseUrl;\n\n // Ensure it starts with /\n if (!normalized.startsWith(\"/\")) {\n normalized = \"/\" + normalized;\n }\n\n // Remove trailing /\n if (normalized.endsWith(\"/\")) {\n normalized = normalized.slice(0, -1);\n }\n\n return normalized;\n }\n\n /**\n * Compile the instance's pattern into a regex and extract parameter names.\n */\n private compilePattern(): { regex: RegExp; keys: string[] } {\n const keys: string[] = [];\n\n // Escape special regex characters except for :param syntax\n let regexPattern = this.pattern\n .replace(/[.+?^${}()|[\\]\\\\]/g, \"\\\\$&\") // Escape special chars\n .replace(/\\\\\\*/g, \".*\"); // Handle * wildcard (already escaped by above)\n\n // Replace :param with capture groups\n regexPattern = regexPattern.replace(/:([a-zA-Z_][a-zA-Z0-9_]*)/g, (_, key) => {\n keys.push(key);\n return \"([^/]+)\"; // Match any non-slash characters\n });\n\n // Create regex that matches the entire pathname\n const regex = new RegExp(`^${regexPattern}$`);\n\n return { regex, keys };\n }\n}\n"],"mappings":"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","ignoreList":[]}
1
+ {"version":3,"file":"features/router/RouteUrl.js","sources":["../../../src/features/router/RouteUrl.ts"],"sourcesContent":["interface Params {\n [key: string]: any;\n}\n\ninterface MatchResult {\n params: Record<string, string>;\n}\n\n/**\n * Utility class for route URL pattern matching and generation.\n * Handles simple route patterns with :param syntax (e.g., /cms/entries/:modelId).\n *\n * Public API uses static methods, which internally construct RouteUrl instances\n * to handle the logic through proper instance methods.\n */\nexport class RouteUrl {\n private static patternCache = new Map<string, { regex: RegExp; keys: string[] }>();\n\n private readonly pattern: string;\n private readonly params: Params | undefined;\n private readonly baseUrl: string;\n\n private constructor(pattern: string, params?: Params, baseUrl?: string) {\n this.pattern = pattern;\n this.params = params;\n this.baseUrl = this.normalizeBaseUrl(baseUrl);\n }\n\n /**\n * Generate a URL from a route pattern and parameters.\n * Path parameters are injected into the pattern, remaining parameters become query strings.\n *\n * @param pattern - Route pattern (e.g., '/security/api-keys/:id')\n * @param params - Parameters object (e.g., { id: 123, new: true })\n * @param baseUrl - Optional base URL to prepend (e.g., '/tenant123')\n * @returns Generated URL string\n *\n * @example\n * RouteUrl.fromPattern('/security/api-keys/:id', { id: 123, new: true })\n * // Returns: '/security/api-keys/123?new=true'\n *\n * @example\n * RouteUrl.fromPattern('/file-manager', {}, '/tenant123')\n * // Returns: '/tenant123/file-manager'\n */\n static fromPattern(pattern: string, params?: Params, baseUrl?: string): string {\n const routeUrl = new RouteUrl(pattern, params, baseUrl);\n return routeUrl.generate();\n }\n\n /**\n * Match a pathname against a route pattern and extract parameters.\n * If baseUrl is provided, it will be stripped from the pathname before matching.\n *\n * @param pathname - The pathname to match (e.g., '/tenant123/security/api-keys/123')\n * @param pattern - Route pattern (e.g., '/security/api-keys/:id')\n * @param baseUrl - Optional base URL to strip before matching (e.g., '/tenant123')\n * @returns Object with extracted params if matched, null otherwise\n *\n * @example\n * RouteUrl.match('/security/api-keys/123', '/security/api-keys/:id')\n * // Returns: { params: { id: '123' } }\n *\n * @example\n * RouteUrl.match('/tenant123/file-manager', '/file-manager', '/tenant123')\n * // Returns: { params: {} }\n */\n static match(pathname: string, pattern: string, baseUrl?: string): MatchResult | null {\n const routeUrl = new RouteUrl(pattern, undefined, baseUrl);\n return routeUrl.matchPathname(pathname);\n }\n\n /**\n * Generate a URL from the instance's pattern, params, and baseUrl.\n */\n private generate(): string {\n if (!this.params) {\n return this.baseUrl + this.pattern;\n }\n\n // Extract path parameter names from pattern\n const pathKeys = this.extractPathKeys();\n\n // Replace :param with actual values\n let url = this.replacePathParams();\n\n // Handle empty url\n if (!url || url === \"\") {\n url = \"/\";\n }\n\n // Prepend baseUrl\n url = this.baseUrl + url;\n\n // Add query parameters for keys not used in the path\n const queryString = this.buildQueryString(pathKeys);\n if (queryString) {\n url += `?${queryString}`;\n }\n\n return url;\n }\n\n /**\n * Match a pathname against the instance's pattern.\n */\n private matchPathname(pathname: string): MatchResult | null {\n // Strip baseUrl from pathname if provided\n if (this.baseUrl && pathname.startsWith(this.baseUrl)) {\n pathname = pathname.slice(this.baseUrl.length) || \"/\";\n } else if (this.baseUrl) {\n // pathname doesn't start with baseUrl, no match\n return null;\n }\n\n // Handle wildcard pattern\n if (this.pattern === \"*\" || this.pattern === \"(.*)\") {\n return { params: {} };\n }\n\n // Get or create cached pattern\n let compiled = RouteUrl.patternCache.get(this.pattern);\n if (!compiled) {\n compiled = this.compilePattern();\n RouteUrl.patternCache.set(this.pattern, compiled);\n }\n\n const match = pathname.match(compiled.regex);\n if (!match) {\n return null;\n }\n\n // Extract parameters from capture groups\n const params: Record<string, string> = {};\n compiled.keys.forEach((key, index) => {\n const value = match[index + 1];\n if (value !== undefined) {\n params[key] = decodeURIComponent(value);\n }\n });\n\n return { params };\n }\n\n /**\n * Extract path parameter names from the pattern.\n */\n private extractPathKeys(): Set<string> {\n const pathKeys = new Set<string>();\n const paramRegex = /:([a-zA-Z_][a-zA-Z0-9_]*)/g;\n let match;\n while ((match = paramRegex.exec(this.pattern)) !== null) {\n pathKeys.add(match[1]);\n }\n return pathKeys;\n }\n\n /**\n * Replace :param placeholders with actual values from params.\n */\n private replacePathParams(): string {\n return this.pattern.replace(/:([a-zA-Z_][a-zA-Z0-9_]*)/g, (_, key) => {\n const value = this.params![key];\n if (value === undefined || value === null) {\n return \"\";\n }\n return encodeURIComponent(String(value));\n });\n }\n\n /**\n * Build query string from params that are not path parameters.\n */\n private buildQueryString(pathKeys: Set<string>): string {\n const queryParams: Record<string, any> = {};\n\n for (const key in this.params) {\n if (!pathKeys.has(key)) {\n queryParams[key] = this.params[key];\n }\n }\n\n if (Object.keys(queryParams).length === 0) {\n return \"\";\n }\n\n const searchParams = new URLSearchParams();\n\n for (const key in queryParams) {\n const value = queryParams[key];\n if (value !== undefined && value !== null) {\n if (Array.isArray(value)) {\n value.forEach(v => searchParams.append(key, String(v)));\n } else {\n searchParams.append(key, String(value));\n }\n }\n }\n\n return searchParams.toString();\n }\n\n /**\n * Normalize baseUrl by ensuring it starts with / and doesn't end with /\n */\n private normalizeBaseUrl(baseUrl?: string): string {\n if (!baseUrl || baseUrl === \"/\") {\n return \"\";\n }\n\n let normalized = baseUrl;\n\n // Ensure it starts with /\n if (!normalized.startsWith(\"/\")) {\n normalized = \"/\" + normalized;\n }\n\n // Remove trailing /\n if (normalized.endsWith(\"/\")) {\n normalized = normalized.slice(0, -1);\n }\n\n return normalized;\n }\n\n /**\n * Compile the instance's pattern into a regex and extract parameter names.\n */\n private compilePattern(): { regex: RegExp; keys: string[] } {\n const keys: string[] = [];\n\n // Escape special regex characters except for :param syntax\n let regexPattern = this.pattern\n .replace(/[.+?^${}()|[\\]\\\\]/g, \"\\\\$&\") // Escape special chars\n .replace(/\\\\\\*/g, \".*\"); // Handle * wildcard (already escaped by above)\n\n // Replace :param with capture groups\n regexPattern = regexPattern.replace(/:([a-zA-Z_][a-zA-Z0-9_]*)/g, (_, key) => {\n keys.push(key);\n return \"([^/]+)\"; // Match any non-slash characters\n });\n\n // Create regex that matches the entire pathname\n const regex = new RegExp(`^${regexPattern}$`);\n\n return { regex, keys };\n }\n}\n"],"names":["RouteUrl","Map","pattern","params","baseUrl","routeUrl","pathname","undefined","pathKeys","url","queryString","compiled","match","key","index","value","decodeURIComponent","Set","paramRegex","_","encodeURIComponent","String","queryParams","Object","searchParams","URLSearchParams","Array","v","normalized","keys","regexPattern","regex","RegExp"],"mappings":"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"}
@@ -1,129 +1,72 @@
1
1
  import { RouteUrl } from "./RouteUrl.js";
2
- /**
3
- * Router manages a collection of route definitions and resolves pathnames to matched routes.
4
- */
5
- export class Router {
6
- routes = [];
7
- constructor(baseUrl = "") {
8
- this.baseUrl = baseUrl;
9
- }
10
-
11
- /**
12
- * Set or update route definitions. Routes with the same name will be replaced.
13
- *
14
- * @param routes - Array of route definitions to add
15
- */
16
- setRoutes(routes) {
17
- routes.forEach(route => {
18
- const index = this.routes.findIndex(r => r.name === route.name);
19
- if (index > -1) {
20
- this.routes[index] = route;
21
- } else {
22
- this.routes.push(route);
23
- }
24
- });
25
- this.sortRoutes();
26
- }
27
-
28
- /**
29
- * Find a route by name.
30
- *
31
- * @param name - Route name to find
32
- * @returns Route definition if found, undefined otherwise
33
- */
34
- findRoute(name) {
35
- return this.routes.find(r => r.name === name);
36
- }
37
-
38
- /**
39
- * Resolve a pathname to a matched route.
40
- * The pathname will be matched against patterns after stripping the baseUrl.
41
- *
42
- * @param pathname - The pathname to resolve (e.g., '/tenant123/cms/entries/123')
43
- * @param queryParams - Optional query parameters to merge into route params
44
- * @returns Matched route with onMatch callback, or null if no route matches
45
- */
46
- resolve(pathname, queryParams) {
47
- for (const route of this.routes) {
48
- const pattern = route.path === "*" ? "(.*)" : route.path;
49
- const matchResult = RouteUrl.match(pathname, pattern, this.baseUrl);
50
- if (matchResult) {
51
- const matchedRoute = {
52
- name: route.name,
53
- path: route.path,
54
- pathname,
55
- params: {
56
- ...matchResult.params,
57
- ...(queryParams || {})
58
- }
59
- };
60
- const onMatch = matched => {
61
- route.onMatch(matched);
62
- };
63
- return {
64
- matchedRoute,
65
- onMatch
66
- };
67
- }
2
+ class Router {
3
+ constructor(baseUrl = ""){
4
+ this.routes = [];
5
+ this.baseUrl = baseUrl;
6
+ }
7
+ setRoutes(routes) {
8
+ routes.forEach((route)=>{
9
+ const index = this.routes.findIndex((r)=>r.name === route.name);
10
+ if (index > -1) this.routes[index] = route;
11
+ else this.routes.push(route);
12
+ });
13
+ this.sortRoutes();
14
+ }
15
+ findRoute(name) {
16
+ return this.routes.find((r)=>r.name === name);
17
+ }
18
+ resolve(pathname, queryParams) {
19
+ for (const route of this.routes){
20
+ const pattern = "*" === route.path ? "(.*)" : route.path;
21
+ const matchResult = RouteUrl.match(pathname, pattern, this.baseUrl);
22
+ if (matchResult) {
23
+ const matchedRoute = {
24
+ name: route.name,
25
+ path: route.path,
26
+ pathname,
27
+ params: {
28
+ ...matchResult.params,
29
+ ...queryParams || {}
30
+ }
31
+ };
32
+ const onMatch = (matched)=>{
33
+ route.onMatch(matched);
34
+ };
35
+ return {
36
+ matchedRoute,
37
+ onMatch
38
+ };
39
+ }
40
+ }
41
+ return null;
42
+ }
43
+ getBaseUrl() {
44
+ return this.baseUrl;
45
+ }
46
+ sortRoutes() {
47
+ const INDEX_PATH = "/";
48
+ const WILDCARD_PATTERNS = [
49
+ "*",
50
+ "(.*)"
51
+ ];
52
+ this.routes.sort((a, b)=>{
53
+ const pathA = a.path || "*";
54
+ const pathB = b.path || "*";
55
+ const isWildcardA = WILDCARD_PATTERNS.includes(pathA);
56
+ const isWildcardB = WILDCARD_PATTERNS.includes(pathB);
57
+ const isIndexA = pathA === INDEX_PATH;
58
+ const isIndexB = pathB === INDEX_PATH;
59
+ if (isWildcardA && isWildcardB || isIndexA && isIndexB) return 0;
60
+ if (isIndexA && isWildcardB) return -1;
61
+ if (isWildcardA && isIndexB) return 1;
62
+ if (isWildcardA) return 1;
63
+ if (isWildcardB) return -1;
64
+ if (isIndexA) return 1;
65
+ if (isIndexB) return -1;
66
+ return 0;
67
+ });
68
68
  }
69
- return null;
70
- }
71
-
72
- /**
73
- * Get the current baseUrl.
74
- */
75
- getBaseUrl() {
76
- return this.baseUrl;
77
- }
78
-
79
- /**
80
- * Sort routes to ensure proper matching order.
81
- * Wildcard routes (*) and index routes (/) are moved to the bottom.
82
- */
83
- sortRoutes() {
84
- const INDEX_PATH = "/";
85
- const WILDCARD_PATTERNS = ["*", "(.*)"];
86
- this.routes.sort((a, b) => {
87
- const pathA = a.path || "*";
88
- const pathB = b.path || "*";
89
- const isWildcardA = WILDCARD_PATTERNS.includes(pathA);
90
- const isWildcardB = WILDCARD_PATTERNS.includes(pathB);
91
- const isIndexA = pathA === INDEX_PATH;
92
- const isIndexB = pathB === INDEX_PATH;
93
-
94
- // Both are wildcards or both are index - maintain order
95
- if (isWildcardA && isWildcardB || isIndexA && isIndexB) {
96
- return 0;
97
- }
98
-
99
- // Index path should come before wildcard
100
- if (isIndexA && isWildcardB) {
101
- return -1;
102
- }
103
- if (isWildcardA && isIndexB) {
104
- return 1;
105
- }
106
-
107
- // Push wildcards to the bottom
108
- if (isWildcardA) {
109
- return 1;
110
- }
111
- if (isWildcardB) {
112
- return -1;
113
- }
114
-
115
- // Push index to the bottom (but above wildcards)
116
- if (isIndexA) {
117
- return 1;
118
- }
119
- if (isIndexB) {
120
- return -1;
121
- }
122
-
123
- // All other routes maintain their order
124
- return 0;
125
- });
126
- }
127
69
  }
70
+ export { Router };
128
71
 
129
72
  //# sourceMappingURL=Router.js.map
@@ -1 +1 @@
1
- {"version":3,"names":["RouteUrl","Router","routes","constructor","baseUrl","setRoutes","forEach","route","index","findIndex","r","name","push","sortRoutes","findRoute","find","resolve","pathname","queryParams","pattern","path","matchResult","match","matchedRoute","params","onMatch","matched","getBaseUrl","INDEX_PATH","WILDCARD_PATTERNS","sort","a","b","pathA","pathB","isWildcardA","includes","isWildcardB","isIndexA","isIndexB"],"sources":["Router.ts"],"sourcesContent":["import type { RouteDefinition, MatchedRoute } from \"./abstractions.js\";\nimport { RouteUrl } from \"./RouteUrl.js\";\n\ninterface ResolveResult {\n matchedRoute: MatchedRoute;\n onMatch: (route: MatchedRoute) => void;\n}\n\n/**\n * Router manages a collection of route definitions and resolves pathnames to matched routes.\n */\nexport class Router {\n private routes: RouteDefinition[] = [];\n private baseUrl: string;\n\n constructor(baseUrl: string = \"\") {\n this.baseUrl = baseUrl;\n }\n\n /**\n * Set or update route definitions. Routes with the same name will be replaced.\n *\n * @param routes - Array of route definitions to add\n */\n setRoutes(routes: RouteDefinition[]): void {\n routes.forEach(route => {\n const index = this.routes.findIndex(r => r.name === route.name);\n\n if (index > -1) {\n this.routes[index] = route;\n } else {\n this.routes.push(route);\n }\n });\n\n this.sortRoutes();\n }\n\n /**\n * Find a route by name.\n *\n * @param name - Route name to find\n * @returns Route definition if found, undefined otherwise\n */\n findRoute(name: string): RouteDefinition | undefined {\n return this.routes.find(r => r.name === name);\n }\n\n /**\n * Resolve a pathname to a matched route.\n * The pathname will be matched against patterns after stripping the baseUrl.\n *\n * @param pathname - The pathname to resolve (e.g., '/tenant123/cms/entries/123')\n * @param queryParams - Optional query parameters to merge into route params\n * @returns Matched route with onMatch callback, or null if no route matches\n */\n resolve(pathname: string, queryParams?: Record<string, unknown>): ResolveResult | null {\n for (const route of this.routes) {\n const pattern = route.path === \"*\" ? \"(.*)\" : route.path;\n const matchResult = RouteUrl.match(pathname, pattern, this.baseUrl);\n\n if (matchResult) {\n const matchedRoute: MatchedRoute = {\n name: route.name,\n path: route.path,\n pathname,\n params: { ...matchResult.params, ...(queryParams || {}) }\n };\n\n const onMatch = (matched: MatchedRoute) => {\n route.onMatch(matched);\n };\n\n return { matchedRoute, onMatch };\n }\n }\n\n return null;\n }\n\n /**\n * Get the current baseUrl.\n */\n getBaseUrl(): string {\n return this.baseUrl;\n }\n\n /**\n * Sort routes to ensure proper matching order.\n * Wildcard routes (*) and index routes (/) are moved to the bottom.\n */\n private sortRoutes(): void {\n const INDEX_PATH = \"/\";\n const WILDCARD_PATTERNS = [\"*\", \"(.*)\"];\n\n this.routes.sort((a, b) => {\n const pathA = a.path || \"*\";\n const pathB = b.path || \"*\";\n\n const isWildcardA = WILDCARD_PATTERNS.includes(pathA);\n const isWildcardB = WILDCARD_PATTERNS.includes(pathB);\n const isIndexA = pathA === INDEX_PATH;\n const isIndexB = pathB === INDEX_PATH;\n\n // Both are wildcards or both are index - maintain order\n if ((isWildcardA && isWildcardB) || (isIndexA && isIndexB)) {\n return 0;\n }\n\n // Index path should come before wildcard\n if (isIndexA && isWildcardB) {\n return -1;\n }\n if (isWildcardA && isIndexB) {\n return 1;\n }\n\n // Push wildcards to the bottom\n if (isWildcardA) {\n return 1;\n }\n if (isWildcardB) {\n return -1;\n }\n\n // Push index to the bottom (but above wildcards)\n if (isIndexA) {\n return 1;\n }\n if (isIndexB) {\n return -1;\n }\n\n // All other routes maintain their order\n return 0;\n });\n }\n}\n"],"mappings":"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","ignoreList":[]}
1
+ {"version":3,"file":"features/router/Router.js","sources":["../../../src/features/router/Router.ts"],"sourcesContent":["import type { RouteDefinition, MatchedRoute } from \"./abstractions.js\";\nimport { RouteUrl } from \"./RouteUrl.js\";\n\ninterface ResolveResult {\n matchedRoute: MatchedRoute;\n onMatch: (route: MatchedRoute) => void;\n}\n\n/**\n * Router manages a collection of route definitions and resolves pathnames to matched routes.\n */\nexport class Router {\n private routes: RouteDefinition[] = [];\n private baseUrl: string;\n\n constructor(baseUrl: string = \"\") {\n this.baseUrl = baseUrl;\n }\n\n /**\n * Set or update route definitions. Routes with the same name will be replaced.\n *\n * @param routes - Array of route definitions to add\n */\n setRoutes(routes: RouteDefinition[]): void {\n routes.forEach(route => {\n const index = this.routes.findIndex(r => r.name === route.name);\n\n if (index > -1) {\n this.routes[index] = route;\n } else {\n this.routes.push(route);\n }\n });\n\n this.sortRoutes();\n }\n\n /**\n * Find a route by name.\n *\n * @param name - Route name to find\n * @returns Route definition if found, undefined otherwise\n */\n findRoute(name: string): RouteDefinition | undefined {\n return this.routes.find(r => r.name === name);\n }\n\n /**\n * Resolve a pathname to a matched route.\n * The pathname will be matched against patterns after stripping the baseUrl.\n *\n * @param pathname - The pathname to resolve (e.g., '/tenant123/cms/entries/123')\n * @param queryParams - Optional query parameters to merge into route params\n * @returns Matched route with onMatch callback, or null if no route matches\n */\n resolve(pathname: string, queryParams?: Record<string, unknown>): ResolveResult | null {\n for (const route of this.routes) {\n const pattern = route.path === \"*\" ? \"(.*)\" : route.path;\n const matchResult = RouteUrl.match(pathname, pattern, this.baseUrl);\n\n if (matchResult) {\n const matchedRoute: MatchedRoute = {\n name: route.name,\n path: route.path,\n pathname,\n params: { ...matchResult.params, ...(queryParams || {}) }\n };\n\n const onMatch = (matched: MatchedRoute) => {\n route.onMatch(matched);\n };\n\n return { matchedRoute, onMatch };\n }\n }\n\n return null;\n }\n\n /**\n * Get the current baseUrl.\n */\n getBaseUrl(): string {\n return this.baseUrl;\n }\n\n /**\n * Sort routes to ensure proper matching order.\n * Wildcard routes (*) and index routes (/) are moved to the bottom.\n */\n private sortRoutes(): void {\n const INDEX_PATH = \"/\";\n const WILDCARD_PATTERNS = [\"*\", \"(.*)\"];\n\n this.routes.sort((a, b) => {\n const pathA = a.path || \"*\";\n const pathB = b.path || \"*\";\n\n const isWildcardA = WILDCARD_PATTERNS.includes(pathA);\n const isWildcardB = WILDCARD_PATTERNS.includes(pathB);\n const isIndexA = pathA === INDEX_PATH;\n const isIndexB = pathB === INDEX_PATH;\n\n // Both are wildcards or both are index - maintain order\n if ((isWildcardA && isWildcardB) || (isIndexA && isIndexB)) {\n return 0;\n }\n\n // Index path should come before wildcard\n if (isIndexA && isWildcardB) {\n return -1;\n }\n if (isWildcardA && isIndexB) {\n return 1;\n }\n\n // Push wildcards to the bottom\n if (isWildcardA) {\n return 1;\n }\n if (isWildcardB) {\n return -1;\n }\n\n // Push index to the bottom (but above wildcards)\n if (isIndexA) {\n return 1;\n }\n if (isIndexB) {\n return -1;\n }\n\n // All other routes maintain their order\n return 0;\n });\n }\n}\n"],"names":["Router","baseUrl","routes","route","index","r","name","pathname","queryParams","pattern","matchResult","RouteUrl","matchedRoute","onMatch","matched","INDEX_PATH","WILDCARD_PATTERNS","a","b","pathA","pathB","isWildcardA","isWildcardB","isIndexA","isIndexB"],"mappings":";AAWO,MAAMA;IAIT,YAAYC,UAAkB,EAAE,CAAE;aAH1B,MAAM,GAAsB,EAAE;QAIlC,IAAI,CAAC,OAAO,GAAGA;IACnB;IAOA,UAAUC,MAAyB,EAAQ;QACvCA,OAAO,OAAO,CAACC,CAAAA;YACX,MAAMC,QAAQ,IAAI,CAAC,MAAM,CAAC,SAAS,CAACC,CAAAA,IAAKA,EAAE,IAAI,KAAKF,MAAM,IAAI;YAE9D,IAAIC,QAAQ,IACR,IAAI,CAAC,MAAM,CAACA,MAAM,GAAGD;iBAErB,IAAI,CAAC,MAAM,CAAC,IAAI,CAACA;QAEzB;QAEA,IAAI,CAAC,UAAU;IACnB;IAQA,UAAUG,IAAY,EAA+B;QACjD,OAAO,IAAI,CAAC,MAAM,CAAC,IAAI,CAACD,CAAAA,IAAKA,EAAE,IAAI,KAAKC;IAC5C;IAUA,QAAQC,QAAgB,EAAEC,WAAqC,EAAwB;QACnF,KAAK,MAAML,SAAS,IAAI,CAAC,MAAM,CAAE;YAC7B,MAAMM,UAAUN,AAAe,QAAfA,MAAM,IAAI,GAAW,SAASA,MAAM,IAAI;YACxD,MAAMO,cAAcC,SAAS,KAAK,CAACJ,UAAUE,SAAS,IAAI,CAAC,OAAO;YAElE,IAAIC,aAAa;gBACb,MAAME,eAA6B;oBAC/B,MAAMT,MAAM,IAAI;oBAChB,MAAMA,MAAM,IAAI;oBAChBI;oBACA,QAAQ;wBAAE,GAAGG,YAAY,MAAM;wBAAE,GAAIF,eAAe,CAAC,CAAC;oBAAE;gBAC5D;gBAEA,MAAMK,UAAU,CAACC;oBACbX,MAAM,OAAO,CAACW;gBAClB;gBAEA,OAAO;oBAAEF;oBAAcC;gBAAQ;YACnC;QACJ;QAEA,OAAO;IACX;IAKA,aAAqB;QACjB,OAAO,IAAI,CAAC,OAAO;IACvB;IAMQ,aAAmB;QACvB,MAAME,aAAa;QACnB,MAAMC,oBAAoB;YAAC;YAAK;SAAO;QAEvC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAACC,GAAGC;YACjB,MAAMC,QAAQF,EAAE,IAAI,IAAI;YACxB,MAAMG,QAAQF,EAAE,IAAI,IAAI;YAExB,MAAMG,cAAcL,kBAAkB,QAAQ,CAACG;YAC/C,MAAMG,cAAcN,kBAAkB,QAAQ,CAACI;YAC/C,MAAMG,WAAWJ,UAAUJ;YAC3B,MAAMS,WAAWJ,UAAUL;YAG3B,IAAKM,eAAeC,eAAiBC,YAAYC,UAC7C,OAAO;YAIX,IAAID,YAAYD,aACZ,OAAO;YAEX,IAAID,eAAeG,UACf,OAAO;YAIX,IAAIH,aACA,OAAO;YAEX,IAAIC,aACA,OAAO;YAIX,IAAIC,UACA,OAAO;YAEX,IAAIC,UACA,OAAO;YAIX,OAAO;QACX;IACJ;AACJ"}