@fragno-dev/db 0.2.2 → 0.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.turbo/turbo-build.log +202 -140
- package/CHANGELOG.md +35 -0
- package/README.md +30 -9
- package/dist/adapters/adapters.d.ts +23 -21
- package/dist/adapters/adapters.d.ts.map +1 -1
- package/dist/adapters/adapters.js.map +1 -1
- package/dist/adapters/generic-sql/driver-config.d.ts +16 -1
- package/dist/adapters/generic-sql/driver-config.d.ts.map +1 -1
- package/dist/adapters/generic-sql/driver-config.js +23 -1
- package/dist/adapters/generic-sql/driver-config.js.map +1 -1
- package/dist/adapters/generic-sql/generic-sql-adapter.d.ts +27 -9
- package/dist/adapters/generic-sql/generic-sql-adapter.d.ts.map +1 -1
- package/dist/adapters/generic-sql/generic-sql-adapter.js +55 -16
- package/dist/adapters/generic-sql/generic-sql-adapter.js.map +1 -1
- package/dist/adapters/generic-sql/generic-sql-uow-executor.js +129 -3
- package/dist/adapters/generic-sql/generic-sql-uow-executor.js.map +1 -1
- package/dist/adapters/generic-sql/migration/dialect/mysql.js +24 -5
- package/dist/adapters/generic-sql/migration/dialect/mysql.js.map +1 -1
- package/dist/adapters/generic-sql/migration/dialect/postgres.js +6 -5
- package/dist/adapters/generic-sql/migration/dialect/postgres.js.map +1 -1
- package/dist/adapters/generic-sql/migration/dialect/sqlite.js +21 -10
- package/dist/adapters/generic-sql/migration/dialect/sqlite.js.map +1 -1
- package/dist/adapters/generic-sql/migration/prepared-migrations.d.ts.map +1 -1
- package/dist/adapters/generic-sql/migration/prepared-migrations.js +8 -8
- package/dist/adapters/generic-sql/migration/prepared-migrations.js.map +1 -1
- package/dist/adapters/generic-sql/migration/sql-generator.js +74 -51
- package/dist/adapters/generic-sql/migration/sql-generator.js.map +1 -1
- package/dist/adapters/generic-sql/query/create-sql-query-compiler.js +6 -5
- package/dist/adapters/generic-sql/query/create-sql-query-compiler.js.map +1 -1
- package/dist/adapters/generic-sql/query/cursor-utils.js +42 -4
- package/dist/adapters/generic-sql/query/cursor-utils.js.map +1 -1
- package/dist/adapters/generic-sql/query/generic-sql-uow-operation-compiler.js +25 -17
- package/dist/adapters/generic-sql/query/generic-sql-uow-operation-compiler.js.map +1 -1
- package/dist/adapters/generic-sql/query/select-builder.js +5 -3
- package/dist/adapters/generic-sql/query/select-builder.js.map +1 -1
- package/dist/adapters/generic-sql/query/sql-query-compiler.js +15 -12
- package/dist/adapters/generic-sql/query/sql-query-compiler.js.map +1 -1
- package/dist/adapters/generic-sql/query/where-builder.js +39 -29
- package/dist/adapters/generic-sql/query/where-builder.js.map +1 -1
- package/dist/adapters/generic-sql/sqlite-storage.d.ts +13 -0
- package/dist/adapters/generic-sql/sqlite-storage.d.ts.map +1 -0
- package/dist/adapters/generic-sql/sqlite-storage.js +15 -0
- package/dist/adapters/generic-sql/sqlite-storage.js.map +1 -0
- package/dist/adapters/generic-sql/uow-decoder.js +7 -3
- package/dist/adapters/generic-sql/uow-decoder.js.map +1 -1
- package/dist/adapters/generic-sql/uow-encoder.js +28 -8
- package/dist/adapters/generic-sql/uow-encoder.js.map +1 -1
- package/dist/adapters/in-memory/condition-evaluator.js +131 -0
- package/dist/adapters/in-memory/condition-evaluator.js.map +1 -0
- package/dist/adapters/in-memory/errors.d.ts +13 -0
- package/dist/adapters/in-memory/errors.d.ts.map +1 -0
- package/dist/adapters/in-memory/errors.js +23 -0
- package/dist/adapters/in-memory/errors.js.map +1 -0
- package/dist/adapters/in-memory/in-memory-adapter.d.ts +27 -0
- package/dist/adapters/in-memory/in-memory-adapter.d.ts.map +1 -0
- package/dist/adapters/in-memory/in-memory-adapter.js +176 -0
- package/dist/adapters/in-memory/in-memory-adapter.js.map +1 -0
- package/dist/adapters/in-memory/in-memory-uow.js +648 -0
- package/dist/adapters/in-memory/in-memory-uow.js.map +1 -0
- package/dist/adapters/in-memory/index.d.ts +4 -0
- package/dist/adapters/in-memory/index.js +4 -0
- package/dist/adapters/in-memory/options.d.ts +28 -0
- package/dist/adapters/in-memory/options.d.ts.map +1 -0
- package/dist/adapters/in-memory/options.js +61 -0
- package/dist/adapters/in-memory/options.js.map +1 -0
- package/dist/adapters/in-memory/reference-resolution.js +26 -0
- package/dist/adapters/in-memory/reference-resolution.js.map +1 -0
- package/dist/adapters/in-memory/sorted-array-index.js +129 -0
- package/dist/adapters/in-memory/sorted-array-index.js.map +1 -0
- package/dist/adapters/in-memory/store.js +71 -0
- package/dist/adapters/in-memory/store.js.map +1 -0
- package/dist/adapters/in-memory/value-comparison.js +28 -0
- package/dist/adapters/in-memory/value-comparison.js.map +1 -0
- package/dist/adapters/shared/from-unit-of-work-compiler.js.map +1 -1
- package/dist/adapters/shared/uow-operation-compiler.js +11 -11
- package/dist/adapters/shared/uow-operation-compiler.js.map +1 -1
- package/dist/adapters/sql/index.d.ts +5 -0
- package/dist/adapters/sql/index.js +4 -0
- package/dist/db-fragment-definition-builder.d.ts +18 -7
- package/dist/db-fragment-definition-builder.d.ts.map +1 -1
- package/dist/db-fragment-definition-builder.js +116 -54
- package/dist/db-fragment-definition-builder.js.map +1 -1
- package/dist/dispatchers/cloudflare-do/index.d.ts +26 -0
- package/dist/dispatchers/cloudflare-do/index.d.ts.map +1 -0
- package/dist/dispatchers/cloudflare-do/index.js +63 -0
- package/dist/dispatchers/cloudflare-do/index.js.map +1 -0
- package/dist/dispatchers/node/index.d.ts +17 -0
- package/dist/dispatchers/node/index.d.ts.map +1 -0
- package/dist/dispatchers/node/index.js +59 -0
- package/dist/dispatchers/node/index.js.map +1 -0
- package/dist/fragments/internal-fragment.d.ts +79 -2
- package/dist/fragments/internal-fragment.d.ts.map +1 -1
- package/dist/fragments/internal-fragment.js +150 -32
- package/dist/fragments/internal-fragment.js.map +1 -1
- package/dist/fragments/internal-fragment.routes.js +29 -0
- package/dist/fragments/internal-fragment.routes.js.map +1 -0
- package/dist/fragments/internal-fragment.schema.d.ts +9 -0
- package/dist/fragments/internal-fragment.schema.d.ts.map +1 -0
- package/dist/fragments/internal-fragment.schema.js +22 -0
- package/dist/fragments/internal-fragment.schema.js.map +1 -0
- package/dist/hooks/durable-hooks-processor.d.ts +14 -0
- package/dist/hooks/durable-hooks-processor.d.ts.map +1 -0
- package/dist/hooks/durable-hooks-processor.js +32 -0
- package/dist/hooks/durable-hooks-processor.js.map +1 -0
- package/dist/hooks/hooks.d.ts +42 -1
- package/dist/hooks/hooks.d.ts.map +1 -1
- package/dist/hooks/hooks.js +72 -6
- package/dist/hooks/hooks.js.map +1 -1
- package/dist/migration-engine/auto-from-schema.js +14 -11
- package/dist/migration-engine/auto-from-schema.js.map +1 -1
- package/dist/migration-engine/generation-engine.d.ts +16 -10
- package/dist/migration-engine/generation-engine.d.ts.map +1 -1
- package/dist/migration-engine/generation-engine.js +72 -33
- package/dist/migration-engine/generation-engine.js.map +1 -1
- package/dist/migration-engine/shared.js.map +1 -1
- package/dist/mod.d.ts +15 -8
- package/dist/mod.d.ts.map +1 -1
- package/dist/mod.js +14 -8
- package/dist/mod.js.map +1 -1
- package/dist/naming/sql-naming.d.ts +19 -0
- package/dist/naming/sql-naming.d.ts.map +1 -0
- package/dist/naming/sql-naming.js +116 -0
- package/dist/naming/sql-naming.js.map +1 -0
- package/dist/node_modules/.pnpm/{rou3@0.7.10 → rou3@0.7.12}/node_modules/rou3/dist/index.js +8 -5
- package/dist/node_modules/.pnpm/rou3@0.7.12/node_modules/rou3/dist/index.js.map +1 -0
- package/dist/outbox/outbox-builder.js +156 -0
- package/dist/outbox/outbox-builder.js.map +1 -0
- package/dist/outbox/outbox.d.ts +52 -0
- package/dist/outbox/outbox.d.ts.map +1 -0
- package/dist/outbox/outbox.js +37 -0
- package/dist/outbox/outbox.js.map +1 -0
- package/dist/packages/fragno/dist/api/fragment-definition-builder.js +3 -2
- package/dist/packages/fragno/dist/api/fragment-definition-builder.js.map +1 -1
- package/dist/packages/fragno/dist/api/fragment-instantiator.js +164 -20
- package/dist/packages/fragno/dist/api/fragment-instantiator.js.map +1 -1
- package/dist/packages/fragno/dist/api/request-input-context.js +67 -0
- package/dist/packages/fragno/dist/api/request-input-context.js.map +1 -1
- package/dist/packages/fragno/dist/api/route.js +14 -1
- package/dist/packages/fragno/dist/api/route.js.map +1 -1
- package/dist/packages/fragno/dist/internal/trace-context.js +12 -0
- package/dist/packages/fragno/dist/internal/trace-context.js.map +1 -0
- package/dist/query/column-defaults.js +20 -4
- package/dist/query/column-defaults.js.map +1 -1
- package/dist/query/cursor.d.ts +3 -1
- package/dist/query/cursor.d.ts.map +1 -1
- package/dist/query/cursor.js +45 -14
- package/dist/query/cursor.js.map +1 -1
- package/dist/query/db-now.d.ts +8 -0
- package/dist/query/db-now.d.ts.map +1 -0
- package/dist/query/db-now.js +7 -0
- package/dist/query/db-now.js.map +1 -0
- package/dist/query/serialize/create-sql-serializer.js +3 -2
- package/dist/query/serialize/create-sql-serializer.js.map +1 -1
- package/dist/query/serialize/dialect/mysql-serializer.js +12 -6
- package/dist/query/serialize/dialect/mysql-serializer.js.map +1 -1
- package/dist/query/serialize/dialect/postgres-serializer.js +25 -7
- package/dist/query/serialize/dialect/postgres-serializer.js.map +1 -1
- package/dist/query/serialize/dialect/sqlite-serializer.js +55 -11
- package/dist/query/serialize/dialect/sqlite-serializer.js.map +1 -1
- package/dist/query/serialize/sql-serializer.js +2 -2
- package/dist/query/serialize/sql-serializer.js.map +1 -1
- package/dist/query/simple-query-interface.d.ts +6 -1
- package/dist/query/simple-query-interface.d.ts.map +1 -1
- package/dist/query/unit-of-work/execute-unit-of-work.d.ts.map +1 -1
- package/dist/query/unit-of-work/execute-unit-of-work.js +11 -6
- package/dist/query/unit-of-work/execute-unit-of-work.js.map +1 -1
- package/dist/query/unit-of-work/unit-of-work.d.ts +50 -14
- package/dist/query/unit-of-work/unit-of-work.d.ts.map +1 -1
- package/dist/query/unit-of-work/unit-of-work.js +86 -5
- package/dist/query/unit-of-work/unit-of-work.js.map +1 -1
- package/dist/query/value-decoding.js +9 -6
- package/dist/query/value-decoding.js.map +1 -1
- package/dist/query/value-encoding.js +29 -9
- package/dist/query/value-encoding.js.map +1 -1
- package/dist/schema/create.d.ts +38 -14
- package/dist/schema/create.d.ts.map +1 -1
- package/dist/schema/create.js +81 -42
- package/dist/schema/create.js.map +1 -1
- package/dist/schema/generate-id.js +2 -2
- package/dist/schema/generate-id.js.map +1 -1
- package/dist/schema/type-conversion/create-sql-type-mapper.js +3 -2
- package/dist/schema/type-conversion/create-sql-type-mapper.js.map +1 -1
- package/dist/schema/type-conversion/dialect/sqlite.js +9 -0
- package/dist/schema/type-conversion/dialect/sqlite.js.map +1 -1
- package/dist/schema/validator.d.ts +10 -0
- package/dist/schema/validator.d.ts.map +1 -0
- package/dist/schema/validator.js +123 -0
- package/dist/schema/validator.js.map +1 -0
- package/dist/schema-output/drizzle.d.ts +30 -0
- package/dist/schema-output/drizzle.d.ts.map +1 -0
- package/dist/{adapters/drizzle/generate.js → schema-output/drizzle.js} +82 -56
- package/dist/schema-output/drizzle.js.map +1 -0
- package/dist/schema-output/prisma.d.ts +17 -0
- package/dist/schema-output/prisma.d.ts.map +1 -0
- package/dist/schema-output/prisma.js +296 -0
- package/dist/schema-output/prisma.js.map +1 -0
- package/dist/util/default-database-adapter.js +61 -0
- package/dist/util/default-database-adapter.js.map +1 -0
- package/dist/with-database.d.ts +1 -1
- package/dist/with-database.d.ts.map +1 -1
- package/dist/with-database.js +12 -3
- package/dist/with-database.js.map +1 -1
- package/package.json +43 -28
- package/src/adapters/adapters.ts +30 -24
- package/src/adapters/drizzle/migrate-drizzle.test.ts +54 -33
- package/src/adapters/drizzle/migration-parity-drizzle-kit.test.ts +599 -0
- package/src/adapters/drizzle/test-utils.ts +12 -8
- package/src/adapters/generic-sql/driver-config.ts +38 -0
- package/src/adapters/generic-sql/generic-sql-adapter.test.ts +5 -5
- package/src/adapters/generic-sql/generic-sql-adapter.ts +110 -24
- package/src/adapters/generic-sql/generic-sql-uow-executor.test.ts +54 -0
- package/src/adapters/generic-sql/generic-sql-uow-executor.ts +231 -3
- package/src/adapters/generic-sql/migration/adapter-migration-parity.test.ts +118 -0
- package/src/adapters/generic-sql/migration/dialect/mysql.test.ts +26 -8
- package/src/adapters/generic-sql/migration/dialect/mysql.ts +46 -8
- package/src/adapters/generic-sql/migration/dialect/postgres.test.ts +25 -7
- package/src/adapters/generic-sql/migration/dialect/postgres.ts +8 -4
- package/src/adapters/generic-sql/migration/dialect/sqlite.test.ts +47 -8
- package/src/adapters/generic-sql/migration/dialect/sqlite.ts +27 -12
- package/src/adapters/generic-sql/migration/prepared-migrations.test.ts +128 -39
- package/src/adapters/generic-sql/migration/prepared-migrations.ts +15 -8
- package/src/adapters/generic-sql/migration/sql-generator.ts +142 -65
- package/src/adapters/generic-sql/query/create-sql-query-compiler.ts +9 -6
- package/src/adapters/generic-sql/query/cursor-utils.test.ts +271 -0
- package/src/adapters/generic-sql/query/cursor-utils.ts +41 -6
- package/src/adapters/generic-sql/query/generic-sql-uow-operation-compiler.test.ts +27 -27
- package/src/adapters/generic-sql/query/generic-sql-uow-operation-compiler.ts +38 -24
- package/src/adapters/generic-sql/query/select-builder.test.ts +15 -11
- package/src/adapters/generic-sql/query/select-builder.ts +6 -2
- package/src/adapters/generic-sql/query/sql-query-compiler.test.ts +52 -2
- package/src/adapters/generic-sql/query/sql-query-compiler.ts +50 -15
- package/src/adapters/generic-sql/query/where-builder.test.ts +91 -17
- package/src/adapters/generic-sql/query/where-builder.ts +90 -38
- package/src/adapters/{kysely/kysely-adapter-pglite.test.ts → generic-sql/sql-adapter-pglite-migrations.test.ts} +6 -6
- package/src/adapters/generic-sql/sql-adapter-pglite-pagination.test.ts +806 -0
- package/src/adapters/{drizzle/drizzle-adapter-pglite.test.ts → generic-sql/sql-adapter-pglite-queries.test.ts} +11 -11
- package/src/adapters/generic-sql/{test/generic-drizzle-adapter-sqlite3.test.ts → sql-adapter-sqlite3-driver.test.ts} +10 -10
- package/src/adapters/{drizzle/drizzle-adapter-sqlite3.test.ts → generic-sql/sql-adapter-sqlite3-uow.test.ts} +7 -7
- package/src/adapters/{kysely/kysely-adapter-sqlocal.test.ts → generic-sql/sql-adapter-sqlocal.test.ts} +6 -6
- package/src/adapters/generic-sql/sqlite-storage.ts +20 -0
- package/src/adapters/generic-sql/uow-decoder.test.ts +1 -1
- package/src/adapters/generic-sql/uow-decoder.ts +21 -3
- package/src/adapters/generic-sql/uow-encoder.test.ts +33 -2
- package/src/adapters/generic-sql/uow-encoder.ts +50 -11
- package/src/adapters/in-memory/condition-evaluator.test.ts +193 -0
- package/src/adapters/in-memory/condition-evaluator.ts +275 -0
- package/src/adapters/in-memory/errors.ts +20 -0
- package/src/adapters/in-memory/in-memory-adapter.ts +277 -0
- package/src/adapters/in-memory/in-memory-uow.mutations.test.ts +296 -0
- package/src/adapters/in-memory/in-memory-uow.retrieval.test.ts +100 -0
- package/src/adapters/in-memory/in-memory-uow.ts +1348 -0
- package/src/adapters/in-memory/index.ts +3 -0
- package/src/adapters/in-memory/options.test.ts +41 -0
- package/src/adapters/in-memory/options.ts +87 -0
- package/src/adapters/in-memory/reference-resolution.test.ts +50 -0
- package/src/adapters/in-memory/reference-resolution.ts +67 -0
- package/src/adapters/in-memory/sorted-array-index.test.ts +123 -0
- package/src/adapters/in-memory/sorted-array-index.ts +228 -0
- package/src/adapters/in-memory/store.test.ts +68 -0
- package/src/adapters/in-memory/store.ts +145 -0
- package/src/adapters/in-memory/value-comparison.ts +53 -0
- package/src/adapters/in-memory/value-normalization.test.ts +57 -0
- package/src/adapters/prisma/prisma-adapter-sqlite3.test.ts +1163 -0
- package/src/adapters/shared/from-unit-of-work-compiler.ts +3 -1
- package/src/adapters/shared/uow-operation-compiler.ts +26 -16
- package/src/adapters/sql/index.ts +12 -0
- package/src/db-fragment-definition-builder.test.ts +30 -12
- package/src/db-fragment-definition-builder.ts +142 -73
- package/src/db-fragment-instantiator.test.ts +105 -13
- package/src/db-fragment-integration.test.ts +9 -7
- package/src/dispatchers/cloudflare-do/index.test.ts +73 -0
- package/src/dispatchers/cloudflare-do/index.ts +104 -0
- package/src/dispatchers/node/index.test.ts +91 -0
- package/src/dispatchers/node/index.ts +87 -0
- package/src/fragments/internal-fragment.routes.ts +42 -0
- package/src/fragments/internal-fragment.schema.ts +51 -0
- package/src/fragments/internal-fragment.test.ts +458 -8
- package/src/fragments/internal-fragment.ts +322 -63
- package/src/hooks/durable-hooks-processor.test.ts +117 -0
- package/src/hooks/durable-hooks-processor.ts +67 -0
- package/src/hooks/hooks.test.ts +165 -5
- package/src/hooks/hooks.ts +197 -9
- package/src/migration-engine/auto-from-schema.test.ts +14 -14
- package/src/migration-engine/auto-from-schema.ts +5 -2
- package/src/migration-engine/create.test.ts +2 -2
- package/src/migration-engine/generation-engine.test.ts +229 -104
- package/src/migration-engine/generation-engine.ts +94 -64
- package/src/migration-engine/shared.ts +1 -0
- package/src/mod.ts +64 -26
- package/src/naming/sql-naming.ts +180 -0
- package/src/outbox/outbox-builder.ts +241 -0
- package/src/outbox/outbox.test.ts +253 -0
- package/src/outbox/outbox.ts +137 -0
- package/src/query/column-defaults.ts +41 -3
- package/src/query/condition-builder.test.ts +3 -3
- package/src/query/cursor.test.ts +116 -18
- package/src/query/cursor.ts +75 -26
- package/src/query/db-now.ts +6 -0
- package/src/query/query-type.test.ts +2 -2
- package/src/query/serialize/create-sql-serializer.ts +7 -2
- package/src/query/serialize/dialect/mysql-serializer.ts +12 -4
- package/src/query/serialize/dialect/postgres-serializer.ts +34 -4
- package/src/query/serialize/dialect/sqlite-serializer.test.ts +51 -1
- package/src/query/serialize/dialect/sqlite-serializer.ts +92 -9
- package/src/query/serialize/sql-serializer.ts +4 -4
- package/src/query/simple-query-interface.ts +5 -0
- package/src/query/unit-of-work/execute-unit-of-work.test.ts +25 -1
- package/src/query/unit-of-work/execute-unit-of-work.ts +25 -8
- package/src/query/unit-of-work/unit-of-work-coordinator.test.ts +12 -12
- package/src/query/unit-of-work/unit-of-work-types.test.ts +1 -1
- package/src/query/unit-of-work/unit-of-work.test.ts +168 -37
- package/src/query/unit-of-work/unit-of-work.ts +203 -18
- package/src/query/value-decoding.test.ts +13 -2
- package/src/query/value-decoding.ts +17 -4
- package/src/query/value-encoding.test.ts +85 -2
- package/src/query/value-encoding.ts +56 -6
- package/src/schema/create.test.ts +129 -42
- package/src/schema/create.ts +185 -47
- package/src/schema/generate-id.test.ts +2 -2
- package/src/schema/generate-id.ts +2 -2
- package/src/schema/serialize.test.ts +14 -2
- package/src/schema/type-conversion/create-sql-type-mapper.ts +7 -2
- package/src/schema/type-conversion/dialect/sqlite.ts +18 -0
- package/src/schema/type-conversion/type-mapping.test.ts +25 -1
- package/src/schema/validator.test.ts +197 -0
- package/src/schema/validator.ts +231 -0
- package/src/{adapters/drizzle/generate.test.ts → schema-output/drizzle.test.ts} +179 -129
- package/src/{adapters/drizzle/generate.ts → schema-output/drizzle.ts} +143 -93
- package/src/schema-output/prisma.test.ts +536 -0
- package/src/schema-output/prisma.ts +573 -0
- package/src/util/default-database-adapter.ts +106 -0
- package/src/with-database.ts +22 -3
- package/tsdown.config.ts +6 -4
- package/dist/adapters/drizzle/drizzle-adapter.d.ts +0 -20
- package/dist/adapters/drizzle/drizzle-adapter.d.ts.map +0 -1
- package/dist/adapters/drizzle/drizzle-adapter.js +0 -27
- package/dist/adapters/drizzle/drizzle-adapter.js.map +0 -1
- package/dist/adapters/drizzle/generate.d.ts +0 -30
- package/dist/adapters/drizzle/generate.d.ts.map +0 -1
- package/dist/adapters/drizzle/generate.js.map +0 -1
- package/dist/adapters/kysely/kysely-adapter.d.ts +0 -19
- package/dist/adapters/kysely/kysely-adapter.d.ts.map +0 -1
- package/dist/adapters/kysely/kysely-adapter.js +0 -17
- package/dist/adapters/kysely/kysely-adapter.js.map +0 -1
- package/dist/adapters/shared/table-name-mapper.d.ts +0 -12
- package/dist/adapters/shared/table-name-mapper.d.ts.map +0 -1
- package/dist/adapters/shared/table-name-mapper.js +0 -43
- package/dist/adapters/shared/table-name-mapper.js.map +0 -1
- package/dist/node_modules/.pnpm/rou3@0.7.10/node_modules/rou3/dist/index.js.map +0 -1
- package/dist/schema-generator/schema-generator.d.ts +0 -15
- package/dist/schema-generator/schema-generator.d.ts.map +0 -1
- package/src/adapters/drizzle/drizzle-adapter.ts +0 -39
- package/src/adapters/kysely/kysely-adapter.ts +0 -27
- package/src/adapters/shared/table-name-mapper.ts +0 -50
- package/src/schema-generator/schema-generator.ts +0 -12
|
@@ -0,0 +1,180 @@
|
|
|
1
|
+
import { createHash } from "node:crypto";
|
|
2
|
+
import type { AnySchema, AnyTable } from "../schema/create";
|
|
3
|
+
|
|
4
|
+
export type NamespaceScope = "suffix" | "schema";
|
|
5
|
+
|
|
6
|
+
export interface SqlNamingStrategy {
|
|
7
|
+
namespaceScope: NamespaceScope;
|
|
8
|
+
namespaceToSchema: (namespace: string) => string;
|
|
9
|
+
|
|
10
|
+
tableName: (logicalTable: string, namespace: string | null) => string;
|
|
11
|
+
columnName: (logicalColumn: string, logicalTable: string) => string;
|
|
12
|
+
|
|
13
|
+
indexName: (logicalIndex: string, logicalTable: string, namespace: string | null) => string;
|
|
14
|
+
uniqueIndexName: (logicalIndex: string, logicalTable: string, namespace: string | null) => string;
|
|
15
|
+
foreignKeyName: (params: {
|
|
16
|
+
logicalTable: string;
|
|
17
|
+
logicalReferencedTable: string;
|
|
18
|
+
referenceName: string;
|
|
19
|
+
namespace: string | null;
|
|
20
|
+
}) => string;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
const normalizeNamespace = (namespace: string | null) =>
|
|
24
|
+
namespace && namespace.length > 0 ? namespace : null;
|
|
25
|
+
|
|
26
|
+
const buildNamespaceSuffix = (namespace: string | null) => {
|
|
27
|
+
const sanitized = normalizeNamespace(namespace);
|
|
28
|
+
return sanitized ? `_${sanitized}` : "";
|
|
29
|
+
};
|
|
30
|
+
|
|
31
|
+
const MAX_IDENTIFIER_LENGTH = 63;
|
|
32
|
+
const HASH_LENGTH = 8;
|
|
33
|
+
|
|
34
|
+
const withHash = (value: string) => {
|
|
35
|
+
const hash = createHash("sha1").update(value).digest("hex").slice(0, 8);
|
|
36
|
+
return `${value}_${hash}`;
|
|
37
|
+
};
|
|
38
|
+
|
|
39
|
+
const truncateWithHash = (value: string) => {
|
|
40
|
+
if (value.length <= MAX_IDENTIFIER_LENGTH) {
|
|
41
|
+
return value;
|
|
42
|
+
}
|
|
43
|
+
const hash = createHash("sha1").update(value).digest("hex").slice(0, HASH_LENGTH);
|
|
44
|
+
return `${value.slice(0, MAX_IDENTIFIER_LENGTH - HASH_LENGTH)}${hash}`;
|
|
45
|
+
};
|
|
46
|
+
|
|
47
|
+
const buildIndexName = (
|
|
48
|
+
prefix: "idx" | "uidx",
|
|
49
|
+
logicalIndex: string,
|
|
50
|
+
logicalTable: string,
|
|
51
|
+
namespace: string | null,
|
|
52
|
+
) => {
|
|
53
|
+
const base = `${logicalTable}_${logicalIndex}${buildNamespaceSuffix(namespace)}`;
|
|
54
|
+
return `${prefix}_${withHash(base)}`;
|
|
55
|
+
};
|
|
56
|
+
|
|
57
|
+
export const suffixNamingStrategy: SqlNamingStrategy = {
|
|
58
|
+
namespaceScope: "suffix",
|
|
59
|
+
namespaceToSchema: (namespace) => namespace,
|
|
60
|
+
tableName: (logicalTable, namespace) => {
|
|
61
|
+
const sanitized = normalizeNamespace(namespace);
|
|
62
|
+
return truncateWithHash(sanitized ? `${logicalTable}_${sanitized}` : logicalTable);
|
|
63
|
+
},
|
|
64
|
+
columnName: (logicalColumn) => logicalColumn,
|
|
65
|
+
indexName: (logicalIndex, logicalTable, namespace) =>
|
|
66
|
+
truncateWithHash(buildIndexName("idx", logicalIndex, logicalTable, namespace)),
|
|
67
|
+
uniqueIndexName: (logicalIndex, logicalTable, namespace) =>
|
|
68
|
+
truncateWithHash(buildIndexName("uidx", logicalIndex, logicalTable, namespace)),
|
|
69
|
+
foreignKeyName: ({ logicalTable, logicalReferencedTable, referenceName, namespace }) => {
|
|
70
|
+
const base = `${logicalTable}_${logicalReferencedTable}_${referenceName}${buildNamespaceSuffix(namespace)}`;
|
|
71
|
+
return truncateWithHash(`fk_${withHash(base)}`);
|
|
72
|
+
},
|
|
73
|
+
};
|
|
74
|
+
|
|
75
|
+
export const schemaNamingStrategy: SqlNamingStrategy = {
|
|
76
|
+
namespaceScope: "schema",
|
|
77
|
+
namespaceToSchema: (namespace) => namespace,
|
|
78
|
+
tableName: (logicalTable) => truncateWithHash(logicalTable),
|
|
79
|
+
columnName: (logicalColumn) => logicalColumn,
|
|
80
|
+
indexName: (logicalIndex) => truncateWithHash(logicalIndex),
|
|
81
|
+
uniqueIndexName: (logicalIndex) => truncateWithHash(logicalIndex),
|
|
82
|
+
foreignKeyName: ({ logicalTable, logicalReferencedTable, referenceName }) =>
|
|
83
|
+
truncateWithHash(`fk_${logicalTable}_${logicalReferencedTable}_${referenceName}`),
|
|
84
|
+
};
|
|
85
|
+
|
|
86
|
+
export class NamingResolver {
|
|
87
|
+
readonly #namespace: string | null;
|
|
88
|
+
readonly #strategy: SqlNamingStrategy;
|
|
89
|
+
readonly #tableNameMap: Record<string, string> = {};
|
|
90
|
+
readonly #columnNameMaps = new Map<string, Record<string, string>>();
|
|
91
|
+
|
|
92
|
+
constructor(schema: AnySchema, namespace: string | null, strategy: SqlNamingStrategy) {
|
|
93
|
+
this.#namespace = namespace;
|
|
94
|
+
this.#strategy = strategy;
|
|
95
|
+
|
|
96
|
+
for (const table of Object.values(schema.tables)) {
|
|
97
|
+
const physicalTable = this.getTableName(table.name);
|
|
98
|
+
this.#tableNameMap[physicalTable] = table.name;
|
|
99
|
+
|
|
100
|
+
const columnMap: Record<string, string> = {};
|
|
101
|
+
for (const column of Object.values(table.columns)) {
|
|
102
|
+
const physicalColumn = this.getColumnName(table.name, column.name);
|
|
103
|
+
columnMap[physicalColumn] = column.name;
|
|
104
|
+
}
|
|
105
|
+
this.#columnNameMaps.set(table.name, columnMap);
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
get namespace(): string | null {
|
|
110
|
+
return this.#namespace;
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
get strategy(): SqlNamingStrategy {
|
|
114
|
+
return this.#strategy;
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
getSchemaName(): string | null {
|
|
118
|
+
if (this.#strategy.namespaceScope !== "schema") {
|
|
119
|
+
return null;
|
|
120
|
+
}
|
|
121
|
+
if (!this.#namespace || this.#namespace.length === 0) {
|
|
122
|
+
return null;
|
|
123
|
+
}
|
|
124
|
+
return this.#strategy.namespaceToSchema(this.#namespace);
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
getTableName(logicalTable: string): string {
|
|
128
|
+
return this.#strategy.tableName(logicalTable, this.#namespace);
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
getColumnName(logicalTable: string, logicalColumn: string): string {
|
|
132
|
+
return this.#strategy.columnName(logicalColumn, logicalTable);
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
getIndexName(logicalIndex: string, logicalTable: string): string {
|
|
136
|
+
return this.#strategy.indexName(logicalIndex, logicalTable, this.#namespace);
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
getUniqueIndexName(logicalIndex: string, logicalTable: string): string {
|
|
140
|
+
return this.#strategy.uniqueIndexName(logicalIndex, logicalTable, this.#namespace);
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
getForeignKeyName(params: {
|
|
144
|
+
logicalTable: string;
|
|
145
|
+
logicalReferencedTable: string;
|
|
146
|
+
referenceName: string;
|
|
147
|
+
}): string {
|
|
148
|
+
return this.#strategy.foreignKeyName({
|
|
149
|
+
...params,
|
|
150
|
+
namespace: this.#namespace,
|
|
151
|
+
});
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
getTableNameMap(): Record<string, string> {
|
|
155
|
+
return { ...this.#tableNameMap };
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
getColumnNameMap(table: AnyTable | string): Record<string, string> {
|
|
159
|
+
const tableName = typeof table === "string" ? table : table.name;
|
|
160
|
+
const map = this.#columnNameMaps.get(tableName);
|
|
161
|
+
return map ? { ...map } : {};
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
export const createNamingResolver = (
|
|
166
|
+
schema: AnySchema,
|
|
167
|
+
namespace: string | null,
|
|
168
|
+
strategy: SqlNamingStrategy,
|
|
169
|
+
): NamingResolver => new NamingResolver(schema, namespace, strategy);
|
|
170
|
+
|
|
171
|
+
/**
|
|
172
|
+
* Sanitizes a namespace for use in SQL identifiers and TypeScript exports.
|
|
173
|
+
* Converts dashes to underscores to ensure compatibility with SQL identifiers.
|
|
174
|
+
*
|
|
175
|
+
* @example
|
|
176
|
+
* sanitizeNamespace("my-fragment") // => "my_fragment"
|
|
177
|
+
*/
|
|
178
|
+
export function sanitizeNamespace(namespace: string): string {
|
|
179
|
+
return namespace.replace(/-/g, "_");
|
|
180
|
+
}
|
|
@@ -0,0 +1,241 @@
|
|
|
1
|
+
import type { MutationOperation } from "../query/unit-of-work/unit-of-work";
|
|
2
|
+
import type { AnySchema, AnyTable } from "../schema/create";
|
|
3
|
+
import { FragnoId, FragnoReference } from "../schema/create";
|
|
4
|
+
import { internalSchema } from "../fragments/internal-fragment.schema";
|
|
5
|
+
import type { OutboxRefLookup, OutboxPayload, OutboxMutation } from "./outbox";
|
|
6
|
+
import { encodeVersionstamp, versionstampToHex } from "./outbox";
|
|
7
|
+
|
|
8
|
+
const INTERNAL_TABLE_NAMES = new Set(Object.keys(internalSchema.tables));
|
|
9
|
+
|
|
10
|
+
type OutboxMutationDraft = OutboxMutation extends infer T
|
|
11
|
+
? T extends OutboxMutation
|
|
12
|
+
? Omit<T, "versionstamp"> & { versionstamp?: string }
|
|
13
|
+
: never
|
|
14
|
+
: never;
|
|
15
|
+
|
|
16
|
+
export type OutboxPlan = {
|
|
17
|
+
drafts: OutboxMutationDraft[];
|
|
18
|
+
lookups: OutboxRefLookup[];
|
|
19
|
+
};
|
|
20
|
+
|
|
21
|
+
export function buildOutboxPlan(operations: MutationOperation<AnySchema>[]): OutboxPlan {
|
|
22
|
+
const drafts: OutboxMutationDraft[] = [];
|
|
23
|
+
const lookups: OutboxRefLookup[] = [];
|
|
24
|
+
|
|
25
|
+
for (const op of operations) {
|
|
26
|
+
if (op.type === "check") {
|
|
27
|
+
continue;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
if (isInternalMutation(op)) {
|
|
31
|
+
continue;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
const table = getTable(op.schema, op.table);
|
|
35
|
+
const schemaName = op.namespace ?? "";
|
|
36
|
+
const namespace = op.namespace ? op.namespace : undefined;
|
|
37
|
+
const mutationIndex = drafts.length;
|
|
38
|
+
|
|
39
|
+
if (op.type === "create") {
|
|
40
|
+
drafts.push({
|
|
41
|
+
op: "create",
|
|
42
|
+
schema: schemaName,
|
|
43
|
+
namespace,
|
|
44
|
+
table: op.table,
|
|
45
|
+
externalId: op.generatedExternalId,
|
|
46
|
+
values: encodeOutboxValues({
|
|
47
|
+
table,
|
|
48
|
+
values: op.values,
|
|
49
|
+
mutationIndex,
|
|
50
|
+
namespace,
|
|
51
|
+
lookups,
|
|
52
|
+
}),
|
|
53
|
+
});
|
|
54
|
+
continue;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
if (op.type === "update") {
|
|
58
|
+
drafts.push({
|
|
59
|
+
op: "update",
|
|
60
|
+
schema: schemaName,
|
|
61
|
+
namespace,
|
|
62
|
+
table: op.table,
|
|
63
|
+
externalId: getExternalId(op.id),
|
|
64
|
+
set: encodeOutboxValues({
|
|
65
|
+
table,
|
|
66
|
+
values: op.set,
|
|
67
|
+
mutationIndex,
|
|
68
|
+
namespace,
|
|
69
|
+
lookups,
|
|
70
|
+
}),
|
|
71
|
+
checkVersion: op.checkVersion && op.id instanceof FragnoId ? op.id.version : undefined,
|
|
72
|
+
});
|
|
73
|
+
continue;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
if (op.type === "delete") {
|
|
77
|
+
drafts.push({
|
|
78
|
+
op: "delete",
|
|
79
|
+
schema: schemaName,
|
|
80
|
+
namespace,
|
|
81
|
+
table: op.table,
|
|
82
|
+
externalId: getExternalId(op.id),
|
|
83
|
+
checkVersion: op.checkVersion && op.id instanceof FragnoId ? op.id.version : undefined,
|
|
84
|
+
});
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
return { drafts, lookups };
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
export function finalizeOutboxPayload(plan: OutboxPlan, transactionVersion: bigint): OutboxPayload {
|
|
92
|
+
const mutations: OutboxMutation[] = plan.drafts.map((draft, index) => {
|
|
93
|
+
const versionstamp = versionstampToHex(encodeVersionstamp(transactionVersion, index));
|
|
94
|
+
return { ...draft, versionstamp } as OutboxMutation;
|
|
95
|
+
});
|
|
96
|
+
|
|
97
|
+
return {
|
|
98
|
+
version: 1,
|
|
99
|
+
mutations,
|
|
100
|
+
};
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
function encodeOutboxValues(options: {
|
|
104
|
+
table: AnyTable;
|
|
105
|
+
values: Record<string, unknown>;
|
|
106
|
+
mutationIndex: number;
|
|
107
|
+
namespace?: string;
|
|
108
|
+
lookups: OutboxRefLookup[];
|
|
109
|
+
}): Record<string, unknown> {
|
|
110
|
+
const { table, values, mutationIndex, namespace, lookups } = options;
|
|
111
|
+
const output: Record<string, unknown> = {};
|
|
112
|
+
|
|
113
|
+
for (const [key, value] of Object.entries(values)) {
|
|
114
|
+
if (value === undefined) {
|
|
115
|
+
continue;
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
const column = table.columns[key];
|
|
119
|
+
|
|
120
|
+
if (column?.role === "reference") {
|
|
121
|
+
const resolved = resolveReferenceValue({
|
|
122
|
+
value,
|
|
123
|
+
mutationIndex,
|
|
124
|
+
columnName: key,
|
|
125
|
+
table,
|
|
126
|
+
namespace,
|
|
127
|
+
lookups,
|
|
128
|
+
});
|
|
129
|
+
output[key] = resolved;
|
|
130
|
+
continue;
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
if (value instanceof FragnoId) {
|
|
134
|
+
output[key] = value.externalId;
|
|
135
|
+
continue;
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
output[key] = value;
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
return output;
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
function resolveReferenceValue(options: {
|
|
145
|
+
value: unknown;
|
|
146
|
+
mutationIndex: number;
|
|
147
|
+
columnName: string;
|
|
148
|
+
table: AnyTable;
|
|
149
|
+
namespace?: string;
|
|
150
|
+
lookups: OutboxRefLookup[];
|
|
151
|
+
}): unknown {
|
|
152
|
+
const { value, mutationIndex, columnName, table, namespace, lookups } = options;
|
|
153
|
+
|
|
154
|
+
if (value === null) {
|
|
155
|
+
return null;
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
if (value instanceof FragnoId) {
|
|
159
|
+
return value.externalId;
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
if (typeof value === "string") {
|
|
163
|
+
return value;
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
if (value instanceof FragnoReference) {
|
|
167
|
+
return createReferencePlaceholder({
|
|
168
|
+
internalId: value.internalId,
|
|
169
|
+
mutationIndex,
|
|
170
|
+
columnName,
|
|
171
|
+
table,
|
|
172
|
+
namespace,
|
|
173
|
+
lookups,
|
|
174
|
+
});
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
if (typeof value === "bigint" || typeof value === "number") {
|
|
178
|
+
return createReferencePlaceholder({
|
|
179
|
+
internalId: value,
|
|
180
|
+
mutationIndex,
|
|
181
|
+
columnName,
|
|
182
|
+
table,
|
|
183
|
+
namespace,
|
|
184
|
+
lookups,
|
|
185
|
+
});
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
return value;
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
function createReferencePlaceholder(options: {
|
|
192
|
+
internalId: bigint | number;
|
|
193
|
+
mutationIndex: number;
|
|
194
|
+
columnName: string;
|
|
195
|
+
table: AnyTable;
|
|
196
|
+
namespace?: string;
|
|
197
|
+
lookups: OutboxRefLookup[];
|
|
198
|
+
}): { __fragno_ref: string } {
|
|
199
|
+
const { internalId, mutationIndex, columnName, table, namespace, lookups } = options;
|
|
200
|
+
const key = `${mutationIndex}.${columnName}`;
|
|
201
|
+
const referencedTable = resolveReferencedTable(table, columnName);
|
|
202
|
+
|
|
203
|
+
lookups.push({
|
|
204
|
+
key,
|
|
205
|
+
internalId,
|
|
206
|
+
table: referencedTable,
|
|
207
|
+
namespace,
|
|
208
|
+
});
|
|
209
|
+
|
|
210
|
+
return { __fragno_ref: key };
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
function resolveReferencedTable(table: AnyTable, columnName: string): AnyTable {
|
|
214
|
+
for (const relation of Object.values(table.relations)) {
|
|
215
|
+
if (relation.on.some(([localColumn]) => localColumn === columnName)) {
|
|
216
|
+
return relation.table;
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
throw new Error(`Reference column ${columnName} not found in table ${table.name}`);
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
function getExternalId(id: FragnoId | string): string {
|
|
224
|
+
return typeof id === "string" ? id : id.externalId;
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
function isInternalMutation(op: MutationOperation<AnySchema>): boolean {
|
|
228
|
+
if (op.schema === internalSchema) {
|
|
229
|
+
return true;
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
return op.namespace === "" && INTERNAL_TABLE_NAMES.has(op.table);
|
|
233
|
+
}
|
|
234
|
+
|
|
235
|
+
function getTable(schema: AnySchema, tableName: string): AnyTable {
|
|
236
|
+
const table = schema.tables[tableName];
|
|
237
|
+
if (!table) {
|
|
238
|
+
throw new Error(`Invalid table name ${tableName}.`);
|
|
239
|
+
}
|
|
240
|
+
return table;
|
|
241
|
+
}
|
|
@@ -0,0 +1,253 @@
|
|
|
1
|
+
import { describe, expect, it } from "vitest";
|
|
2
|
+
import { defineFragment, instantiate } from "@fragno-dev/core";
|
|
3
|
+
import { SQLocalKysely } from "sqlocal/kysely";
|
|
4
|
+
import { KyselyPGlite } from "kysely-pglite";
|
|
5
|
+
import { PGlite } from "@electric-sql/pglite";
|
|
6
|
+
import superjson, { type SuperJSONResult } from "superjson";
|
|
7
|
+
import { SqlAdapter } from "../adapters/generic-sql/generic-sql-adapter";
|
|
8
|
+
import { PGLiteDriverConfig, SQLocalDriverConfig } from "../adapters/generic-sql/driver-config";
|
|
9
|
+
import type { SimpleQueryInterface } from "../query/simple-query-interface";
|
|
10
|
+
import { withDatabase } from "../with-database";
|
|
11
|
+
import { internalSchema, type InternalFragmentInstance } from "../fragments/internal-fragment";
|
|
12
|
+
import { schema, idColumn, column, referenceColumn, FragnoReference } from "../schema/create";
|
|
13
|
+
import type { OutboxConfig, OutboxEntry, OutboxPayload } from "./outbox";
|
|
14
|
+
|
|
15
|
+
const outboxSchema = schema("outbox", (s) => {
|
|
16
|
+
return s
|
|
17
|
+
.addTable("users", (t) => {
|
|
18
|
+
return t
|
|
19
|
+
.addColumn("id", idColumn())
|
|
20
|
+
.addColumn("email", column("string"))
|
|
21
|
+
.createIndex("idx_users_email", ["email"], { unique: true });
|
|
22
|
+
})
|
|
23
|
+
.addTable("posts", (t) => {
|
|
24
|
+
return t
|
|
25
|
+
.addColumn("id", idColumn())
|
|
26
|
+
.addColumn("authorId", referenceColumn())
|
|
27
|
+
.addColumn("title", column("string"));
|
|
28
|
+
})
|
|
29
|
+
.addReference("author", {
|
|
30
|
+
type: "one",
|
|
31
|
+
from: { table: "posts", column: "authorId" },
|
|
32
|
+
to: { table: "users", column: "id" },
|
|
33
|
+
});
|
|
34
|
+
});
|
|
35
|
+
|
|
36
|
+
const outboxFragmentName = "outbox-test";
|
|
37
|
+
const outboxFragmentDef = defineFragment(outboxFragmentName)
|
|
38
|
+
.extend(withDatabase(outboxSchema))
|
|
39
|
+
.build();
|
|
40
|
+
|
|
41
|
+
type OutboxAdapterConfig =
|
|
42
|
+
| { type: "kysely-sqlite"; outbox?: OutboxConfig }
|
|
43
|
+
| { type: "kysely-pglite"; outbox?: OutboxConfig };
|
|
44
|
+
|
|
45
|
+
type OutboxTestContext = {
|
|
46
|
+
db: SimpleQueryInterface<typeof outboxSchema>;
|
|
47
|
+
internalFragment: InternalFragmentInstance;
|
|
48
|
+
cleanup: () => Promise<void>;
|
|
49
|
+
};
|
|
50
|
+
|
|
51
|
+
async function migrateSchema(
|
|
52
|
+
adapter: SqlAdapter,
|
|
53
|
+
schemaToMigrate: typeof outboxSchema | typeof internalSchema,
|
|
54
|
+
namespace: string,
|
|
55
|
+
): Promise<void> {
|
|
56
|
+
const migrations = adapter.prepareMigrations(schemaToMigrate, namespace);
|
|
57
|
+
await migrations.executeWithDriver(adapter.driver, 0);
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
async function createAdapter(config: OutboxAdapterConfig): Promise<{
|
|
61
|
+
adapter: SqlAdapter;
|
|
62
|
+
cleanup: () => Promise<void>;
|
|
63
|
+
}> {
|
|
64
|
+
if (config.type === "kysely-sqlite") {
|
|
65
|
+
const { dialect } = new SQLocalKysely(":memory:");
|
|
66
|
+
const adapter = new SqlAdapter({
|
|
67
|
+
dialect,
|
|
68
|
+
driverConfig: new SQLocalDriverConfig(),
|
|
69
|
+
outbox: config.outbox,
|
|
70
|
+
});
|
|
71
|
+
|
|
72
|
+
await migrateSchema(adapter, internalSchema, "");
|
|
73
|
+
await migrateSchema(adapter, outboxSchema, outboxSchema.name);
|
|
74
|
+
|
|
75
|
+
return {
|
|
76
|
+
adapter,
|
|
77
|
+
cleanup: async () => {
|
|
78
|
+
await adapter.close();
|
|
79
|
+
},
|
|
80
|
+
};
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
const pgliteDatabase = new PGlite();
|
|
84
|
+
const { dialect } = new KyselyPGlite(pgliteDatabase);
|
|
85
|
+
const adapter = new SqlAdapter({
|
|
86
|
+
dialect,
|
|
87
|
+
driverConfig: new PGLiteDriverConfig(),
|
|
88
|
+
outbox: config.outbox,
|
|
89
|
+
});
|
|
90
|
+
|
|
91
|
+
await migrateSchema(adapter, internalSchema, "");
|
|
92
|
+
await migrateSchema(adapter, outboxSchema, outboxSchema.name);
|
|
93
|
+
|
|
94
|
+
return {
|
|
95
|
+
adapter,
|
|
96
|
+
cleanup: async () => {
|
|
97
|
+
await adapter.close();
|
|
98
|
+
},
|
|
99
|
+
};
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
async function buildOutboxTest(adapterConfig: OutboxAdapterConfig): Promise<OutboxTestContext> {
|
|
103
|
+
const { adapter, cleanup } = await createAdapter(adapterConfig);
|
|
104
|
+
const fragment = instantiate(outboxFragmentDef)
|
|
105
|
+
.withConfig({})
|
|
106
|
+
.withRoutes([])
|
|
107
|
+
.withOptions({ databaseAdapter: adapter })
|
|
108
|
+
.build();
|
|
109
|
+
const deps = fragment.$internal.deps as { db: SimpleQueryInterface<typeof outboxSchema> };
|
|
110
|
+
|
|
111
|
+
return {
|
|
112
|
+
db: deps.db,
|
|
113
|
+
internalFragment: fragment.$internal.linkedFragments._fragno_internal,
|
|
114
|
+
cleanup,
|
|
115
|
+
};
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
async function listOutbox(
|
|
119
|
+
internalFragment: InternalFragmentInstance,
|
|
120
|
+
options?: { afterVersionstamp?: string; limit?: number },
|
|
121
|
+
): Promise<OutboxEntry[]> {
|
|
122
|
+
return internalFragment.inContext(async function () {
|
|
123
|
+
return (await this.handlerTx()
|
|
124
|
+
.withServiceCalls(() => [internalFragment.services.outboxService.list(options)] as const)
|
|
125
|
+
.transform(({ serviceResult: [result] }) => result)
|
|
126
|
+
.execute()) as OutboxEntry[];
|
|
127
|
+
});
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
const adapterConfigs = [{ type: "kysely-sqlite" as const }, { type: "kysely-pglite" as const }];
|
|
131
|
+
|
|
132
|
+
describe("Fragno DB Outbox", () => {
|
|
133
|
+
it("does not write outbox entries when disabled", async () => {
|
|
134
|
+
const { db, internalFragment, cleanup } = await buildOutboxTest({
|
|
135
|
+
type: "kysely-sqlite",
|
|
136
|
+
});
|
|
137
|
+
|
|
138
|
+
await db.create("users", { email: "disabled@example.com" });
|
|
139
|
+
|
|
140
|
+
const entries = await listOutbox(internalFragment);
|
|
141
|
+
expect(entries).toHaveLength(0);
|
|
142
|
+
|
|
143
|
+
await cleanup();
|
|
144
|
+
});
|
|
145
|
+
|
|
146
|
+
it("stores refMap placeholders and lists entries in order", async () => {
|
|
147
|
+
const { db, internalFragment, cleanup } = await buildOutboxTest({
|
|
148
|
+
type: "kysely-sqlite",
|
|
149
|
+
outbox: { enabled: true },
|
|
150
|
+
});
|
|
151
|
+
|
|
152
|
+
await db.create("users", { email: "alpha@example.com" });
|
|
153
|
+
const user = await db.findFirst("users", (b) =>
|
|
154
|
+
b.whereIndex("idx_users_email", (eb) => eb("email", "=", "alpha@example.com")),
|
|
155
|
+
);
|
|
156
|
+
expect(user).not.toBeNull();
|
|
157
|
+
expect(user?.id.internalId).toBeDefined();
|
|
158
|
+
|
|
159
|
+
await db.create("posts", {
|
|
160
|
+
title: "Hello",
|
|
161
|
+
authorId: FragnoReference.fromInternal(user!.id.internalId!),
|
|
162
|
+
});
|
|
163
|
+
|
|
164
|
+
const entries = await listOutbox(internalFragment);
|
|
165
|
+
expect(entries).toHaveLength(2);
|
|
166
|
+
expect(entries[0].versionstamp < entries[1].versionstamp).toBe(true);
|
|
167
|
+
|
|
168
|
+
const filtered = await listOutbox(internalFragment, {
|
|
169
|
+
afterVersionstamp: entries[0].versionstamp,
|
|
170
|
+
limit: 1,
|
|
171
|
+
});
|
|
172
|
+
expect(filtered).toHaveLength(1);
|
|
173
|
+
expect(filtered[0].versionstamp).toBe(entries[1].versionstamp);
|
|
174
|
+
|
|
175
|
+
const payload = superjson.deserialize(entries[1].payload as SuperJSONResult) as OutboxPayload;
|
|
176
|
+
expect(payload.version).toBe(1);
|
|
177
|
+
expect(payload.mutations).toHaveLength(1);
|
|
178
|
+
const [mutation] = payload.mutations;
|
|
179
|
+
if (mutation.op !== "create") {
|
|
180
|
+
throw new Error("Expected create mutation in outbox payload.");
|
|
181
|
+
}
|
|
182
|
+
expect(mutation.values).toMatchObject({
|
|
183
|
+
authorId: { __fragno_ref: "0.authorId" },
|
|
184
|
+
});
|
|
185
|
+
expect(entries[1].refMap).toEqual({
|
|
186
|
+
"0.authorId": user!.id.externalId,
|
|
187
|
+
});
|
|
188
|
+
|
|
189
|
+
await internalFragment.inContext(async function () {
|
|
190
|
+
await this.handlerTx()
|
|
191
|
+
.withServiceCalls(() => [
|
|
192
|
+
internalFragment.services.settingsService.set("outbox-test", "noop", "1"),
|
|
193
|
+
])
|
|
194
|
+
.execute();
|
|
195
|
+
});
|
|
196
|
+
|
|
197
|
+
const afterInternal = await listOutbox(internalFragment);
|
|
198
|
+
expect(afterInternal).toHaveLength(2);
|
|
199
|
+
|
|
200
|
+
await cleanup();
|
|
201
|
+
});
|
|
202
|
+
|
|
203
|
+
it("orders outbox entries by commit order across concurrent UOWs", async () => {
|
|
204
|
+
const { db, internalFragment, cleanup } = await buildOutboxTest({
|
|
205
|
+
type: "kysely-sqlite",
|
|
206
|
+
outbox: { enabled: true },
|
|
207
|
+
});
|
|
208
|
+
|
|
209
|
+
const uow1 = db.createUnitOfWork("uow-1");
|
|
210
|
+
const uow2 = db.createUnitOfWork("uow-2");
|
|
211
|
+
const uow1Id = uow1.idempotencyKey;
|
|
212
|
+
const uow2Id = uow2.idempotencyKey;
|
|
213
|
+
|
|
214
|
+
uow1.create("users", { email: "order-1@example.com" });
|
|
215
|
+
uow2.create("users", { email: "order-2@example.com" });
|
|
216
|
+
|
|
217
|
+
const completionOrder: string[] = [];
|
|
218
|
+
await Promise.all([
|
|
219
|
+
uow1.executeMutations().then(() => completionOrder.push(uow1Id)),
|
|
220
|
+
uow2.executeMutations().then(() => completionOrder.push(uow2Id)),
|
|
221
|
+
]);
|
|
222
|
+
|
|
223
|
+
const entries = await listOutbox(internalFragment);
|
|
224
|
+
expect(entries.map((entry) => entry.uowId)).toEqual(completionOrder);
|
|
225
|
+
expect(entries[0].versionstamp < entries[1].versionstamp).toBe(true);
|
|
226
|
+
|
|
227
|
+
await cleanup();
|
|
228
|
+
});
|
|
229
|
+
|
|
230
|
+
describe.each(adapterConfigs)("adapter opt-in (%s)", (adapterConfig) => {
|
|
231
|
+
it("writes outbox rows only when enabled", async () => {
|
|
232
|
+
const { db, internalFragment, cleanup } = await buildOutboxTest(adapterConfig);
|
|
233
|
+
|
|
234
|
+
await db.create("users", { email: "disabled@example.com" });
|
|
235
|
+
const disabledEntries = await listOutbox(internalFragment);
|
|
236
|
+
expect(disabledEntries).toHaveLength(0);
|
|
237
|
+
await cleanup();
|
|
238
|
+
|
|
239
|
+
const {
|
|
240
|
+
db: enabledDb,
|
|
241
|
+
internalFragment: enabledInternal,
|
|
242
|
+
cleanup: enabledCleanup,
|
|
243
|
+
} = await buildOutboxTest({
|
|
244
|
+
...adapterConfig,
|
|
245
|
+
outbox: { enabled: true },
|
|
246
|
+
});
|
|
247
|
+
await enabledDb.create("users", { email: "enabled@example.com" });
|
|
248
|
+
const enabledEntries = await listOutbox(enabledInternal);
|
|
249
|
+
expect(enabledEntries).toHaveLength(1);
|
|
250
|
+
await enabledCleanup();
|
|
251
|
+
}, 10_000);
|
|
252
|
+
});
|
|
253
|
+
});
|