instance-state.ts 2.2 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061
  1. import { Effect, ScopedCache, Scope, ServiceMap } from "effect"
  2. import { Instance, type InstanceContext } from "@/project/instance"
  3. import { registerDisposer } from "./instance-registry"
  4. const TypeId = "~opencode/InstanceState"
  5. export const InstanceRef = ServiceMap.Reference<InstanceContext | undefined>("~opencode/InstanceRef", {
  6. defaultValue: () => undefined,
  7. })
  8. const context = Effect.gen(function* () {
  9. const ref = yield* InstanceRef
  10. return ref ?? Instance.current
  11. })
  12. const directory = Effect.gen(function* () {
  13. const ref = yield* InstanceRef
  14. return ref ? ref.directory : Instance.directory
  15. })
  16. export interface InstanceState<A, E = never, R = never> {
  17. readonly [TypeId]: typeof TypeId
  18. readonly cache: ScopedCache.ScopedCache<string, A, E, R>
  19. }
  20. export namespace InstanceState {
  21. export const make = <A, E = never, R = never>(
  22. init: (ctx: InstanceContext) => Effect.Effect<A, E, R | Scope.Scope>,
  23. ): Effect.Effect<InstanceState<A, E, Exclude<R, Scope.Scope>>, never, R | Scope.Scope> =>
  24. Effect.gen(function* () {
  25. const cache = yield* ScopedCache.make<string, A, E, R>({
  26. capacity: Number.POSITIVE_INFINITY,
  27. lookup: () => Effect.gen(function* () { return yield* init(yield* context) }),
  28. })
  29. const off = registerDisposer((directory) => Effect.runPromise(ScopedCache.invalidate(cache, directory)))
  30. yield* Effect.addFinalizer(() => Effect.sync(off))
  31. return {
  32. [TypeId]: TypeId,
  33. cache,
  34. }
  35. })
  36. export const get = <A, E, R>(self: InstanceState<A, E, R>) =>
  37. Effect.gen(function* () { return yield* ScopedCache.get(self.cache, yield* directory) })
  38. export const use = <A, E, R, B>(self: InstanceState<A, E, R>, select: (value: A) => B) =>
  39. Effect.map(get(self), select)
  40. export const useEffect = <A, E, R, B, E2, R2>(
  41. self: InstanceState<A, E, R>,
  42. select: (value: A) => Effect.Effect<B, E2, R2>,
  43. ) => Effect.flatMap(get(self), select)
  44. export const has = <A, E, R>(self: InstanceState<A, E, R>) =>
  45. Effect.gen(function* () { return yield* ScopedCache.has(self.cache, yield* directory) })
  46. export const invalidate = <A, E, R>(self: InstanceState<A, E, R>) =>
  47. Effect.gen(function* () { return yield* ScopedCache.invalidate(self.cache, yield* directory) })
  48. }