@dxos/functions 0.8.2-main.fbd8ed0 → 0.8.2-staging.7ac8446

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 (176) hide show
  1. package/dist/lib/browser/bundler/index.mjs +3 -0
  2. package/dist/lib/browser/bundler/index.mjs.map +1 -1
  3. package/dist/lib/browser/chunk-HI7YZO2K.mjs +482 -0
  4. package/dist/lib/browser/chunk-HI7YZO2K.mjs.map +7 -0
  5. package/dist/lib/browser/chunk-LT4LR4VU.mjs +72 -0
  6. package/dist/lib/browser/chunk-LT4LR4VU.mjs.map +7 -0
  7. package/dist/lib/browser/chunk-RVSG6WTL.mjs +358 -0
  8. package/dist/lib/browser/chunk-RVSG6WTL.mjs.map +7 -0
  9. package/dist/lib/browser/chunk-XRCXIG74.mjs +12 -0
  10. package/dist/lib/browser/chunk-XRCXIG74.mjs.map +7 -0
  11. package/dist/lib/browser/edge/index.mjs +7 -63
  12. package/dist/lib/browser/edge/index.mjs.map +4 -4
  13. package/dist/lib/browser/index.mjs +101 -371
  14. package/dist/lib/browser/index.mjs.map +4 -4
  15. package/dist/lib/browser/meta.json +1 -1
  16. package/dist/lib/browser/testing/index.mjs +670 -0
  17. package/dist/lib/browser/testing/index.mjs.map +7 -0
  18. package/dist/lib/browser/types/index.mjs +49 -0
  19. package/dist/lib/browser/types/index.mjs.map +7 -0
  20. package/dist/lib/node/bundler/index.cjs +1 -0
  21. package/dist/lib/node/bundler/index.cjs.map +1 -1
  22. package/dist/lib/node/chunk-DSUGRAAL.cjs +392 -0
  23. package/dist/lib/node/chunk-DSUGRAAL.cjs.map +7 -0
  24. package/dist/lib/node/chunk-JEQ2X3Z6.cjs +34 -0
  25. package/dist/lib/node/chunk-JEQ2X3Z6.cjs.map +7 -0
  26. package/dist/lib/node/chunk-NXZNXVT3.cjs +94 -0
  27. package/dist/lib/node/chunk-NXZNXVT3.cjs.map +7 -0
  28. package/dist/lib/node/chunk-RXMCVAMJ.cjs +496 -0
  29. package/dist/lib/node/chunk-RXMCVAMJ.cjs.map +7 -0
  30. package/dist/lib/node/edge/index.cjs +5 -65
  31. package/dist/lib/node/edge/index.cjs.map +4 -4
  32. package/dist/lib/node/index.cjs +93 -383
  33. package/dist/lib/node/index.cjs.map +4 -4
  34. package/dist/lib/node/meta.json +1 -1
  35. package/dist/lib/node/testing/index.cjs +687 -0
  36. package/dist/lib/node/testing/index.cjs.map +7 -0
  37. package/dist/lib/node/types/index.cjs +70 -0
  38. package/dist/lib/node/types/index.cjs.map +7 -0
  39. package/dist/lib/node-esm/bundler/index.mjs +1 -0
  40. package/dist/lib/node-esm/bundler/index.mjs.map +1 -1
  41. package/dist/lib/node-esm/chunk-DHGBFXSZ.mjs +12 -0
  42. package/dist/lib/node-esm/chunk-DHGBFXSZ.mjs.map +7 -0
  43. package/dist/lib/node-esm/chunk-HBD2FZXO.mjs +358 -0
  44. package/dist/lib/node-esm/chunk-HBD2FZXO.mjs.map +7 -0
  45. package/dist/lib/node-esm/chunk-O2SXVYU5.mjs +72 -0
  46. package/dist/lib/node-esm/chunk-O2SXVYU5.mjs.map +7 -0
  47. package/dist/lib/node-esm/chunk-SQSJO5HI.mjs +482 -0
  48. package/dist/lib/node-esm/chunk-SQSJO5HI.mjs.map +7 -0
  49. package/dist/lib/node-esm/edge/index.mjs +6 -64
  50. package/dist/lib/node-esm/edge/index.mjs.map +4 -4
  51. package/dist/lib/node-esm/index.mjs +99 -371
  52. package/dist/lib/node-esm/index.mjs.map +4 -4
  53. package/dist/lib/node-esm/meta.json +1 -1
  54. package/dist/lib/node-esm/testing/index.mjs +670 -0
  55. package/dist/lib/node-esm/testing/index.mjs.map +7 -0
  56. package/dist/lib/node-esm/types/index.mjs +49 -0
  57. package/dist/lib/node-esm/types/index.mjs.map +7 -0
  58. package/dist/types/src/browser/index.d.ts +2 -0
  59. package/dist/types/src/browser/index.d.ts.map +1 -0
  60. package/dist/types/src/bundler/bundler.d.ts.map +1 -1
  61. package/dist/types/src/edge/functions.d.ts +3 -3
  62. package/dist/types/src/edge/functions.d.ts.map +1 -1
  63. package/dist/types/src/edge/index.d.ts.map +1 -1
  64. package/dist/types/src/function/function-registry.d.ts +25 -0
  65. package/dist/types/src/function/function-registry.d.ts.map +1 -0
  66. package/dist/types/src/function/function-registry.test.d.ts +2 -0
  67. package/dist/types/src/function/function-registry.test.d.ts.map +1 -0
  68. package/dist/types/src/function/index.d.ts +2 -0
  69. package/dist/types/src/function/index.d.ts.map +1 -0
  70. package/dist/types/src/handler.d.ts +66 -18
  71. package/dist/types/src/handler.d.ts.map +1 -1
  72. package/dist/types/src/index.d.ts +3 -3
  73. package/dist/types/src/index.d.ts.map +1 -1
  74. package/dist/types/src/runtime/dev-server.d.ts +52 -0
  75. package/dist/types/src/runtime/dev-server.d.ts.map +1 -0
  76. package/dist/types/src/runtime/dev-server.test.d.ts +2 -0
  77. package/dist/types/src/runtime/dev-server.test.d.ts.map +1 -0
  78. package/dist/types/src/runtime/index.d.ts +3 -0
  79. package/dist/types/src/runtime/index.d.ts.map +1 -0
  80. package/dist/types/src/runtime/scheduler.d.ts +34 -0
  81. package/dist/types/src/runtime/scheduler.d.ts.map +1 -0
  82. package/dist/types/src/runtime/scheduler.test.d.ts +2 -0
  83. package/dist/types/src/runtime/scheduler.test.d.ts.map +1 -0
  84. package/dist/types/src/testing/functions-integration.test.d.ts +2 -0
  85. package/dist/types/src/testing/functions-integration.test.d.ts.map +1 -0
  86. package/dist/types/src/testing/index.d.ts +5 -0
  87. package/dist/types/src/testing/index.d.ts.map +1 -0
  88. package/dist/types/src/testing/manifest.d.ts +3 -0
  89. package/dist/types/src/testing/manifest.d.ts.map +1 -0
  90. package/dist/types/src/testing/plugin-init.d.ts +6 -0
  91. package/dist/types/src/testing/plugin-init.d.ts.map +1 -0
  92. package/dist/types/src/testing/setup.d.ts +15 -0
  93. package/dist/types/src/testing/setup.d.ts.map +1 -0
  94. package/dist/types/src/testing/test/handler.d.ts +4 -0
  95. package/dist/types/src/testing/test/handler.d.ts.map +1 -0
  96. package/dist/types/src/testing/test/index.d.ts +3 -0
  97. package/dist/types/src/testing/test/index.d.ts.map +1 -0
  98. package/dist/types/src/testing/types.d.ts +10 -0
  99. package/dist/types/src/testing/types.d.ts.map +1 -0
  100. package/dist/types/src/testing/util.d.ts +5 -0
  101. package/dist/types/src/testing/util.d.ts.map +1 -0
  102. package/dist/types/src/translations.d.ts +1 -2
  103. package/dist/types/src/translations.d.ts.map +1 -1
  104. package/dist/types/src/trigger/index.d.ts +3 -0
  105. package/dist/types/src/trigger/index.d.ts.map +1 -0
  106. package/dist/types/src/trigger/trigger-registry.d.ts +38 -0
  107. package/dist/types/src/trigger/trigger-registry.d.ts.map +1 -0
  108. package/dist/types/src/trigger/trigger-registry.test.d.ts +2 -0
  109. package/dist/types/src/trigger/trigger-registry.test.d.ts.map +1 -0
  110. package/dist/types/src/trigger/type/index.d.ts +3 -0
  111. package/dist/types/src/trigger/type/index.d.ts.map +1 -0
  112. package/dist/types/src/trigger/type/subscription-trigger.d.ts +4 -0
  113. package/dist/types/src/trigger/type/subscription-trigger.d.ts.map +1 -0
  114. package/dist/types/src/trigger/type/timer-trigger.d.ts +4 -0
  115. package/dist/types/src/trigger/type/timer-trigger.d.ts.map +1 -0
  116. package/dist/types/src/trigger/type/webhook-trigger.d.ts +4 -0
  117. package/dist/types/src/trigger/type/webhook-trigger.d.ts.map +1 -0
  118. package/dist/types/src/types/index.d.ts +5 -0
  119. package/dist/types/src/types/index.d.ts.map +1 -0
  120. package/dist/types/src/types/schema.d.ts +53 -0
  121. package/dist/types/src/types/schema.d.ts.map +1 -0
  122. package/dist/types/src/types/trace.d.ts +146 -0
  123. package/dist/types/src/types/trace.d.ts.map +1 -0
  124. package/dist/types/src/types/types.d.ts +265 -0
  125. package/dist/types/src/types/types.d.ts.map +1 -0
  126. package/dist/types/src/{url.d.ts → types/url.d.ts} +0 -6
  127. package/dist/types/src/types/url.d.ts.map +1 -0
  128. package/dist/types/tools/schema.d.ts +2 -0
  129. package/dist/types/tools/schema.d.ts.map +1 -0
  130. package/dist/types/tsconfig.tsbuildinfo +1 -1
  131. package/package.json +35 -20
  132. package/schema/functions.json +211 -0
  133. package/src/browser/index.ts +5 -0
  134. package/src/edge/functions.ts +4 -7
  135. package/src/edge/index.ts +0 -4
  136. package/src/function/function-registry.test.ts +118 -0
  137. package/src/function/function-registry.ts +104 -0
  138. package/src/function/index.ts +5 -0
  139. package/src/handler.ts +124 -23
  140. package/src/index.ts +5 -4
  141. package/src/runtime/dev-server.test.ts +79 -0
  142. package/src/runtime/dev-server.ts +240 -0
  143. package/src/runtime/index.ts +6 -0
  144. package/src/runtime/scheduler.test.ts +152 -0
  145. package/src/runtime/scheduler.ts +170 -0
  146. package/src/testing/functions-integration.test.ts +65 -0
  147. package/src/testing/index.ts +8 -0
  148. package/src/testing/manifest.ts +15 -0
  149. package/src/testing/plugin-init.ts +20 -0
  150. package/src/testing/setup.ts +109 -0
  151. package/src/testing/test/handler.ts +15 -0
  152. package/src/testing/test/index.ts +7 -0
  153. package/src/testing/types.ts +9 -0
  154. package/src/testing/util.ts +26 -0
  155. package/src/translations.ts +1 -1
  156. package/src/trigger/index.ts +6 -0
  157. package/src/trigger/trigger-registry.test.ts +278 -0
  158. package/src/trigger/trigger-registry.ts +218 -0
  159. package/src/trigger/type/index.ts +7 -0
  160. package/src/trigger/type/subscription-trigger.ts +84 -0
  161. package/src/trigger/type/timer-trigger.ts +48 -0
  162. package/src/trigger/type/webhook-trigger.ts +48 -0
  163. package/src/types/index.ts +8 -0
  164. package/src/types/schema.ts +46 -0
  165. package/src/{trace.ts → types/trace.ts} +31 -33
  166. package/src/types/types.ts +163 -0
  167. package/src/{url.ts → types/url.ts} +0 -5
  168. package/dist/types/src/schema.d.ts +0 -57
  169. package/dist/types/src/schema.d.ts.map +0 -1
  170. package/dist/types/src/trace.d.ts +0 -148
  171. package/dist/types/src/trace.d.ts.map +0 -1
  172. package/dist/types/src/types.d.ts +0 -407
  173. package/dist/types/src/types.d.ts.map +0 -1
  174. package/dist/types/src/url.d.ts.map +0 -1
  175. package/src/schema.ts +0 -53
  176. package/src/types.ts +0 -210
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../src/types/schema.ts", "../../../src/types/types.ts", "../../../src/types/trace.ts", "../../../src/types/url.ts"],
4
+ "sourcesContent": ["//\n// Copyright 2024 DXOS.org\n//\n\nimport { EchoObject, JsonSchemaType, LabelAnnotationId, Ref, S, TypedObject } from '@dxos/echo-schema';\nimport { TextType } from '@dxos/schema';\n\n/**\n * Source script.\n */\nexport const ScriptType = S.Struct({\n name: S.optional(S.String),\n description: S.optional(S.String),\n // TODO(burdon): Change to hash of deployed content.\n // Whether source has changed since last deploy.\n changed: S.optional(S.Boolean),\n source: Ref(TextType),\n})\n .annotations({ [LabelAnnotationId]: 'name' })\n .pipe(EchoObject({ typename: 'dxos.org/type/Script', version: '0.1.0' }));\n\nexport type ScriptType = S.Schema.Type<typeof ScriptType>;\n\n/**\n * Function deployment.\n */\nexport class FunctionType extends TypedObject({\n typename: 'dxos.org/type/Function',\n version: '0.1.0',\n})({\n // TODO(burdon): Rename to id/uri?\n name: S.NonEmptyString,\n version: S.String,\n\n description: S.optional(S.String),\n\n // Reference to a source script if it exists within ECHO.\n // TODO(burdon): Don't ref ScriptType directly (core).\n source: S.optional(Ref(ScriptType)),\n\n inputSchema: S.optional(JsonSchemaType),\n outputSchema: S.optional(JsonSchemaType),\n\n // Local binding to a function name.\n binding: S.optional(S.String),\n}) {}\n", "//\n// Copyright 2023 DXOS.org\n//\n\nimport { AST, OptionsAnnotationId, RawObject, S, TypedObject, DXN } from '@dxos/echo-schema';\n\n/**\n * Type discriminator for TriggerType.\n * Every spec has a type field of type TriggerKind that we can use to understand which type we're working with.\n * https://www.typescriptlang.org/docs/handbook/2/narrowing.html#discriminated-unions\n */\nexport enum TriggerKind {\n Timer = 'timer',\n Webhook = 'webhook',\n Subscription = 'subscription',\n Email = 'email',\n Queue = 'queue',\n}\n\n// TODO(burdon): Rename prop kind.\nconst typeLiteralAnnotations = { [AST.TitleAnnotationId]: 'Type' };\n\n/**\n * Cron timer.\n */\nconst TimerTriggerSchema = S.Struct({\n type: S.Literal(TriggerKind.Timer).annotations(typeLiteralAnnotations),\n cron: S.String.annotations({\n [AST.TitleAnnotationId]: 'Cron',\n [AST.ExamplesAnnotationId]: ['0 0 * * *'],\n }),\n}).pipe(S.mutable);\n\nexport type TimerTrigger = S.Schema.Type<typeof TimerTriggerSchema>;\n\nconst EmailTriggerSchema = S.Struct({\n type: S.Literal(TriggerKind.Email).annotations(typeLiteralAnnotations),\n}).pipe(S.mutable);\n\nexport type EmailTrigger = S.Schema.Type<typeof EmailTriggerSchema>;\n\nconst QueueTriggerSchema = S.Struct({\n type: S.Literal(TriggerKind.Queue).annotations(typeLiteralAnnotations),\n queue: DXN,\n}).pipe(S.mutable);\n\nexport type QueueTrigger = S.Schema.Type<typeof QueueTriggerSchema>;\n\n/**\n * Webhook.\n */\nconst WebhookTriggerSchema = S.Struct({\n type: S.Literal(TriggerKind.Webhook).annotations(typeLiteralAnnotations),\n method: S.optional(\n S.String.annotations({\n [AST.TitleAnnotationId]: 'Method',\n [OptionsAnnotationId]: ['GET', 'POST'],\n }),\n ),\n port: S.optional(\n S.Number.annotations({\n [AST.TitleAnnotationId]: 'Port',\n }),\n ),\n}).pipe(S.mutable);\n\nexport type WebhookTrigger = S.Schema.Type<typeof WebhookTriggerSchema>;\n\n// TODO(burdon): Use ECHO definition (from https://github.com/dxos/dxos/pull/8233).\nconst QuerySchema = S.Struct({\n type: S.optional(S.String.annotations({ [AST.TitleAnnotationId]: 'Type' })),\n props: S.optional(S.Record({ key: S.String, value: S.Any })),\n}).annotations({ [AST.TitleAnnotationId]: 'Query' });\n\n/**\n * Subscription.\n */\nconst SubscriptionTriggerSchema = S.Struct({\n type: S.Literal(TriggerKind.Subscription).annotations(typeLiteralAnnotations),\n // TODO(burdon): Define query DSL (from ECHO). Reconcile with Table.Query.\n filter: QuerySchema,\n options: S.optional(\n S.Struct({\n // Watch changes to object (not just creation).\n deep: S.optional(S.Boolean.annotations({ [AST.TitleAnnotationId]: 'Nested' })),\n // Debounce changes (delay in ms).\n delay: S.optional(S.Number.annotations({ [AST.TitleAnnotationId]: 'Delay' })),\n }).annotations({ [AST.TitleAnnotationId]: 'Options' }),\n ),\n}).pipe(S.mutable);\n\nexport type SubscriptionTrigger = S.Schema.Type<typeof SubscriptionTriggerSchema>;\n\n/**\n * Trigger schema (discriminated union).\n */\nexport const TriggerSchema = S.Union(\n TimerTriggerSchema,\n WebhookTriggerSchema,\n SubscriptionTriggerSchema,\n EmailTriggerSchema,\n QueueTriggerSchema,\n).annotations({\n [AST.TitleAnnotationId]: 'Trigger',\n});\n\nexport type TriggerType = S.Schema.Type<typeof TriggerSchema>;\n\n/**\n * Function trigger.\n */\nexport const FunctionTriggerSchema = S.Struct({\n // TODO(burdon): What type does this reference.\n // TODO(wittjosiah): This should probably be a Ref?\n function: S.optional(S.String.annotations({ [AST.TitleAnnotationId]: 'Function' })),\n\n enabled: S.optional(S.Boolean.annotations({ [AST.TitleAnnotationId]: 'Enabled' })),\n\n // TODO(burdon): Flatten entire schema.\n spec: S.optional(TriggerSchema),\n\n // TODO(burdon): Get schema as partial from function.\n // TODO(wittjosiah): Rename to payload.\n // The `meta` property is merged into the event data passed to the function.\n meta: S.optional(S.mutable(S.Record({ key: S.String, value: S.Any }))),\n});\n\nexport type FunctionTriggerType = S.Schema.Type<typeof FunctionTriggerSchema>;\n\n/**\n * Function trigger.\n */\nexport class FunctionTrigger extends TypedObject({\n typename: 'dxos.org/type/FunctionTrigger',\n version: '0.1.0',\n})(FunctionTriggerSchema.fields) {}\n\n/**\n * Function definition.\n * @deprecated (Use dxos.org/type/Function)\n */\n// TODO(burdon): Reconcile with FunctionType.\nexport class FunctionDef extends TypedObject({\n typename: 'dxos.org/type/FunctionDef',\n version: '0.1.0',\n})({\n uri: S.String,\n description: S.optional(S.String),\n route: S.String,\n handler: S.String,\n}) {}\n\n/**\n * Function manifest file.\n */\nexport const FunctionManifestSchema = S.Struct({\n functions: S.optional(S.mutable(S.Array(RawObject(FunctionDef)))),\n triggers: S.optional(S.mutable(S.Array(RawObject(FunctionTrigger)))),\n});\n\nexport type FunctionManifest = S.Schema.Type<typeof FunctionManifestSchema>;\n\nexport const FUNCTION_TYPES = [FunctionDef, FunctionTrigger];\n", "//\n// Copyright 2025 DXOS.org\n//\n\nimport { EchoObject, Expando, ObjectId, Ref, S } from '@dxos/echo-schema';\nimport { log } from '@dxos/log';\n\nimport { FunctionTrigger, type FunctionTriggerType } from './types';\n\nexport enum InvocationOutcome {\n SUCCESS = 'success',\n FAILURE = 'failure',\n PENDING = 'pending',\n}\n\n// TODO(burdon): Convert to extensible discriminated union of EDGE events.\nexport enum InvocationTraceEventType {\n START = 'start',\n END = 'end',\n}\n\nexport const TraceEventException = S.Struct({\n timestampMs: S.Number,\n message: S.String,\n name: S.String,\n stack: S.optional(S.String),\n});\nexport type TraceEventException = S.Schema.Type<typeof TraceEventException>;\n\nexport const InvocationTraceStartEvent = S.Struct({\n /**\n * Queue message id.\n */\n id: ObjectId,\n type: S.Literal(InvocationTraceEventType.START),\n /**\n * Invocation id, the same for invocation start and end events.\n */\n invocationId: ObjectId,\n /**\n * Event generation time.\n */\n timestampMs: S.Number,\n /**\n * Data passed to function / workflow as an argument.\n */\n // TODO(burdon): Input schema?\n input: S.Object,\n /**\n * Queue DXN for function/workflow invocation events.\n */\n // TODO(burdon): Need reference type for queue. vs. string?\n invocationTraceQueue: Ref(Expando),\n /**\n * DXN of the invoked function/workflow.\n */\n invocationTarget: Ref(Expando),\n /**\n * Present for automatic invocations.\n */\n trigger: S.optional(Ref(FunctionTrigger)),\n}).pipe(EchoObject({ typename: 'dxos.org/type/InvocationTraceStart', version: '0.1.0' }));\n\nexport type InvocationTraceStartEvent = S.Schema.Type<typeof InvocationTraceStartEvent>;\n\nexport const InvocationTraceEndEvent = S.Struct({\n /**\n * Trace event id.\n */\n id: ObjectId,\n type: S.Literal(InvocationTraceEventType.END),\n /**\n * Invocation id, will be the same for invocation start and end.\n */\n invocationId: ObjectId,\n /**\n * Event generation time.\n */\n // TODO(burdon): Remove ms suffix.\n timestampMs: S.Number,\n outcome: S.Enums(InvocationOutcome),\n exception: S.optional(TraceEventException),\n}).pipe(EchoObject({ typename: 'dxos.org/type/InvocationTraceEnd', version: '0.1.0' }));\n\nexport type InvocationTraceEndEvent = S.Schema.Type<typeof InvocationTraceEndEvent>;\n\nexport type InvocationTraceEvent = InvocationTraceStartEvent | InvocationTraceEndEvent;\n\nexport const TraceEventLog = S.Struct({\n timestampMs: S.Number,\n level: S.String,\n message: S.String,\n context: S.optional(S.Object),\n});\n\nexport const TraceEvent = S.Struct({\n id: ObjectId,\n // TODO(burdon): Need enum/numeric result (not string).\n outcome: S.String,\n truncated: S.Boolean,\n /**\n * Time when the event was persisted.\n */\n ingestionTimestampMs: S.Number,\n logs: S.Array(TraceEventLog),\n exceptions: S.Array(TraceEventException),\n}).pipe(EchoObject({ typename: 'dxos.org/type/TraceEvent', version: '0.1.0' }));\n\nexport type TraceEvent = S.Schema.Type<typeof TraceEvent>;\n\n/**\n * Deprecated InvocationTrace event format.\n * @deprecated\n */\n// TODO(burdon): Remove.\nexport type InvocationSpan = {\n id: string;\n timestampMs: number;\n outcome: InvocationOutcome;\n input: object;\n durationMs: number;\n invocationTraceQueue: Ref<Expando>;\n invocationTarget: Ref<Expando>;\n trigger?: Ref<FunctionTriggerType>;\n exception?: TraceEventException;\n};\n\nexport const createInvocationSpans = (items?: InvocationTraceEvent[]): InvocationSpan[] => {\n if (!items) {\n return [];\n }\n\n const eventsByInvocationId = new Map<string, { start?: InvocationTraceStartEvent; end?: InvocationTraceEndEvent }>();\n for (const event of items) {\n if (!('invocationId' in event)) {\n // Skip legacy format entries.\n continue;\n }\n\n const invocationId = event.invocationId;\n const entry = eventsByInvocationId.get(invocationId) || { start: undefined, end: undefined };\n if (event.type === InvocationTraceEventType.START) {\n entry.start = event as InvocationTraceStartEvent;\n } else if (event.type === InvocationTraceEventType.END) {\n entry.end = event as InvocationTraceEndEvent;\n }\n\n eventsByInvocationId.set(invocationId, entry);\n }\n\n const now = Date.now();\n const result: InvocationSpan[] = [];\n\n // Create spans for each invocation\n for (const [invocationId, { start, end }] of eventsByInvocationId.entries()) {\n if (!start) {\n // No start event, can't create a meaningful span\n log.warn('found end event without matching start', { invocationId });\n continue;\n }\n\n const isInProgress = end === undefined;\n\n result.push({\n id: invocationId,\n timestampMs: start.timestampMs,\n durationMs: isInProgress ? now - start.timestampMs : end!.timestampMs - start.timestampMs,\n outcome: end?.outcome ?? InvocationOutcome.PENDING,\n exception: end?.exception,\n input: start.input,\n invocationTraceQueue: start.invocationTraceQueue,\n invocationTarget: start.invocationTarget,\n trigger: start.trigger,\n });\n }\n\n return result;\n};\n", "//\n// Copyright 2025 DXOS.org\n//\n\nimport { type ObjectMeta } from '@dxos/echo-schema';\nimport { type SpaceId } from '@dxos/keys';\n\n// TODO: use URL scheme for source?\nconst FUNCTIONS_META_KEY = 'dxos.org/service/function';\n\nexport const FUNCTIONS_PRESET_META_KEY = 'dxos.org/service/function-preset';\n\nconst isSecure = (protocol: string) => {\n return protocol === 'https:' || protocol === 'wss:';\n};\n\nexport const getUserFunctionUrlInMetadata = (meta: ObjectMeta) => {\n return meta.keys.find((key) => key.source === FUNCTIONS_META_KEY)?.id;\n};\n\nexport const setUserFunctionUrlInMetadata = (meta: ObjectMeta, functionUrl: string) => {\n const key = meta.keys.find((key) => key.source === FUNCTIONS_META_KEY);\n if (key) {\n if (key.id !== functionUrl) {\n throw new Error('Metadata mismatch');\n }\n } else {\n meta.keys.push({ source: FUNCTIONS_META_KEY, id: functionUrl });\n }\n};\n\nexport const getInvocationUrl = (functionUrl: string, edgeUrl: string, options: InvocationOptions = {}) => {\n const baseUrl = new URL('functions/', edgeUrl);\n\n // Leading slashes cause the URL to be treated as an absolute path.\n const relativeUrl = functionUrl.replace(/^\\//, '');\n const url = new URL(`./${relativeUrl}`, baseUrl.toString());\n options.spaceId && url.searchParams.set('spaceId', options.spaceId);\n options.subjectId && url.searchParams.set('subjectId', options.subjectId);\n url.protocol = isSecure(url.protocol) ? 'https' : 'http';\n return url.toString();\n};\n\nexport type InvocationOptions = {\n spaceId?: SpaceId;\n subjectId?: string;\n};\n"],
5
+ "mappings": "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",
6
+ "names": ["EchoObject", "JsonSchemaType", "LabelAnnotationId", "Ref", "S", "TypedObject", "TextType", "ScriptType", "S", "Struct", "name", "optional", "String", "description", "changed", "Boolean", "source", "Ref", "TextType", "annotations", "LabelAnnotationId", "pipe", "EchoObject", "typename", "version", "FunctionType", "TypedObject", "NonEmptyString", "inputSchema", "JsonSchemaType", "outputSchema", "binding", "AST", "OptionsAnnotationId", "RawObject", "S", "TypedObject", "DXN", "TriggerKind", "typeLiteralAnnotations", "AST", "TitleAnnotationId", "TimerTriggerSchema", "S", "Struct", "type", "Literal", "annotations", "cron", "String", "ExamplesAnnotationId", "pipe", "mutable", "EmailTriggerSchema", "QueueTriggerSchema", "queue", "DXN", "WebhookTriggerSchema", "method", "optional", "OptionsAnnotationId", "port", "Number", "QuerySchema", "props", "Record", "key", "value", "Any", "SubscriptionTriggerSchema", "filter", "options", "deep", "Boolean", "delay", "TriggerSchema", "Union", "FunctionTriggerSchema", "function", "enabled", "spec", "meta", "FunctionTrigger", "TypedObject", "typename", "version", "fields", "FunctionDef", "uri", "description", "route", "handler", "FunctionManifestSchema", "functions", "Array", "RawObject", "triggers", "FUNCTION_TYPES", "EchoObject", "Expando", "ObjectId", "Ref", "S", "log", "InvocationOutcome", "InvocationTraceEventType", "TraceEventException", "S", "Struct", "timestampMs", "Number", "message", "String", "name", "stack", "optional", "InvocationTraceStartEvent", "id", "ObjectId", "type", "Literal", "invocationId", "input", "Object", "invocationTraceQueue", "Ref", "Expando", "invocationTarget", "trigger", "FunctionTrigger", "pipe", "EchoObject", "typename", "version", "InvocationTraceEndEvent", "outcome", "Enums", "exception", "TraceEventLog", "level", "context", "TraceEvent", "truncated", "Boolean", "ingestionTimestampMs", "logs", "Array", "exceptions", "createInvocationSpans", "items", "eventsByInvocationId", "Map", "event", "entry", "get", "start", "undefined", "end", "set", "now", "Date", "result", "entries", "log", "warn", "isInProgress", "push", "durationMs", "FUNCTIONS_META_KEY", "FUNCTIONS_PRESET_META_KEY", "isSecure", "protocol", "getUserFunctionUrlInMetadata", "meta", "keys", "find", "key", "source", "id", "setUserFunctionUrlInMetadata", "functionUrl", "Error", "push", "getInvocationUrl", "edgeUrl", "options", "baseUrl", "URL", "relativeUrl", "replace", "url", "toString", "spaceId", "searchParams", "set", "subjectId"]
7
+ }
@@ -0,0 +1,72 @@
1
+ import { createRequire } from 'node:module';const require = createRequire(import.meta.url);
2
+
3
+ // packages/core/functions/src/edge/functions.ts
4
+ import { createEdgeIdentity } from "@dxos/client/edge";
5
+ import { EdgeHttpClient } from "@dxos/edge-client";
6
+ import { invariant } from "@dxos/invariant";
7
+ import { log } from "@dxos/log";
8
+ var __dxlog_file = "/home/runner/work/dxos/dxos/packages/core/functions/src/edge/functions.ts";
9
+ var uploadWorkerFunction = async ({ client, spaceId, version, source, name, functionId }) => {
10
+ const edgeUrl = client.config.values.runtime?.services?.edge?.url;
11
+ invariant(edgeUrl, "Edge is not configured.", {
12
+ F: __dxlog_file,
13
+ L: 33,
14
+ S: void 0,
15
+ A: [
16
+ "edgeUrl",
17
+ "'Edge is not configured.'"
18
+ ]
19
+ });
20
+ const edgeClient = new EdgeHttpClient(edgeUrl);
21
+ const edgeIdentity = createEdgeIdentity(client);
22
+ edgeClient.setIdentity(edgeIdentity);
23
+ const response = await edgeClient.uploadFunction({
24
+ spaceId,
25
+ functionId
26
+ }, {
27
+ name,
28
+ version,
29
+ script: source
30
+ });
31
+ log.info("Uploaded", {
32
+ identityKey: edgeIdentity.identityKey,
33
+ functionId,
34
+ name,
35
+ source: source.length,
36
+ response
37
+ }, {
38
+ F: __dxlog_file,
39
+ L: 40,
40
+ S: void 0,
41
+ C: (f, a) => f(...a)
42
+ });
43
+ return response;
44
+ };
45
+ var incrementSemverPatch = (version) => {
46
+ const [major, minor, patch] = version.split(".");
47
+ const patchNum = Number(patch);
48
+ invariant(!Number.isNaN(patchNum), `Unexpected function version format: ${version}`, {
49
+ F: __dxlog_file,
50
+ L: 54,
51
+ S: void 0,
52
+ A: [
53
+ "!Number.isNaN(patchNum)",
54
+ "`Unexpected function version format: ${version}`"
55
+ ]
56
+ });
57
+ return [
58
+ major,
59
+ minor,
60
+ String(patchNum + 1)
61
+ ].join(".");
62
+ };
63
+ var publicKeyToDid = (key) => {
64
+ return `did:key:${key.toHex()}`;
65
+ };
66
+
67
+ export {
68
+ uploadWorkerFunction,
69
+ incrementSemverPatch,
70
+ publicKeyToDid
71
+ };
72
+ //# sourceMappingURL=chunk-O2SXVYU5.mjs.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../src/edge/functions.ts"],
4
+ "sourcesContent": ["//\n// Copyright 2024 DXOS.org\n//\n\nimport { type DID } from 'iso-did/types';\n\nimport { type Client } from '@dxos/client';\nimport { createEdgeIdentity } from '@dxos/client/edge';\nimport { EdgeHttpClient } from '@dxos/edge-client';\nimport { invariant } from '@dxos/invariant';\nimport type { PublicKey, SpaceId } from '@dxos/keys';\nimport { log } from '@dxos/log';\nimport { type UploadFunctionResponseBody } from '@dxos/protocols';\n\nexport type UploadWorkerArgs = {\n client: Client;\n spaceId: SpaceId;\n source: string;\n version: string;\n name?: string;\n functionId?: string;\n};\n\nexport const uploadWorkerFunction = async ({\n client,\n spaceId,\n version,\n source,\n name,\n functionId,\n}: UploadWorkerArgs): Promise<UploadFunctionResponseBody> => {\n const edgeUrl = client.config.values.runtime?.services?.edge?.url;\n invariant(edgeUrl, 'Edge is not configured.');\n const edgeClient = new EdgeHttpClient(edgeUrl);\n const edgeIdentity = createEdgeIdentity(client);\n edgeClient.setIdentity(edgeIdentity);\n const response = await edgeClient.uploadFunction({ spaceId, functionId }, { name, version, script: source });\n\n // TODO(burdon): Edge service log.\n log.info('Uploaded', {\n identityKey: edgeIdentity.identityKey,\n functionId,\n name,\n source: source.length,\n response,\n });\n\n return response;\n};\n\nexport const incrementSemverPatch = (version: string): string => {\n const [major, minor, patch] = version.split('.');\n const patchNum = Number(patch);\n invariant(!Number.isNaN(patchNum), `Unexpected function version format: ${version}`);\n return [major, minor, String(patchNum + 1)].join('.');\n};\n\n// TODO(burdon): Factor out.\nexport const publicKeyToDid = (key: PublicKey): DID => {\n return `did:key:${key.toHex()}`;\n};\n"],
5
+ "mappings": ";;;AAOA,SAASA,0BAA0B;AACnC,SAASC,sBAAsB;AAC/B,SAASC,iBAAiB;AAE1B,SAASC,WAAW;;AAYb,IAAMC,uBAAuB,OAAO,EACzCC,QACAC,SACAC,SACAC,QACAC,MACAC,WAAU,MACO;AACjB,QAAMC,UAAUN,OAAOO,OAAOC,OAAOC,SAASC,UAAUC,MAAMC;AAC9Df,YAAUS,SAAS,2BAAA;;;;;;;;;AACnB,QAAMO,aAAa,IAAIjB,eAAeU,OAAAA;AACtC,QAAMQ,eAAenB,mBAAmBK,MAAAA;AACxCa,aAAWE,YAAYD,YAAAA;AACvB,QAAME,WAAW,MAAMH,WAAWI,eAAe;IAAEhB;IAASI;EAAW,GAAG;IAAED;IAAMF;IAASgB,QAAQf;EAAO,CAAA;AAG1GL,MAAIqB,KAAK,YAAY;IACnBC,aAAaN,aAAaM;IAC1Bf;IACAD;IACAD,QAAQA,OAAOkB;IACfL;EACF,GAAA;;;;;;AAEA,SAAOA;AACT;AAEO,IAAMM,uBAAuB,CAACpB,YAAAA;AACnC,QAAM,CAACqB,OAAOC,OAAOC,KAAAA,IAASvB,QAAQwB,MAAM,GAAA;AAC5C,QAAMC,WAAWC,OAAOH,KAAAA;AACxB5B,YAAU,CAAC+B,OAAOC,MAAMF,QAAAA,GAAW,uCAAuCzB,OAAAA,IAAS;;;;;;;;;AACnF,SAAO;IAACqB;IAAOC;IAAOM,OAAOH,WAAW,CAAA;IAAII,KAAK,GAAA;AACnD;AAGO,IAAMC,iBAAiB,CAACC,QAAAA;AAC7B,SAAO,WAAWA,IAAIC,MAAK,CAAA;AAC7B;",
6
+ "names": ["createEdgeIdentity", "EdgeHttpClient", "invariant", "log", "uploadWorkerFunction", "client", "spaceId", "version", "source", "name", "functionId", "edgeUrl", "config", "values", "runtime", "services", "edge", "url", "edgeClient", "edgeIdentity", "setIdentity", "response", "uploadFunction", "script", "info", "identityKey", "length", "incrementSemverPatch", "major", "minor", "patch", "split", "patchNum", "Number", "isNaN", "String", "join", "publicKeyToDid", "key", "toHex"]
7
+ }
@@ -0,0 +1,482 @@
1
+ import { createRequire } from 'node:module';const require = createRequire(import.meta.url);
2
+ import {
3
+ FunctionDef,
4
+ FunctionTrigger
5
+ } from "./chunk-HBD2FZXO.mjs";
6
+
7
+ // packages/core/functions/src/function/function-registry.ts
8
+ import { Event } from "@dxos/async";
9
+ import { create, Filter } from "@dxos/client/echo";
10
+ import { Resource } from "@dxos/context";
11
+ import { PublicKey } from "@dxos/keys";
12
+ import { log } from "@dxos/log";
13
+ import { ComplexMap, diff } from "@dxos/util";
14
+ var __dxlog_file = "/home/runner/work/dxos/dxos/packages/core/functions/src/function/function-registry.ts";
15
+ var FunctionRegistry = class extends Resource {
16
+ constructor(_client) {
17
+ super();
18
+ this._client = _client;
19
+ this._functionBySpaceKey = new ComplexMap(PublicKey.hash);
20
+ this.registered = new Event();
21
+ }
22
+ getFunctions(space) {
23
+ return this._functionBySpaceKey.get(space.key) ?? [];
24
+ }
25
+ getUniqueByUri() {
26
+ const uniqueByUri = [
27
+ ...this._functionBySpaceKey.values()
28
+ ].flatMap((defs) => defs).reduce((acc, v) => {
29
+ acc.set(v.uri, v);
30
+ return acc;
31
+ }, /* @__PURE__ */ new Map());
32
+ return [
33
+ ...uniqueByUri.values()
34
+ ];
35
+ }
36
+ /**
37
+ * Loads function definitions from the manifest into the space.
38
+ * We first load all the definitions from the space to deduplicate by functionId.
39
+ */
40
+ async register(space, functions) {
41
+ log("register", {
42
+ space: space.key,
43
+ functions: functions?.length ?? 0
44
+ }, {
45
+ F: __dxlog_file,
46
+ L: 48,
47
+ S: this,
48
+ C: (f, a) => f(...a)
49
+ });
50
+ if (!functions?.length) {
51
+ return;
52
+ }
53
+ if (!space.db.graph.schemaRegistry.hasSchema(FunctionDef)) {
54
+ space.db.graph.schemaRegistry.addSchema([
55
+ FunctionDef
56
+ ]);
57
+ }
58
+ const { objects: existing } = await space.db.query(Filter.schema(FunctionDef)).run();
59
+ const { added } = diff(existing, functions, (a, b) => a.uri === b.uri);
60
+ added.forEach((def) => space.db.add(create(FunctionDef, def)));
61
+ if (added.length > 0) {
62
+ await space.db.flush({
63
+ indexes: true,
64
+ updates: true
65
+ });
66
+ }
67
+ }
68
+ async _open() {
69
+ log.info("opening...", void 0, {
70
+ F: __dxlog_file,
71
+ L: 68,
72
+ S: this,
73
+ C: (f, a) => f(...a)
74
+ });
75
+ const spacesSubscription = this._client.spaces.subscribe(async (spaces) => {
76
+ for (const space of spaces) {
77
+ if (this._functionBySpaceKey.has(space.key)) {
78
+ continue;
79
+ }
80
+ const registered = [];
81
+ this._functionBySpaceKey.set(space.key, registered);
82
+ await space.waitUntilReady();
83
+ if (this._ctx.disposed) {
84
+ break;
85
+ }
86
+ this._ctx.onDispose(space.db.query(Filter.schema(FunctionDef)).subscribe(({ objects }) => {
87
+ const { added } = diff(registered, objects, (a, b) => a.uri === b.uri);
88
+ if (added.length > 0) {
89
+ registered.push(...added);
90
+ this.registered.emit({
91
+ space,
92
+ added
93
+ });
94
+ }
95
+ }));
96
+ }
97
+ });
98
+ this._ctx.onDispose(() => spacesSubscription.unsubscribe());
99
+ }
100
+ async _close(_) {
101
+ log.info("closing...", void 0, {
102
+ F: __dxlog_file,
103
+ L: 101,
104
+ S: this,
105
+ C: (f, a) => f(...a)
106
+ });
107
+ this._functionBySpaceKey.clear();
108
+ }
109
+ };
110
+
111
+ // packages/core/functions/src/trigger/type/subscription-trigger.ts
112
+ import { debounce, UpdateScheduler } from "@dxos/async";
113
+ import { Filter as Filter2 } from "@dxos/client/echo";
114
+ import { createSubscription } from "@dxos/echo-db";
115
+ import { log as log2 } from "@dxos/log";
116
+ var __dxlog_file2 = "/home/runner/work/dxos/dxos/packages/core/functions/src/trigger/type/subscription-trigger.ts";
117
+ var createSubscriptionTrigger = async (ctx, space, spec, callback) => {
118
+ const objectIds = /* @__PURE__ */ new Set();
119
+ const task = new UpdateScheduler(ctx, async () => {
120
+ if (objectIds.size > 0) {
121
+ const objects = Array.from(objectIds);
122
+ objectIds.clear();
123
+ await callback({
124
+ objects
125
+ });
126
+ }
127
+ }, {
128
+ maxFrequency: 4
129
+ });
130
+ const subscriptions = [];
131
+ const subscription = createSubscription(({ added, updated }) => {
132
+ const sizeBefore = objectIds.size;
133
+ for (const object of added) {
134
+ objectIds.add(object.id);
135
+ }
136
+ for (const object of updated) {
137
+ objectIds.add(object.id);
138
+ }
139
+ if (objectIds.size > sizeBefore) {
140
+ log2.info("updated", {
141
+ added: added.length,
142
+ updated: updated.length
143
+ }, {
144
+ F: __dxlog_file2,
145
+ L: 46,
146
+ S: void 0,
147
+ C: (f, a) => f(...a)
148
+ });
149
+ task.trigger();
150
+ }
151
+ });
152
+ subscriptions.push(() => subscription.unsubscribe());
153
+ const { filter, options: { deep, delay } = {} } = spec;
154
+ const update = ({ objects }) => {
155
+ log2.info("update", {
156
+ objects: objects.length
157
+ }, {
158
+ F: __dxlog_file2,
159
+ L: 56,
160
+ S: void 0,
161
+ C: (f, a) => f(...a)
162
+ });
163
+ subscription.update(objects);
164
+ if (deep) {
165
+ }
166
+ };
167
+ log2.info("subscription", {
168
+ filter
169
+ }, {
170
+ F: __dxlog_file2,
171
+ L: 74,
172
+ S: void 0,
173
+ C: (f, a) => f(...a)
174
+ });
175
+ if (filter.type) {
176
+ const query = space.db.query(Filter2.typename(filter.type, filter.props));
177
+ subscriptions.push(query.subscribe(delay ? debounce(update, delay) : update));
178
+ }
179
+ ctx.onDispose(() => {
180
+ subscriptions.forEach((unsubscribe) => unsubscribe());
181
+ });
182
+ };
183
+
184
+ // packages/core/functions/src/trigger/type/timer-trigger.ts
185
+ import { parseCronExpression } from "cron-schedule";
186
+ import { DeferredTask } from "@dxos/async";
187
+ import { log as log3 } from "@dxos/log";
188
+ var __dxlog_file3 = "/home/runner/work/dxos/dxos/packages/core/functions/src/trigger/type/timer-trigger.ts";
189
+ var createTimerTrigger = async (ctx, space, spec, callback) => {
190
+ const task = new DeferredTask(ctx, async () => {
191
+ await callback({});
192
+ });
193
+ let last = 0;
194
+ let run = 0;
195
+ const schedule = parseCronExpression(spec.cron);
196
+ const getRunTimeout = () => Date.now() - schedule.getNextDate().getTime();
197
+ const runCron = () => {
198
+ if (ctx.disposed) {
199
+ return;
200
+ }
201
+ const now = Date.now();
202
+ const delta = last ? now - last : 0;
203
+ last = now;
204
+ run++;
205
+ log3.info("tick", {
206
+ space: space.key.truncate(),
207
+ count: run,
208
+ delta
209
+ }, {
210
+ F: __dxlog_file3,
211
+ L: 39,
212
+ S: void 0,
213
+ C: (f, a) => f(...a)
214
+ });
215
+ task.schedule();
216
+ timeout = setTimeout(runCron, getRunTimeout());
217
+ };
218
+ let timeout = setTimeout(runCron, getRunTimeout());
219
+ ctx.onDispose(() => clearTimeout(timeout));
220
+ };
221
+
222
+ // packages/core/functions/src/trigger/trigger-registry.ts
223
+ import { Event as Event2 } from "@dxos/async";
224
+ import { create as create2, Filter as Filter3, getMeta, compareForeignKeys } from "@dxos/client/echo";
225
+ import { Context, Resource as Resource2 } from "@dxos/context";
226
+ import { ECHO_ATTR_META, foreignKey } from "@dxos/echo-schema";
227
+ import { invariant } from "@dxos/invariant";
228
+ import { PublicKey as PublicKey2 } from "@dxos/keys";
229
+ import { log as log4 } from "@dxos/log";
230
+ import { ComplexMap as ComplexMap2, diff as diff2 } from "@dxos/util";
231
+ var __dxlog_file4 = "/home/runner/work/dxos/dxos/packages/core/functions/src/trigger/trigger-registry.ts";
232
+ var triggerFactory = {
233
+ timer: createTimerTrigger,
234
+ // TODO(burdon): Cannot use in browser.
235
+ // webhook: createWebhookTrigger,
236
+ subscription: createSubscriptionTrigger
237
+ };
238
+ var TriggerRegistry = class extends Resource2 {
239
+ constructor(_client, _options) {
240
+ super();
241
+ this._client = _client;
242
+ this._options = _options;
243
+ this._triggersBySpaceKey = new ComplexMap2(PublicKey2.hash);
244
+ this.registered = new Event2();
245
+ this.removed = new Event2();
246
+ }
247
+ getActiveTriggers(space) {
248
+ return this._getTriggers(space, (t) => t.activationCtx != null);
249
+ }
250
+ getInactiveTriggers(space) {
251
+ return this._getTriggers(space, (t) => t.activationCtx == null);
252
+ }
253
+ /**
254
+ * Set callback for trigger.
255
+ */
256
+ async activate(space, trigger, callback) {
257
+ log4("activate", {
258
+ space: space.key,
259
+ trigger
260
+ }, {
261
+ F: __dxlog_file4,
262
+ L: 75,
263
+ S: this,
264
+ C: (f, a) => f(...a)
265
+ });
266
+ const activationCtx = new Context({
267
+ name: `FunctionTrigger-${trigger.function}`
268
+ }, {
269
+ F: __dxlog_file4,
270
+ L: 77
271
+ });
272
+ this._ctx.onDispose(() => activationCtx.dispose());
273
+ const registeredTrigger = this._triggersBySpaceKey.get(space.key)?.find((reg) => reg.trigger.id === trigger.id);
274
+ invariant(registeredTrigger, `Trigger is not registered: ${trigger.function}`, {
275
+ F: __dxlog_file4,
276
+ L: 80,
277
+ S: this,
278
+ A: [
279
+ "registeredTrigger",
280
+ "`Trigger is not registered: ${trigger.function}`"
281
+ ]
282
+ });
283
+ registeredTrigger.activationCtx = activationCtx;
284
+ try {
285
+ invariant(trigger.spec, void 0, {
286
+ F: __dxlog_file4,
287
+ L: 85,
288
+ S: this,
289
+ A: [
290
+ "trigger.spec",
291
+ ""
292
+ ]
293
+ });
294
+ const options = this._options?.[trigger.spec.type];
295
+ const createTrigger = triggerFactory[trigger.spec.type];
296
+ invariant(createTrigger, `Trigger factory not found: ${trigger.spec.type}`, {
297
+ F: __dxlog_file4,
298
+ L: 88,
299
+ S: this,
300
+ A: [
301
+ "createTrigger",
302
+ "`Trigger factory not found: ${trigger.spec.type}`"
303
+ ]
304
+ });
305
+ await createTrigger(activationCtx, space, trigger.spec, callback, options);
306
+ } catch (err) {
307
+ delete registeredTrigger.activationCtx;
308
+ throw err;
309
+ }
310
+ }
311
+ /**
312
+ * Loads triggers from the manifest into the space.
313
+ */
314
+ async register(space, manifest) {
315
+ log4("register", {
316
+ space: space.key
317
+ }, {
318
+ F: __dxlog_file4,
319
+ L: 100,
320
+ S: this,
321
+ C: (f, a) => f(...a)
322
+ });
323
+ if (!manifest.triggers?.length) {
324
+ return;
325
+ }
326
+ if (!space.db.graph.schemaRegistry.hasSchema(FunctionTrigger)) {
327
+ space.db.graph.schemaRegistry.addSchema([
328
+ FunctionTrigger
329
+ ]);
330
+ }
331
+ const manifestTriggers = manifest.triggers.map((trigger) => {
332
+ let keys = trigger[ECHO_ATTR_META]?.keys;
333
+ delete trigger[ECHO_ATTR_META];
334
+ if (!keys?.length) {
335
+ keys = [
336
+ foreignKey("manifest", [
337
+ trigger.function,
338
+ trigger.spec?.type
339
+ ].join(":"))
340
+ ];
341
+ }
342
+ return create2(FunctionTrigger, trigger, {
343
+ keys
344
+ });
345
+ });
346
+ const { objects: existing } = await space.db.query(Filter3.schema(FunctionTrigger)).run();
347
+ const { added } = diff2(existing, manifestTriggers, compareForeignKeys);
348
+ added.forEach((trigger) => {
349
+ space.db.add(trigger);
350
+ log4.info("added", {
351
+ meta: getMeta(trigger)
352
+ }, {
353
+ F: __dxlog_file4,
354
+ L: 127,
355
+ S: this,
356
+ C: (f, a) => f(...a)
357
+ });
358
+ });
359
+ if (added.length > 0) {
360
+ await space.db.flush();
361
+ }
362
+ }
363
+ async _open() {
364
+ log4.info("open...", void 0, {
365
+ F: __dxlog_file4,
366
+ L: 136,
367
+ S: this,
368
+ C: (f, a) => f(...a)
369
+ });
370
+ const spaceListSubscription = this._client.spaces.subscribe(async (spaces) => {
371
+ for (const space of spaces) {
372
+ if (this._triggersBySpaceKey.has(space.key)) {
373
+ continue;
374
+ }
375
+ const registered = [];
376
+ this._triggersBySpaceKey.set(space.key, registered);
377
+ await space.waitUntilReady();
378
+ if (this._ctx.disposed) {
379
+ break;
380
+ }
381
+ this._ctx.onDispose(space.db.query(Filter3.schema(FunctionTrigger)).subscribe(async ({ objects: current }) => {
382
+ log4.info("update", {
383
+ space: space.key,
384
+ registered: registered.length,
385
+ current: current.length
386
+ }, {
387
+ F: __dxlog_file4,
388
+ L: 153,
389
+ S: this,
390
+ C: (f, a) => f(...a)
391
+ });
392
+ await this._handleRemovedTriggers(space, current, registered);
393
+ this._handleNewTriggers(space, current, registered);
394
+ }));
395
+ }
396
+ });
397
+ this._ctx.onDispose(() => spaceListSubscription.unsubscribe());
398
+ log4.info("opened", void 0, {
399
+ F: __dxlog_file4,
400
+ L: 162,
401
+ S: this,
402
+ C: (f, a) => f(...a)
403
+ });
404
+ }
405
+ async _close(_) {
406
+ log4.info("close...", void 0, {
407
+ F: __dxlog_file4,
408
+ L: 166,
409
+ S: this,
410
+ C: (f, a) => f(...a)
411
+ });
412
+ this._triggersBySpaceKey.clear();
413
+ log4.info("closed", void 0, {
414
+ F: __dxlog_file4,
415
+ L: 168,
416
+ S: this,
417
+ C: (f, a) => f(...a)
418
+ });
419
+ }
420
+ _handleNewTriggers(space, current, registered) {
421
+ const added = current.filter((candidate) => {
422
+ return candidate.enabled && registered.find((reg) => reg.trigger.id === candidate.id) == null;
423
+ });
424
+ if (added.length > 0) {
425
+ const newRegisteredTriggers = added.map((trigger) => ({
426
+ trigger
427
+ }));
428
+ registered.push(...newRegisteredTriggers);
429
+ log4.info("added", () => ({
430
+ spaceKey: space.key,
431
+ triggers: added.map((trigger) => trigger.function)
432
+ }), {
433
+ F: __dxlog_file4,
434
+ L: 179,
435
+ S: this,
436
+ C: (f, a) => f(...a)
437
+ });
438
+ this.registered.emit({
439
+ space,
440
+ triggers: added
441
+ });
442
+ }
443
+ }
444
+ async _handleRemovedTriggers(space, current, registered) {
445
+ const removed = [];
446
+ for (let i = registered.length - 1; i >= 0; i--) {
447
+ const wasRemoved = current.filter((trigger) => trigger.enabled).find((trigger) => trigger.id === registered[i].trigger.id) == null;
448
+ if (wasRemoved) {
449
+ const unregistered = registered.splice(i, 1)[0];
450
+ await unregistered.activationCtx?.dispose();
451
+ removed.push(unregistered.trigger);
452
+ }
453
+ }
454
+ if (removed.length > 0) {
455
+ log4.info("removed", () => ({
456
+ spaceKey: space.key,
457
+ triggers: removed.map((trigger) => trigger.function)
458
+ }), {
459
+ F: __dxlog_file4,
460
+ L: 205,
461
+ S: this,
462
+ C: (f, a) => f(...a)
463
+ });
464
+ this.removed.emit({
465
+ space,
466
+ triggers: removed
467
+ });
468
+ }
469
+ }
470
+ _getTriggers(space, predicate) {
471
+ const allSpaceTriggers = this._triggersBySpaceKey.get(space.key) ?? [];
472
+ return allSpaceTriggers.filter(predicate).map((trigger) => trigger.trigger);
473
+ }
474
+ };
475
+
476
+ export {
477
+ FunctionRegistry,
478
+ createSubscriptionTrigger,
479
+ createTimerTrigger,
480
+ TriggerRegistry
481
+ };
482
+ //# sourceMappingURL=chunk-SQSJO5HI.mjs.map