@effect-app/infra 4.0.0-beta.11 → 4.0.0-beta.111

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 (157) hide show
  1. package/CHANGELOG.md +729 -0
  2. package/dist/CUPS.d.ts +3 -3
  3. package/dist/CUPS.d.ts.map +1 -1
  4. package/dist/CUPS.js +3 -3
  5. package/dist/Emailer/service.d.ts +3 -3
  6. package/dist/Emailer/service.d.ts.map +1 -1
  7. package/dist/Emailer/service.js +3 -3
  8. package/dist/MainFiberSet.d.ts +2 -2
  9. package/dist/MainFiberSet.d.ts.map +1 -1
  10. package/dist/MainFiberSet.js +3 -3
  11. package/dist/Model/Repository/internal/internal.d.ts +3 -3
  12. package/dist/Model/Repository/internal/internal.d.ts.map +1 -1
  13. package/dist/Model/Repository/internal/internal.js +21 -16
  14. package/dist/Model/Repository/makeRepo.d.ts +2 -2
  15. package/dist/Model/Repository/makeRepo.d.ts.map +1 -1
  16. package/dist/Model/Repository/makeRepo.js +1 -1
  17. package/dist/Model/Repository/validation.d.ts +5 -4
  18. package/dist/Model/Repository/validation.d.ts.map +1 -1
  19. package/dist/Model/query/dsl.d.ts +9 -9
  20. package/dist/Operations.d.ts +2 -2
  21. package/dist/Operations.d.ts.map +1 -1
  22. package/dist/Operations.js +3 -3
  23. package/dist/OperationsRepo.d.ts +2 -2
  24. package/dist/OperationsRepo.d.ts.map +1 -1
  25. package/dist/OperationsRepo.js +3 -3
  26. package/dist/QueueMaker/SQLQueue.d.ts +2 -4
  27. package/dist/QueueMaker/SQLQueue.d.ts.map +1 -1
  28. package/dist/QueueMaker/SQLQueue.js +8 -6
  29. package/dist/QueueMaker/errors.d.ts +1 -1
  30. package/dist/QueueMaker/errors.d.ts.map +1 -1
  31. package/dist/QueueMaker/memQueue.js +3 -3
  32. package/dist/QueueMaker/sbqueue.js +3 -3
  33. package/dist/RequestContext.d.ts +19 -14
  34. package/dist/RequestContext.d.ts.map +1 -1
  35. package/dist/RequestContext.js +5 -5
  36. package/dist/RequestFiberSet.d.ts +2 -2
  37. package/dist/RequestFiberSet.d.ts.map +1 -1
  38. package/dist/RequestFiberSet.js +5 -5
  39. package/dist/Store/ContextMapContainer.d.ts +19 -3
  40. package/dist/Store/ContextMapContainer.d.ts.map +1 -1
  41. package/dist/Store/ContextMapContainer.js +13 -3
  42. package/dist/Store/Cosmos.d.ts.map +1 -1
  43. package/dist/Store/Cosmos.js +136 -68
  44. package/dist/Store/Disk.d.ts.map +1 -1
  45. package/dist/Store/Disk.js +3 -4
  46. package/dist/Store/Memory.d.ts +2 -2
  47. package/dist/Store/Memory.d.ts.map +1 -1
  48. package/dist/Store/Memory.js +4 -4
  49. package/dist/Store/SQL/Pg.d.ts +4 -0
  50. package/dist/Store/SQL/Pg.d.ts.map +1 -0
  51. package/dist/Store/SQL/Pg.js +186 -0
  52. package/dist/Store/SQL/query.d.ts +38 -0
  53. package/dist/Store/SQL/query.d.ts.map +1 -0
  54. package/dist/Store/SQL/query.js +367 -0
  55. package/dist/Store/SQL.d.ts +20 -0
  56. package/dist/Store/SQL.d.ts.map +1 -0
  57. package/dist/Store/SQL.js +370 -0
  58. package/dist/Store/index.d.ts +4 -1
  59. package/dist/Store/index.d.ts.map +1 -1
  60. package/dist/Store/index.js +12 -2
  61. package/dist/Store/service.d.ts +11 -5
  62. package/dist/Store/service.d.ts.map +1 -1
  63. package/dist/Store/service.js +24 -6
  64. package/dist/adapters/ServiceBus.d.ts +6 -6
  65. package/dist/adapters/ServiceBus.d.ts.map +1 -1
  66. package/dist/adapters/ServiceBus.js +9 -9
  67. package/dist/adapters/cosmos-client.d.ts +2 -2
  68. package/dist/adapters/cosmos-client.d.ts.map +1 -1
  69. package/dist/adapters/cosmos-client.js +3 -3
  70. package/dist/adapters/logger.d.ts.map +1 -1
  71. package/dist/adapters/memQueue.d.ts +2 -2
  72. package/dist/adapters/memQueue.d.ts.map +1 -1
  73. package/dist/adapters/memQueue.js +3 -3
  74. package/dist/adapters/mongo-client.d.ts +2 -2
  75. package/dist/adapters/mongo-client.d.ts.map +1 -1
  76. package/dist/adapters/mongo-client.js +3 -3
  77. package/dist/adapters/redis-client.d.ts +3 -3
  78. package/dist/adapters/redis-client.d.ts.map +1 -1
  79. package/dist/adapters/redis-client.js +3 -3
  80. package/dist/api/ContextProvider.d.ts +6 -6
  81. package/dist/api/ContextProvider.d.ts.map +1 -1
  82. package/dist/api/ContextProvider.js +6 -6
  83. package/dist/api/internal/auth.d.ts +1 -1
  84. package/dist/api/internal/events.d.ts +2 -2
  85. package/dist/api/internal/events.d.ts.map +1 -1
  86. package/dist/api/internal/events.js +6 -4
  87. package/dist/api/layerUtils.d.ts +5 -5
  88. package/dist/api/layerUtils.d.ts.map +1 -1
  89. package/dist/api/layerUtils.js +5 -5
  90. package/dist/api/routing/middleware/RouterMiddleware.d.ts +3 -3
  91. package/dist/api/routing/middleware/RouterMiddleware.d.ts.map +1 -1
  92. package/dist/api/routing/middleware/middleware.d.ts +35 -1
  93. package/dist/api/routing/middleware/middleware.d.ts.map +1 -1
  94. package/dist/api/routing/middleware/middleware.js +39 -1
  95. package/dist/api/routing.d.ts +1 -5
  96. package/dist/api/routing.d.ts.map +1 -1
  97. package/dist/api/routing.js +3 -2
  98. package/dist/api/setupRequest.d.ts +6 -3
  99. package/dist/api/setupRequest.d.ts.map +1 -1
  100. package/dist/api/setupRequest.js +11 -6
  101. package/dist/logger.d.ts.map +1 -1
  102. package/examples/query.ts +30 -26
  103. package/package.json +32 -18
  104. package/src/CUPS.ts +2 -2
  105. package/src/Emailer/service.ts +2 -2
  106. package/src/MainFiberSet.ts +2 -2
  107. package/src/Model/Repository/internal/internal.ts +75 -59
  108. package/src/Model/Repository/makeRepo.ts +2 -2
  109. package/src/Operations.ts +2 -2
  110. package/src/OperationsRepo.ts +2 -2
  111. package/src/QueueMaker/SQLQueue.ts +8 -7
  112. package/src/QueueMaker/memQueue.ts +2 -2
  113. package/src/QueueMaker/sbqueue.ts +2 -2
  114. package/src/RequestContext.ts +4 -4
  115. package/src/RequestFiberSet.ts +4 -4
  116. package/src/Store/ContextMapContainer.ts +41 -2
  117. package/src/Store/Cosmos.ts +352 -255
  118. package/src/Store/Disk.ts +2 -3
  119. package/src/Store/Memory.ts +4 -4
  120. package/src/Store/SQL/Pg.ts +328 -0
  121. package/src/Store/SQL/query.ts +409 -0
  122. package/src/Store/SQL.ts +686 -0
  123. package/src/Store/index.ts +15 -1
  124. package/src/Store/service.ts +26 -7
  125. package/src/adapters/ServiceBus.ts +8 -8
  126. package/src/adapters/cosmos-client.ts +2 -2
  127. package/src/adapters/memQueue.ts +2 -2
  128. package/src/adapters/mongo-client.ts +2 -2
  129. package/src/adapters/redis-client.ts +2 -2
  130. package/src/api/ContextProvider.ts +11 -11
  131. package/src/api/internal/events.ts +6 -5
  132. package/src/api/layerUtils.ts +8 -8
  133. package/src/api/routing/middleware/RouterMiddleware.ts +4 -4
  134. package/src/api/routing/middleware/middleware.ts +43 -0
  135. package/src/api/routing.ts +3 -3
  136. package/src/api/setupRequest.ts +27 -7
  137. package/test/contextProvider.test.ts +11 -11
  138. package/test/controller.test.ts +12 -9
  139. package/test/dist/contextProvider.test.d.ts.map +1 -1
  140. package/test/dist/controller.test.d.ts.map +1 -1
  141. package/test/dist/date-query.test.d.ts.map +1 -0
  142. package/test/dist/fixtures.d.ts +18 -8
  143. package/test/dist/fixtures.d.ts.map +1 -1
  144. package/test/dist/fixtures.js +11 -9
  145. package/test/dist/query.test.d.ts.map +1 -1
  146. package/test/dist/rawQuery.test.d.ts.map +1 -1
  147. package/test/dist/requires.test.d.ts.map +1 -1
  148. package/test/dist/rpc-multi-middleware.test.d.ts.map +1 -1
  149. package/test/dist/sql-store.test.d.ts.map +1 -0
  150. package/test/fixtures.ts +10 -8
  151. package/test/query.test.ts +162 -16
  152. package/test/rawQuery.test.ts +19 -17
  153. package/test/requires.test.ts +6 -5
  154. package/test/rpc-multi-middleware.test.ts +72 -3
  155. package/test/sql-store.test.ts +1064 -0
  156. package/test/validateSample.test.ts +1 -1
  157. package/tsconfig.json +0 -1
@@ -1,5 +1,5 @@
1
1
  import { describe, expect, it } from "@effect/vitest"
2
- import { Array, Config, Effect, flow, Layer, ManagedRuntime, Redacted, References, Result, S, ServiceMap } from "effect-app"
2
+ import { Array, Config, Effect, flow, Layer, ManagedRuntime, Redacted, References, Result, S, Context } from "effect-app"
3
3
  import { LogLevels } from "effect-app/utils"
4
4
  import { setupRequestContextFromCurrent } from "../src/api/setupRequest.js"
5
5
  import { and, or, project, where, whereEvery, whereSome } from "../src/Model/query.js"
@@ -24,7 +24,7 @@ class Something extends S.Class<Something>("Something")({
24
24
  id: S.String,
25
25
  name: S.String,
26
26
  description: S.String,
27
- items: S.Array(S.Struct({ id: S.String, value: S.Number, description: S.String }))
27
+ items: S.Array(S.Struct({ id: S.String, value: S.Finite, description: S.String }))
28
28
  }) {}
29
29
 
30
30
  const items = [
@@ -49,7 +49,7 @@ const items = [
49
49
  ]
50
50
 
51
51
  // @effect-diagnostics-next-line missingEffectServiceDependency:off
52
- class SomethingRepo extends ServiceMap.Service<SomethingRepo>()(
52
+ class SomethingRepo extends Context.Service<SomethingRepo>()(
53
53
  "SomethingRepo",
54
54
  {
55
55
  make: Effect.gen(function*() {
@@ -83,21 +83,23 @@ class SomethingRepo extends ServiceMap.Service<SomethingRepo>()(
83
83
  .layer
84
84
  .pipe(
85
85
  Layer.provide(
86
- Effect.gen(function*() {
87
- const url = yield* Config.redacted("STORAGE_URL").pipe(
88
- Config.withDefault(
89
- Redacted.make(
90
- // the emulator doesn't implement array projections :/ so you need an actual cloud instance!
91
- "AccountEndpoint=http://localhost:8081/;AccountKey=C2y6yDjf5/R+ob0N8A7Cgv30VRDJIWEHLM+4QDU5DE2nQ9nDuVTqobD4b8mGGyPMbIZnqyMsEcaGQy67XIw/Jw=="
86
+ Effect
87
+ .gen(function*() {
88
+ const url = yield* Config.redacted("STORAGE_URL").pipe(
89
+ Config.withDefault(
90
+ Redacted.make(
91
+ // the emulator doesn't implement array projections :/ so you need an actual cloud instance!
92
+ "AccountEndpoint=http://localhost:8081/;AccountKey=C2y6yDjf5/R+ob0N8A7Cgv30VRDJIWEHLM+4QDU5DE2nQ9nDuVTqobD4b8mGGyPMbIZnqyMsEcaGQy67XIw/Jw=="
93
+ )
92
94
  )
93
95
  )
94
- )
95
- return CosmosStoreLayer({
96
- dbName: "test",
97
- prefix: "",
98
- url
96
+ return CosmosStoreLayer({
97
+ dbName: "test",
98
+ prefix: "",
99
+ url
100
+ })
99
101
  })
100
- }).pipe(Layer.unwrap)
102
+ .pipe(Layer.unwrap)
101
103
  )
102
104
  )
103
105
  }
@@ -107,7 +109,7 @@ describe("select first-level array fields", () => {
107
109
  .gen(function*() {
108
110
  const repo = yield* SomethingRepo
109
111
 
110
- const projected = S.Struct({ name: S.String, items: S.Array(S.Struct({ id: S.String, value: S.Number })) })
112
+ const projected = S.Struct({ name: S.String, items: S.Array(S.Struct({ id: S.String, value: S.Finite })) })
111
113
 
112
114
  // ok crazy lol, "value" is a reserved word in CosmosDB, so we have to use t["value"] as a field name instead of t.value
113
115
  const items = yield* repo.queryRaw(projected, {
@@ -159,7 +161,7 @@ describe("select first-level array fields", () => {
159
161
  .pipe(Effect.provide(SomethingRepo.Test), rt.runPromise))
160
162
  })
161
163
 
162
- const projected = S.Struct({ name: S.String, items: S.Array(S.Struct({ id: S.String, value: S.Number })) })
164
+ const projected = S.Struct({ name: S.String, items: S.Array(S.Struct({ id: S.String, value: S.Finite })) })
163
165
 
164
166
  const expected = [
165
167
  {
@@ -1,5 +1,5 @@
1
1
  import { describe, expect, expectTypeOf, it } from "@effect/vitest"
2
- import { Effect, Layer, Result, S, ServiceMap } from "effect-app"
2
+ import { Context, Effect, Layer, Result, S } from "effect-app"
3
3
  import { NotLoggedInError, UnauthorizedError } from "effect-app/client"
4
4
  import { HttpHeaders } from "effect-app/http"
5
5
  import * as RpcX from "effect-app/rpc"
@@ -63,11 +63,12 @@ const testSuite = (_mw: typeof middleware3) =>
63
63
  "works",
64
64
  Effect.fn(function*() {
65
65
  const defaultOpts = {
66
+ client: null as any, // TODO?
66
67
  headers: HttpHeaders.fromRecordUnsafe({}),
67
68
  payload: { _tag: "Test" },
68
69
  clientId: 0,
69
70
  requestId: "test-id" as any,
70
- rpc: { ...TestRpc, annotations: ServiceMap.make(_mw.requestContext, {}) }
71
+ rpc: { ...TestRpc, annotations: Context.make(_mw.requestContext, {}) }
71
72
  }
72
73
  const next = Effect.void as unknown as Effect.Effect<SuccessValue, unhandled, never>
73
74
  const layer = _mw.layer.pipe(
@@ -89,7 +90,7 @@ const testSuite = (_mw: typeof middleware3) =>
89
90
  headers: HttpHeaders.fromRecordUnsafe({ "x-user": "test-user", "x-is-manager": "true" }),
90
91
  rpc: {
91
92
  ...defaultOpts.rpc,
92
- annotations: ServiceMap.make(_mw.requestContext, { requireRoles: ["manager"] })
93
+ annotations: Context.make(_mw.requestContext, { requireRoles: ["manager"] })
93
94
  }
94
95
  })
95
96
  )
@@ -127,7 +128,7 @@ const testSuite = (_mw: typeof middleware3) =>
127
128
  Object.assign({ ...defaultOpts }, {
128
129
  rpc: {
129
130
  ...defaultOpts.rpc,
130
- annotations: ServiceMap.make(_mw.requestContext, { requireRoles: ["manager"] })
131
+ annotations: Context.make(_mw.requestContext, { requireRoles: ["manager"] })
131
132
  }
132
133
  })
133
134
  )
@@ -153,7 +154,7 @@ const testSuite = (_mw: typeof middleware3) =>
153
154
  {
154
155
  rpc: {
155
156
  ...defaultOpts.rpc,
156
- annotations: ServiceMap.make(_mw.requestContext, { requireRoles: ["manager"] })
157
+ annotations: Context.make(_mw.requestContext, { requireRoles: ["manager"] })
157
158
  }
158
159
  }
159
160
  )
@@ -1,14 +1,14 @@
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, 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
8
  import { MiddlewareMaker } from "effect-app/rpc"
9
9
  import { middlewareGroup } from "effect-app/rpc/MiddlewareMaker"
10
10
  import { FetchHttpClient } from "effect/unstable/http"
11
- import { RpcClient, RpcGroup, RpcSerialization, RpcServer, RpcTest } from "effect/unstable/rpc"
11
+ import { Rpc, RpcClient, RpcGroup, RpcSerialization, RpcServer, RpcTest } from "effect/unstable/rpc"
12
12
  import { createServer } from "http"
13
13
  import { DefaultGenericMiddlewaresLive } from "../src/api/routing.js"
14
14
  import { AllowAnonymous, AllowAnonymousLive, RequestContextMap, RequireRoles, RequireRolesLive, Some, SomeElseMiddleware, SomeElseMiddlewareLive, SomeMiddleware, SomeMiddlewareLive, SomeService, Test, TestLive, UserProfile } from "./fixtures.js"
@@ -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
+ )