@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
+ DriverRegistry,
3
+ DefaultDriverRegistry,
4
+ DEFAULT_DRIVER_ID
5
+ } from "../src/services/driver-registry";
6
+ import { DataDriver } from "@rebasepro/types";
7
+
8
+ /**
9
+ * Mock DataDriver for testing
10
+ */
11
+ function createMockDataDriverDelegate(key: string): DataDriver {
12
+ return {
13
+ key,
14
+ initialised: true,
15
+ fetchCollection: jest.fn().mockResolvedValue([]),
16
+ fetchEntity: jest.fn().mockResolvedValue(undefined),
17
+ saveEntity: jest.fn().mockResolvedValue({ id: "test-id", path: "test", values: {} }),
18
+ deleteEntity: jest.fn().mockResolvedValue(undefined),
19
+ checkUniqueField: jest.fn().mockResolvedValue(true)
20
+ };
21
+ }
22
+
23
+ describe("DriverRegistry", () => {
24
+ describe("DEFAULT_DRIVER_ID", () => {
25
+ it("should be '(default)'", () => {
26
+ expect(DEFAULT_DRIVER_ID).toBe("(default)");
27
+ });
28
+ });
29
+
30
+ describe("DefaultDriverRegistry", () => {
31
+ describe("constructor and basic operations", () => {
32
+ it("should create an empty registry", () => {
33
+ const registry = new DefaultDriverRegistry();
34
+ expect(registry.size()).toBe(0);
35
+ expect(registry.list()).toEqual([]);
36
+ });
37
+
38
+ it("should register a driver", () => {
39
+ const registry = new DefaultDriverRegistry();
40
+ const mockDelegate = createMockDataDriverDelegate("postgres");
41
+
42
+ registry.register("test-db", mockDelegate);
43
+
44
+ expect(registry.has("test-db")).toBe(true);
45
+ expect(registry.size()).toBe(1);
46
+ expect(registry.list()).toContain("test-db");
47
+ });
48
+
49
+ it("should get a registered driver", () => {
50
+ const registry = new DefaultDriverRegistry();
51
+ const mockDelegate = createMockDataDriverDelegate("postgres");
52
+
53
+ registry.register("my-db", mockDelegate);
54
+
55
+ const retrieved = registry.get("my-db");
56
+ expect(retrieved).toBe(mockDelegate);
57
+ });
58
+
59
+ it("should return undefined for non-existent driver", () => {
60
+ const registry = new DefaultDriverRegistry();
61
+ expect(registry.get("non-existent")).toBeUndefined();
62
+ });
63
+ });
64
+
65
+ describe("default driver handling", () => {
66
+ it("should get default driver with get(undefined)", () => {
67
+ const registry = new DefaultDriverRegistry();
68
+ const mockDelegate = createMockDataDriverDelegate("postgres");
69
+
70
+ registry.register(DEFAULT_DRIVER_ID, mockDelegate);
71
+
72
+ expect(registry.get(undefined)).toBe(mockDelegate);
73
+ expect(registry.get(null)).toBe(mockDelegate);
74
+ });
75
+
76
+ it("should get default driver with getDefault()", () => {
77
+ const registry = new DefaultDriverRegistry();
78
+ const mockDelegate = createMockDataDriverDelegate("postgres");
79
+
80
+ registry.register(DEFAULT_DRIVER_ID, mockDelegate);
81
+
82
+ expect(registry.getDefault()).toBe(mockDelegate);
83
+ });
84
+
85
+ it("should throw error when no default driver exists", () => {
86
+ const registry = new DefaultDriverRegistry();
87
+
88
+ expect(() => registry.getDefault()).toThrow(
89
+ `[DriverRegistry] No default driver registered.`
90
+ );
91
+ });
92
+ });
93
+
94
+ describe("getOrDefault", () => {
95
+ let registry: DefaultDriverRegistry;
96
+ let defaultDelegate: DataDriver;
97
+ let analyticsDelegate: DataDriver;
98
+
99
+ beforeEach(() => {
100
+ registry = new DefaultDriverRegistry();
101
+ defaultDelegate = createMockDataDriverDelegate("default-postgres");
102
+ analyticsDelegate = createMockDataDriverDelegate("analytics-postgres");
103
+
104
+ registry.register(DEFAULT_DRIVER_ID, defaultDelegate);
105
+ registry.register("analytics", analyticsDelegate);
106
+ });
107
+
108
+ it("should return specific driver when found", () => {
109
+ expect(registry.getOrDefault("analytics")).toBe(analyticsDelegate);
110
+ });
111
+
112
+ it("should return default when id is undefined", () => {
113
+ expect(registry.getOrDefault(undefined)).toBe(defaultDelegate);
114
+ });
115
+
116
+ it("should return default when id is null", () => {
117
+ expect(registry.getOrDefault(null)).toBe(defaultDelegate);
118
+ });
119
+
120
+ it("should fallback to default when id not found", () => {
121
+ // This should log a warning and return the default
122
+ const consoleSpy = jest.spyOn(console, "warn").mockImplementation();
123
+
124
+ expect(registry.getOrDefault("non-existent")).toBe(defaultDelegate);
125
+ expect(consoleSpy).toHaveBeenCalledWith(
126
+ expect.stringContaining('Driver "non-existent" not found')
127
+ );
128
+
129
+ consoleSpy.mockRestore();
130
+ });
131
+
132
+ it("should throw when fallback fails (no default)", () => {
133
+ const emptyRegistry = new DefaultDriverRegistry();
134
+
135
+ expect(() => emptyRegistry.getOrDefault("anything")).toThrow();
136
+ });
137
+ });
138
+
139
+ describe("overwriting drivers", () => {
140
+ it("should overwrite existing driver with same id", () => {
141
+ const registry = new DefaultDriverRegistry();
142
+ const original = createMockDataDriverDelegate("original");
143
+ const replacement = createMockDataDriverDelegate("replacement");
144
+
145
+ const consoleSpy = jest.spyOn(console, "warn").mockImplementation();
146
+
147
+ registry.register("my-db", original);
148
+ registry.register("my-db", replacement);
149
+
150
+ expect(registry.get("my-db")).toBe(replacement);
151
+ expect(consoleSpy).toHaveBeenCalledWith(
152
+ expect.stringContaining('Overwriting driver with id "my-db"')
153
+ );
154
+
155
+ consoleSpy.mockRestore();
156
+ });
157
+ });
158
+
159
+ describe("list and size", () => {
160
+ it("should list all registered drivers", () => {
161
+ const registry = new DefaultDriverRegistry();
162
+
163
+ registry.register("db-1", createMockDataDriverDelegate("pg1"));
164
+ registry.register("db-2", createMockDataDriverDelegate("pg2"));
165
+ registry.register(DEFAULT_DRIVER_ID, createMockDataDriverDelegate("default"));
166
+
167
+ const list = registry.list();
168
+ expect(list).toHaveLength(3);
169
+ expect(list).toContain("db-1");
170
+ expect(list).toContain("db-2");
171
+ expect(list).toContain(DEFAULT_DRIVER_ID);
172
+ });
173
+
174
+ it("should return correct size", () => {
175
+ const registry = new DefaultDriverRegistry();
176
+
177
+ expect(registry.size()).toBe(0);
178
+
179
+ registry.register("db-1", createMockDataDriverDelegate("pg1"));
180
+ expect(registry.size()).toBe(1);
181
+
182
+ registry.register("db-2", createMockDataDriverDelegate("pg2"));
183
+ expect(registry.size()).toBe(2);
184
+ });
185
+ });
186
+ });
187
+
188
+ describe("DefaultDriverRegistry.create() factory", () => {
189
+ describe("with single DataDriver", () => {
190
+ it('should register single delegate as "(default)"', () => {
191
+ const mockDelegate = createMockDataDriverDelegate("postgres");
192
+
193
+ const registry = DefaultDriverRegistry.create(mockDelegate);
194
+
195
+ expect(registry.has(DEFAULT_DRIVER_ID)).toBe(true);
196
+ expect(registry.getDefault()).toBe(mockDelegate);
197
+ expect(registry.size()).toBe(1);
198
+ });
199
+ });
200
+
201
+ describe("with map of DataDriverDelegates", () => {
202
+ it("should register all delegates from map", () => {
203
+ const defaultDelegate = createMockDataDriverDelegate("default-pg");
204
+ const analyticsDelegate = createMockDataDriverDelegate("analytics-pg");
205
+
206
+ const registry = DefaultDriverRegistry.create({
207
+ [DEFAULT_DRIVER_ID]: defaultDelegate,
208
+ "analytics": analyticsDelegate
209
+ });
210
+
211
+ expect(registry.size()).toBe(2);
212
+ expect(registry.getDefault()).toBe(defaultDelegate);
213
+ expect(registry.get("analytics")).toBe(analyticsDelegate);
214
+ });
215
+
216
+ it("should use first entry as default if no explicit default provided", () => {
217
+ const db1 = createMockDataDriverDelegate("db1");
218
+ const db2 = createMockDataDriverDelegate("db2");
219
+
220
+ const consoleSpy = jest.spyOn(console, "warn").mockImplementation();
221
+
222
+ const registry = DefaultDriverRegistry.create({
223
+ "primary": db1,
224
+ "secondary": db2
225
+ });
226
+
227
+ // Should have registered both + created default pointing to first
228
+ expect(registry.size()).toBe(3); // primary, secondary, (default)
229
+ expect(registry.has(DEFAULT_DRIVER_ID)).toBe(true);
230
+ expect(consoleSpy).toHaveBeenCalledWith(
231
+ expect.stringContaining('No "(default)" driver provided')
232
+ );
233
+
234
+ consoleSpy.mockRestore();
235
+ });
236
+
237
+ it("should handle empty map gracefully", () => {
238
+ const registry = DefaultDriverRegistry.create({});
239
+
240
+ expect(registry.size()).toBe(0);
241
+ expect(() => registry.getDefault()).toThrow();
242
+ });
243
+ });
244
+ });
245
+
246
+ describe("type detection (isDataDriverDelegate)", () => {
247
+ it("should correctly identify a DataDriver", () => {
248
+ const mockDelegate = createMockDataDriverDelegate("postgres");
249
+
250
+ // The factory should recognize it as a single delegate
251
+ const registry = DefaultDriverRegistry.create(mockDelegate);
252
+ expect(registry.size()).toBe(1);
253
+ expect(registry.has(DEFAULT_DRIVER_ID)).toBe(true);
254
+ });
255
+
256
+ it("should correctly identify a map of DataDriverDelegates", () => {
257
+ const delegates = {
258
+ [DEFAULT_DRIVER_ID]: createMockDataDriverDelegate("pg1"),
259
+ "other": createMockDataDriverDelegate("pg2")
260
+ };
261
+
262
+ // The factory should recognize it as a map
263
+ const registry = DefaultDriverRegistry.create(delegates);
264
+ expect(registry.size()).toBe(2);
265
+ });
266
+
267
+ it("should not mistakenly identify a map as a single delegate", () => {
268
+ // A map doesn't have the required DataDriver methods
269
+ const map = {
270
+ key: "not-a-delegate", // This looks like the key property but...
271
+ db1: createMockDataDriverDelegate("pg1")
272
+ };
273
+
274
+ // This should be treated as a map (and show a warning because 'key' isn't a valid delegate)
275
+ const registry = DefaultDriverRegistry.create(map as any);
276
+ // 'key' entry will be ignored since it's not a valid delegate
277
+ expect(registry.has("db1")).toBe(true);
278
+ });
279
+ });
280
+ });
@@ -0,0 +1,133 @@
1
+ import { Hono } from "hono";
2
+ import { ApiError, errorHandler } from "../src/api/errors";
3
+ import { HonoEnv } from "../src/api/types";
4
+
5
+ describe("Error Handler (Hono)", () => {
6
+ function createApp() {
7
+ const app = new Hono<HonoEnv>();
8
+ app.onError(errorHandler);
9
+
10
+ // Test routes that throw different errors
11
+ app.get("/bad-request", () => {
12
+ throw ApiError.badRequest("Missing required field", "MISSING_FIELD", { field: "email" });
13
+ });
14
+ app.get("/unauthorized", () => {
15
+ throw ApiError.unauthorized("Token expired", "TOKEN_EXPIRED");
16
+ });
17
+ app.get("/forbidden", () => {
18
+ throw ApiError.forbidden("Admin only", "FORBIDDEN");
19
+ });
20
+ app.get("/not-found", () => {
21
+ throw ApiError.notFound("Entity not found");
22
+ });
23
+ app.get("/conflict", () => {
24
+ throw ApiError.conflict("Email already exists", "EMAIL_EXISTS");
25
+ });
26
+ app.get("/internal", () => {
27
+ throw ApiError.internal("Database connection failed");
28
+ });
29
+ app.get("/service-unavailable", () => {
30
+ throw ApiError.serviceUnavailable("Feature not configured");
31
+ });
32
+ app.get("/generic-error", () => {
33
+ throw new Error("Something went wrong");
34
+ });
35
+ app.get("/error-with-code", () => {
36
+ const err = new Error("Rate limited") as Error & { code: string };
37
+ err.code = "RATE_LIMITED";
38
+ throw err;
39
+ });
40
+
41
+ return app;
42
+ }
43
+
44
+ it("formats ApiError with correct status and body structure", async () => {
45
+ const app = createApp();
46
+ const res = await app.request("/bad-request");
47
+ expect(res.status).toBe(400);
48
+ const body = await res.json() as any;
49
+ expect(body.error.message).toBe("Missing required field");
50
+ expect(body.error.code).toBe("MISSING_FIELD");
51
+ expect(body.error.details).toEqual({ field: "email" });
52
+ });
53
+
54
+ it("handles 401 Unauthorized", async () => {
55
+ const app = createApp();
56
+ const res = await app.request("/unauthorized");
57
+ expect(res.status).toBe(401);
58
+ const body = await res.json() as any;
59
+ expect(body.error.code).toBe("TOKEN_EXPIRED");
60
+ });
61
+
62
+ it("handles 403 Forbidden", async () => {
63
+ const app = createApp();
64
+ const res = await app.request("/forbidden");
65
+ expect(res.status).toBe(403);
66
+ const body = await res.json() as any;
67
+ expect(body.error.code).toBe("FORBIDDEN");
68
+ });
69
+
70
+ it("handles 404 Not Found", async () => {
71
+ const app = createApp();
72
+ const res = await app.request("/not-found");
73
+ expect(res.status).toBe(404);
74
+ const body = await res.json() as any;
75
+ expect(body.error.code).toBe("NOT_FOUND");
76
+ });
77
+
78
+ it("handles 409 Conflict", async () => {
79
+ const app = createApp();
80
+ const res = await app.request("/conflict");
81
+ expect(res.status).toBe(409);
82
+ const body = await res.json() as any;
83
+ expect(body.error.code).toBe("EMAIL_EXISTS");
84
+ });
85
+
86
+ it("handles 500 Internal", async () => {
87
+ const app = createApp();
88
+ const res = await app.request("/internal");
89
+ expect(res.status).toBe(500);
90
+ });
91
+
92
+ it("handles 503 Service Unavailable", async () => {
93
+ const app = createApp();
94
+ const res = await app.request("/service-unavailable");
95
+ expect(res.status).toBe(503);
96
+ });
97
+
98
+ it("converts generic Error to 500 with INTERNAL_ERROR code", async () => {
99
+ const app = createApp();
100
+ const res = await app.request("/generic-error");
101
+ expect(res.status).toBe(500);
102
+ const body = await res.json() as any;
103
+ expect(body.error.code).toBe("INTERNAL_ERROR");
104
+ expect(body.error.message).toBe("Internal Server Error");
105
+ });
106
+
107
+ it("maps known error codes to HTTP status codes", async () => {
108
+ const app = createApp();
109
+ const res = await app.request("/error-with-code");
110
+ // RATE_LIMITED is not in the code-to-status map, so it should default to 500
111
+ expect(res.status).toBe(500);
112
+ });
113
+
114
+ it("omits details when not provided", async () => {
115
+ const app = createApp();
116
+ const res = await app.request("/unauthorized");
117
+ const body = await res.json() as any;
118
+ expect(body.error.details).toBeUndefined();
119
+ });
120
+
121
+ it("returns consistent error shape for all error types", async () => {
122
+ const app = createApp();
123
+ const paths = ["/bad-request", "/unauthorized", "/forbidden", "/not-found", "/internal", "/generic-error"];
124
+
125
+ for (const path of paths) {
126
+ const res = await app.request(path);
127
+ const body = await res.json() as any;
128
+ expect(body).toHaveProperty("error");
129
+ expect(body.error).toHaveProperty("message");
130
+ expect(body.error).toHaveProperty("code");
131
+ }
132
+ });
133
+ });
@@ -0,0 +1,150 @@
1
+ import { ApiError, errorHandler } from "../src/api/errors";
2
+
3
+ // ── Minimal Hono-context mock ────────────────────────────────────────────
4
+ function createMockContext(method = "GET", path = "/test") {
5
+ let capturedStatus: number | undefined;
6
+ let capturedBody: any;
7
+
8
+ const c = {
9
+ req: { method, path },
10
+ json: (body: any, status?: number) => {
11
+ capturedBody = body;
12
+ capturedStatus = status ?? 200;
13
+ return new Response(JSON.stringify(body), { status: capturedStatus });
14
+ },
15
+ } as any;
16
+
17
+ return {
18
+ c,
19
+ getStatus: () => capturedStatus,
20
+ getBody: () => capturedBody,
21
+ };
22
+ }
23
+
24
+ // ── ApiError class ────────────────────────────────────────────────────────
25
+ describe("ApiError", () => {
26
+ describe("constructor", () => {
27
+ it("should create an error with statusCode, code, message, and details", () => {
28
+ const err = new ApiError(422, "VALIDATION_ERROR", "Invalid field", { field: "email" });
29
+ expect(err).toBeInstanceOf(Error);
30
+ expect(err).toBeInstanceOf(ApiError);
31
+ expect(err.statusCode).toBe(422);
32
+ expect(err.code).toBe("VALIDATION_ERROR");
33
+ expect(err.message).toBe("Invalid field");
34
+ expect(err.details).toEqual({ field: "email" });
35
+ expect(err.name).toBe("ApiError");
36
+ });
37
+
38
+ it("should default details to undefined", () => {
39
+ const err = new ApiError(400, "BAD_REQUEST", "Bad");
40
+ expect(err.details).toBeUndefined();
41
+ });
42
+ });
43
+
44
+ describe("factory methods", () => {
45
+ it("badRequest → 400", () => {
46
+ const err = ApiError.badRequest("Missing field", "MISSING_FIELD");
47
+ expect(err.statusCode).toBe(400);
48
+ expect(err.code).toBe("MISSING_FIELD");
49
+ });
50
+
51
+ it("badRequest uses default code", () => {
52
+ const err = ApiError.badRequest("Oops");
53
+ expect(err.code).toBe("BAD_REQUEST");
54
+ });
55
+
56
+ it("unauthorized → 401", () => {
57
+ const err = ApiError.unauthorized("Bad token");
58
+ expect(err.statusCode).toBe(401);
59
+ expect(err.code).toBe("UNAUTHORIZED");
60
+ });
61
+
62
+ it("forbidden → 403", () => {
63
+ const err = ApiError.forbidden("No access");
64
+ expect(err.statusCode).toBe(403);
65
+ expect(err.code).toBe("FORBIDDEN");
66
+ });
67
+
68
+ it("notFound → 404", () => {
69
+ const err = ApiError.notFound("Entity not found");
70
+ expect(err.statusCode).toBe(404);
71
+ expect(err.code).toBe("NOT_FOUND");
72
+ });
73
+
74
+ it("conflict → 409", () => {
75
+ const err = ApiError.conflict("Already exists", "EMAIL_EXISTS");
76
+ expect(err.statusCode).toBe(409);
77
+ expect(err.code).toBe("EMAIL_EXISTS");
78
+ });
79
+
80
+ it("internal → 500", () => {
81
+ const err = ApiError.internal("Boom");
82
+ expect(err.statusCode).toBe(500);
83
+ expect(err.code).toBe("INTERNAL_ERROR");
84
+ });
85
+
86
+ it("serviceUnavailable → 503", () => {
87
+ const err = ApiError.serviceUnavailable("Down");
88
+ expect(err.statusCode).toBe(503);
89
+ expect(err.code).toBe("SERVICE_UNAVAILABLE");
90
+ });
91
+ });
92
+ });
93
+
94
+ // ── errorHandler (Hono ErrorHandler) ──────────────────────────────────────
95
+ describe("errorHandler", () => {
96
+ it("should format ApiError with statusCode, code, message", () => {
97
+ const { c, getStatus, getBody } = createMockContext();
98
+ const err = ApiError.notFound("User not found");
99
+ errorHandler(err, c);
100
+
101
+ expect(getStatus()).toBe(404);
102
+ expect(getBody()).toEqual({
103
+ error: { message: "User not found", code: "NOT_FOUND" }
104
+ });
105
+ });
106
+
107
+ it("should include details when present", () => {
108
+ const { c, getBody } = createMockContext();
109
+ const err = ApiError.badRequest("Validation failed", "VALIDATION", { fields: ["email"] });
110
+ errorHandler(err, c);
111
+
112
+ expect(getBody()).toEqual({
113
+ error: {
114
+ message: "Validation failed",
115
+ code: "VALIDATION",
116
+ details: { fields: ["email"] }
117
+ }
118
+ });
119
+ });
120
+
121
+ it("should handle plain Error with code property", () => {
122
+ const { c, getStatus, getBody } = createMockContext();
123
+ const err = Object.assign(new Error("Not found"), { code: "NOT_FOUND" });
124
+ errorHandler(err, c);
125
+
126
+ expect(getStatus()).toBe(404);
127
+ expect(getBody()).toEqual({
128
+ error: { message: "Not found", code: "NOT_FOUND" }
129
+ });
130
+ });
131
+
132
+ it("should default to 500 for unknown errors", () => {
133
+ const { c, getStatus, getBody } = createMockContext();
134
+ const err = new Error("Something broke");
135
+ errorHandler(err, c);
136
+
137
+ expect(getStatus()).toBe(500);
138
+ expect(getBody()).toEqual({
139
+ error: { message: "Internal Server Error", code: "INTERNAL_ERROR" }
140
+ });
141
+ });
142
+
143
+ it("should use statusCode from error if present", () => {
144
+ const { c, getStatus } = createMockContext();
145
+ const err = Object.assign(new Error("Rate limited"), { statusCode: 429, code: "RATE_LIMITED" });
146
+ errorHandler(err, c);
147
+
148
+ expect(getStatus()).toBe(429);
149
+ });
150
+ });