@rebasepro/server-core 0.0.1-canary.4d4fb3e

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 (254) hide show
  1. package/LICENSE +6 -0
  2. package/README.md +40 -0
  3. package/build-errors.txt +52 -0
  4. package/coverage/clover.xml +3739 -0
  5. package/coverage/coverage-final.json +31 -0
  6. package/coverage/lcov-report/base.css +224 -0
  7. package/coverage/lcov-report/block-navigation.js +87 -0
  8. package/coverage/lcov-report/favicon.png +0 -0
  9. package/coverage/lcov-report/index.html +266 -0
  10. package/coverage/lcov-report/prettify.css +1 -0
  11. package/coverage/lcov-report/prettify.js +2 -0
  12. package/coverage/lcov-report/sort-arrow-sprite.png +0 -0
  13. package/coverage/lcov-report/sorter.js +210 -0
  14. package/coverage/lcov-report/src/api/ast-schema-editor.ts.html +952 -0
  15. package/coverage/lcov-report/src/api/errors.ts.html +472 -0
  16. package/coverage/lcov-report/src/api/graphql/graphql-schema-generator.ts.html +1069 -0
  17. package/coverage/lcov-report/src/api/graphql/index.html +116 -0
  18. package/coverage/lcov-report/src/api/index.html +176 -0
  19. package/coverage/lcov-report/src/api/openapi-generator.ts.html +565 -0
  20. package/coverage/lcov-report/src/api/rest/api-generator.ts.html +994 -0
  21. package/coverage/lcov-report/src/api/rest/index.html +131 -0
  22. package/coverage/lcov-report/src/api/rest/query-parser.ts.html +550 -0
  23. package/coverage/lcov-report/src/api/schema-editor-routes.ts.html +202 -0
  24. package/coverage/lcov-report/src/api/server.ts.html +823 -0
  25. package/coverage/lcov-report/src/auth/admin-routes.ts.html +973 -0
  26. package/coverage/lcov-report/src/auth/index.html +176 -0
  27. package/coverage/lcov-report/src/auth/jwt.ts.html +574 -0
  28. package/coverage/lcov-report/src/auth/middleware.ts.html +745 -0
  29. package/coverage/lcov-report/src/auth/password.ts.html +310 -0
  30. package/coverage/lcov-report/src/auth/services.ts.html +2074 -0
  31. package/coverage/lcov-report/src/collections/index.html +116 -0
  32. package/coverage/lcov-report/src/collections/loader.ts.html +232 -0
  33. package/coverage/lcov-report/src/db/auth-schema.ts.html +523 -0
  34. package/coverage/lcov-report/src/db/data-transformer.ts.html +1753 -0
  35. package/coverage/lcov-report/src/db/entityService.ts.html +700 -0
  36. package/coverage/lcov-report/src/db/index.html +146 -0
  37. package/coverage/lcov-report/src/db/services/EntityFetchService.ts.html +4048 -0
  38. package/coverage/lcov-report/src/db/services/EntityPersistService.ts.html +883 -0
  39. package/coverage/lcov-report/src/db/services/RelationService.ts.html +3121 -0
  40. package/coverage/lcov-report/src/db/services/entity-helpers.ts.html +442 -0
  41. package/coverage/lcov-report/src/db/services/index.html +176 -0
  42. package/coverage/lcov-report/src/db/services/index.ts.html +124 -0
  43. package/coverage/lcov-report/src/generate-drizzle-schema-logic.ts.html +1960 -0
  44. package/coverage/lcov-report/src/index.html +116 -0
  45. package/coverage/lcov-report/src/services/driver-registry.ts.html +631 -0
  46. package/coverage/lcov-report/src/services/index.html +131 -0
  47. package/coverage/lcov-report/src/services/postgresDataDriver.ts.html +3025 -0
  48. package/coverage/lcov-report/src/storage/LocalStorageController.ts.html +1189 -0
  49. package/coverage/lcov-report/src/storage/S3StorageController.ts.html +970 -0
  50. package/coverage/lcov-report/src/storage/index.html +161 -0
  51. package/coverage/lcov-report/src/storage/storage-registry.ts.html +646 -0
  52. package/coverage/lcov-report/src/storage/types.ts.html +451 -0
  53. package/coverage/lcov-report/src/utils/drizzle-conditions.ts.html +3082 -0
  54. package/coverage/lcov-report/src/utils/index.html +116 -0
  55. package/coverage/lcov.info +7179 -0
  56. package/dist/common/src/collections/CollectionRegistry.d.ts +48 -0
  57. package/dist/common/src/collections/index.d.ts +1 -0
  58. package/dist/common/src/data/buildRebaseData.d.ts +14 -0
  59. package/dist/common/src/index.d.ts +3 -0
  60. package/dist/common/src/util/builders.d.ts +57 -0
  61. package/dist/common/src/util/callbacks.d.ts +6 -0
  62. package/dist/common/src/util/collections.d.ts +11 -0
  63. package/dist/common/src/util/common.d.ts +2 -0
  64. package/dist/common/src/util/conditions.d.ts +26 -0
  65. package/dist/common/src/util/entities.d.ts +36 -0
  66. package/dist/common/src/util/enums.d.ts +3 -0
  67. package/dist/common/src/util/index.d.ts +16 -0
  68. package/dist/common/src/util/navigation_from_path.d.ts +34 -0
  69. package/dist/common/src/util/navigation_utils.d.ts +20 -0
  70. package/dist/common/src/util/parent_references_from_path.d.ts +6 -0
  71. package/dist/common/src/util/paths.d.ts +14 -0
  72. package/dist/common/src/util/permissions.d.ts +5 -0
  73. package/dist/common/src/util/references.d.ts +2 -0
  74. package/dist/common/src/util/relations.d.ts +12 -0
  75. package/dist/common/src/util/resolutions.d.ts +72 -0
  76. package/dist/common/src/util/storage.d.ts +24 -0
  77. package/dist/index-BeMqpmfQ.js +239 -0
  78. package/dist/index-BeMqpmfQ.js.map +1 -0
  79. package/dist/index-bl4J3lNb.js +55823 -0
  80. package/dist/index-bl4J3lNb.js.map +1 -0
  81. package/dist/index.es.js +58 -0
  82. package/dist/index.es.js.map +1 -0
  83. package/dist/index.umd.js +56062 -0
  84. package/dist/index.umd.js.map +1 -0
  85. package/dist/server-core/src/api/ast-schema-editor.d.ts +21 -0
  86. package/dist/server-core/src/api/collections_for_test/callbacks_test_collection.d.ts +2 -0
  87. package/dist/server-core/src/api/errors.d.ts +35 -0
  88. package/dist/server-core/src/api/graphql/graphql-schema-generator.d.ts +35 -0
  89. package/dist/server-core/src/api/graphql/index.d.ts +1 -0
  90. package/dist/server-core/src/api/index.d.ts +9 -0
  91. package/dist/server-core/src/api/openapi-generator.d.ts +2 -0
  92. package/dist/server-core/src/api/rest/api-generator.d.ts +64 -0
  93. package/dist/server-core/src/api/rest/index.d.ts +1 -0
  94. package/dist/server-core/src/api/rest/query-parser.d.ts +9 -0
  95. package/dist/server-core/src/api/schema-editor-routes.d.ts +3 -0
  96. package/dist/server-core/src/api/server.d.ts +40 -0
  97. package/dist/server-core/src/api/types.d.ts +90 -0
  98. package/dist/server-core/src/auth/admin-routes.d.ts +7 -0
  99. package/dist/server-core/src/auth/google-oauth.d.ts +20 -0
  100. package/dist/server-core/src/auth/index.d.ts +12 -0
  101. package/dist/server-core/src/auth/interfaces.d.ts +270 -0
  102. package/dist/server-core/src/auth/jwt.d.ts +42 -0
  103. package/dist/server-core/src/auth/middleware.d.ts +56 -0
  104. package/dist/server-core/src/auth/password.d.ts +22 -0
  105. package/dist/server-core/src/auth/rate-limiter.d.ts +31 -0
  106. package/dist/server-core/src/auth/routes.d.ts +17 -0
  107. package/dist/server-core/src/bootstrappers/index.d.ts +0 -0
  108. package/dist/server-core/src/collections/BackendCollectionRegistry.d.ts +13 -0
  109. package/dist/server-core/src/collections/loader.d.ts +5 -0
  110. package/dist/server-core/src/db/interfaces.d.ts +18 -0
  111. package/dist/server-core/src/email/index.d.ts +6 -0
  112. package/dist/server-core/src/email/smtp-email-service.d.ts +25 -0
  113. package/dist/server-core/src/email/templates.d.ts +33 -0
  114. package/dist/server-core/src/email/types.d.ts +110 -0
  115. package/dist/server-core/src/functions/function-loader.d.ts +17 -0
  116. package/dist/server-core/src/functions/function-routes.d.ts +10 -0
  117. package/dist/server-core/src/functions/index.d.ts +3 -0
  118. package/dist/server-core/src/history/history-routes.d.ts +23 -0
  119. package/dist/server-core/src/history/index.d.ts +1 -0
  120. package/dist/server-core/src/index.d.ts +24 -0
  121. package/dist/server-core/src/init.d.ts +49 -0
  122. package/dist/server-core/src/serve-spa.d.ts +30 -0
  123. package/dist/server-core/src/services/driver-registry.d.ts +78 -0
  124. package/dist/server-core/src/storage/LocalStorageController.d.ts +46 -0
  125. package/dist/server-core/src/storage/S3StorageController.d.ts +36 -0
  126. package/dist/server-core/src/storage/index.d.ts +18 -0
  127. package/dist/server-core/src/storage/routes.d.ts +38 -0
  128. package/dist/server-core/src/storage/storage-registry.d.ts +78 -0
  129. package/dist/server-core/src/storage/types.d.ts +91 -0
  130. package/dist/server-core/src/types/index.d.ts +11 -0
  131. package/dist/server-core/src/utils/logging.d.ts +9 -0
  132. package/dist/server-core/src/utils/sql.d.ts +27 -0
  133. package/dist/types/src/controllers/analytics_controller.d.ts +7 -0
  134. package/dist/types/src/controllers/auth.d.ts +117 -0
  135. package/dist/types/src/controllers/client.d.ts +58 -0
  136. package/dist/types/src/controllers/collection_registry.d.ts +44 -0
  137. package/dist/types/src/controllers/customization_controller.d.ts +54 -0
  138. package/dist/types/src/controllers/data.d.ts +141 -0
  139. package/dist/types/src/controllers/data_driver.d.ts +168 -0
  140. package/dist/types/src/controllers/database_admin.d.ts +11 -0
  141. package/dist/types/src/controllers/dialogs_controller.d.ts +36 -0
  142. package/dist/types/src/controllers/effective_role.d.ts +4 -0
  143. package/dist/types/src/controllers/index.d.ts +17 -0
  144. package/dist/types/src/controllers/local_config_persistence.d.ts +20 -0
  145. package/dist/types/src/controllers/navigation.d.ts +213 -0
  146. package/dist/types/src/controllers/registry.d.ts +51 -0
  147. package/dist/types/src/controllers/side_dialogs_controller.d.ts +67 -0
  148. package/dist/types/src/controllers/side_entity_controller.d.ts +89 -0
  149. package/dist/types/src/controllers/snackbar.d.ts +24 -0
  150. package/dist/types/src/controllers/storage.d.ts +173 -0
  151. package/dist/types/src/index.d.ts +4 -0
  152. package/dist/types/src/rebase_context.d.ts +101 -0
  153. package/dist/types/src/types/backend.d.ts +533 -0
  154. package/dist/types/src/types/builders.d.ts +14 -0
  155. package/dist/types/src/types/chips.d.ts +5 -0
  156. package/dist/types/src/types/collections.d.ts +812 -0
  157. package/dist/types/src/types/data_source.d.ts +64 -0
  158. package/dist/types/src/types/entities.d.ts +145 -0
  159. package/dist/types/src/types/entity_actions.d.ts +98 -0
  160. package/dist/types/src/types/entity_callbacks.d.ts +173 -0
  161. package/dist/types/src/types/entity_link_builder.d.ts +7 -0
  162. package/dist/types/src/types/entity_overrides.d.ts +9 -0
  163. package/dist/types/src/types/entity_views.d.ts +61 -0
  164. package/dist/types/src/types/export_import.d.ts +21 -0
  165. package/dist/types/src/types/index.d.ts +22 -0
  166. package/dist/types/src/types/locales.d.ts +4 -0
  167. package/dist/types/src/types/modify_collections.d.ts +5 -0
  168. package/dist/types/src/types/plugins.d.ts +225 -0
  169. package/dist/types/src/types/properties.d.ts +1091 -0
  170. package/dist/types/src/types/property_config.d.ts +70 -0
  171. package/dist/types/src/types/relations.d.ts +336 -0
  172. package/dist/types/src/types/slots.d.ts +228 -0
  173. package/dist/types/src/types/translations.d.ts +826 -0
  174. package/dist/types/src/types/user_management_delegate.d.ts +120 -0
  175. package/dist/types/src/types/websockets.d.ts +78 -0
  176. package/dist/types/src/users/index.d.ts +2 -0
  177. package/dist/types/src/users/roles.d.ts +22 -0
  178. package/dist/types/src/users/user.d.ts +46 -0
  179. package/history_diff.log +385 -0
  180. package/jest.config.cjs +16 -0
  181. package/package.json +86 -0
  182. package/scratch.ts +8 -0
  183. package/src/api/ast-schema-editor.ts +289 -0
  184. package/src/api/collections_for_test/callbacks_test_collection.ts +57 -0
  185. package/src/api/errors.ts +155 -0
  186. package/src/api/graphql/graphql-schema-generator.ts +334 -0
  187. package/src/api/graphql/index.ts +2 -0
  188. package/src/api/index.ts +11 -0
  189. package/src/api/openapi-generator.ts +160 -0
  190. package/src/api/rest/api-generator.ts +466 -0
  191. package/src/api/rest/index.ts +2 -0
  192. package/src/api/rest/query-parser.ts +155 -0
  193. package/src/api/schema-editor-routes.ts +39 -0
  194. package/src/api/server.ts +245 -0
  195. package/src/api/types.ts +90 -0
  196. package/src/auth/admin-routes.ts +488 -0
  197. package/src/auth/google-oauth.ts +60 -0
  198. package/src/auth/index.ts +21 -0
  199. package/src/auth/interfaces.ts +316 -0
  200. package/src/auth/jwt.ts +164 -0
  201. package/src/auth/middleware.ts +235 -0
  202. package/src/auth/password.ts +75 -0
  203. package/src/auth/rate-limiter.ts +129 -0
  204. package/src/auth/routes.ts +730 -0
  205. package/src/bootstrappers/index.ts +1 -0
  206. package/src/collections/BackendCollectionRegistry.ts +20 -0
  207. package/src/collections/loader.ts +49 -0
  208. package/src/db/interfaces.ts +60 -0
  209. package/src/email/index.ts +17 -0
  210. package/src/email/smtp-email-service.ts +88 -0
  211. package/src/email/templates.ts +301 -0
  212. package/src/email/types.ts +112 -0
  213. package/src/functions/function-loader.ts +91 -0
  214. package/src/functions/function-routes.ts +31 -0
  215. package/src/functions/index.ts +3 -0
  216. package/src/history/history-routes.ts +128 -0
  217. package/src/history/index.ts +2 -0
  218. package/src/index.ts +56 -0
  219. package/src/init.ts +309 -0
  220. package/src/serve-spa.ts +81 -0
  221. package/src/services/driver-registry.ts +182 -0
  222. package/src/storage/LocalStorageController.ts +368 -0
  223. package/src/storage/S3StorageController.ts +295 -0
  224. package/src/storage/index.ts +32 -0
  225. package/src/storage/routes.ts +247 -0
  226. package/src/storage/storage-registry.ts +187 -0
  227. package/src/storage/types.ts +122 -0
  228. package/src/types/index.ts +27 -0
  229. package/src/utils/logging.ts +35 -0
  230. package/src/utils/sql.ts +38 -0
  231. package/test/admin-routes.test.ts +591 -0
  232. package/test/api-generator.test.ts +458 -0
  233. package/test/ast-schema-editor.test.ts +61 -0
  234. package/test/auth-middleware-hono.test.ts +321 -0
  235. package/test/auth-routes.test.ts +868 -0
  236. package/test/driver-registry.test.ts +280 -0
  237. package/test/errors-hono.test.ts +133 -0
  238. package/test/errors.test.ts +150 -0
  239. package/test/jwt-security.test.ts +173 -0
  240. package/test/jwt.test.ts +311 -0
  241. package/test/middleware.test.ts +295 -0
  242. package/test/password.test.ts +165 -0
  243. package/test/query-parser.test.ts +258 -0
  244. package/test/rate-limiter.test.ts +102 -0
  245. package/test/storage-local.test.ts +278 -0
  246. package/test/storage-registry.test.ts +280 -0
  247. package/test/storage-routes.test.ts +218 -0
  248. package/test/storage-s3.test.ts +301 -0
  249. package/test-ast.ts +28 -0
  250. package/test_output.txt +1133 -0
  251. package/tsconfig.json +49 -0
  252. package/tsconfig.prod.json +20 -0
  253. package/vite.config.ts +78 -0
  254. package/vite.config.ts.timestamp-1775065397568-8a853255edf6e.mjs +46 -0
@@ -0,0 +1,280 @@
1
+ import {
2
+ StorageRegistry,
3
+ DefaultStorageRegistry,
4
+ DEFAULT_STORAGE_ID
5
+ } from "../src/storage/storage-registry";
6
+ import { StorageController } from "../src/storage/types";
7
+
8
+ /**
9
+ * Mock StorageController for testing
10
+ */
11
+ function createMockStorageController(type: 'local' | 's3'): StorageController {
12
+ return {
13
+ uploadFile: jest.fn().mockResolvedValue({ path: "test/file.txt" }),
14
+ getDownloadURL: jest.fn().mockResolvedValue({ url: "http://example.com/file.txt" }),
15
+ getFile: jest.fn().mockResolvedValue(null),
16
+ deleteFile: jest.fn().mockResolvedValue(undefined),
17
+ list: jest.fn().mockResolvedValue({ items: [], prefixes: [] }),
18
+ getType: jest.fn().mockReturnValue(type)
19
+ };
20
+ }
21
+
22
+ describe("StorageRegistry", () => {
23
+ describe("DEFAULT_STORAGE_ID", () => {
24
+ it("should be '(default)'", () => {
25
+ expect(DEFAULT_STORAGE_ID).toBe("(default)");
26
+ });
27
+ });
28
+
29
+ describe("DefaultStorageRegistry", () => {
30
+ describe("constructor and basic operations", () => {
31
+ it("should create an empty registry", () => {
32
+ const registry = new DefaultStorageRegistry();
33
+ expect(registry.size()).toBe(0);
34
+ expect(registry.list()).toEqual([]);
35
+ });
36
+
37
+ it("should register a storage controller", () => {
38
+ const registry = new DefaultStorageRegistry();
39
+ const mockController = createMockStorageController('local');
40
+
41
+ registry.register("test-storage", mockController);
42
+
43
+ expect(registry.has("test-storage")).toBe(true);
44
+ expect(registry.size()).toBe(1);
45
+ expect(registry.list()).toContain("test-storage");
46
+ });
47
+
48
+ it("should get a registered storage controller", () => {
49
+ const registry = new DefaultStorageRegistry();
50
+ const mockController = createMockStorageController('s3');
51
+
52
+ registry.register("my-storage", mockController);
53
+
54
+ const retrieved = registry.get("my-storage");
55
+ expect(retrieved).toBe(mockController);
56
+ });
57
+
58
+ it("should return undefined for non-existent storage", () => {
59
+ const registry = new DefaultStorageRegistry();
60
+ expect(registry.get("non-existent")).toBeUndefined();
61
+ });
62
+ });
63
+
64
+ describe("default storage handling", () => {
65
+ it("should get default storage with get(undefined)", () => {
66
+ const registry = new DefaultStorageRegistry();
67
+ const mockController = createMockStorageController('local');
68
+
69
+ registry.register(DEFAULT_STORAGE_ID, mockController);
70
+
71
+ expect(registry.get(undefined)).toBe(mockController);
72
+ expect(registry.get(null)).toBe(mockController);
73
+ });
74
+
75
+ it("should get default storage with getDefault()", () => {
76
+ const registry = new DefaultStorageRegistry();
77
+ const mockController = createMockStorageController('local');
78
+
79
+ registry.register(DEFAULT_STORAGE_ID, mockController);
80
+
81
+ expect(registry.getDefault()).toBe(mockController);
82
+ });
83
+
84
+ it("should throw error when no default storage exists", () => {
85
+ const registry = new DefaultStorageRegistry();
86
+
87
+ expect(() => registry.getDefault()).toThrow(
88
+ `[StorageRegistry] No default storage registered.`
89
+ );
90
+ });
91
+ });
92
+
93
+ describe("getOrDefault", () => {
94
+ let registry: DefaultStorageRegistry;
95
+ let defaultController: StorageController;
96
+ let mediaController: StorageController;
97
+
98
+ beforeEach(() => {
99
+ registry = new DefaultStorageRegistry();
100
+ defaultController = createMockStorageController('local');
101
+ mediaController = createMockStorageController('s3');
102
+
103
+ registry.register(DEFAULT_STORAGE_ID, defaultController);
104
+ registry.register("media", mediaController);
105
+ });
106
+
107
+ it("should return specific storage when found", () => {
108
+ expect(registry.getOrDefault("media")).toBe(mediaController);
109
+ });
110
+
111
+ it("should return default when id is undefined", () => {
112
+ expect(registry.getOrDefault(undefined)).toBe(defaultController);
113
+ });
114
+
115
+ it("should return default when id is null", () => {
116
+ expect(registry.getOrDefault(null)).toBe(defaultController);
117
+ });
118
+
119
+ it("should fallback to default when id not found", () => {
120
+ const consoleSpy = jest.spyOn(console, "warn").mockImplementation();
121
+
122
+ expect(registry.getOrDefault("non-existent")).toBe(defaultController);
123
+ expect(consoleSpy).toHaveBeenCalledWith(
124
+ expect.stringContaining('Storage "non-existent" not found')
125
+ );
126
+
127
+ consoleSpy.mockRestore();
128
+ });
129
+
130
+ it("should throw when fallback fails (no default)", () => {
131
+ const emptyRegistry = new DefaultStorageRegistry();
132
+
133
+ expect(() => emptyRegistry.getOrDefault("anything")).toThrow();
134
+ });
135
+ });
136
+
137
+ describe("overwriting storages", () => {
138
+ it("should overwrite existing storage with same id", () => {
139
+ const registry = new DefaultStorageRegistry();
140
+ const original = createMockStorageController('local');
141
+ const replacement = createMockStorageController('s3');
142
+
143
+ const consoleSpy = jest.spyOn(console, "warn").mockImplementation();
144
+
145
+ registry.register("my-storage", original);
146
+ registry.register("my-storage", replacement);
147
+
148
+ expect(registry.get("my-storage")).toBe(replacement);
149
+ expect(consoleSpy).toHaveBeenCalledWith(
150
+ expect.stringContaining('Overwriting storage with id "my-storage"')
151
+ );
152
+
153
+ consoleSpy.mockRestore();
154
+ });
155
+ });
156
+
157
+ describe("list and size", () => {
158
+ it("should list all registered storages", () => {
159
+ const registry = new DefaultStorageRegistry();
160
+
161
+ registry.register("storage-1", createMockStorageController('local'));
162
+ registry.register("storage-2", createMockStorageController('s3'));
163
+ registry.register(DEFAULT_STORAGE_ID, createMockStorageController('local'));
164
+
165
+ const list = registry.list();
166
+ expect(list).toHaveLength(3);
167
+ expect(list).toContain("storage-1");
168
+ expect(list).toContain("storage-2");
169
+ expect(list).toContain(DEFAULT_STORAGE_ID);
170
+ });
171
+
172
+ it("should return correct size", () => {
173
+ const registry = new DefaultStorageRegistry();
174
+
175
+ expect(registry.size()).toBe(0);
176
+
177
+ registry.register("storage-1", createMockStorageController('local'));
178
+ expect(registry.size()).toBe(1);
179
+
180
+ registry.register("storage-2", createMockStorageController('s3'));
181
+ expect(registry.size()).toBe(2);
182
+ });
183
+ });
184
+ });
185
+
186
+ describe("DefaultStorageRegistry.create() factory", () => {
187
+ describe("with single StorageController", () => {
188
+ it('should register single controller as "(default)"', () => {
189
+ const mockController = createMockStorageController('local');
190
+
191
+ const registry = DefaultStorageRegistry.create(mockController);
192
+
193
+ expect(registry.has(DEFAULT_STORAGE_ID)).toBe(true);
194
+ expect(registry.getDefault()).toBe(mockController);
195
+ expect(registry.size()).toBe(1);
196
+ });
197
+ });
198
+
199
+ describe("with map of StorageControllers", () => {
200
+ it("should register all controllers from map", () => {
201
+ const defaultController = createMockStorageController('local');
202
+ const mediaController = createMockStorageController('s3');
203
+
204
+ const registry = DefaultStorageRegistry.create({
205
+ [DEFAULT_STORAGE_ID]: defaultController,
206
+ "media": mediaController
207
+ });
208
+
209
+ expect(registry.size()).toBe(2);
210
+ expect(registry.getDefault()).toBe(defaultController);
211
+ expect(registry.get("media")).toBe(mediaController);
212
+ });
213
+
214
+ it("should use first entry as default if no explicit default provided", () => {
215
+ const local = createMockStorageController('local');
216
+ const s3 = createMockStorageController('s3');
217
+
218
+ const consoleSpy = jest.spyOn(console, "warn").mockImplementation();
219
+
220
+ const registry = DefaultStorageRegistry.create({
221
+ "primary": local,
222
+ "secondary": s3
223
+ });
224
+
225
+ // Should have registered both + created default pointing to first
226
+ expect(registry.size()).toBe(3); // primary, secondary, (default)
227
+ expect(registry.has(DEFAULT_STORAGE_ID)).toBe(true);
228
+ expect(consoleSpy).toHaveBeenCalledWith(
229
+ expect.stringContaining('No "(default)" storage provided')
230
+ );
231
+
232
+ consoleSpy.mockRestore();
233
+ });
234
+
235
+ it("should handle empty map gracefully", () => {
236
+ const registry = DefaultStorageRegistry.create({});
237
+
238
+ expect(registry.size()).toBe(0);
239
+ expect(() => registry.getDefault()).toThrow();
240
+ });
241
+ });
242
+ });
243
+
244
+ describe("type detection (isStorageController)", () => {
245
+ it("should correctly identify a StorageController", () => {
246
+ const mockController = createMockStorageController('local');
247
+
248
+ // The factory should recognize it as a single controller
249
+ const registry = DefaultStorageRegistry.create(mockController);
250
+ expect(registry.size()).toBe(1);
251
+ expect(registry.has(DEFAULT_STORAGE_ID)).toBe(true);
252
+ });
253
+
254
+ it("should correctly identify a map of StorageControllers", () => {
255
+ const controllers = {
256
+ [DEFAULT_STORAGE_ID]: createMockStorageController('local'),
257
+ "other": createMockStorageController('s3')
258
+ };
259
+
260
+ // The factory should recognize it as a map
261
+ const registry = DefaultStorageRegistry.create(controllers);
262
+ expect(registry.size()).toBe(2);
263
+ });
264
+ });
265
+
266
+ describe("integration with storage types", () => {
267
+ it("should correctly report storage types", () => {
268
+ const localController = createMockStorageController('local');
269
+ const s3Controller = createMockStorageController('s3');
270
+
271
+ const registry = DefaultStorageRegistry.create({
272
+ [DEFAULT_STORAGE_ID]: localController,
273
+ "media": s3Controller
274
+ });
275
+
276
+ expect(registry.getDefault().getType()).toBe('local');
277
+ expect(registry.get("media")?.getType()).toBe('s3');
278
+ });
279
+ });
280
+ });
@@ -0,0 +1,218 @@
1
+ /**
2
+ * Tests for storage routes — specifically the sub-router wildcard extraction
3
+ * that broke when Hono's c.req.param('*') stopped working in mounted sub-routers.
4
+ *
5
+ * These tests use Hono's built-in `app.fetch()` to simulate requests without
6
+ * needing a running HTTP server, which keeps them fast and deterministic.
7
+ */
8
+ import * as fs from "fs";
9
+ import * as path from "path";
10
+ import * as os from "os";
11
+ import { Hono } from "hono";
12
+ import { HonoEnv } from "../src/api/types";
13
+ import { errorHandler } from "../src/api/errors";
14
+ import { LocalStorageController } from "../src/storage/LocalStorageController";
15
+ import { createStorageRoutes, extractWildcardPath } from "../src/storage/routes";
16
+
17
+ // ──────────────────────────────────────────────────────────────────────
18
+ // Unit tests for extractWildcardPath
19
+ // ──────────────────────────────────────────────────────────────────────
20
+
21
+ describe("extractWildcardPath", () => {
22
+ it("should extract path after the route prefix", () => {
23
+ const result = extractWildcardPath({
24
+ req: {
25
+ path: "/api/storage/metadata/default/author_pictures/photo.jpg",
26
+ routePath: "/api/storage/metadata/*"
27
+ }
28
+ });
29
+ expect(result).toBe("default/author_pictures/photo.jpg");
30
+ });
31
+
32
+ it("should extract simple file path", () => {
33
+ const result = extractWildcardPath({
34
+ req: {
35
+ path: "/api/storage/file/testfile.jpg",
36
+ routePath: "/api/storage/file/*"
37
+ }
38
+ });
39
+ expect(result).toBe("testfile.jpg");
40
+ });
41
+
42
+ it("should return empty string for trailing-slash-only path", () => {
43
+ const result = extractWildcardPath({
44
+ req: {
45
+ path: "/api/storage/metadata/",
46
+ routePath: "/api/storage/metadata/*"
47
+ }
48
+ });
49
+ expect(result).toBe("");
50
+ });
51
+
52
+ it("should return empty string when path equals prefix (no trailing slash)", () => {
53
+ const result = extractWildcardPath({
54
+ req: {
55
+ path: "/api/storage/metadata",
56
+ routePath: "/api/storage/metadata/*"
57
+ }
58
+ });
59
+ expect(result).toBe("");
60
+ });
61
+
62
+ it("should handle deeply nested paths", () => {
63
+ const result = extractWildcardPath({
64
+ req: {
65
+ path: "/api/storage/file/bucket/a/b/c/d/file.png",
66
+ routePath: "/api/storage/file/*"
67
+ }
68
+ });
69
+ expect(result).toBe("bucket/a/b/c/d/file.png");
70
+ });
71
+
72
+ it("should handle paths with spaces and special chars", () => {
73
+ const result = extractWildcardPath({
74
+ req: {
75
+ path: "/api/storage/file/default/photos/my%20file%20(1).png",
76
+ routePath: "/api/storage/file/*"
77
+ }
78
+ });
79
+ expect(result).toBe("default/photos/my%20file%20(1).png");
80
+ });
81
+ });
82
+
83
+ // ──────────────────────────────────────────────────────────────────────
84
+ // Integration tests: storage routes mounted as Hono sub-router
85
+ // ──────────────────────────────────────────────────────────────────────
86
+
87
+ describe("Storage routes (sub-router integration)", () => {
88
+ let app: Hono<HonoEnv>;
89
+ let tempDir: string;
90
+ let controller: LocalStorageController;
91
+
92
+ beforeEach(async () => {
93
+ tempDir = await fs.promises.mkdtemp(path.join(os.tmpdir(), "rebase-routes-test-"));
94
+ controller = new LocalStorageController({ basePath: tempDir });
95
+
96
+ // Upload a test file so we have something to serve
97
+ const content = Buffer.from("Hello test file");
98
+ const file = new File([content], "test.txt", { type: "text/plain" });
99
+ await controller.uploadFile({ file, fileName: "test.txt", path: "photos" });
100
+
101
+ // Create the Hono app with storage routes mounted as a SUB-ROUTER
102
+ // (this is the exact pattern that caused the bug)
103
+ app = new Hono<HonoEnv>();
104
+ app.onError(errorHandler); // required to convert ApiError throws to proper HTTP responses
105
+ const storageRoutes = createStorageRoutes({
106
+ controller,
107
+ requireAuth: false // skip auth for tests
108
+ });
109
+ app.route("/api/storage", storageRoutes);
110
+ });
111
+
112
+ afterEach(async () => {
113
+ await fs.promises.rm(tempDir, { recursive: true, force: true });
114
+ });
115
+
116
+ describe("GET /metadata/*", () => {
117
+ it("should return metadata for a valid file path", async () => {
118
+ const res = await app.fetch(
119
+ new Request("http://localhost/api/storage/metadata/default/photos/test.txt")
120
+ );
121
+
122
+ expect(res.status).toBe(200);
123
+ const body = await res.json() as { success: boolean; data: { contentType: string } };
124
+ expect(body.success).toBe(true);
125
+ expect(body.data).toBeDefined();
126
+ expect(body.data.contentType).toBe("text/plain");
127
+ });
128
+
129
+ it("should return metadata without explicit bucket prefix", async () => {
130
+ const res = await app.fetch(
131
+ new Request("http://localhost/api/storage/metadata/photos/test.txt")
132
+ );
133
+
134
+ expect(res.status).toBe(200);
135
+ const body = await res.json() as { success: boolean; data: { contentType: string } };
136
+ expect(body.success).toBe(true);
137
+ expect(body.data).toBeDefined();
138
+ });
139
+
140
+ it("should return 404 for empty path", async () => {
141
+ const res = await app.fetch(
142
+ new Request("http://localhost/api/storage/metadata/")
143
+ );
144
+ expect(res.status).toBe(404);
145
+ });
146
+
147
+ it("should return 404 for non-existent file", async () => {
148
+ const res = await app.fetch(
149
+ new Request("http://localhost/api/storage/metadata/default/nope/missing.txt")
150
+ );
151
+ expect(res.status).toBe(404);
152
+ });
153
+ });
154
+
155
+ describe("GET /file/*", () => {
156
+ it("should serve file content for a valid path", async () => {
157
+ const res = await app.fetch(
158
+ new Request("http://localhost/api/storage/file/default/photos/test.txt")
159
+ );
160
+
161
+ expect(res.status).toBe(200);
162
+ const body = await res.text();
163
+ expect(body).toBe("Hello test file");
164
+ });
165
+
166
+ it("should serve file without explicit bucket prefix", async () => {
167
+ const res = await app.fetch(
168
+ new Request("http://localhost/api/storage/file/photos/test.txt")
169
+ );
170
+
171
+ expect(res.status).toBe(200);
172
+ const body = await res.text();
173
+ expect(body).toBe("Hello test file");
174
+ });
175
+
176
+ it("should return 404 for empty path", async () => {
177
+ const res = await app.fetch(
178
+ new Request("http://localhost/api/storage/file/")
179
+ );
180
+ expect(res.status).toBe(404);
181
+ });
182
+
183
+ it("should return 404 for non-existent file", async () => {
184
+ const res = await app.fetch(
185
+ new Request("http://localhost/api/storage/file/default/nope/missing.txt")
186
+ );
187
+ expect(res.status).toBe(404);
188
+ });
189
+ });
190
+
191
+ describe("DELETE /file/*", () => {
192
+ it("should delete an existing file", async () => {
193
+ // Upload another file to delete
194
+ const file = new File([Buffer.from("delete me")], "deleteme.txt", { type: "text/plain" });
195
+ await controller.uploadFile({ file, fileName: "deleteme.txt", path: "photos" });
196
+
197
+ const res = await app.fetch(
198
+ new Request("http://localhost/api/storage/file/default/photos/deleteme.txt", { method: "DELETE" })
199
+ );
200
+
201
+ expect(res.status).toBe(200);
202
+ const body = await res.json() as { success: boolean };
203
+ expect(body.success).toBe(true);
204
+
205
+ // Verify the file is actually gone
206
+ const filePath = path.join(tempDir, "default", "photos", "deleteme.txt");
207
+ const exists = await fs.promises.access(filePath).then(() => true).catch(() => false);
208
+ expect(exists).toBe(false);
209
+ });
210
+
211
+ it("should handle empty path gracefully", async () => {
212
+ const res = await app.fetch(
213
+ new Request("http://localhost/api/storage/file/", { method: "DELETE" })
214
+ );
215
+ expect(res.status).toBe(200);
216
+ });
217
+ });
218
+ });