@fragno-dev/db 0.1.14 → 0.1.15

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 (183) hide show
  1. package/.turbo/turbo-build.log +179 -139
  2. package/CHANGELOG.md +24 -0
  3. package/dist/adapters/adapters.d.ts +15 -1
  4. package/dist/adapters/adapters.d.ts.map +1 -1
  5. package/dist/adapters/adapters.js.map +1 -1
  6. package/dist/adapters/drizzle/drizzle-adapter.d.ts +3 -1
  7. package/dist/adapters/drizzle/drizzle-adapter.d.ts.map +1 -1
  8. package/dist/adapters/drizzle/drizzle-adapter.js +9 -2
  9. package/dist/adapters/drizzle/drizzle-adapter.js.map +1 -1
  10. package/dist/adapters/drizzle/drizzle-query.js +2 -2
  11. package/dist/adapters/drizzle/drizzle-query.js.map +1 -1
  12. package/dist/adapters/drizzle/drizzle-uow-compiler.js +27 -8
  13. package/dist/adapters/drizzle/drizzle-uow-compiler.js.map +1 -1
  14. package/dist/adapters/drizzle/drizzle-uow-decoder.js +22 -15
  15. package/dist/adapters/drizzle/drizzle-uow-decoder.js.map +1 -1
  16. package/dist/adapters/drizzle/drizzle-uow-executor.js +18 -7
  17. package/dist/adapters/drizzle/drizzle-uow-executor.js.map +1 -1
  18. package/dist/adapters/drizzle/generate.d.ts +4 -1
  19. package/dist/adapters/drizzle/generate.d.ts.map +1 -1
  20. package/dist/adapters/drizzle/generate.js +11 -18
  21. package/dist/adapters/drizzle/generate.js.map +1 -1
  22. package/dist/adapters/kysely/kysely-adapter.d.ts +3 -1
  23. package/dist/adapters/kysely/kysely-adapter.d.ts.map +1 -1
  24. package/dist/adapters/kysely/kysely-adapter.js +7 -1
  25. package/dist/adapters/kysely/kysely-adapter.js.map +1 -1
  26. package/dist/adapters/kysely/kysely-query-builder.js +1 -1
  27. package/dist/adapters/kysely/kysely-query-compiler.js +3 -2
  28. package/dist/adapters/kysely/kysely-query-compiler.js.map +1 -1
  29. package/dist/adapters/kysely/kysely-query.d.ts +1 -0
  30. package/dist/adapters/kysely/kysely-query.d.ts.map +1 -1
  31. package/dist/adapters/kysely/kysely-query.js +25 -18
  32. package/dist/adapters/kysely/kysely-query.js.map +1 -1
  33. package/dist/adapters/kysely/kysely-shared.d.ts +3 -0
  34. package/dist/adapters/kysely/kysely-shared.d.ts.map +1 -1
  35. package/dist/adapters/kysely/kysely-shared.js +16 -1
  36. package/dist/adapters/kysely/kysely-shared.js.map +1 -1
  37. package/dist/adapters/kysely/kysely-uow-compiler.js +34 -11
  38. package/dist/adapters/kysely/kysely-uow-compiler.js.map +1 -1
  39. package/dist/adapters/kysely/kysely-uow-executor.js +8 -4
  40. package/dist/adapters/kysely/kysely-uow-executor.js.map +1 -1
  41. package/dist/adapters/kysely/migration/execute-base.js +1 -1
  42. package/dist/adapters/kysely/migration/execute-base.js.map +1 -1
  43. package/dist/db-fragment-definition-builder.d.ts +152 -0
  44. package/dist/db-fragment-definition-builder.d.ts.map +1 -0
  45. package/dist/db-fragment-definition-builder.js +137 -0
  46. package/dist/db-fragment-definition-builder.js.map +1 -0
  47. package/dist/fragments/internal-fragment.d.ts +19 -0
  48. package/dist/fragments/internal-fragment.d.ts.map +1 -0
  49. package/dist/fragments/internal-fragment.js +39 -0
  50. package/dist/fragments/internal-fragment.js.map +1 -0
  51. package/dist/migration-engine/generation-engine.d.ts.map +1 -1
  52. package/dist/migration-engine/generation-engine.js +35 -15
  53. package/dist/migration-engine/generation-engine.js.map +1 -1
  54. package/dist/mod.d.ts +8 -20
  55. package/dist/mod.d.ts.map +1 -1
  56. package/dist/mod.js +7 -35
  57. package/dist/mod.js.map +1 -1
  58. package/dist/node_modules/.pnpm/rou3@0.7.8/node_modules/rou3/dist/index.js +165 -0
  59. package/dist/node_modules/.pnpm/rou3@0.7.8/node_modules/rou3/dist/index.js.map +1 -0
  60. package/dist/packages/fragno/dist/api/bind-services.js +20 -0
  61. package/dist/packages/fragno/dist/api/bind-services.js.map +1 -0
  62. package/dist/packages/fragno/dist/api/error.js +48 -0
  63. package/dist/packages/fragno/dist/api/error.js.map +1 -0
  64. package/dist/packages/fragno/dist/api/fragment-definition-builder.js +320 -0
  65. package/dist/packages/fragno/dist/api/fragment-definition-builder.js.map +1 -0
  66. package/dist/packages/fragno/dist/api/fragment-instantiator.js +487 -0
  67. package/dist/packages/fragno/dist/api/fragment-instantiator.js.map +1 -0
  68. package/dist/packages/fragno/dist/api/fragno-response.js +73 -0
  69. package/dist/packages/fragno/dist/api/fragno-response.js.map +1 -0
  70. package/dist/packages/fragno/dist/api/internal/response-stream.js +81 -0
  71. package/dist/packages/fragno/dist/api/internal/response-stream.js.map +1 -0
  72. package/dist/packages/fragno/dist/api/internal/route.js +10 -0
  73. package/dist/packages/fragno/dist/api/internal/route.js.map +1 -0
  74. package/dist/packages/fragno/dist/api/mutable-request-state.js +97 -0
  75. package/dist/packages/fragno/dist/api/mutable-request-state.js.map +1 -0
  76. package/dist/packages/fragno/dist/api/request-context-storage.js +43 -0
  77. package/dist/packages/fragno/dist/api/request-context-storage.js.map +1 -0
  78. package/dist/packages/fragno/dist/api/request-input-context.js +118 -0
  79. package/dist/packages/fragno/dist/api/request-input-context.js.map +1 -0
  80. package/dist/packages/fragno/dist/api/request-middleware.js +83 -0
  81. package/dist/packages/fragno/dist/api/request-middleware.js.map +1 -0
  82. package/dist/packages/fragno/dist/api/request-output-context.js +119 -0
  83. package/dist/packages/fragno/dist/api/request-output-context.js.map +1 -0
  84. package/dist/packages/fragno/dist/api/route.js +17 -0
  85. package/dist/packages/fragno/dist/api/route.js.map +1 -0
  86. package/dist/packages/fragno/dist/internal/symbols.js +10 -0
  87. package/dist/packages/fragno/dist/internal/symbols.js.map +1 -0
  88. package/dist/query/cursor.d.ts +10 -2
  89. package/dist/query/cursor.d.ts.map +1 -1
  90. package/dist/query/cursor.js +11 -4
  91. package/dist/query/cursor.js.map +1 -1
  92. package/dist/query/execute-unit-of-work.d.ts +123 -0
  93. package/dist/query/execute-unit-of-work.d.ts.map +1 -0
  94. package/dist/query/execute-unit-of-work.js +184 -0
  95. package/dist/query/execute-unit-of-work.js.map +1 -0
  96. package/dist/query/query.d.ts +2 -2
  97. package/dist/query/query.d.ts.map +1 -1
  98. package/dist/query/result-transform.js +4 -2
  99. package/dist/query/result-transform.js.map +1 -1
  100. package/dist/query/retry-policy.d.ts +88 -0
  101. package/dist/query/retry-policy.d.ts.map +1 -0
  102. package/dist/query/retry-policy.js +61 -0
  103. package/dist/query/retry-policy.js.map +1 -0
  104. package/dist/query/unit-of-work.d.ts +104 -50
  105. package/dist/query/unit-of-work.d.ts.map +1 -1
  106. package/dist/query/unit-of-work.js +384 -194
  107. package/dist/query/unit-of-work.js.map +1 -1
  108. package/dist/schema/serialize.js +12 -7
  109. package/dist/schema/serialize.js.map +1 -1
  110. package/dist/with-database.d.ts +28 -0
  111. package/dist/with-database.d.ts.map +1 -0
  112. package/dist/with-database.js +34 -0
  113. package/dist/with-database.js.map +1 -0
  114. package/package.json +9 -2
  115. package/src/adapters/adapters.ts +16 -0
  116. package/src/adapters/drizzle/drizzle-adapter-pglite.test.ts +80 -16
  117. package/src/adapters/drizzle/drizzle-adapter-sqlite.test.ts +158 -2
  118. package/src/adapters/drizzle/drizzle-adapter.test.ts +3 -51
  119. package/src/adapters/drizzle/drizzle-adapter.ts +20 -7
  120. package/src/adapters/drizzle/drizzle-query.ts +1 -2
  121. package/src/adapters/drizzle/drizzle-uow-compiler-mysql.test.ts +1442 -0
  122. package/src/adapters/drizzle/drizzle-uow-compiler-sqlite.test.ts +1414 -0
  123. package/src/adapters/drizzle/drizzle-uow-compiler.test.ts +21 -4
  124. package/src/adapters/drizzle/drizzle-uow-compiler.ts +44 -3
  125. package/src/adapters/drizzle/drizzle-uow-decoder.ts +32 -22
  126. package/src/adapters/drizzle/drizzle-uow-executor.ts +41 -8
  127. package/src/adapters/drizzle/generate.test.ts +102 -269
  128. package/src/adapters/drizzle/generate.ts +12 -30
  129. package/src/adapters/drizzle/test-utils.ts +36 -5
  130. package/src/adapters/kysely/kysely-adapter-pglite.test.ts +64 -20
  131. package/src/adapters/kysely/kysely-adapter-sqlite.test.ts +156 -0
  132. package/src/adapters/kysely/kysely-adapter.ts +9 -1
  133. package/src/adapters/kysely/kysely-query-compiler.ts +3 -8
  134. package/src/adapters/kysely/kysely-query.ts +34 -25
  135. package/src/adapters/kysely/kysely-shared.ts +34 -0
  136. package/src/adapters/kysely/kysely-uow-compiler.test.ts +61 -73
  137. package/src/adapters/kysely/kysely-uow-compiler.ts +44 -12
  138. package/src/adapters/kysely/kysely-uow-executor.ts +26 -7
  139. package/src/adapters/kysely/kysely-uow-joins.test.ts +31 -48
  140. package/src/adapters/kysely/migration/execute-base.ts +1 -1
  141. package/src/db-fragment-definition-builder.test.ts +887 -0
  142. package/src/db-fragment-definition-builder.ts +506 -0
  143. package/src/db-fragment-instantiator.test.ts +467 -0
  144. package/src/db-fragment-integration.test.ts +408 -0
  145. package/src/fragments/internal-fragment.test.ts +160 -0
  146. package/src/fragments/internal-fragment.ts +85 -0
  147. package/src/migration-engine/generation-engine.test.ts +58 -15
  148. package/src/migration-engine/generation-engine.ts +78 -25
  149. package/src/mod.ts +25 -52
  150. package/src/query/cursor.test.ts +119 -0
  151. package/src/query/cursor.ts +17 -4
  152. package/src/query/execute-unit-of-work.test.ts +1310 -0
  153. package/src/query/execute-unit-of-work.ts +463 -0
  154. package/src/query/query.ts +2 -2
  155. package/src/query/result-transform.test.ts +129 -0
  156. package/src/query/result-transform.ts +4 -1
  157. package/src/query/retry-policy.test.ts +217 -0
  158. package/src/query/retry-policy.ts +141 -0
  159. package/src/query/unit-of-work-coordinator.test.ts +833 -0
  160. package/src/query/unit-of-work-types.test.ts +2 -2
  161. package/src/query/unit-of-work.test.ts +873 -191
  162. package/src/query/unit-of-work.ts +602 -409
  163. package/src/schema/serialize.ts +22 -11
  164. package/src/with-database.ts +140 -0
  165. package/tsdown.config.ts +1 -0
  166. package/dist/bind-services.d.ts +0 -7
  167. package/dist/bind-services.d.ts.map +0 -1
  168. package/dist/bind-services.js +0 -14
  169. package/dist/bind-services.js.map +0 -1
  170. package/dist/fragment.d.ts +0 -173
  171. package/dist/fragment.d.ts.map +0 -1
  172. package/dist/fragment.js +0 -191
  173. package/dist/fragment.js.map +0 -1
  174. package/dist/shared/settings-schema.js +0 -36
  175. package/dist/shared/settings-schema.js.map +0 -1
  176. package/src/bind-services.test.ts +0 -214
  177. package/src/bind-services.ts +0 -37
  178. package/src/db-fragment.test.ts +0 -800
  179. package/src/fragment.ts +0 -727
  180. package/src/query/unit-of-work-multi-schema.test.ts +0 -64
  181. package/src/shared/settings-schema.ts +0 -61
  182. package/src/uow-context-integration.test.ts +0 -102
  183. package/src/uow-context.test.ts +0 -182
@@ -1 +0,0 @@
1
- {"version":3,"file":"fragment.js","names":["serviceContext: DatabaseRequestThisContext","#name","#schema","#namespace","#dependencies","#services","#usedServices","#providedServices","#createDatabaseContext","services","implementation: TService"],"sources":["../src/fragment.ts"],"sourcesContent":["import type { AnySchema } from \"./schema/create\";\nimport type { AbstractQuery } from \"./query/query\";\nimport type { DatabaseAdapter } from \"./adapters/adapters\";\nimport { bindServicesToContext, type BoundServices } from \"./bind-services\";\nimport { AsyncLocalStorage } from \"async_hooks\";\nimport type { IUnitOfWorkBase, UnitOfWorkSchemaView } from \"./query/unit-of-work\";\nimport type { RequestThisContext } from \"@fragno-dev/core/api\";\n\nexport const uowStorage = new AsyncLocalStorage<IUnitOfWorkBase>();\n\n/**\n * Service context for database fragments, providing access to the Unit of Work.\n */\nexport interface DatabaseRequestThisContext extends RequestThisContext {\n /**\n * Get the Unit of Work from the current context.\n * @param schema - Optional schema to get a typed view. If not provided, returns the base UOW.\n * @returns IUnitOfWorkBase if no schema provided, or typed UnitOfWorkSchemaView if schema provided.\n */\n getUnitOfWork(): IUnitOfWorkBase;\n getUnitOfWork<TSchema extends AnySchema>(\n schema: TSchema,\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n ): UnitOfWorkSchemaView<TSchema, [], any>;\n}\n\nexport const serviceContext: DatabaseRequestThisContext = {\n getUnitOfWork<TSchema extends AnySchema>(\n schema?: TSchema,\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n ): any {\n const uow = uowStorage.getStore();\n if (!uow) {\n throw new Error(\"No UnitOfWork in context. Service must be called within a route handler.\");\n }\n if (schema) {\n return uow.forSchema(schema);\n }\n return uow;\n },\n};\n\nexport function withUnitOfWork<T>(uow: IUnitOfWorkBase, callback: () => T): Promise<T> {\n return Promise.resolve(uowStorage.run(uow, callback));\n}\n\n/**\n * Type helper that enforces DatabaseRequestThisContext on all functions in a service object\n */\ntype WithDatabaseThis<T> = {\n [K in keyof T]: T[K] extends (...args: infer A) => infer R\n ? (this: DatabaseRequestThisContext, ...args: A) => R\n : T[K] extends Record<string, unknown>\n ? WithDatabaseThis<T[K]>\n : T[K];\n};\n\n// Import types from fragno package\nimport type {\n FragmentDefinition,\n RouteHandler,\n FragnoPublicConfig,\n RequestInputContext,\n RequestOutputContext,\n} from \"@fragno-dev/core\";\n\nexport { bindServicesToContext, type BoundServices };\n\n/**\n * Route handler type for database fragments with access to Unit of Work.\n */\nexport type DatabaseRouteHandler = (\n this: DatabaseRequestThisContext,\n inputContext: RequestInputContext,\n outputContext: RequestOutputContext,\n) => Promise<Response>;\n\n/**\n * Extended FragnoPublicConfig that includes a database adapter.\n * Use this type when creating fragments with database support.\n */\nexport type FragnoPublicConfigWithDatabase = FragnoPublicConfig & {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n databaseAdapter: DatabaseAdapter<any>;\n};\n\n/**\n * Additional context provided to database fragments containing the database adapter and ORM instance.\n */\nexport type DatabaseFragmentContext<TSchema extends AnySchema> = {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n databaseAdapter: DatabaseAdapter<any>;\n orm: AbstractQuery<TSchema>;\n};\n\nexport class DatabaseFragmentBuilder<\n const TSchema extends AnySchema,\n const TConfig,\n const TDeps = {},\n const TServices = {},\n const TUsedServices = {},\n const TProvidedServices = {},\n> {\n // Type-only property to expose type parameters for better inference\n readonly $types!: {\n schema: TSchema;\n config: TConfig;\n deps: TDeps;\n services: TServices;\n usedServices: TUsedServices;\n providedServices: TProvidedServices;\n };\n\n #name: string;\n #schema?: TSchema;\n #namespace?: string;\n #dependencies?: (\n context: {\n config: TConfig;\n fragnoConfig: FragnoPublicConfig;\n } & DatabaseFragmentContext<TSchema>,\n ) => TDeps;\n #services?: (\n context: {\n config: TConfig;\n fragnoConfig: FragnoPublicConfig;\n deps: TDeps & TUsedServices;\n defineService: <T extends Record<string, unknown>>(\n services: WithDatabaseThis<T>,\n ) => WithDatabaseThis<T>;\n } & DatabaseFragmentContext<TSchema>,\n ) => TServices;\n #usedServices?: Record<string, { name: string; required: boolean }>;\n #providedServices?: Record<string, unknown>;\n\n constructor(options: {\n name: string;\n schema?: TSchema;\n namespace?: string;\n dependencies?: (\n context: {\n config: TConfig;\n fragnoConfig: FragnoPublicConfig;\n } & DatabaseFragmentContext<TSchema>,\n ) => TDeps;\n services?: (\n context: {\n config: TConfig;\n fragnoConfig: FragnoPublicConfig;\n deps: TDeps & TUsedServices;\n defineService: <T extends Record<string, unknown>>(\n services: WithDatabaseThis<T>,\n ) => WithDatabaseThis<T>;\n } & DatabaseFragmentContext<TSchema>,\n ) => TServices;\n usedServices?: Record<string, { name: string; required: boolean }>;\n providedServices?: Record<string, unknown>;\n }) {\n this.#name = options.name;\n this.#schema = options.schema;\n this.#namespace = options.namespace;\n this.#dependencies = options.dependencies;\n this.#services = options.services;\n this.#usedServices = options.usedServices;\n this.#providedServices = options.providedServices;\n }\n\n get $requiredOptions(): FragnoPublicConfigWithDatabase {\n throw new Error(\"Type only method. Do not call.\");\n }\n\n get definition(): FragmentDefinition<\n TConfig,\n TDeps,\n BoundServices<TServices>,\n { databaseSchema?: TSchema; databaseNamespace: string },\n BoundServices<TUsedServices>,\n BoundServices<TProvidedServices>,\n DatabaseRequestThisContext\n > {\n const schema = this.#schema;\n const namespace = this.#namespace ?? \"\";\n const name = this.#name;\n const dependencies = this.#dependencies;\n const services = this.#services;\n const providedServices = this.#providedServices;\n\n return {\n name,\n dependencies: (config: TConfig, options: FragnoPublicConfig) => {\n const dbContext = this.#createDatabaseContext(options, schema, namespace, name);\n return dependencies?.({ config, fragnoConfig: options, ...dbContext }) ?? ({} as TDeps);\n },\n services: (\n config: TConfig,\n options: FragnoPublicConfig,\n deps: TDeps & BoundServices<TUsedServices>,\n ) => {\n const dbContext = this.#createDatabaseContext(options, schema, namespace, name);\n // Cast deps back to raw type for internal services function.\n // This is safe because:\n // 1. deps are already bound (their 'this' parameters are stripped)\n // 2. The services function expects raw types but only uses the public API\n // 3. BoundServices<T> has the same runtime shape as T (just without 'this')\n\n // defineService provides typing for service functions\n // It expects the input to already have proper 'this' types on functions\n const defineService = <T extends Record<string, unknown>>(\n services: WithDatabaseThis<T>,\n ): WithDatabaseThis<T> => services;\n\n const rawServices =\n services?.({\n config,\n fragnoConfig: options,\n deps: deps as TDeps & TUsedServices,\n defineService,\n ...dbContext,\n }) ?? ({} as TServices);\n\n // Bind all service methods to serviceContext\n return bindServicesToContext(\n rawServices as Record<string, unknown>,\n ) as BoundServices<TServices>;\n },\n additionalContext: {\n databaseSchema: schema,\n databaseNamespace: namespace,\n },\n createHandlerWrapper: schema\n ? (options: FragnoPublicConfig) => {\n const dbContext = this.#createDatabaseContext(options, schema, namespace, name);\n const { orm } = dbContext;\n\n // Return handler wrapper function\n return (handler: DatabaseRouteHandler): RouteHandler => {\n return async (inputContext, outputContext) => {\n // Create UOW for this request\n const uow = orm.createUnitOfWork();\n\n // Execute handler within AsyncLocalStorage context\n return withUnitOfWork(uow, async () => {\n // Bind handler to serviceContext so it has access to getUnitOfWork via 'this'\n const boundHandler = handler.bind(serviceContext);\n return boundHandler(inputContext, outputContext);\n });\n };\n };\n }\n : undefined,\n usedServices: this.#usedServices as\n | {\n [K in keyof TUsedServices]: { name: string; required: boolean };\n }\n | undefined,\n // Pass providedServices as-is - let fragment-instantiation.ts handle resolution\n // The factory functions will be called by createFragment\n providedServices: providedServices as\n | {\n [K in keyof BoundServices<TProvidedServices>]: BoundServices<TProvidedServices>[K];\n }\n | ((\n config: TConfig,\n options: FragnoPublicConfig,\n deps: TDeps & BoundServices<TUsedServices>,\n ) => BoundServices<TProvidedServices>)\n | undefined,\n };\n }\n\n #createDatabaseContext(\n options: FragnoPublicConfig,\n schema: TSchema | undefined,\n namespace: string,\n name: string,\n ): DatabaseFragmentContext<TSchema> {\n // Safe cast: FragnoPublicConfig is extended with databaseAdapter by the user\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const databaseAdapter = (options as any).databaseAdapter as DatabaseAdapter<any> | undefined;\n\n if (!databaseAdapter) {\n throw new Error(`Fragment '${name}' requires a database adapter in options.databaseAdapter`);\n }\n if (!schema) {\n throw new Error(`Fragment '${name}' requires a schema. Use withDatabase() to provide one.`);\n }\n\n // Safe cast: we create a query engine for TSchema and know it will be AbstractQuery<TSchema>\n const orm = databaseAdapter.createQueryEngine(\n schema,\n namespace,\n ) as unknown as AbstractQuery<TSchema>;\n\n return { databaseAdapter, orm };\n }\n\n withDatabase<TNewSchema extends AnySchema>(\n schema: TNewSchema,\n namespace?: string,\n ): DatabaseFragmentBuilder<\n TNewSchema,\n TConfig,\n TDeps,\n TServices,\n TUsedServices,\n TProvidedServices\n > {\n return new DatabaseFragmentBuilder<\n TNewSchema,\n TConfig,\n TDeps,\n TServices,\n TUsedServices,\n TProvidedServices\n >({\n name: this.#name,\n schema,\n namespace: namespace ?? this.#name + \"-db\",\n dependencies: this.#dependencies as\n | ((\n context: {\n config: TConfig;\n fragnoConfig: FragnoPublicConfig;\n } & DatabaseFragmentContext<TNewSchema>,\n ) => TDeps)\n | undefined,\n services: this.#services as\n | ((\n context: {\n config: TConfig;\n fragnoConfig: FragnoPublicConfig;\n deps: TDeps & TUsedServices;\n defineService: <T extends Record<string, unknown>>(\n services: WithDatabaseThis<T>,\n ) => WithDatabaseThis<T>;\n } & DatabaseFragmentContext<TNewSchema>,\n ) => TServices)\n | undefined,\n usedServices: this.#usedServices,\n providedServices: this.#providedServices,\n });\n }\n\n withDependencies<TNewDeps>(\n fn: (\n context: {\n config: TConfig;\n fragnoConfig: FragnoPublicConfig;\n } & DatabaseFragmentContext<TSchema>,\n ) => TNewDeps,\n ): DatabaseFragmentBuilder<TSchema, TConfig, TNewDeps, {}, TUsedServices, TProvidedServices> {\n return new DatabaseFragmentBuilder<\n TSchema,\n TConfig,\n TNewDeps,\n {},\n TUsedServices,\n TProvidedServices\n >({\n name: this.#name,\n schema: this.#schema,\n namespace: this.#namespace,\n dependencies: fn,\n services: undefined,\n usedServices: this.#usedServices,\n providedServices: this.#providedServices,\n });\n }\n\n /**\n * Declare that this fragment uses a service.\n * @param serviceName - The name of the service to use\n * @param options - Optional configuration: { optional: boolean } (defaults to required)\n */\n usesService<TServiceName extends string, TService>(\n serviceName: TServiceName,\n options?: { optional?: false },\n ): DatabaseFragmentBuilder<\n TSchema,\n TConfig,\n TDeps,\n TServices,\n TUsedServices & { [K in TServiceName]: TService },\n TProvidedServices\n >;\n usesService<TServiceName extends string, TService>(\n serviceName: TServiceName,\n options: { optional: true },\n ): DatabaseFragmentBuilder<\n TSchema,\n TConfig,\n TDeps,\n TServices,\n TUsedServices & { [K in TServiceName]: TService | undefined },\n TProvidedServices\n >;\n usesService<TServiceName extends string, TService>(\n serviceName: TServiceName,\n options?: { optional?: boolean },\n ): DatabaseFragmentBuilder<\n TSchema,\n TConfig,\n TDeps,\n TServices,\n TUsedServices & { [K in TServiceName]: TService | TService | undefined },\n TProvidedServices\n > {\n const isOptional = options?.optional ?? false;\n return new DatabaseFragmentBuilder<\n TSchema,\n TConfig,\n TDeps,\n TServices,\n TUsedServices & { [K in TServiceName]: TService | (TService | undefined) },\n TProvidedServices\n >({\n name: this.#name,\n schema: this.#schema,\n namespace: this.#namespace,\n dependencies: this.#dependencies as unknown as\n | ((\n context: {\n config: TConfig;\n fragnoConfig: FragnoPublicConfig;\n } & DatabaseFragmentContext<TSchema>,\n ) => TDeps)\n | undefined,\n services: this.#services as unknown as\n | ((\n context: {\n config: TConfig;\n fragnoConfig: FragnoPublicConfig;\n deps: TDeps &\n (TUsedServices & { [K in TServiceName]: TService | (TService | undefined) });\n } & DatabaseFragmentContext<TSchema>,\n ) => TServices)\n | undefined,\n usedServices: {\n ...this.#usedServices,\n [serviceName]: { name: serviceName, required: !isOptional },\n },\n providedServices: this.#providedServices,\n });\n }\n\n /**\n * Define services for this fragment (unnamed).\n * Functions in the service will have access to DatabaseRequestThisContext via `this` if using `defineService`.\n *\n * @example\n * With `this` context:\n * ```ts\n * .providesService(({ defineService }) => defineService({\n * createUser: function(name: string) {\n * const uow = this.getUnitOfWork(mySchema);\n * return uow.create('user', { name });\n * }\n * }))\n * ```\n *\n * Without `this` context:\n * ```ts\n * .providesService(({ db }) => ({\n * createUser: async (name: string) => {\n * return db.create('user', { name });\n * }\n * }))\n * ```\n */\n providesService<TNewServices>(\n fn: (context: {\n config: TConfig;\n fragnoConfig: FragnoPublicConfig;\n deps: TDeps & TUsedServices;\n db: AbstractQuery<TSchema>;\n defineService: <T extends Record<string, unknown>>(\n services: WithDatabaseThis<T>,\n ) => WithDatabaseThis<T>;\n }) => TNewServices,\n ): DatabaseFragmentBuilder<\n TSchema,\n TConfig,\n TDeps,\n TNewServices,\n TUsedServices,\n TProvidedServices\n >;\n\n /**\n * Provide a named service that other fragments can use.\n * Functions in the service will have access to DatabaseRequestThisContext via `this` if using `defineService`.\n * You can also pass a service object directly instead of a callback.\n *\n * @example\n * With callback and `this` context:\n * ```ts\n * .providesService(\"myService\", ({ defineService }) => defineService({\n * createUser: function(name: string) {\n * const uow = this.getUnitOfWork(mySchema);\n * return uow.create('user', { name });\n * }\n * }))\n * ```\n *\n * With callback, no `this` context:\n * ```ts\n * .providesService(\"myService\", ({ db }) => ({\n * createUser: async (name: string) => {\n * return db.create('user', { name });\n * }\n * }))\n * ```\n *\n * With direct object:\n * ```ts\n * .providesService(\"myService\", {\n * createUser: async (name: string) => { ... }\n * })\n * ```\n */\n providesService<TServiceName extends string, TService>(\n serviceName: TServiceName,\n fnOrService:\n | ((context: {\n config: TConfig;\n fragnoConfig: FragnoPublicConfig;\n deps: TDeps & TUsedServices;\n db: AbstractQuery<TSchema>;\n defineService: <T extends Record<string, unknown>>(\n services: WithDatabaseThis<T>,\n ) => WithDatabaseThis<T>;\n }) => TService)\n | TService,\n ): DatabaseFragmentBuilder<\n TSchema,\n TConfig,\n TDeps,\n TServices,\n TUsedServices,\n TProvidedServices & { [K in TServiceName]: BoundServices<TService> }\n >;\n\n providesService<TServiceName extends string, TService>(\n ...args:\n | [\n fn: (context: {\n config: TConfig;\n fragnoConfig: FragnoPublicConfig;\n deps: TDeps & TUsedServices;\n db: AbstractQuery<TSchema>;\n defineService: <T extends Record<string, unknown>>(\n services: WithDatabaseThis<T>,\n ) => WithDatabaseThis<T>;\n }) => TService,\n ]\n | [\n serviceName: TServiceName,\n fnOrService:\n | ((context: {\n config: TConfig;\n fragnoConfig: FragnoPublicConfig;\n deps: TDeps & TUsedServices;\n db: AbstractQuery<TSchema>;\n defineService: <T extends Record<string, unknown>>(\n services: WithDatabaseThis<T>,\n ) => WithDatabaseThis<T>;\n }) => TService)\n | TService,\n ]\n ):\n | DatabaseFragmentBuilder<TSchema, TConfig, TDeps, TService, TUsedServices, TProvidedServices>\n | DatabaseFragmentBuilder<\n TSchema,\n TConfig,\n TDeps,\n TServices,\n TUsedServices,\n TProvidedServices & { [K in TServiceName]: BoundServices<TService> }\n > {\n if (args.length === 1) {\n // Unnamed service - replaces withServices\n const [fn] = args;\n\n // Create a callback that takes a single context object (matching #services signature)\n // Note: We don't explicitly type the return to preserve the WithDatabaseThis wrapper\n const servicesCallback = (\n context: {\n config: TConfig;\n fragnoConfig: FragnoPublicConfig;\n deps: TDeps & TUsedServices;\n } & DatabaseFragmentContext<TSchema>,\n ) => {\n // defineService provides typing for service functions\n // It expects the input to already have proper 'this' types on functions\n const defineService = <T extends Record<string, unknown>>(\n services: WithDatabaseThis<T>,\n ): WithDatabaseThis<T> => services;\n\n const services = fn({\n config: context.config,\n fragnoConfig: context.fragnoConfig,\n deps: context.deps,\n db: context.orm,\n defineService,\n });\n\n // Return without casting to preserve the WithDatabaseThis wrapper\n return services;\n };\n\n return new DatabaseFragmentBuilder<\n TSchema,\n TConfig,\n TDeps,\n TService,\n TUsedServices,\n TProvidedServices\n >({\n name: this.#name,\n schema: this.#schema,\n namespace: this.#namespace,\n dependencies: this.#dependencies,\n // Safe cast: servicesCallback returns WithDatabaseThis<TService> but we store it as TService.\n // At runtime, bindServicesToContext will handle the 'this' binding properly.\n services: servicesCallback as (\n context: {\n config: TConfig;\n fragnoConfig: FragnoPublicConfig;\n deps: TDeps & TUsedServices;\n defineService: <T extends Record<string, unknown>>(\n services: WithDatabaseThis<T>,\n ) => WithDatabaseThis<T>;\n } & DatabaseFragmentContext<TSchema>,\n ) => TService,\n usedServices: this.#usedServices,\n providedServices: this.#providedServices,\n });\n } else {\n // Named service\n const [serviceName, fnOrService] = args;\n\n // Create a callback that provides the full context\n const createService = (\n config: TConfig,\n options: FragnoPublicConfig,\n deps: TDeps & TUsedServices,\n ): BoundServices<TService> => {\n const dbContext = this.#createDatabaseContext(\n options,\n this.#schema,\n this.#namespace ?? \"\",\n this.#name,\n );\n\n // Check if fnOrService is a function or a direct object\n let implementation: TService;\n if (typeof fnOrService === \"function\") {\n // It's a callback - call it with context\n // defineService provides typing for service functions\n // It expects the input to already have proper 'this' types on functions\n const defineService = <T extends Record<string, unknown>>(\n services: WithDatabaseThis<T>,\n ): WithDatabaseThis<T> => services;\n\n // Safe cast: we checked that fnOrService is a function\n implementation = (\n fnOrService as (context: {\n config: TConfig;\n fragnoConfig: FragnoPublicConfig;\n deps: TDeps & TUsedServices;\n db: AbstractQuery<TSchema>;\n defineService: <T extends Record<string, unknown>>(\n services: WithDatabaseThis<T>,\n ) => WithDatabaseThis<T>;\n }) => TService\n )({\n config,\n fragnoConfig: options,\n deps,\n db: dbContext.orm,\n defineService,\n });\n } else {\n // It's a direct object\n implementation = fnOrService;\n }\n\n // Bind the service implementation so methods have access to serviceContext\n return bindServicesToContext(\n implementation as Record<string, unknown>,\n ) as BoundServices<TService>;\n };\n\n // We need to evaluate this immediately to store in providedServices\n // For now, we'll create a placeholder that will be evaluated when fragment is instantiated\n // Actually, we need to defer this until fragment instantiation\n // Let's store a function that creates the service\n return new DatabaseFragmentBuilder<\n TSchema,\n TConfig,\n TDeps,\n TServices,\n TUsedServices,\n TProvidedServices & { [K in TServiceName]: BoundServices<TService> }\n >({\n name: this.#name,\n schema: this.#schema,\n namespace: this.#namespace,\n dependencies: this.#dependencies,\n services: this.#services,\n usedServices: this.#usedServices,\n providedServices: {\n ...this.#providedServices,\n [serviceName]: createService,\n } as Record<string, unknown>,\n });\n }\n }\n}\n\nexport function defineFragmentWithDatabase<TConfig = {}>(\n name: string,\n): DatabaseFragmentBuilder<never, TConfig, {}, {}, {}, {}> {\n return new DatabaseFragmentBuilder<never, TConfig, {}, {}, {}, {}>({\n name,\n });\n}\n"],"mappings":"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"}
@@ -1,36 +0,0 @@
1
- import { column, idColumn, schema } from "../schema/create.js";
2
-
3
- //#region src/shared/settings-schema.ts
4
- const SETTINGS_TABLE_NAME = "fragno_db_settings";
5
- const SETTINGS_NAMESPACE = "fragno-db-settings";
6
- const settingsSchema = schema((s) => {
7
- return s.addTable(SETTINGS_TABLE_NAME, (t) => {
8
- return t.addColumn("id", idColumn()).addColumn("key", column("string")).addColumn("value", column("string")).createIndex("unique_key", ["key"], { unique: true });
9
- });
10
- });
11
- function createSettingsManager(queryEngine, namespace) {
12
- return {
13
- async get(key) {
14
- const [[result]] = await queryEngine.createUnitOfWork().find(SETTINGS_TABLE_NAME, (b) => b.whereIndex("unique_key", (eb) => eb("key", "=", `${namespace}.${key}`))).executeRetrieve();
15
- return result;
16
- },
17
- async set(key, value) {
18
- const uow = queryEngine.createUnitOfWork("createSettingsManager#set").find(SETTINGS_TABLE_NAME, (b) => b.whereIndex("unique_key", (eb) => eb("key", "=", `${namespace}.${key}`)));
19
- const [[existing]] = await uow.executeRetrieve();
20
- if (existing) uow.update(SETTINGS_TABLE_NAME, existing.id, (b) => b.set({ value }).check());
21
- else uow.create(SETTINGS_TABLE_NAME, {
22
- key: `${namespace}.${key}`,
23
- value
24
- });
25
- const { success } = await uow.executeMutations();
26
- if (!success) throw new Error("Failed to set schema version");
27
- },
28
- async delete(id) {
29
- await queryEngine.delete(SETTINGS_TABLE_NAME, id);
30
- }
31
- };
32
- }
33
-
34
- //#endregion
35
- export { SETTINGS_NAMESPACE, SETTINGS_TABLE_NAME, createSettingsManager, settingsSchema };
36
- //# sourceMappingURL=settings-schema.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"settings-schema.js","names":[],"sources":["../../src/shared/settings-schema.ts"],"sourcesContent":["import type { AbstractQuery } from \"../query/query\";\nimport { schema, idColumn, column, type FragnoId } from \"../schema/create\";\n\nexport const SETTINGS_TABLE_NAME = \"fragno_db_settings\" as const;\nexport const SETTINGS_NAMESPACE = \"fragno-db-settings\" as const;\n\nexport const settingsSchema = schema((s) => {\n return s.addTable(SETTINGS_TABLE_NAME, (t) => {\n return t\n .addColumn(\"id\", idColumn())\n .addColumn(\"key\", column(\"string\"))\n .addColumn(\"value\", column(\"string\"))\n .createIndex(\"unique_key\", [\"key\"], { unique: true });\n });\n});\n\nexport function createSettingsManager(\n // oxlint-disable-next-line no-explicit-any\n queryEngine: AbstractQuery<typeof settingsSchema, any>,\n namespace: string,\n) {\n return {\n async get(key: string): Promise<{ id: FragnoId; key: string; value: string } | undefined> {\n const uow = queryEngine\n .createUnitOfWork()\n .find(SETTINGS_TABLE_NAME, (b) =>\n b.whereIndex(\"unique_key\", (eb) => eb(\"key\", \"=\", `${namespace}.${key}`)),\n );\n const [[result]] = await uow.executeRetrieve();\n return result; // Safe: result can be undefined if key doesn't exist\n },\n\n async set(key: string, value: string) {\n const uow = queryEngine\n .createUnitOfWork(\"createSettingsManager#set\")\n .find(SETTINGS_TABLE_NAME, (b) =>\n b.whereIndex(\"unique_key\", (eb) => eb(\"key\", \"=\", `${namespace}.${key}`)),\n );\n const [[existing]] = await uow.executeRetrieve();\n\n if (existing) {\n uow.update(SETTINGS_TABLE_NAME, existing.id, (b) => b.set({ value }).check());\n } else {\n uow.create(SETTINGS_TABLE_NAME, {\n key: `${namespace}.${key}`,\n value,\n });\n }\n\n const { success } = await uow.executeMutations();\n\n if (!success) {\n throw new Error(\"Failed to set schema version\");\n }\n },\n\n async delete(id: FragnoId) {\n await queryEngine.delete(SETTINGS_TABLE_NAME, id);\n },\n };\n}\n"],"mappings":";;;AAGA,MAAa,sBAAsB;AACnC,MAAa,qBAAqB;AAElC,MAAa,iBAAiB,QAAQ,MAAM;AAC1C,QAAO,EAAE,SAAS,sBAAsB,MAAM;AAC5C,SAAO,EACJ,UAAU,MAAM,UAAU,CAAC,CAC3B,UAAU,OAAO,OAAO,SAAS,CAAC,CAClC,UAAU,SAAS,OAAO,SAAS,CAAC,CACpC,YAAY,cAAc,CAAC,MAAM,EAAE,EAAE,QAAQ,MAAM,CAAC;GACvD;EACF;AAEF,SAAgB,sBAEd,aACA,WACA;AACA,QAAO;EACL,MAAM,IAAI,KAAgF;GAMxF,MAAM,CAAC,CAAC,WAAW,MALP,YACT,kBAAkB,CAClB,KAAK,sBAAsB,MAC1B,EAAE,WAAW,eAAe,OAAO,GAAG,OAAO,KAAK,GAAG,UAAU,GAAG,MAAM,CAAC,CAC1E,CAC0B,iBAAiB;AAC9C,UAAO;;EAGT,MAAM,IAAI,KAAa,OAAe;GACpC,MAAM,MAAM,YACT,iBAAiB,4BAA4B,CAC7C,KAAK,sBAAsB,MAC1B,EAAE,WAAW,eAAe,OAAO,GAAG,OAAO,KAAK,GAAG,UAAU,GAAG,MAAM,CAAC,CAC1E;GACH,MAAM,CAAC,CAAC,aAAa,MAAM,IAAI,iBAAiB;AAEhD,OAAI,SACF,KAAI,OAAO,qBAAqB,SAAS,KAAK,MAAM,EAAE,IAAI,EAAE,OAAO,CAAC,CAAC,OAAO,CAAC;OAE7E,KAAI,OAAO,qBAAqB;IAC9B,KAAK,GAAG,UAAU,GAAG;IACrB;IACD,CAAC;GAGJ,MAAM,EAAE,YAAY,MAAM,IAAI,kBAAkB;AAEhD,OAAI,CAAC,QACH,OAAM,IAAI,MAAM,+BAA+B;;EAInD,MAAM,OAAO,IAAc;AACzB,SAAM,YAAY,OAAO,qBAAqB,GAAG;;EAEpD"}
@@ -1,214 +0,0 @@
1
- import { describe, it, expect } from "vitest";
2
- import { bindServicesToContext } from "./bind-services";
3
- import { withUnitOfWork, type DatabaseRequestThisContext } from "./fragment";
4
- import type { IUnitOfWorkBase } from "./query/unit-of-work";
5
- import { schema, idColumn } from "./schema/create";
6
-
7
- // Create a simple test schema for tests
8
- const testSchema = schema((s) => {
9
- return s.addTable("test", (t) => {
10
- return t.addColumn("id", idColumn());
11
- });
12
- });
13
-
14
- describe("bindServicesToContext", () => {
15
- it("should bind simple function to context", async () => {
16
- const mockSchemaView = { test: "schema-view" };
17
- const mockUow = {
18
- test: "uow",
19
- forSchema: () => mockSchemaView,
20
- } as unknown as IUnitOfWorkBase;
21
-
22
- const services = {
23
- testMethod: function (this: DatabaseRequestThisContext) {
24
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
25
- return this.getUnitOfWork(testSchema as any);
26
- },
27
- };
28
-
29
- const bound = bindServicesToContext(services);
30
-
31
- const result = await withUnitOfWork(mockUow, () => bound.testMethod());
32
-
33
- expect(result).toBe(mockSchemaView);
34
- });
35
-
36
- it("should bind multiple functions", async () => {
37
- const mockSchemaView = { test: "schema-view" };
38
- const mockUow = {
39
- test: "uow",
40
- forSchema: () => mockSchemaView,
41
- } as unknown as IUnitOfWorkBase;
42
-
43
- const services = {
44
- method1: function (this: DatabaseRequestThisContext) {
45
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
46
- return this.getUnitOfWork(testSchema as any);
47
- },
48
- method2: function (this: DatabaseRequestThisContext) {
49
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
50
- return this.getUnitOfWork(testSchema as any);
51
- },
52
- };
53
-
54
- const bound = bindServicesToContext(services);
55
-
56
- await withUnitOfWork(mockUow, () => {
57
- expect(bound.method1()).toBe(mockSchemaView);
58
- expect(bound.method2()).toBe(mockSchemaView);
59
- });
60
- });
61
-
62
- it("should bind nested service objects", async () => {
63
- const mockSchemaView = { test: "schema-view" };
64
- const mockUow = {
65
- test: "uow",
66
- forSchema: () => mockSchemaView,
67
- } as unknown as IUnitOfWorkBase;
68
-
69
- const services = {
70
- nested: {
71
- method: function (this: DatabaseRequestThisContext) {
72
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
73
- return this.getUnitOfWork(testSchema as any);
74
- },
75
- },
76
- };
77
-
78
- const bound = bindServicesToContext(services);
79
-
80
- const result = await withUnitOfWork(mockUow, () => bound.nested.method());
81
-
82
- expect(result).toBe(mockSchemaView);
83
- });
84
-
85
- it("should preserve non-function properties", () => {
86
- const services = {
87
- method: function () {
88
- return "test";
89
- },
90
- constant: "value",
91
- number: 42,
92
- };
93
-
94
- const bound = bindServicesToContext(services);
95
-
96
- expect(bound.constant).toBe("value");
97
- expect(bound.number).toBe(42);
98
- expect(bound.method()).toBe("test");
99
- });
100
-
101
- it("should handle functions with parameters", async () => {
102
- const mockSchemaView = { test: "schema-view" };
103
- const mockUow = {
104
- test: "uow",
105
- forSchema: () => mockSchemaView,
106
- } as unknown as IUnitOfWorkBase;
107
-
108
- const services = {
109
- testMethod: function (this: DatabaseRequestThisContext, param1: string, param2: number) {
110
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
111
- const uow = this.getUnitOfWork(testSchema as any);
112
- return { uow, param1, param2 };
113
- },
114
- };
115
-
116
- const bound = bindServicesToContext(services);
117
-
118
- const result = await withUnitOfWork(mockUow, () => bound.testMethod("hello", 123));
119
-
120
- expect(result.uow).toBe(mockSchemaView);
121
- expect(result.param1).toBe("hello");
122
- expect(result.param2).toBe(123);
123
- });
124
-
125
- it("should handle async functions", async () => {
126
- const mockSchemaView = { test: "schema-view" };
127
- const mockUow = {
128
- test: "uow",
129
- forSchema: () => mockSchemaView,
130
- } as unknown as IUnitOfWorkBase;
131
-
132
- const services = {
133
- asyncMethod: async function (this: DatabaseRequestThisContext) {
134
- await new Promise((resolve) => setTimeout(resolve, 10));
135
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
136
- return this.getUnitOfWork(testSchema as any);
137
- },
138
- };
139
-
140
- const bound = bindServicesToContext(services);
141
-
142
- const result = await withUnitOfWork(mockUow, async () => await bound.asyncMethod());
143
-
144
- expect(result).toBe(mockSchemaView);
145
- });
146
-
147
- it("should not bind arrays", () => {
148
- const services = {
149
- array: [1, 2, 3],
150
- method: function () {
151
- return "test";
152
- },
153
- };
154
-
155
- const bound = bindServicesToContext(services);
156
-
157
- expect(bound.array).toEqual([1, 2, 3]);
158
- expect(Array.isArray(bound.array)).toBe(true);
159
- });
160
-
161
- it("should handle deeply nested objects", async () => {
162
- const mockSchemaView = { test: "schema-view" };
163
- const mockUow = {
164
- test: "uow",
165
- forSchema: () => mockSchemaView,
166
- } as unknown as IUnitOfWorkBase;
167
-
168
- const services = {
169
- level1: {
170
- level2: {
171
- level3: {
172
- method: function (this: DatabaseRequestThisContext) {
173
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
174
- return this.getUnitOfWork(testSchema as any);
175
- },
176
- },
177
- },
178
- },
179
- };
180
-
181
- const bound = bindServicesToContext(services);
182
-
183
- const result = await withUnitOfWork(mockUow, () => bound.level1.level2.level3.method());
184
-
185
- expect(result).toBe(mockSchemaView);
186
- });
187
-
188
- it("should allow bound services to access UOW independently", async () => {
189
- const mockSchemaView = { test: "schema-view" };
190
- const mockUow = {
191
- test: "uow",
192
- forSchema: () => mockSchemaView,
193
- } as unknown as IUnitOfWorkBase;
194
-
195
- const services = {
196
- getUow: function (this: DatabaseRequestThisContext) {
197
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
198
- return this.getUnitOfWork(testSchema as any);
199
- },
200
- callOther: function (this: DatabaseRequestThisContext) {
201
- // Both methods can access UOW via their own `this` context
202
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
203
- return this.getUnitOfWork(testSchema as any);
204
- },
205
- };
206
-
207
- const bound = bindServicesToContext(services);
208
-
209
- await withUnitOfWork(mockUow, () => {
210
- expect(bound.getUow()).toBe(mockSchemaView);
211
- expect(bound.callOther()).toBe(mockSchemaView);
212
- });
213
- });
214
- });
@@ -1,37 +0,0 @@
1
- import { serviceContext } from "./fragment";
2
-
3
- // Type helper to remove 'this' parameter from functions
4
- type OmitThisParameter<T> = T extends (this: infer _This, ...args: infer A) => infer R
5
- ? (...args: A) => R
6
- : T;
7
-
8
- // Recursively remove 'this' parameter from all functions in an object
9
- export type BoundServices<T> = {
10
- [K in keyof T]: T[K] extends (...args: never[]) => unknown
11
- ? OmitThisParameter<T[K]>
12
- : T[K] extends Record<string, unknown>
13
- ? BoundServices<T[K]>
14
- : T[K];
15
- };
16
-
17
- export function bindServicesToContext<T extends Record<string, unknown>>(
18
- services: T,
19
- ): BoundServices<T> {
20
- const bound = {} as BoundServices<T>;
21
-
22
- for (const [key, value] of Object.entries(services)) {
23
- if (typeof value === "function") {
24
- // Bind function to serviceContext
25
- bound[key as keyof T] = value.bind(serviceContext) as BoundServices<T>[keyof T];
26
- } else if (value && typeof value === "object" && !Array.isArray(value)) {
27
- // Recursively bind nested service objects
28
- bound[key as keyof T] = bindServicesToContext(
29
- value as Record<string, unknown>,
30
- ) as BoundServices<T>[keyof T];
31
- } else {
32
- bound[key as keyof T] = value as BoundServices<T>[keyof T];
33
- }
34
- }
35
-
36
- return bound;
37
- }