@effect-app/infra 4.0.0-beta.16 → 4.0.0-beta.161
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/CHANGELOG.md +1060 -0
- package/dist/CUPS.d.ts +15 -7
- package/dist/CUPS.d.ts.map +1 -1
- package/dist/CUPS.js +10 -12
- package/dist/Emailer/Sendgrid.d.ts +14 -14
- package/dist/Emailer/Sendgrid.d.ts.map +1 -1
- package/dist/Emailer/Sendgrid.js +16 -15
- package/dist/Emailer/fake.d.ts +1 -1
- package/dist/Emailer/service.d.ts +9 -3
- package/dist/Emailer/service.d.ts.map +1 -1
- package/dist/Emailer/service.js +3 -3
- package/dist/Emailer.d.ts +1 -1
- package/dist/MainFiberSet.d.ts +5 -5
- package/dist/MainFiberSet.d.ts.map +1 -1
- package/dist/MainFiberSet.js +3 -3
- package/dist/Model/Repository/Registry.d.ts +20 -0
- package/dist/Model/Repository/Registry.d.ts.map +1 -0
- package/dist/Model/Repository/Registry.js +17 -0
- package/dist/Model/Repository/ext.d.ts +33 -15
- package/dist/Model/Repository/ext.d.ts.map +1 -1
- package/dist/Model/Repository/ext.js +54 -2
- package/dist/Model/Repository/internal/internal.d.ts +6 -6
- package/dist/Model/Repository/internal/internal.d.ts.map +1 -1
- package/dist/Model/Repository/internal/internal.js +43 -32
- package/dist/Model/Repository/legacy.d.ts +1 -1
- package/dist/Model/Repository/makeRepo.d.ts +7 -6
- package/dist/Model/Repository/makeRepo.d.ts.map +1 -1
- package/dist/Model/Repository/makeRepo.js +5 -1
- package/dist/Model/Repository/service.d.ts +28 -23
- package/dist/Model/Repository/service.d.ts.map +1 -1
- package/dist/Model/Repository/validation.d.ts +142 -17
- package/dist/Model/Repository/validation.d.ts.map +1 -1
- package/dist/Model/Repository/validation.js +5 -5
- package/dist/Model/Repository.d.ts +2 -1
- package/dist/Model/Repository.d.ts.map +1 -1
- package/dist/Model/Repository.js +2 -1
- package/dist/Model/dsl.d.ts +4 -4
- package/dist/Model/dsl.d.ts.map +1 -1
- package/dist/Model/filter/filterApi.d.ts +5 -5
- package/dist/Model/filter/filterApi.d.ts.map +1 -1
- package/dist/Model/filter/types/errors.d.ts +1 -1
- package/dist/Model/filter/types/fields.d.ts +1 -1
- package/dist/Model/filter/types/path/common.d.ts +1 -1
- package/dist/Model/filter/types/path/eager.d.ts +1 -1
- package/dist/Model/filter/types/path/eager.d.ts.map +1 -1
- package/dist/Model/filter/types/path/index.d.ts +1 -1
- package/dist/Model/filter/types/utils.d.ts +1 -1
- package/dist/Model/filter/types/validator.d.ts +1 -1
- package/dist/Model/filter/types.d.ts +1 -1
- package/dist/Model/query/dsl.d.ts +1 -1
- package/dist/Model/query/dsl.d.ts.map +1 -1
- package/dist/Model/query/new-kid-interpreter.d.ts +6 -6
- package/dist/Model/query/new-kid-interpreter.d.ts.map +1 -1
- package/dist/Model/query/new-kid-interpreter.js +3 -3
- package/dist/Model/query.d.ts +1 -1
- package/dist/Model.d.ts +2 -1
- package/dist/Model.d.ts.map +1 -1
- package/dist/Model.js +2 -1
- package/dist/Operations.d.ts +6 -6
- package/dist/Operations.d.ts.map +1 -1
- package/dist/Operations.js +56 -59
- package/dist/OperationsRepo.d.ts +11 -29
- package/dist/OperationsRepo.d.ts.map +1 -1
- package/dist/OperationsRepo.js +3 -3
- package/dist/QueueMaker/SQLQueue.d.ts +5 -7
- package/dist/QueueMaker/SQLQueue.d.ts.map +1 -1
- package/dist/QueueMaker/SQLQueue.js +105 -114
- package/dist/QueueMaker/errors.d.ts +2 -2
- package/dist/QueueMaker/errors.d.ts.map +1 -1
- package/dist/QueueMaker/memQueue.d.ts +7 -4
- package/dist/QueueMaker/memQueue.d.ts.map +1 -1
- package/dist/QueueMaker/memQueue.js +51 -62
- package/dist/QueueMaker/sbqueue.d.ts +6 -3
- package/dist/QueueMaker/sbqueue.d.ts.map +1 -1
- package/dist/QueueMaker/sbqueue.js +37 -53
- package/dist/QueueMaker/service.d.ts +1 -1
- package/dist/RequestContext.d.ts +114 -26
- package/dist/RequestContext.d.ts.map +1 -1
- package/dist/RequestContext.js +7 -7
- package/dist/RequestFiberSet.d.ts +7 -7
- package/dist/RequestFiberSet.d.ts.map +1 -1
- package/dist/RequestFiberSet.js +5 -5
- package/dist/Store/ContextMapContainer.d.ts +19 -3
- package/dist/Store/ContextMapContainer.d.ts.map +1 -1
- package/dist/Store/ContextMapContainer.js +13 -3
- package/dist/Store/Cosmos/query.d.ts +1 -1
- package/dist/Store/Cosmos/query.d.ts.map +1 -1
- package/dist/Store/Cosmos/query.js +8 -10
- package/dist/Store/Cosmos.d.ts +1 -1
- package/dist/Store/Cosmos.d.ts.map +1 -1
- package/dist/Store/Cosmos.js +308 -242
- package/dist/Store/Disk.d.ts +2 -2
- package/dist/Store/Disk.d.ts.map +1 -1
- package/dist/Store/Disk.js +25 -22
- package/dist/Store/Memory.d.ts +4 -4
- package/dist/Store/Memory.d.ts.map +1 -1
- package/dist/Store/Memory.js +27 -22
- package/dist/Store/SQL/Pg.d.ts +4 -0
- package/dist/Store/SQL/Pg.d.ts.map +1 -0
- package/dist/Store/SQL/Pg.js +189 -0
- package/dist/Store/SQL/query.d.ts +38 -0
- package/dist/Store/SQL/query.d.ts.map +1 -0
- package/dist/Store/SQL/query.js +367 -0
- package/dist/Store/SQL.d.ts +20 -0
- package/dist/Store/SQL.d.ts.map +1 -0
- package/dist/Store/SQL.js +381 -0
- package/dist/Store/codeFilter.d.ts +1 -1
- package/dist/Store/codeFilter.d.ts.map +1 -1
- package/dist/Store/codeFilter.js +2 -1
- package/dist/Store/index.d.ts +5 -2
- package/dist/Store/index.d.ts.map +1 -1
- package/dist/Store/index.js +15 -3
- package/dist/Store/service.d.ts +17 -6
- package/dist/Store/service.d.ts.map +1 -1
- package/dist/Store/service.js +24 -6
- package/dist/Store/utils.d.ts +1 -1
- package/dist/Store/utils.d.ts.map +1 -1
- package/dist/Store/utils.js +3 -4
- package/dist/Store.d.ts +1 -1
- package/dist/adapters/SQL/Model.d.ts +28 -42
- package/dist/adapters/SQL/Model.d.ts.map +1 -1
- package/dist/adapters/SQL/Model.js +2 -2
- package/dist/adapters/SQL.d.ts +1 -1
- package/dist/adapters/ServiceBus.d.ts +9 -9
- package/dist/adapters/ServiceBus.d.ts.map +1 -1
- package/dist/adapters/ServiceBus.js +13 -15
- package/dist/adapters/cosmos-client.d.ts +3 -3
- package/dist/adapters/cosmos-client.d.ts.map +1 -1
- package/dist/adapters/cosmos-client.js +3 -3
- package/dist/adapters/index.d.ts +8 -2
- package/dist/adapters/index.d.ts.map +1 -1
- package/dist/adapters/index.js +8 -2
- package/dist/adapters/logger.d.ts +1 -1
- package/dist/adapters/logger.d.ts.map +1 -1
- package/dist/adapters/memQueue.d.ts +3 -3
- package/dist/adapters/memQueue.d.ts.map +1 -1
- package/dist/adapters/memQueue.js +3 -3
- package/dist/adapters/mongo-client.d.ts +3 -3
- package/dist/adapters/mongo-client.d.ts.map +1 -1
- package/dist/adapters/mongo-client.js +3 -3
- package/dist/adapters/redis-client.d.ts +3 -3
- package/dist/adapters/redis-client.d.ts.map +1 -1
- package/dist/adapters/redis-client.js +3 -3
- package/dist/api/ContextProvider.d.ts +7 -7
- package/dist/api/ContextProvider.d.ts.map +1 -1
- package/dist/api/ContextProvider.js +6 -6
- package/dist/api/codec.d.ts +1 -1
- package/dist/api/internal/RequestContextMiddleware.d.ts +2 -2
- package/dist/api/internal/RequestContextMiddleware.d.ts.map +1 -1
- package/dist/api/internal/RequestContextMiddleware.js +2 -2
- package/dist/api/internal/auth.d.ts +44 -6
- package/dist/api/internal/auth.d.ts.map +1 -1
- package/dist/api/internal/auth.js +160 -29
- package/dist/api/internal/events.d.ts +3 -3
- package/dist/api/internal/events.d.ts.map +1 -1
- package/dist/api/internal/events.js +9 -7
- package/dist/api/internal/health.d.ts +1 -1
- package/dist/api/layerUtils.d.ts +6 -6
- package/dist/api/layerUtils.d.ts.map +1 -1
- package/dist/api/layerUtils.js +5 -5
- package/dist/api/middlewares.d.ts +1 -1
- package/dist/api/reportError.d.ts +1 -1
- package/dist/api/routing/middleware/RouterMiddleware.d.ts +4 -4
- package/dist/api/routing/middleware/RouterMiddleware.d.ts.map +1 -1
- package/dist/api/routing/middleware/middleware.d.ts +39 -3
- package/dist/api/routing/middleware/middleware.d.ts.map +1 -1
- package/dist/api/routing/middleware/middleware.js +45 -14
- package/dist/api/routing/middleware.d.ts +1 -2
- package/dist/api/routing/middleware.d.ts.map +1 -1
- package/dist/api/routing/middleware.js +1 -2
- package/dist/api/routing/schema/jwt.d.ts +1 -1
- package/dist/api/routing/schema/jwt.d.ts.map +1 -1
- package/dist/api/routing/tsort.d.ts +1 -1
- package/dist/api/routing/tsort.d.ts.map +1 -1
- package/dist/api/routing/utils.d.ts +3 -3
- package/dist/api/routing/utils.d.ts.map +1 -1
- package/dist/api/routing.d.ts +12 -14
- package/dist/api/routing.d.ts.map +1 -1
- package/dist/api/routing.js +17 -6
- package/dist/api/setupRequest.d.ts +8 -5
- package/dist/api/setupRequest.d.ts.map +1 -1
- package/dist/api/setupRequest.js +12 -7
- package/dist/api/util.d.ts +1 -1
- package/dist/arbs.d.ts +1 -1
- package/dist/arbs.d.ts.map +1 -1
- package/dist/arbs.js +5 -3
- package/dist/errorReporter.d.ts +4 -4
- package/dist/errorReporter.d.ts.map +1 -1
- package/dist/errorReporter.js +16 -23
- package/dist/errors.d.ts +1 -1
- package/dist/fileUtil.d.ts +1 -1
- package/dist/fileUtil.d.ts.map +1 -1
- package/dist/index.d.ts +1 -1
- package/dist/logger/jsonLogger.d.ts +1 -1
- package/dist/logger/logFmtLogger.d.ts +1 -1
- package/dist/logger/shared.d.ts +1 -1
- package/dist/logger/shared.js +2 -2
- package/dist/logger.d.ts +1 -1
- package/dist/logger.d.ts.map +1 -1
- package/dist/rateLimit.d.ts +9 -3
- package/dist/rateLimit.d.ts.map +1 -1
- package/dist/rateLimit.js +5 -11
- package/dist/test.d.ts +1 -1
- package/dist/test.d.ts.map +1 -1
- package/dist/vitest.d.ts +1 -1
- package/eslint.config.mjs +3 -3
- package/examples/query.ts +39 -35
- package/package.json +42 -28
- package/src/CUPS.ts +9 -11
- package/src/Emailer/Sendgrid.ts +17 -14
- package/src/Emailer/service.ts +8 -2
- package/src/MainFiberSet.ts +3 -3
- package/src/Model/Repository/Registry.ts +33 -0
- package/src/Model/Repository/ext.ts +93 -6
- package/src/Model/Repository/internal/internal.ts +97 -88
- package/src/Model/Repository/makeRepo.ts +12 -10
- package/src/Model/Repository/service.ts +31 -22
- package/src/Model/Repository/validation.ts +4 -4
- package/src/Model/Repository.ts +1 -0
- package/src/Model/dsl.ts +3 -3
- package/src/Model/query/new-kid-interpreter.ts +2 -2
- package/src/Model.ts +1 -0
- package/src/Operations.ts +78 -113
- package/src/OperationsRepo.ts +2 -2
- package/src/QueueMaker/SQLQueue.ts +121 -151
- package/src/QueueMaker/memQueue.ts +82 -103
- package/src/QueueMaker/sbqueue.ts +56 -86
- package/src/RequestContext.ts +8 -8
- package/src/RequestFiberSet.ts +4 -4
- package/src/Store/ContextMapContainer.ts +41 -2
- package/src/Store/Cosmos/query.ts +9 -11
- package/src/Store/Cosmos.ts +437 -343
- package/src/Store/Disk.ts +52 -49
- package/src/Store/Memory.ts +54 -48
- package/src/Store/SQL/Pg.ts +318 -0
- package/src/Store/SQL/query.ts +409 -0
- package/src/Store/SQL.ts +668 -0
- package/src/Store/codeFilter.ts +1 -0
- package/src/Store/index.ts +17 -2
- package/src/Store/service.ts +31 -7
- package/src/Store/utils.ts +23 -22
- package/src/adapters/SQL/Model.ts +10 -4
- package/src/adapters/ServiceBus.ts +111 -115
- package/src/adapters/cosmos-client.ts +2 -2
- package/src/adapters/index.ts +7 -0
- package/src/adapters/memQueue.ts +2 -2
- package/src/adapters/mongo-client.ts +2 -2
- package/src/adapters/redis-client.ts +2 -2
- package/src/api/ContextProvider.ts +11 -11
- package/src/api/internal/RequestContextMiddleware.ts +1 -1
- package/src/api/internal/auth.ts +246 -44
- package/src/api/internal/events.ts +12 -8
- package/src/api/layerUtils.ts +8 -8
- package/src/api/routing/middleware/RouterMiddleware.ts +4 -4
- package/src/api/routing/middleware/middleware.ts +52 -12
- package/src/api/routing/middleware.ts +0 -2
- package/src/api/routing.ts +21 -7
- package/src/api/setupRequest.ts +28 -8
- package/src/arbs.ts +4 -2
- package/src/errorReporter.ts +58 -72
- package/src/logger/shared.ts +1 -1
- package/src/rateLimit.ts +30 -22
- package/test/auth.test.ts +101 -0
- package/test/contextProvider.test.ts +11 -11
- package/test/controller.test.ts +18 -14
- package/test/dist/auth.test.d.ts.map +1 -0
- package/test/dist/contextProvider.test.d.ts.map +1 -1
- package/test/dist/controller.test.d.ts.map +1 -1
- package/test/dist/date-query.test.d.ts.map +1 -0
- package/test/dist/fixtures.d.ts +26 -12
- package/test/dist/fixtures.d.ts.map +1 -1
- package/test/dist/fixtures.js +12 -10
- package/test/dist/query.test.d.ts.map +1 -1
- package/test/dist/rawQuery.test.d.ts.map +1 -1
- package/test/dist/repository-ext.test.d.ts.map +1 -0
- package/test/dist/requires.test.d.ts.map +1 -1
- package/test/dist/router-generator.test.d.ts.map +1 -0
- package/test/dist/routing-interruptibility.test.d.ts.map +1 -0
- package/test/dist/rpc-multi-middleware.test.d.ts.map +1 -1
- package/test/dist/sql-store.test.d.ts.map +1 -0
- package/test/fixtures.ts +11 -9
- package/test/query.test.ts +216 -34
- package/test/rawQuery.test.ts +23 -19
- package/test/repository-ext.test.ts +60 -0
- package/test/requires.test.ts +6 -6
- package/test/router-generator.test.ts +180 -0
- package/test/routing-interruptibility.test.ts +63 -0
- package/test/rpc-multi-middleware.test.ts +78 -9
- package/test/sql-store.test.ts +1064 -0
- package/test/validateSample.test.ts +15 -12
- package/tsconfig.examples.json +1 -1
- package/tsconfig.json +0 -1
- package/tsconfig.json.bak +2 -2
- package/tsconfig.src.json +35 -35
- package/tsconfig.test.json +2 -2
|
@@ -0,0 +1,180 @@
|
|
|
1
|
+
import { type MakeContext, type MakeErrors, makeRouter } from "@effect-app/infra/api/routing"
|
|
2
|
+
import { makeAllDSL, makeOneDSL } from "@effect-app/infra/Model"
|
|
3
|
+
import { expectTypeOf, it } from "@effect/vitest"
|
|
4
|
+
import { Context, Effect, Layer, RpcX, S } from "effect-app"
|
|
5
|
+
import { InvalidStateError, makeRpcClient, UnauthorizedError } from "effect-app/client"
|
|
6
|
+
import { DefaultGenericMiddlewares } from "effect-app/middleware"
|
|
7
|
+
import { type FixEnv } from "effect-app/Pure"
|
|
8
|
+
import { MiddlewareMaker } from "effect-app/rpc"
|
|
9
|
+
import { type TypeTestId } from "effect-app/TypeTest"
|
|
10
|
+
import { type ConfigError } from "effect/Config"
|
|
11
|
+
import { type RpcSerialization } from "effect/unstable/rpc/RpcSerialization"
|
|
12
|
+
import { DefaultGenericMiddlewaresLive, DevModeMiddlewareLive } from "../src/api/routing/middleware.js"
|
|
13
|
+
import { AllowAnonymous, AllowAnonymousLive, RequestContextMap, RequireRoles, RequireRolesLive, Some, SomeElse, SomeService, Test, TestLive } from "./fixtures.js"
|
|
14
|
+
|
|
15
|
+
// Inline minimal context provider (provides `Some`)
|
|
16
|
+
class CtxProvider extends RpcX.RpcMiddleware.Tag<CtxProvider, { provides: Some }>()("CtxProvider") {
|
|
17
|
+
static Default = Layer.make(this, {
|
|
18
|
+
*make() {
|
|
19
|
+
return Effect.fnUntraced(function*(effect) {
|
|
20
|
+
return yield* Effect.provideService(effect, Some, Some.of({ a: 1 }))
|
|
21
|
+
})
|
|
22
|
+
}
|
|
23
|
+
})
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
// Provides `SomeElse` so AllowAnonymous's requirement is met.
|
|
27
|
+
class SomeElseProvider extends RpcX.RpcMiddleware.Tag<SomeElseProvider, { provides: SomeElse }>()("SomeElseProvider") {
|
|
28
|
+
static Default = Layer.make(this, {
|
|
29
|
+
*make() {
|
|
30
|
+
return Effect.fnUntraced(function*(effect) {
|
|
31
|
+
return yield* Effect.provideService(effect, SomeElse, SomeElse.of({ b: 2 }))
|
|
32
|
+
})
|
|
33
|
+
}
|
|
34
|
+
})
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
class mw extends MiddlewareMaker
|
|
38
|
+
.Tag<mw>()("mw", RequestContextMap)
|
|
39
|
+
.middleware(RequireRoles, Test)
|
|
40
|
+
.middleware(AllowAnonymous)
|
|
41
|
+
.middleware(CtxProvider)
|
|
42
|
+
.middleware(...DefaultGenericMiddlewares, SomeElseProvider)
|
|
43
|
+
{
|
|
44
|
+
static Default = this.layer.pipe(
|
|
45
|
+
Layer.provide([
|
|
46
|
+
RequireRolesLive,
|
|
47
|
+
TestLive,
|
|
48
|
+
AllowAnonymousLive,
|
|
49
|
+
CtxProvider.Default,
|
|
50
|
+
SomeElseProvider.Default,
|
|
51
|
+
DefaultGenericMiddlewaresLive,
|
|
52
|
+
DevModeMiddlewareLive,
|
|
53
|
+
SomeService.Default
|
|
54
|
+
])
|
|
55
|
+
)
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
const { TaggedRequestFor } = makeRpcClient(RequestContextMap)
|
|
59
|
+
const Req = TaggedRequestFor("GenRouter")
|
|
60
|
+
|
|
61
|
+
class GetThing extends Req.Query<GetThing>()("GetThing", { id: S.String }, { success: S.String }) {}
|
|
62
|
+
class DoThing extends Req.Command<DoThing>()("DoThing", { id: S.String }, { success: S.Void }) {}
|
|
63
|
+
|
|
64
|
+
const Resource = { GetThing, DoThing }
|
|
65
|
+
|
|
66
|
+
const { Router, matchAll } = makeRouter(mw)
|
|
67
|
+
|
|
68
|
+
class ThingRepo extends Context.Service<ThingRepo>()("ThingRepo", {
|
|
69
|
+
make: Effect.succeed({ get: (id: string) => Effect.succeed(id + "!") })
|
|
70
|
+
}) {
|
|
71
|
+
static Default = Layer.effect(this, this.make)
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
// Case under test:
|
|
75
|
+
// `match({})` is given handlers as **shorthand generator methods** (`*GetThing(req) { ... }`).
|
|
76
|
+
// tsgo (>= 7 dev) infers `TNext = unknown` for these shorthand generators while TS6 infers `never`.
|
|
77
|
+
// `HandlerWithInputGen` in routing.ts must accept both — see the structural fix.
|
|
78
|
+
const router = Router(Resource)({
|
|
79
|
+
dependencies: [ThingRepo.Default],
|
|
80
|
+
*effect(match) {
|
|
81
|
+
const repo = yield* ThingRepo
|
|
82
|
+
|
|
83
|
+
if (Math.random() > 0.5) return yield* new InvalidStateError("nope")
|
|
84
|
+
|
|
85
|
+
return match({
|
|
86
|
+
*GetThing(req) {
|
|
87
|
+
const some = yield* Some
|
|
88
|
+
if (req.id === "boom") {
|
|
89
|
+
return yield* Effect.fail(new UnauthorizedError())
|
|
90
|
+
}
|
|
91
|
+
return yield* repo.get(req.id + String(some.a))
|
|
92
|
+
},
|
|
93
|
+
*DoThing(_req) {
|
|
94
|
+
yield* Effect.succeed(1)
|
|
95
|
+
}
|
|
96
|
+
})
|
|
97
|
+
}
|
|
98
|
+
})
|
|
99
|
+
|
|
100
|
+
// Same scenario but using the `raw:` variant — exercises the `raw` path of `HandlerWithInputGen`.
|
|
101
|
+
const routerRaw = Router({ GetThing })({
|
|
102
|
+
*effect(match) {
|
|
103
|
+
return match({
|
|
104
|
+
GetThing: {
|
|
105
|
+
*raw(req) {
|
|
106
|
+
const some = yield* Some
|
|
107
|
+
return yield* Effect.succeed(req.id + String(some.a))
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
})
|
|
111
|
+
}
|
|
112
|
+
})
|
|
113
|
+
|
|
114
|
+
it("router with generator-method handlers compiles", () => {
|
|
115
|
+
expectTypeOf(router).toMatchTypeOf<
|
|
116
|
+
Layer.Layer<never, ConfigError | InvalidStateError, SomeService | RpcSerialization>
|
|
117
|
+
>()
|
|
118
|
+
expectTypeOf(routerRaw).toMatchTypeOf<Layer.Layer<never, ConfigError, SomeService | RpcSerialization>>()
|
|
119
|
+
})
|
|
120
|
+
|
|
121
|
+
// Type-level assertions: verify generator yields propagate to MakeErrors / MakeContext
|
|
122
|
+
type Errors = MakeErrors<typeof router[TypeTestId]>
|
|
123
|
+
type Ctx = MakeContext<typeof router[TypeTestId]>
|
|
124
|
+
expectTypeOf<Errors>().toEqualTypeOf<InvalidStateError>()
|
|
125
|
+
expectTypeOf<Ctx>().toEqualTypeOf<ThingRepo>()
|
|
126
|
+
|
|
127
|
+
const matched = matchAll({ router })
|
|
128
|
+
expectTypeOf(matched).toMatchTypeOf<
|
|
129
|
+
Layer.Layer<never, ConfigError | InvalidStateError, SomeService | RpcSerialization>
|
|
130
|
+
>()
|
|
131
|
+
|
|
132
|
+
// ---------------------------------------------------------------------------
|
|
133
|
+
// DSL R-inference regression
|
|
134
|
+
// ---------------------------------------------------------------------------
|
|
135
|
+
// `OneDSL`/`OneDSLExt.update`/`.modify` previously annotated the callback's
|
|
136
|
+
// effect R as `FixEnv<R, Evt, S1, S2>`. That deadlocked inference of `R`
|
|
137
|
+
// (TS6 → `never`, tsgo → `unknown`), causing yielded effects to leak
|
|
138
|
+
// `unknown` in the R slot when consumed by generator handlers.
|
|
139
|
+
// The fix uses bare `R` in the callback and `FixEnv<R, …>` only on the return.
|
|
140
|
+
class Item extends S.Class<Item>("Item")({ id: S.String, label: S.String }) {}
|
|
141
|
+
class Dep extends Context.Service<Dep>()("Dep", { make: Effect.succeed({ tag: "dep" as const }) }) {}
|
|
142
|
+
|
|
143
|
+
type Evt = { _tag: "Updated"; id: string }
|
|
144
|
+
|
|
145
|
+
const Items$ = makeAllDSL<Item, Evt>()
|
|
146
|
+
const Item$ = makeOneDSL<Item, Evt>()
|
|
147
|
+
|
|
148
|
+
// Callback body uses generator syntax (TNext = unknown under tsgo) and yields
|
|
149
|
+
// a service-dependent effect — R must be inferred as `Dep` (plus the
|
|
150
|
+
// canonical PureEnvEnv contributed by FixEnv on the return).
|
|
151
|
+
const oneUpdate = Item$.update((item) =>
|
|
152
|
+
Effect.gen(function*() {
|
|
153
|
+
const dep = yield* Dep
|
|
154
|
+
return new Item({ id: item.id, label: item.label + dep.tag })
|
|
155
|
+
})
|
|
156
|
+
)
|
|
157
|
+
|
|
158
|
+
const allUpdate = Items$.update((items) =>
|
|
159
|
+
Effect.gen(function*() {
|
|
160
|
+
const dep = yield* Dep
|
|
161
|
+
return items.map((_) => new Item({ id: _.id, label: _.label + dep.tag }))
|
|
162
|
+
})
|
|
163
|
+
)
|
|
164
|
+
|
|
165
|
+
const oneModify = Item$.modify((item, _dsl) =>
|
|
166
|
+
Effect.gen(function*() {
|
|
167
|
+
const dep = yield* Dep
|
|
168
|
+
return { ...item, tag: dep.tag }
|
|
169
|
+
})
|
|
170
|
+
)
|
|
171
|
+
|
|
172
|
+
// `R` should be `FixEnv<Dep, Evt, …>` — never collapsed to `unknown`/`never`.
|
|
173
|
+
// The regression manifested as `unknown` here, breaking `Dep` assignability.
|
|
174
|
+
expectTypeOf(oneUpdate).toMatchTypeOf<Effect.Effect<Item, never, FixEnv<Dep, Evt, Item, Item>>>()
|
|
175
|
+
expectTypeOf(allUpdate).toMatchTypeOf<
|
|
176
|
+
Effect.Effect<readonly Item[], never, FixEnv<Dep, Evt, readonly Item[], readonly Item[]>>
|
|
177
|
+
>()
|
|
178
|
+
expectTypeOf(oneModify).toMatchTypeOf<
|
|
179
|
+
Effect.Effect<{ tag: "dep"; id: string; label: string }, never, FixEnv<Dep, Evt, Item, Item>>
|
|
180
|
+
>()
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
import { describe, expect, it } from "@effect/vitest"
|
|
2
|
+
import { Effect, Fiber, Layer, Ref } from "effect"
|
|
3
|
+
import { S } from "effect-app"
|
|
4
|
+
import { ConfigureInterruptibilityMiddleware } from "effect-app/middleware"
|
|
5
|
+
import { Rpc, RpcGroup, RpcTest } from "effect/unstable/rpc"
|
|
6
|
+
import { applyRequestTypeInterruptibility } from "../src/api/routing.js"
|
|
7
|
+
import { ConfigureInterruptibilityMiddlewareLive, RequestType } from "../src/api/routing/middleware.js"
|
|
8
|
+
|
|
9
|
+
const InterruptibilityRpcs = RpcGroup.make(
|
|
10
|
+
Rpc
|
|
11
|
+
.make("doCommand", { success: S.Void })
|
|
12
|
+
.annotate(RequestType, "command")
|
|
13
|
+
.middleware(ConfigureInterruptibilityMiddleware),
|
|
14
|
+
Rpc
|
|
15
|
+
.make("doQuery", { success: S.Void })
|
|
16
|
+
.annotate(RequestType, "query")
|
|
17
|
+
.middleware(ConfigureInterruptibilityMiddleware)
|
|
18
|
+
)
|
|
19
|
+
|
|
20
|
+
const makeImplLayer = (commandDone: Ref.Ref<boolean>, queryDone: Ref.Ref<boolean>) =>
|
|
21
|
+
InterruptibilityRpcs.toLayer({
|
|
22
|
+
doCommand: () =>
|
|
23
|
+
applyRequestTypeInterruptibility(
|
|
24
|
+
"command",
|
|
25
|
+
Effect.sleep("120 millis").pipe(Effect.andThen(Ref.set(commandDone, true)))
|
|
26
|
+
),
|
|
27
|
+
doQuery: () =>
|
|
28
|
+
applyRequestTypeInterruptibility(
|
|
29
|
+
"query",
|
|
30
|
+
Effect.sleep("120 millis").pipe(Effect.andThen(Ref.set(queryDone, true)))
|
|
31
|
+
)
|
|
32
|
+
})
|
|
33
|
+
|
|
34
|
+
describe("routing interruptibility", () => {
|
|
35
|
+
it.live(
|
|
36
|
+
"e2e: command continues after client interrupt, query does not",
|
|
37
|
+
() =>
|
|
38
|
+
Effect.gen(function*() {
|
|
39
|
+
const commandDone = yield* Ref.make(false)
|
|
40
|
+
const queryDone = yield* Ref.make(false)
|
|
41
|
+
|
|
42
|
+
const client = yield* RpcTest
|
|
43
|
+
.makeClient(InterruptibilityRpcs)
|
|
44
|
+
.pipe(
|
|
45
|
+
Effect.provide(
|
|
46
|
+
Layer.mergeAll(makeImplLayer(commandDone, queryDone), ConfigureInterruptibilityMiddlewareLive)
|
|
47
|
+
)
|
|
48
|
+
)
|
|
49
|
+
|
|
50
|
+
const commandFiber = yield* Effect.forkDetach(client.doCommand())
|
|
51
|
+
yield* Effect.sleep("20 millis")
|
|
52
|
+
yield* Fiber.interrupt(commandFiber)
|
|
53
|
+
yield* Effect.sleep("180 millis")
|
|
54
|
+
expect(yield* Ref.get(commandDone)).toBe(true)
|
|
55
|
+
|
|
56
|
+
const queryFiber = yield* Effect.forkDetach(client.doQuery())
|
|
57
|
+
yield* Effect.sleep("20 millis")
|
|
58
|
+
yield* Fiber.interrupt(queryFiber)
|
|
59
|
+
yield* Effect.sleep("180 millis")
|
|
60
|
+
expect(yield* Ref.get(queryDone)).toBe(false)
|
|
61
|
+
})
|
|
62
|
+
)
|
|
63
|
+
})
|
|
@@ -1,19 +1,18 @@
|
|
|
1
1
|
import { NodeHttpServer } from "@effect/platform-node"
|
|
2
2
|
import { expect, expectTypeOf, it } from "@effect/vitest"
|
|
3
|
-
import { Console, Effect, Layer, Result } from "effect"
|
|
4
|
-
import { S } from "effect-app"
|
|
3
|
+
import { Console, Effect, Layer, Ref, Result } from "effect"
|
|
4
|
+
import { Context, RpcX, S } from "effect-app"
|
|
5
5
|
import { NotLoggedInError } from "effect-app/client"
|
|
6
6
|
import { HttpRouter } from "effect-app/http"
|
|
7
7
|
import { DefaultGenericMiddlewares } from "effect-app/middleware"
|
|
8
|
-
import { MiddlewareMaker } from "effect-app/rpc"
|
|
9
|
-
import { middlewareGroup } from "effect-app/rpc/MiddlewareMaker"
|
|
10
8
|
import { FetchHttpClient } from "effect/unstable/http"
|
|
11
|
-
import { RpcClient, RpcGroup, RpcSerialization, RpcServer, RpcTest } from "effect/unstable/rpc"
|
|
9
|
+
import { Rpc, RpcClient, RpcGroup, RpcSerialization, RpcServer, RpcTest } from "effect/unstable/rpc"
|
|
12
10
|
import { createServer } from "http"
|
|
13
11
|
import { DefaultGenericMiddlewaresLive } from "../src/api/routing.js"
|
|
14
12
|
import { AllowAnonymous, AllowAnonymousLive, RequestContextMap, RequireRoles, RequireRolesLive, Some, SomeElseMiddleware, SomeElseMiddlewareLive, SomeMiddleware, SomeMiddlewareLive, SomeService, Test, TestLive, UserProfile } from "./fixtures.js"
|
|
15
13
|
|
|
16
|
-
const incomplete =
|
|
14
|
+
const incomplete = RpcX
|
|
15
|
+
.MiddlewareMaker
|
|
17
16
|
.Tag<middleware>()("MiddlewareMaker", RequestContextMap)
|
|
18
17
|
.middleware(RequireRoles)
|
|
19
18
|
.middleware(AllowAnonymous, Test)
|
|
@@ -21,7 +20,8 @@ const incomplete = MiddlewareMaker
|
|
|
21
20
|
// this extension is allowed otherwise the error is quite obscure
|
|
22
21
|
export class incompleteMiddleware extends incomplete {}
|
|
23
22
|
|
|
24
|
-
class middleware extends
|
|
23
|
+
class middleware extends RpcX
|
|
24
|
+
.MiddlewareMaker
|
|
25
25
|
.Tag<middleware>()("MiddlewareMaker", RequestContextMap)
|
|
26
26
|
.middleware(RequireRoles)
|
|
27
27
|
.middleware(AllowAnonymous, Test)
|
|
@@ -29,7 +29,7 @@ class middleware extends MiddlewareMaker
|
|
|
29
29
|
.middleware(...DefaultGenericMiddlewares)
|
|
30
30
|
{}
|
|
31
31
|
|
|
32
|
-
const UserRpcs = middlewareGroup(middleware)(
|
|
32
|
+
const UserRpcs = RpcX.MiddlewareMaker.middlewareGroup(middleware)(
|
|
33
33
|
RpcGroup.make(
|
|
34
34
|
middleware.rpc("getUser", {
|
|
35
35
|
success: S.Literal("awesome")
|
|
@@ -56,7 +56,7 @@ const impl = UserRpcs
|
|
|
56
56
|
|
|
57
57
|
expectTypeOf<Layer.Services<typeof impl>>().toEqualTypeOf<never>()
|
|
58
58
|
|
|
59
|
-
const UserRpcsBad = middlewareGroup(middleware)(
|
|
59
|
+
const UserRpcsBad = RpcX.MiddlewareMaker.middlewareGroup(middleware)(
|
|
60
60
|
RpcGroup.make(
|
|
61
61
|
middleware.rpc("doSomethingElse", {
|
|
62
62
|
success: S.Literal("also-awesome2"),
|
|
@@ -136,3 +136,72 @@ it.live(
|
|
|
136
136
|
Effect.provide(RpcTestLayer)
|
|
137
137
|
)
|
|
138
138
|
)
|
|
139
|
+
|
|
140
|
+
// Per-request service isolation test
|
|
141
|
+
|
|
142
|
+
class PerRequestCounter extends Context.Service<PerRequestCounter>()(
|
|
143
|
+
"PerRequestCounter",
|
|
144
|
+
{ make: Effect.sync(() => ({ a: 0 })) }
|
|
145
|
+
) {
|
|
146
|
+
static Default = Layer.effect(this, this.make)
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
class GlobalCounter extends Context.Service<GlobalCounter, {
|
|
150
|
+
readonly ref: Ref.Ref<number>
|
|
151
|
+
}>()("GlobalCounter") {}
|
|
152
|
+
|
|
153
|
+
const CounterRpcs = RpcGroup.make(
|
|
154
|
+
Rpc.make("incrementA", {
|
|
155
|
+
success: S.Number
|
|
156
|
+
}),
|
|
157
|
+
Rpc.make("incrementB", {
|
|
158
|
+
success: S.Number
|
|
159
|
+
})
|
|
160
|
+
)
|
|
161
|
+
|
|
162
|
+
const counterImpl = CounterRpcs
|
|
163
|
+
.toLayer({
|
|
164
|
+
incrementA: Effect.fn(function*() {
|
|
165
|
+
const counter = yield* PerRequestCounter
|
|
166
|
+
counter.a++
|
|
167
|
+
const global = yield* GlobalCounter
|
|
168
|
+
yield* Ref.update(global.ref, (n) => n + 1)
|
|
169
|
+
return counter.a
|
|
170
|
+
}, Effect.provide(PerRequestCounter.Default)),
|
|
171
|
+
incrementB: Effect.fn(function*() {
|
|
172
|
+
const counter = yield* PerRequestCounter
|
|
173
|
+
counter.a++
|
|
174
|
+
const global = yield* GlobalCounter
|
|
175
|
+
yield* Ref.update(global.ref, (n) => n + 1)
|
|
176
|
+
return counter.a
|
|
177
|
+
}, Effect.provide(PerRequestCounter.Default))
|
|
178
|
+
})
|
|
179
|
+
|
|
180
|
+
const GlobalCounterLive = Layer.effect(
|
|
181
|
+
GlobalCounter,
|
|
182
|
+
Ref.make(0).pipe(Effect.map((ref) => ({ ref })))
|
|
183
|
+
)
|
|
184
|
+
|
|
185
|
+
const CounterTestLayer = counterImpl.pipe(Layer.provideMerge(GlobalCounterLive))
|
|
186
|
+
|
|
187
|
+
it.live(
|
|
188
|
+
"per-request service isolation with shared global counter",
|
|
189
|
+
Effect.fnUntraced(
|
|
190
|
+
function*() {
|
|
191
|
+
const client = yield* RpcTest.makeClient(CounterRpcs)
|
|
192
|
+
const global = yield* GlobalCounter
|
|
193
|
+
|
|
194
|
+
const r1 = yield* client.incrementA()
|
|
195
|
+
const r2 = yield* client.incrementB()
|
|
196
|
+
|
|
197
|
+
// per-request counter is fresh each time → both return 1
|
|
198
|
+
expect(r1).toBe(1)
|
|
199
|
+
expect(r2).toBe(1)
|
|
200
|
+
|
|
201
|
+
// global counter is shared across requests → accumulates to 2
|
|
202
|
+
const globalCount = yield* Ref.get(global.ref)
|
|
203
|
+
expect(globalCount).toBe(2)
|
|
204
|
+
},
|
|
205
|
+
Effect.provide(CounterTestLayer)
|
|
206
|
+
)
|
|
207
|
+
)
|