@codemation/host 1.0.0 → 1.0.2
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 +31 -0
- package/dist/{AppConfigFactory-BPp02HMv.js → AppConfigFactory-C4OXGOs2.js} +16 -7
- package/dist/{AppConfigFactory-BPp02HMv.js.map → AppConfigFactory-C4OXGOs2.js.map} +1 -1
- package/dist/{AppConfigFactory-PFmDg5Sg.d.ts → AppConfigFactory-D3k-R3Ch.d.ts} +338 -6
- package/dist/{AppContainerFactory-Cr3JeVmg.js → AppContainerFactory-CKRDz8kQ.js} +409 -92
- package/dist/AppContainerFactory-CKRDz8kQ.js.map +1 -0
- package/dist/{CodemationAppContext-DP_-56c6.d.ts → CodemationAppContext-YgJRUHWF.d.ts} +2 -2
- package/dist/{CodemationAuthoring.types-zJ2t73Bn.d.ts → CodemationAuthoring.types-lUdxXYq-.d.ts} +7 -6
- package/dist/{CodemationConfigNormalizer-B8RGUwAe.d.ts → CodemationConfigNormalizer-BWBp7mFB.d.ts} +2 -2
- package/dist/{CodemationConsumerConfigLoader-C_QVwcI3.d.ts → CodemationConsumerConfigLoader-Bka3v6lh.d.ts} +2 -2
- package/dist/{CodemationPluginListMerger-Bgn1CIX9.d.ts → CodemationPluginListMerger-Oz-GAkxz.d.ts} +17 -5
- package/dist/{CredentialServices-95DPogx-.d.ts → CredentialServices-CKXPg5xu.d.ts} +3 -3
- package/dist/{PublicFrontendBootstrapFactory-C_iLgPV-.d.ts → PublicFrontendBootstrapFactory-DkQoSYDo.d.ts} +2 -2
- package/dist/authoring.d.ts +3 -3
- package/dist/consumer.d.ts +4 -4
- package/dist/credentials.d.ts +3 -3
- package/dist/devServerSidecar.d.ts +1 -1
- package/dist/{index-W4eSjdCM.d.ts → index-BxIc_L4D.d.ts} +233 -151
- package/dist/index.d.ts +11 -11
- package/dist/index.js +4 -4
- package/dist/nextServer.d.ts +7 -7
- package/dist/nextServer.js +2 -2
- package/dist/{persistenceServer-_pqP_0nw.d.ts → persistenceServer-BLG7_6B5.d.ts} +2 -2
- package/dist/{persistenceServer-CA0_q0D7.js → persistenceServer-KyHL0u01.js} +2 -2
- package/dist/{persistenceServer-CA0_q0D7.js.map → persistenceServer-KyHL0u01.js.map} +1 -1
- package/dist/persistenceServer.d.ts +5 -5
- package/dist/persistenceServer.js +2 -2
- package/dist/{server-Q5uwa6iR.d.ts → server-B0SD6Nvk.d.ts} +5 -5
- package/dist/{server-BE4PLhcb.js → server-CMUVhYIc.js} +3 -3
- package/dist/{server-BE4PLhcb.js.map → server-CMUVhYIc.js.map} +1 -1
- package/dist/server.d.ts +8 -8
- package/dist/server.js +4 -4
- package/package.json +5 -5
- package/prisma/migrations/20260430120000_telemetry_iteration_identity/migration.sql +17 -0
- package/prisma/migrations/20260430130000_execution_instance_iteration_identity/migration.sql +11 -0
- package/prisma/migrations.sqlite/20260430120000_telemetry_iteration_identity/migration.sql +14 -0
- package/prisma/migrations.sqlite/20260430130000_execution_instance_iteration_identity/migration.sql +10 -0
- package/prisma/schema.postgresql.prisma +12 -0
- package/prisma/schema.sqlite.prisma +12 -0
- package/src/application/contracts/IterationCostContracts.ts +11 -0
- package/src/application/queries/GetIterationCostQuery.ts +14 -0
- package/src/application/queries/GetIterationCostQueryHandler.ts +92 -0
- package/src/application/queries/GetWorkflowRunDetailQueryHandler.ts +44 -2
- package/src/application/queries/RunIterationProjectionFactory.ts +123 -0
- package/src/application/queries/WorkflowQueryHandlers.ts +1 -0
- package/src/application/telemetry/OtelExecutionTelemetry.types.ts +3 -0
- package/src/application/telemetry/RunEventBusTelemetryReporter.ts +7 -0
- package/src/application/telemetry/StoredNodeExecutionTelemetry.ts +14 -0
- package/src/application/telemetry/StoredTelemetrySpanScope.ts +90 -1
- package/src/bootstrap/AppContainerFactory.ts +5 -0
- package/src/domain/telemetry/TelemetryContracts.ts +12 -0
- package/src/infrastructure/persistence/InMemoryTelemetryMetricPointStore.ts +3 -0
- package/src/infrastructure/persistence/InMemoryTelemetrySpanStore.ts +3 -0
- package/src/infrastructure/persistence/InMemoryWorkflowRunRepository.ts +23 -0
- package/src/infrastructure/persistence/PrismaTelemetryMetricPointStore.ts +9 -0
- package/src/infrastructure/persistence/PrismaTelemetrySpanStore.ts +6 -0
- package/src/infrastructure/persistence/PrismaWorkflowRunRepository.ts +12 -0
- package/src/infrastructure/persistence/generated/prisma-postgresql-client/edge.js +15 -6
- package/src/infrastructure/persistence/generated/prisma-postgresql-client/index-browser.js +11 -2
- package/src/infrastructure/persistence/generated/prisma-postgresql-client/index.d.ts +343 -5
- package/src/infrastructure/persistence/generated/prisma-postgresql-client/index.js +15 -6
- package/src/infrastructure/persistence/generated/prisma-postgresql-client/package.json +1 -1
- package/src/infrastructure/persistence/generated/prisma-postgresql-client/schema.prisma +12 -0
- package/src/infrastructure/persistence/generated/prisma-sqlite-client/edge.js +15 -6
- package/src/infrastructure/persistence/generated/prisma-sqlite-client/index-browser.js +11 -2
- package/src/infrastructure/persistence/generated/prisma-sqlite-client/index.d.ts +343 -5
- package/src/infrastructure/persistence/generated/prisma-sqlite-client/index.js +15 -6
- package/src/infrastructure/persistence/generated/prisma-sqlite-client/package.json +1 -1
- package/src/infrastructure/persistence/generated/prisma-sqlite-client/schema.prisma +12 -0
- package/dist/AppContainerFactory-Cr3JeVmg.js.map +0 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"server-BE4PLhcb.js","names":["config: CodemationConfig","consumerRoot: string","configSource?: string","workflowSources: ReadonlyArray<string>","env?: Readonly<NodeJS.ProcessEnv>","consumerConfigLoader: CodemationConsumerConfigLoader","appConfigFactory: AppConfigFactory","discoveredPackages: CodemationDiscoveredPluginPackage[]","resolvedPackages: CodemationResolvedPluginPackage[]","packageRoots: string[]"],"sources":["../src/presentation/http/CodemationServerGatewayFactory.ts","../src/presentation/server/AppConfigLoader.ts","../src/presentation/server/CodemationPluginDiscovery.ts"],"sourcesContent":["import { accessSync } from \"node:fs\";\nimport path from \"node:path\";\nimport type { QueryBus } from \"../../application/bus/QueryBus\";\nimport type { WorkflowDto, WorkflowSummary } from \"../../application/contracts/WorkflowViewContracts\";\nimport { WorkflowDefinitionMapper } from \"../../application/mapping/WorkflowDefinitionMapper\";\nimport { GetWorkflowDetailQuery } from \"../../application/queries/GetWorkflowDetailQuery\";\nimport { GetWorkflowSummariesQuery } from \"../../application/queries/GetWorkflowSummariesQuery\";\nimport { ApplicationTokens } from \"../../applicationTokens\";\nimport { AppContainerFactory } from \"../../bootstrap/AppContainerFactory\";\nimport { AppContainerLifecycle } from \"../../bootstrap/AppContainerLifecycle\";\nimport { FrontendRuntime } from \"../../bootstrap/runtime/FrontendRuntime\";\nimport { AppConfigFactory } from \"../../bootstrap/runtime/AppConfigFactory\";\nimport type { CodemationConfig } from \"../config/CodemationConfig\";\nimport { CodemationConfigNormalizer } from \"../config/CodemationConfigNormalizer\";\nimport { CodemationHonoApiApp } from \"./hono/CodemationHonoApiAppFactory\";\n\ntype ServerGatewayContext = Readonly<{\n container: import(\"@codemation/core\").Container;\n httpApi: CodemationHonoApiApp;\n queryBus: QueryBus;\n workflowDefinitionMapper: WorkflowDefinitionMapper;\n}>;\n\nexport class CodemationServerGateway {\n private static readonly contextsByConfig = new WeakMap<object, Promise<ServerGatewayContext>>();\n\n constructor(\n private readonly config: CodemationConfig,\n private readonly consumerRoot: string,\n private readonly configSource?: string,\n private readonly workflowSources: ReadonlyArray<string> = [],\n private readonly env?: Readonly<NodeJS.ProcessEnv>,\n ) {}\n\n async dispatch(request: Request): Promise<Response> {\n return await (await this.getContext()).httpApi.fetch(request);\n }\n\n async prepare(): Promise<void> {\n await this.getContext();\n }\n\n async close(): Promise<void> {\n const cachedContext = CodemationServerGateway.contextsByConfig.get(this.config as object);\n if (!cachedContext) {\n return;\n }\n CodemationServerGateway.contextsByConfig.delete(this.config as object);\n await (await cachedContext).container.resolve(AppContainerLifecycle).stop();\n }\n\n async loadWorkflowSummaries(): Promise<ReadonlyArray<WorkflowSummary>> {\n const context = await this.getContext();\n const workflows = await context.queryBus.execute(new GetWorkflowSummariesQuery());\n return workflows.map((workflow) => context.workflowDefinitionMapper.toSummary(workflow));\n }\n\n async loadWorkflowDetail(workflowId: string): Promise<WorkflowDto> {\n const context = await this.getContext();\n const workflow = await context.queryBus.execute(new GetWorkflowDetailQuery(workflowId));\n if (!workflow) {\n throw new Error(`Unknown workflowId: ${workflowId}`);\n }\n return await context.workflowDefinitionMapper.map(workflow);\n }\n\n private getContext(): Promise<ServerGatewayContext> {\n const cachedContext = CodemationServerGateway.contextsByConfig.get(this.config as object);\n if (cachedContext) {\n return cachedContext;\n }\n const nextContext = this.createContext();\n CodemationServerGateway.contextsByConfig.set(this.config as object, nextContext);\n return nextContext;\n }\n\n private async createContext(): Promise<ServerGatewayContext> {\n const repoRoot = this.detectWorkspaceRoot(this.consumerRoot);\n // This gateway is the config/env boundary that materializes AppConfig from raw inputs.\n // eslint-disable-next-line no-restricted-properties\n const env = this.env ?? process.env;\n const appConfig = new AppConfigFactory().create({\n repoRoot,\n consumerRoot: this.consumerRoot,\n env,\n config: new CodemationConfigNormalizer().normalize(this.config),\n workflowSources: this.resolveWorkflowSources(),\n });\n const container = await new AppContainerFactory().create({\n appConfig,\n sharedWorkflowWebsocketServer: null,\n });\n await container.resolve(FrontendRuntime).start();\n return {\n container,\n httpApi: container.resolve(CodemationHonoApiApp),\n queryBus: container.resolve(ApplicationTokens.QueryBus),\n workflowDefinitionMapper: container.resolve(WorkflowDefinitionMapper),\n };\n }\n\n private resolveWorkflowSources(): ReadonlyArray<string> {\n if (this.workflowSources.length > 0) {\n return [...this.workflowSources];\n }\n if (!this.configSource || !this.config.workflows || this.config.workflows.length === 0) {\n return [];\n }\n return [this.configSource];\n }\n private detectWorkspaceRoot(startDirectory: string): string {\n let currentDirectory = path.resolve(startDirectory);\n while (true) {\n try {\n accessSync(path.resolve(currentDirectory, \"pnpm-workspace.yaml\"));\n return currentDirectory;\n } catch {\n const parentDirectory = path.dirname(currentDirectory);\n if (parentDirectory === currentDirectory) {\n return startDirectory;\n }\n currentDirectory = parentDirectory;\n }\n }\n }\n}\n","import type { AppConfig } from \"../config/AppConfig\";\nimport { CodemationConsumerConfigLoader } from \"./CodemationConsumerConfigLoader\";\nimport { AppConfigFactory } from \"../../bootstrap/runtime/AppConfigFactory\";\n\nexport type AppConfigLoadResult = Readonly<{\n appConfig: AppConfig;\n bootstrapSource: string | null;\n}>;\n\nexport class AppConfigLoader {\n constructor(\n private readonly consumerConfigLoader: CodemationConsumerConfigLoader = new CodemationConsumerConfigLoader(),\n private readonly appConfigFactory: AppConfigFactory = new AppConfigFactory(),\n ) {}\n\n async load(\n args: Readonly<{\n consumerRoot: string;\n repoRoot: string;\n env: NodeJS.ProcessEnv;\n configPathOverride?: string;\n }>,\n ): Promise<AppConfigLoadResult> {\n const resolution = await this.consumerConfigLoader.load({\n consumerRoot: args.consumerRoot,\n configPathOverride: args.configPathOverride,\n });\n return {\n appConfig: this.appConfigFactory.create({\n repoRoot: args.repoRoot,\n consumerRoot: args.consumerRoot,\n env: args.env,\n config: resolution.config,\n workflowSources: resolution.workflowSources,\n }),\n bootstrapSource: resolution.bootstrapSource,\n };\n }\n}\n","import type { CodemationPackageManifest } from \"../config/CodemationPackageManifest\";\nimport { CodemationPluginPackageMetadata, type CodemationPlugin } from \"../config/CodemationPlugin\";\nimport { readFile, readdir } from \"node:fs/promises\";\nimport path from \"node:path\";\nimport { pathToFileURL } from \"node:url\";\n\nexport type CodemationDiscoveredPluginPackage = Readonly<{\n packageName: string;\n packageRoot: string;\n pluginEntry: string;\n developmentEntry?: string;\n}>;\n\nexport type CodemationResolvedPluginPackage = Readonly<\n CodemationDiscoveredPluginPackage & {\n plugin: CodemationPlugin;\n }\n>;\n\ntype PackageJsonShape = Readonly<{\n codemation?: CodemationPackageManifest;\n name?: string;\n exports?: Readonly<Record<string, unknown>>;\n}>;\n\nexport class CodemationPluginDiscovery {\n private readonly pluginPackageMetadata = new CodemationPluginPackageMetadata();\n\n async discover(consumerRoot: string): Promise<ReadonlyArray<CodemationDiscoveredPluginPackage>> {\n const nodeModulesRoot = path.resolve(consumerRoot, \"node_modules\");\n const packageRoots = await this.collectPackageRoots(nodeModulesRoot);\n const discoveredPackages: CodemationDiscoveredPluginPackage[] = [];\n for (const packageRoot of packageRoots) {\n const packageJson = await this.readPackageJson(path.resolve(packageRoot, \"package.json\"));\n const pluginManifest = packageJson.codemation?.plugin;\n if (!packageJson.name || typeof pluginManifest !== \"string\" || pluginManifest.trim().length === 0) {\n continue;\n }\n discoveredPackages.push({\n packageName: packageJson.name,\n packageRoot,\n pluginEntry: pluginManifest,\n developmentEntry: await this.resolveDevelopmentPluginEntry(packageRoot),\n });\n }\n return discoveredPackages.sort((left, right) => left.packageName.localeCompare(right.packageName));\n }\n\n async resolvePlugins(consumerRoot: string): Promise<ReadonlyArray<CodemationResolvedPluginPackage>> {\n const discoveredPackages = await this.discover(consumerRoot);\n return await this.resolveDiscoveredPackages(discoveredPackages);\n }\n\n async resolveDiscoveredPackages(\n discoveredPackages: ReadonlyArray<CodemationDiscoveredPluginPackage>,\n ): Promise<ReadonlyArray<CodemationResolvedPluginPackage>> {\n const resolvedPackages: CodemationResolvedPluginPackage[] = [];\n for (const discoveredPackage of discoveredPackages) {\n resolvedPackages.push({\n ...discoveredPackage,\n plugin: await this.loadPlugin(discoveredPackage),\n });\n }\n return resolvedPackages;\n }\n\n private async collectPackageRoots(nodeModulesRoot: string): Promise<ReadonlyArray<string>> {\n try {\n const entries = await readdir(nodeModulesRoot, { withFileTypes: true });\n const packageRoots: string[] = [];\n for (const entry of entries) {\n if (!entry.isDirectory() && !entry.isSymbolicLink()) {\n continue;\n }\n if (entry.name.startsWith(\"@\")) {\n const scopedEntries = await readdir(path.resolve(nodeModulesRoot, entry.name), { withFileTypes: true });\n for (const scopedEntry of scopedEntries) {\n if (scopedEntry.isDirectory() || scopedEntry.isSymbolicLink()) {\n packageRoots.push(path.resolve(nodeModulesRoot, entry.name, scopedEntry.name));\n }\n }\n continue;\n }\n packageRoots.push(path.resolve(nodeModulesRoot, entry.name));\n }\n return packageRoots;\n } catch {\n return [];\n }\n }\n\n private async readPackageJson(packageJsonPath: string): Promise<PackageJsonShape> {\n try {\n const rawPackageJson = await readFile(packageJsonPath, \"utf8\");\n return JSON.parse(rawPackageJson) as PackageJsonShape;\n } catch {\n return {};\n }\n }\n\n private async loadPlugin(discoveredPackage: CodemationDiscoveredPluginPackage): Promise<CodemationPlugin> {\n const pluginModulePath = path.resolve(discoveredPackage.packageRoot, this.resolvePluginEntry(discoveredPackage));\n const importedModule = (await import(\n /* webpackIgnore: true */ this.resolvePluginModuleSpecifier(pluginModulePath)\n )) as Record<string, unknown>;\n const exportedValue = importedModule.default;\n const plugin = this.resolvePluginValue(exportedValue);\n if (!plugin) {\n throw new Error(`Plugin package \"${discoveredPackage.packageName}\" did not default-export a Codemation plugin.`);\n }\n return this.pluginPackageMetadata.attachPackageName(plugin, discoveredPackage.packageName);\n }\n\n private resolvePluginValue(value: unknown): CodemationPlugin | null {\n if (this.isPluginConfig(value)) {\n return value;\n }\n return null;\n }\n\n private isPluginConfig(value: unknown): value is CodemationPlugin {\n if (!value || typeof value !== \"object\" || Array.isArray(value)) {\n return false;\n }\n const pluginValue = value as {\n credentialTypes?: unknown;\n register?: unknown;\n sandbox?: unknown;\n };\n if (pluginValue.register !== undefined && typeof pluginValue.register !== \"function\") {\n return false;\n }\n if (pluginValue.credentialTypes !== undefined && !Array.isArray(pluginValue.credentialTypes)) {\n return false;\n }\n return (\n pluginValue.register !== undefined ||\n pluginValue.credentialTypes !== undefined ||\n pluginValue.sandbox !== undefined ||\n Object.keys(pluginValue).length === 0\n );\n }\n\n private resolvePluginEntry(discoveredPackage: CodemationDiscoveredPluginPackage): string {\n const preferSource =\n process.env.CODEMATION_PREFER_PLUGIN_SOURCE_ENTRY === \"true\" &&\n typeof discoveredPackage.developmentEntry === \"string\" &&\n discoveredPackage.developmentEntry.trim().length > 0;\n const selectedEntry = preferSource ? discoveredPackage.developmentEntry : discoveredPackage.pluginEntry;\n return selectedEntry;\n }\n\n private async resolveDevelopmentPluginEntry(packageRoot: string): Promise<string | undefined> {\n const candidates = [\n path.resolve(packageRoot, \"codemation.plugin.ts\"),\n path.resolve(packageRoot, \"codemation.plugin.js\"),\n path.resolve(packageRoot, \"src\", \"codemation.plugin.ts\"),\n path.resolve(packageRoot, \"src\", \"codemation.plugin.js\"),\n ];\n for (const candidate of candidates) {\n if (await this.exists(candidate)) {\n return path.relative(packageRoot, candidate);\n }\n }\n return undefined;\n }\n\n private async exists(filePath: string): Promise<boolean> {\n try {\n await readFile(filePath, \"utf8\");\n return true;\n } catch {\n return false;\n }\n }\n\n private resolvePluginModuleSpecifier(pluginModulePath: string): string {\n return pathToFileURL(pluginModulePath).href;\n }\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"server-CMUVhYIc.js","names":["config: CodemationConfig","consumerRoot: string","configSource?: string","workflowSources: ReadonlyArray<string>","env?: Readonly<NodeJS.ProcessEnv>","consumerConfigLoader: CodemationConsumerConfigLoader","appConfigFactory: AppConfigFactory","discoveredPackages: CodemationDiscoveredPluginPackage[]","resolvedPackages: CodemationResolvedPluginPackage[]","packageRoots: string[]"],"sources":["../src/presentation/http/CodemationServerGatewayFactory.ts","../src/presentation/server/AppConfigLoader.ts","../src/presentation/server/CodemationPluginDiscovery.ts"],"sourcesContent":["import { accessSync } from \"node:fs\";\nimport path from \"node:path\";\nimport type { QueryBus } from \"../../application/bus/QueryBus\";\nimport type { WorkflowDto, WorkflowSummary } from \"../../application/contracts/WorkflowViewContracts\";\nimport { WorkflowDefinitionMapper } from \"../../application/mapping/WorkflowDefinitionMapper\";\nimport { GetWorkflowDetailQuery } from \"../../application/queries/GetWorkflowDetailQuery\";\nimport { GetWorkflowSummariesQuery } from \"../../application/queries/GetWorkflowSummariesQuery\";\nimport { ApplicationTokens } from \"../../applicationTokens\";\nimport { AppContainerFactory } from \"../../bootstrap/AppContainerFactory\";\nimport { AppContainerLifecycle } from \"../../bootstrap/AppContainerLifecycle\";\nimport { FrontendRuntime } from \"../../bootstrap/runtime/FrontendRuntime\";\nimport { AppConfigFactory } from \"../../bootstrap/runtime/AppConfigFactory\";\nimport type { CodemationConfig } from \"../config/CodemationConfig\";\nimport { CodemationConfigNormalizer } from \"../config/CodemationConfigNormalizer\";\nimport { CodemationHonoApiApp } from \"./hono/CodemationHonoApiAppFactory\";\n\ntype ServerGatewayContext = Readonly<{\n container: import(\"@codemation/core\").Container;\n httpApi: CodemationHonoApiApp;\n queryBus: QueryBus;\n workflowDefinitionMapper: WorkflowDefinitionMapper;\n}>;\n\nexport class CodemationServerGateway {\n private static readonly contextsByConfig = new WeakMap<object, Promise<ServerGatewayContext>>();\n\n constructor(\n private readonly config: CodemationConfig,\n private readonly consumerRoot: string,\n private readonly configSource?: string,\n private readonly workflowSources: ReadonlyArray<string> = [],\n private readonly env?: Readonly<NodeJS.ProcessEnv>,\n ) {}\n\n async dispatch(request: Request): Promise<Response> {\n return await (await this.getContext()).httpApi.fetch(request);\n }\n\n async prepare(): Promise<void> {\n await this.getContext();\n }\n\n async close(): Promise<void> {\n const cachedContext = CodemationServerGateway.contextsByConfig.get(this.config as object);\n if (!cachedContext) {\n return;\n }\n CodemationServerGateway.contextsByConfig.delete(this.config as object);\n await (await cachedContext).container.resolve(AppContainerLifecycle).stop();\n }\n\n async loadWorkflowSummaries(): Promise<ReadonlyArray<WorkflowSummary>> {\n const context = await this.getContext();\n const workflows = await context.queryBus.execute(new GetWorkflowSummariesQuery());\n return workflows.map((workflow) => context.workflowDefinitionMapper.toSummary(workflow));\n }\n\n async loadWorkflowDetail(workflowId: string): Promise<WorkflowDto> {\n const context = await this.getContext();\n const workflow = await context.queryBus.execute(new GetWorkflowDetailQuery(workflowId));\n if (!workflow) {\n throw new Error(`Unknown workflowId: ${workflowId}`);\n }\n return await context.workflowDefinitionMapper.map(workflow);\n }\n\n private getContext(): Promise<ServerGatewayContext> {\n const cachedContext = CodemationServerGateway.contextsByConfig.get(this.config as object);\n if (cachedContext) {\n return cachedContext;\n }\n const nextContext = this.createContext();\n CodemationServerGateway.contextsByConfig.set(this.config as object, nextContext);\n return nextContext;\n }\n\n private async createContext(): Promise<ServerGatewayContext> {\n const repoRoot = this.detectWorkspaceRoot(this.consumerRoot);\n // This gateway is the config/env boundary that materializes AppConfig from raw inputs.\n // eslint-disable-next-line no-restricted-properties\n const env = this.env ?? process.env;\n const appConfig = new AppConfigFactory().create({\n repoRoot,\n consumerRoot: this.consumerRoot,\n env,\n config: new CodemationConfigNormalizer().normalize(this.config),\n workflowSources: this.resolveWorkflowSources(),\n });\n const container = await new AppContainerFactory().create({\n appConfig,\n sharedWorkflowWebsocketServer: null,\n });\n await container.resolve(FrontendRuntime).start();\n return {\n container,\n httpApi: container.resolve(CodemationHonoApiApp),\n queryBus: container.resolve(ApplicationTokens.QueryBus),\n workflowDefinitionMapper: container.resolve(WorkflowDefinitionMapper),\n };\n }\n\n private resolveWorkflowSources(): ReadonlyArray<string> {\n if (this.workflowSources.length > 0) {\n return [...this.workflowSources];\n }\n if (!this.configSource || !this.config.workflows || this.config.workflows.length === 0) {\n return [];\n }\n return [this.configSource];\n }\n private detectWorkspaceRoot(startDirectory: string): string {\n let currentDirectory = path.resolve(startDirectory);\n while (true) {\n try {\n accessSync(path.resolve(currentDirectory, \"pnpm-workspace.yaml\"));\n return currentDirectory;\n } catch {\n const parentDirectory = path.dirname(currentDirectory);\n if (parentDirectory === currentDirectory) {\n return startDirectory;\n }\n currentDirectory = parentDirectory;\n }\n }\n }\n}\n","import type { AppConfig } from \"../config/AppConfig\";\nimport { CodemationConsumerConfigLoader } from \"./CodemationConsumerConfigLoader\";\nimport { AppConfigFactory } from \"../../bootstrap/runtime/AppConfigFactory\";\n\nexport type AppConfigLoadResult = Readonly<{\n appConfig: AppConfig;\n bootstrapSource: string | null;\n}>;\n\nexport class AppConfigLoader {\n constructor(\n private readonly consumerConfigLoader: CodemationConsumerConfigLoader = new CodemationConsumerConfigLoader(),\n private readonly appConfigFactory: AppConfigFactory = new AppConfigFactory(),\n ) {}\n\n async load(\n args: Readonly<{\n consumerRoot: string;\n repoRoot: string;\n env: NodeJS.ProcessEnv;\n configPathOverride?: string;\n }>,\n ): Promise<AppConfigLoadResult> {\n const resolution = await this.consumerConfigLoader.load({\n consumerRoot: args.consumerRoot,\n configPathOverride: args.configPathOverride,\n });\n return {\n appConfig: this.appConfigFactory.create({\n repoRoot: args.repoRoot,\n consumerRoot: args.consumerRoot,\n env: args.env,\n config: resolution.config,\n workflowSources: resolution.workflowSources,\n }),\n bootstrapSource: resolution.bootstrapSource,\n };\n }\n}\n","import type { CodemationPackageManifest } from \"../config/CodemationPackageManifest\";\nimport { CodemationPluginPackageMetadata, type CodemationPlugin } from \"../config/CodemationPlugin\";\nimport { readFile, readdir } from \"node:fs/promises\";\nimport path from \"node:path\";\nimport { pathToFileURL } from \"node:url\";\n\nexport type CodemationDiscoveredPluginPackage = Readonly<{\n packageName: string;\n packageRoot: string;\n pluginEntry: string;\n developmentEntry?: string;\n}>;\n\nexport type CodemationResolvedPluginPackage = Readonly<\n CodemationDiscoveredPluginPackage & {\n plugin: CodemationPlugin;\n }\n>;\n\ntype PackageJsonShape = Readonly<{\n codemation?: CodemationPackageManifest;\n name?: string;\n exports?: Readonly<Record<string, unknown>>;\n}>;\n\nexport class CodemationPluginDiscovery {\n private readonly pluginPackageMetadata = new CodemationPluginPackageMetadata();\n\n async discover(consumerRoot: string): Promise<ReadonlyArray<CodemationDiscoveredPluginPackage>> {\n const nodeModulesRoot = path.resolve(consumerRoot, \"node_modules\");\n const packageRoots = await this.collectPackageRoots(nodeModulesRoot);\n const discoveredPackages: CodemationDiscoveredPluginPackage[] = [];\n for (const packageRoot of packageRoots) {\n const packageJson = await this.readPackageJson(path.resolve(packageRoot, \"package.json\"));\n const pluginManifest = packageJson.codemation?.plugin;\n if (!packageJson.name || typeof pluginManifest !== \"string\" || pluginManifest.trim().length === 0) {\n continue;\n }\n discoveredPackages.push({\n packageName: packageJson.name,\n packageRoot,\n pluginEntry: pluginManifest,\n developmentEntry: await this.resolveDevelopmentPluginEntry(packageRoot),\n });\n }\n return discoveredPackages.sort((left, right) => left.packageName.localeCompare(right.packageName));\n }\n\n async resolvePlugins(consumerRoot: string): Promise<ReadonlyArray<CodemationResolvedPluginPackage>> {\n const discoveredPackages = await this.discover(consumerRoot);\n return await this.resolveDiscoveredPackages(discoveredPackages);\n }\n\n async resolveDiscoveredPackages(\n discoveredPackages: ReadonlyArray<CodemationDiscoveredPluginPackage>,\n ): Promise<ReadonlyArray<CodemationResolvedPluginPackage>> {\n const resolvedPackages: CodemationResolvedPluginPackage[] = [];\n for (const discoveredPackage of discoveredPackages) {\n resolvedPackages.push({\n ...discoveredPackage,\n plugin: await this.loadPlugin(discoveredPackage),\n });\n }\n return resolvedPackages;\n }\n\n private async collectPackageRoots(nodeModulesRoot: string): Promise<ReadonlyArray<string>> {\n try {\n const entries = await readdir(nodeModulesRoot, { withFileTypes: true });\n const packageRoots: string[] = [];\n for (const entry of entries) {\n if (!entry.isDirectory() && !entry.isSymbolicLink()) {\n continue;\n }\n if (entry.name.startsWith(\"@\")) {\n const scopedEntries = await readdir(path.resolve(nodeModulesRoot, entry.name), { withFileTypes: true });\n for (const scopedEntry of scopedEntries) {\n if (scopedEntry.isDirectory() || scopedEntry.isSymbolicLink()) {\n packageRoots.push(path.resolve(nodeModulesRoot, entry.name, scopedEntry.name));\n }\n }\n continue;\n }\n packageRoots.push(path.resolve(nodeModulesRoot, entry.name));\n }\n return packageRoots;\n } catch {\n return [];\n }\n }\n\n private async readPackageJson(packageJsonPath: string): Promise<PackageJsonShape> {\n try {\n const rawPackageJson = await readFile(packageJsonPath, \"utf8\");\n return JSON.parse(rawPackageJson) as PackageJsonShape;\n } catch {\n return {};\n }\n }\n\n private async loadPlugin(discoveredPackage: CodemationDiscoveredPluginPackage): Promise<CodemationPlugin> {\n const pluginModulePath = path.resolve(discoveredPackage.packageRoot, this.resolvePluginEntry(discoveredPackage));\n const importedModule = (await import(\n /* webpackIgnore: true */ this.resolvePluginModuleSpecifier(pluginModulePath)\n )) as Record<string, unknown>;\n const exportedValue = importedModule.default;\n const plugin = this.resolvePluginValue(exportedValue);\n if (!plugin) {\n throw new Error(`Plugin package \"${discoveredPackage.packageName}\" did not default-export a Codemation plugin.`);\n }\n return this.pluginPackageMetadata.attachPackageName(plugin, discoveredPackage.packageName);\n }\n\n private resolvePluginValue(value: unknown): CodemationPlugin | null {\n if (this.isPluginConfig(value)) {\n return value;\n }\n return null;\n }\n\n private isPluginConfig(value: unknown): value is CodemationPlugin {\n if (!value || typeof value !== \"object\" || Array.isArray(value)) {\n return false;\n }\n const pluginValue = value as {\n credentialTypes?: unknown;\n register?: unknown;\n sandbox?: unknown;\n };\n if (pluginValue.register !== undefined && typeof pluginValue.register !== \"function\") {\n return false;\n }\n if (pluginValue.credentialTypes !== undefined && !Array.isArray(pluginValue.credentialTypes)) {\n return false;\n }\n return (\n pluginValue.register !== undefined ||\n pluginValue.credentialTypes !== undefined ||\n pluginValue.sandbox !== undefined ||\n Object.keys(pluginValue).length === 0\n );\n }\n\n private resolvePluginEntry(discoveredPackage: CodemationDiscoveredPluginPackage): string {\n const preferSource =\n process.env.CODEMATION_PREFER_PLUGIN_SOURCE_ENTRY === \"true\" &&\n typeof discoveredPackage.developmentEntry === \"string\" &&\n discoveredPackage.developmentEntry.trim().length > 0;\n const selectedEntry = preferSource ? discoveredPackage.developmentEntry : discoveredPackage.pluginEntry;\n return selectedEntry;\n }\n\n private async resolveDevelopmentPluginEntry(packageRoot: string): Promise<string | undefined> {\n const candidates = [\n path.resolve(packageRoot, \"codemation.plugin.ts\"),\n path.resolve(packageRoot, \"codemation.plugin.js\"),\n path.resolve(packageRoot, \"src\", \"codemation.plugin.ts\"),\n path.resolve(packageRoot, \"src\", \"codemation.plugin.js\"),\n ];\n for (const candidate of candidates) {\n if (await this.exists(candidate)) {\n return path.relative(packageRoot, candidate);\n }\n }\n return undefined;\n }\n\n private async exists(filePath: string): Promise<boolean> {\n try {\n await readFile(filePath, \"utf8\");\n return true;\n } catch {\n return false;\n }\n }\n\n private resolvePluginModuleSpecifier(pluginModulePath: string): string {\n return pathToFileURL(pluginModulePath).href;\n }\n}\n"],"mappings":"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"}
|
package/dist/server.d.ts
CHANGED
|
@@ -1,12 +1,12 @@
|
|
|
1
|
-
import "./index-
|
|
2
|
-
import "./CodemationAppContext-
|
|
1
|
+
import "./index-BxIc_L4D.js";
|
|
2
|
+
import "./CodemationAppContext-YgJRUHWF.js";
|
|
3
3
|
import "./CodemationWhitelabelConfig-D5rYcLlj.js";
|
|
4
4
|
import { n as InternalAuthBootstrap, r as FrontendAppConfig, t as PublicFrontendBootstrap } from "./PublicFrontendBootstrap-ci0Vwxrb.js";
|
|
5
5
|
import { i as CodemationFrontendAuthSnapshotJsonCodec, n as InternalAuthBootstrapJsonCodec, r as FrontendAppConfigJsonCodec, t as PublicFrontendBootstrapJsonCodec } from "./PublicFrontendBootstrapJsonCodec-Ja0GQgpp.js";
|
|
6
|
-
import "./CodemationConfigNormalizer-
|
|
7
|
-
import { i as CodemationConsumerAppResolver, n as CodemationConsumerConfigResolution, r as CodemationConsumerApp, t as CodemationConsumerConfigLoader } from "./CodemationConsumerConfigLoader-
|
|
8
|
-
import { r as PrismaDatabaseClient } from "./AppConfigFactory-
|
|
9
|
-
import { t as CodemationPostgresPrismaClientFactory } from "./persistenceServer-
|
|
10
|
-
import { a as CodemationResolvedPluginPackage, c as CodemationServerGateway, i as CodemationPluginDiscovery, l as ApiPaths, n as WorkflowModulePathFinder, o as AppConfigLoadResult, r as CodemationDiscoveredPluginPackage, s as AppConfigLoader, t as WorkflowDiscoveryPathSegmentsComputer } from "./server-
|
|
11
|
-
import { i as CodemationFrontendAuthSnapshotFactory, n as InternalAuthBootstrapFactory, r as FrontendAppConfigFactory, t as PublicFrontendBootstrapFactory } from "./PublicFrontendBootstrapFactory-
|
|
6
|
+
import "./CodemationConfigNormalizer-BWBp7mFB.js";
|
|
7
|
+
import { i as CodemationConsumerAppResolver, n as CodemationConsumerConfigResolution, r as CodemationConsumerApp, t as CodemationConsumerConfigLoader } from "./CodemationConsumerConfigLoader-Bka3v6lh.js";
|
|
8
|
+
import { r as PrismaDatabaseClient } from "./AppConfigFactory-D3k-R3Ch.js";
|
|
9
|
+
import { t as CodemationPostgresPrismaClientFactory } from "./persistenceServer-BLG7_6B5.js";
|
|
10
|
+
import { a as CodemationResolvedPluginPackage, c as CodemationServerGateway, i as CodemationPluginDiscovery, l as ApiPaths, n as WorkflowModulePathFinder, o as AppConfigLoadResult, r as CodemationDiscoveredPluginPackage, s as AppConfigLoader, t as WorkflowDiscoveryPathSegmentsComputer } from "./server-B0SD6Nvk.js";
|
|
11
|
+
import { i as CodemationFrontendAuthSnapshotFactory, n as InternalAuthBootstrapFactory, r as FrontendAppConfigFactory, t as PublicFrontendBootstrapFactory } from "./PublicFrontendBootstrapFactory-DkQoSYDo.js";
|
|
12
12
|
export { ApiPaths, AppConfigLoadResult, AppConfigLoader, CodemationConsumerApp, CodemationConsumerAppResolver, CodemationConsumerConfigLoader, CodemationConsumerConfigResolution, CodemationDiscoveredPluginPackage, CodemationFrontendAuthSnapshotFactory, CodemationFrontendAuthSnapshotJsonCodec, CodemationPluginDiscovery, CodemationPostgresPrismaClientFactory, CodemationResolvedPluginPackage, CodemationServerGateway, FrontendAppConfig, FrontendAppConfigFactory, FrontendAppConfigJsonCodec, InternalAuthBootstrap, InternalAuthBootstrapFactory, InternalAuthBootstrapJsonCodec, PrismaDatabaseClient as PrismaClient, PublicFrontendBootstrap, PublicFrontendBootstrapFactory, PublicFrontendBootstrapJsonCodec, WorkflowDiscoveryPathSegmentsComputer, WorkflowModulePathFinder };
|
package/dist/server.js
CHANGED
|
@@ -5,9 +5,9 @@ import "./ServerLoggerFactory-CsNW5qhz.js";
|
|
|
5
5
|
import "./decorateParam-D7WPDFAf.js";
|
|
6
6
|
import "./decorate-Dq0XLibd.js";
|
|
7
7
|
import "./CredentialServices-BNBMFOPt.js";
|
|
8
|
-
import { _ as FrontendAppConfigFactory, b as ApiPaths, g as InternalAuthBootstrapFactory, h as PublicFrontendBootstrapFactory, v as CodemationFrontendAuthSnapshotFactory } from "./AppContainerFactory-
|
|
9
|
-
import "./AppConfigFactory-
|
|
10
|
-
import { t as CodemationPostgresPrismaClientFactory } from "./persistenceServer-
|
|
11
|
-
import { n as AppConfigLoader, r as CodemationServerGateway, t as CodemationPluginDiscovery } from "./server-
|
|
8
|
+
import { _ as FrontendAppConfigFactory, b as ApiPaths, g as InternalAuthBootstrapFactory, h as PublicFrontendBootstrapFactory, v as CodemationFrontendAuthSnapshotFactory } from "./AppContainerFactory-CKRDz8kQ.js";
|
|
9
|
+
import "./AppConfigFactory-C4OXGOs2.js";
|
|
10
|
+
import { t as CodemationPostgresPrismaClientFactory } from "./persistenceServer-KyHL0u01.js";
|
|
11
|
+
import { n as AppConfigLoader, r as CodemationServerGateway, t as CodemationPluginDiscovery } from "./server-CMUVhYIc.js";
|
|
12
12
|
|
|
13
13
|
export { ApiPaths, AppConfigLoader, CodemationConsumerAppResolver, CodemationConsumerConfigLoader, CodemationFrontendAuthSnapshotFactory, CodemationFrontendAuthSnapshotJsonCodec, CodemationPluginDiscovery, CodemationPostgresPrismaClientFactory, CodemationServerGateway, FrontendAppConfigFactory, FrontendAppConfigJsonCodec, InternalAuthBootstrapFactory, InternalAuthBootstrapJsonCodec, PublicFrontendBootstrapFactory, PublicFrontendBootstrapJsonCodec, WorkflowDiscoveryPathSegmentsComputer, WorkflowModulePathFinder };
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@codemation/host",
|
|
3
|
-
"version": "1.0.
|
|
3
|
+
"version": "1.0.2",
|
|
4
4
|
"publishConfig": {
|
|
5
5
|
"access": "public"
|
|
6
6
|
},
|
|
@@ -121,9 +121,9 @@
|
|
|
121
121
|
"tsx": "^4.21.0",
|
|
122
122
|
"ws": "^8.19.0",
|
|
123
123
|
"zxcvbn": "^4.4.2",
|
|
124
|
-
"@codemation/core": "1.0.
|
|
125
|
-
"@codemation/eventbus-redis": "0.0.
|
|
126
|
-
"@codemation/core-nodes": "1.0.
|
|
124
|
+
"@codemation/core": "1.0.1",
|
|
125
|
+
"@codemation/eventbus-redis": "0.0.33",
|
|
126
|
+
"@codemation/core-nodes": "1.0.2"
|
|
127
127
|
},
|
|
128
128
|
"devDependencies": {
|
|
129
129
|
"@playwright/test": "^1.58.2",
|
|
@@ -149,7 +149,7 @@
|
|
|
149
149
|
"typescript": "^5.9.3",
|
|
150
150
|
"vitest": "^4.0.18",
|
|
151
151
|
"zod": "^4.3.6",
|
|
152
|
-
"@codemation/core-nodes-gmail": "0.1.
|
|
152
|
+
"@codemation/core-nodes-gmail": "0.1.7"
|
|
153
153
|
},
|
|
154
154
|
"scripts": {
|
|
155
155
|
"changeset:verify": "pnpm --workspace-root run changeset:verify",
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
-- Add per-item iteration identity columns to telemetry tables.
|
|
2
|
+
-- See docs in @codemation/host telemetry attribute names: codemation.iteration.id,
|
|
3
|
+
-- codemation.iteration.index, codemation.parent.invocation_id.
|
|
4
|
+
|
|
5
|
+
ALTER TABLE "TelemetrySpan"
|
|
6
|
+
ADD COLUMN "iteration_id" TEXT,
|
|
7
|
+
ADD COLUMN "item_index" INTEGER,
|
|
8
|
+
ADD COLUMN "parent_invocation_id" TEXT;
|
|
9
|
+
|
|
10
|
+
CREATE INDEX "TelemetrySpan_iteration_id_idx" ON "TelemetrySpan"("iteration_id");
|
|
11
|
+
|
|
12
|
+
ALTER TABLE "TelemetryMetricPoint"
|
|
13
|
+
ADD COLUMN "iteration_id" TEXT,
|
|
14
|
+
ADD COLUMN "item_index" INTEGER,
|
|
15
|
+
ADD COLUMN "parent_invocation_id" TEXT;
|
|
16
|
+
|
|
17
|
+
CREATE INDEX "TelemetryMetricPoint_iteration_id_idx" ON "TelemetryMetricPoint"("iteration_id");
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
-- Add per-item iteration identity columns to ExecutionInstance (Postgres).
|
|
2
|
+
-- These are required so connection invocations restored from disk preserve the
|
|
3
|
+
-- per-item identity that the engine stamps on them at runtime; the bottom
|
|
4
|
+
-- execution tree groups invocations by iterationId to render synthetic
|
|
5
|
+
-- "Item N" parent rows when an agent processed 2+ items.
|
|
6
|
+
ALTER TABLE "ExecutionInstance"
|
|
7
|
+
ADD COLUMN "iteration_id" TEXT,
|
|
8
|
+
ADD COLUMN "item_index" INTEGER,
|
|
9
|
+
ADD COLUMN "parent_invocation_id" TEXT;
|
|
10
|
+
|
|
11
|
+
CREATE INDEX "ExecutionInstance_iteration_id_idx" ON "ExecutionInstance"("iteration_id");
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
-- Add per-item iteration identity columns to telemetry tables (SQLite).
|
|
2
|
+
-- See @codemation/host telemetry attribute names: codemation.iteration.id,
|
|
3
|
+
-- codemation.iteration.index, codemation.parent.invocation_id.
|
|
4
|
+
ALTER TABLE "TelemetrySpan" ADD COLUMN "iteration_id" TEXT;
|
|
5
|
+
ALTER TABLE "TelemetrySpan" ADD COLUMN "item_index" INTEGER;
|
|
6
|
+
ALTER TABLE "TelemetrySpan" ADD COLUMN "parent_invocation_id" TEXT;
|
|
7
|
+
|
|
8
|
+
CREATE INDEX "TelemetrySpan_iteration_id_idx" ON "TelemetrySpan"("iteration_id");
|
|
9
|
+
|
|
10
|
+
ALTER TABLE "TelemetryMetricPoint" ADD COLUMN "iteration_id" TEXT;
|
|
11
|
+
ALTER TABLE "TelemetryMetricPoint" ADD COLUMN "item_index" INTEGER;
|
|
12
|
+
ALTER TABLE "TelemetryMetricPoint" ADD COLUMN "parent_invocation_id" TEXT;
|
|
13
|
+
|
|
14
|
+
CREATE INDEX "TelemetryMetricPoint_iteration_id_idx" ON "TelemetryMetricPoint"("iteration_id");
|
package/prisma/migrations.sqlite/20260430130000_execution_instance_iteration_identity/migration.sql
ADDED
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
-- Add per-item iteration identity columns to ExecutionInstance (SQLite).
|
|
2
|
+
-- These are required so connection invocations restored from disk preserve the
|
|
3
|
+
-- per-item identity that the engine stamps on them at runtime; the bottom
|
|
4
|
+
-- execution tree groups invocations by iterationId to render synthetic
|
|
5
|
+
-- "Item N" parent rows when an agent processed 2+ items.
|
|
6
|
+
ALTER TABLE "ExecutionInstance" ADD COLUMN "iteration_id" TEXT;
|
|
7
|
+
ALTER TABLE "ExecutionInstance" ADD COLUMN "item_index" INTEGER;
|
|
8
|
+
ALTER TABLE "ExecutionInstance" ADD COLUMN "parent_invocation_id" TEXT;
|
|
9
|
+
|
|
10
|
+
CREATE INDEX "ExecutionInstance_iteration_id_idx" ON "ExecutionInstance"("iteration_id");
|
|
@@ -95,12 +95,16 @@ model ExecutionInstance {
|
|
|
95
95
|
inputTruncated Boolean? @map("input_truncated")
|
|
96
96
|
outputTruncated Boolean? @map("output_truncated")
|
|
97
97
|
usedPinnedOutput Boolean? @map("used_pinned_output")
|
|
98
|
+
iterationId String? @map("iteration_id")
|
|
99
|
+
itemIndex Int? @map("item_index")
|
|
100
|
+
parentInvocationId String? @map("parent_invocation_id")
|
|
98
101
|
|
|
99
102
|
run Run @relation(fields: [runId], references: [runId], onDelete: Cascade)
|
|
100
103
|
|
|
101
104
|
@@index([runId, slotNodeId, updatedAt])
|
|
102
105
|
@@index([runId, parentInstanceId, updatedAt])
|
|
103
106
|
@@index([runId, kind, updatedAt])
|
|
107
|
+
@@index([iterationId])
|
|
104
108
|
@@unique([runId, slotNodeId, runIndex])
|
|
105
109
|
}
|
|
106
110
|
|
|
@@ -173,6 +177,9 @@ model TelemetrySpan {
|
|
|
173
177
|
attributesJson String? @map("attributes_json")
|
|
174
178
|
eventsJson String? @map("events_json")
|
|
175
179
|
retentionExpiresAt String? @map("retention_expires_at")
|
|
180
|
+
iterationId String? @map("iteration_id")
|
|
181
|
+
itemIndex Int? @map("item_index")
|
|
182
|
+
parentInvocationId String? @map("parent_invocation_id")
|
|
176
183
|
updatedAt String @map("updated_at")
|
|
177
184
|
|
|
178
185
|
@@unique([traceId, spanId])
|
|
@@ -182,6 +189,7 @@ model TelemetrySpan {
|
|
|
182
189
|
@@index([runId, endTime])
|
|
183
190
|
@@index([modelName, endTime])
|
|
184
191
|
@@index([connectionInvocationId])
|
|
192
|
+
@@index([iterationId])
|
|
185
193
|
@@index([retentionExpiresAt])
|
|
186
194
|
}
|
|
187
195
|
|
|
@@ -229,12 +237,16 @@ model TelemetryMetricPoint {
|
|
|
229
237
|
modelName String? @map("model_name")
|
|
230
238
|
dimensionsJson String? @map("dimensions_json")
|
|
231
239
|
retentionExpiresAt String? @map("retention_expires_at")
|
|
240
|
+
iterationId String? @map("iteration_id")
|
|
241
|
+
itemIndex Int? @map("item_index")
|
|
242
|
+
parentInvocationId String? @map("parent_invocation_id")
|
|
232
243
|
|
|
233
244
|
@@index([workflowId, observedAt])
|
|
234
245
|
@@index([workflowId, metricName, observedAt])
|
|
235
246
|
@@index([runId, observedAt])
|
|
236
247
|
@@index([traceId, observedAt])
|
|
237
248
|
@@index([modelName, observedAt])
|
|
249
|
+
@@index([iterationId])
|
|
238
250
|
@@index([retentionExpiresAt])
|
|
239
251
|
}
|
|
240
252
|
|
|
@@ -96,12 +96,16 @@ model ExecutionInstance {
|
|
|
96
96
|
inputTruncated Boolean? @map("input_truncated")
|
|
97
97
|
outputTruncated Boolean? @map("output_truncated")
|
|
98
98
|
usedPinnedOutput Boolean? @map("used_pinned_output")
|
|
99
|
+
iterationId String? @map("iteration_id")
|
|
100
|
+
itemIndex Int? @map("item_index")
|
|
101
|
+
parentInvocationId String? @map("parent_invocation_id")
|
|
99
102
|
|
|
100
103
|
run Run @relation(fields: [runId], references: [runId], onDelete: Cascade)
|
|
101
104
|
|
|
102
105
|
@@index([runId, slotNodeId, updatedAt])
|
|
103
106
|
@@index([runId, parentInstanceId, updatedAt])
|
|
104
107
|
@@index([runId, kind, updatedAt])
|
|
108
|
+
@@index([iterationId])
|
|
105
109
|
@@unique([runId, slotNodeId, runIndex])
|
|
106
110
|
}
|
|
107
111
|
|
|
@@ -174,6 +178,9 @@ model TelemetrySpan {
|
|
|
174
178
|
attributesJson String? @map("attributes_json")
|
|
175
179
|
eventsJson String? @map("events_json")
|
|
176
180
|
retentionExpiresAt String? @map("retention_expires_at")
|
|
181
|
+
iterationId String? @map("iteration_id")
|
|
182
|
+
itemIndex Int? @map("item_index")
|
|
183
|
+
parentInvocationId String? @map("parent_invocation_id")
|
|
177
184
|
updatedAt String @map("updated_at")
|
|
178
185
|
|
|
179
186
|
@@unique([traceId, spanId])
|
|
@@ -183,6 +190,7 @@ model TelemetrySpan {
|
|
|
183
190
|
@@index([runId, endTime])
|
|
184
191
|
@@index([modelName, endTime])
|
|
185
192
|
@@index([connectionInvocationId])
|
|
193
|
+
@@index([iterationId])
|
|
186
194
|
@@index([retentionExpiresAt])
|
|
187
195
|
}
|
|
188
196
|
|
|
@@ -230,12 +238,16 @@ model TelemetryMetricPoint {
|
|
|
230
238
|
modelName String? @map("model_name")
|
|
231
239
|
dimensionsJson String? @map("dimensions_json")
|
|
232
240
|
retentionExpiresAt String? @map("retention_expires_at")
|
|
241
|
+
iterationId String? @map("iteration_id")
|
|
242
|
+
itemIndex Int? @map("item_index")
|
|
243
|
+
parentInvocationId String? @map("parent_invocation_id")
|
|
233
244
|
|
|
234
245
|
@@index([workflowId, observedAt])
|
|
235
246
|
@@index([workflowId, metricName, observedAt])
|
|
236
247
|
@@index([runId, observedAt])
|
|
237
248
|
@@index([traceId, observedAt])
|
|
238
249
|
@@index([modelName, observedAt])
|
|
250
|
+
@@index([iterationId])
|
|
239
251
|
@@index([retentionExpiresAt])
|
|
240
252
|
}
|
|
241
253
|
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Per-iteration cost rollup, derived from `codemation.cost.estimated` metric points
|
|
3
|
+
* grouped by `iterationId` and currency.
|
|
4
|
+
*/
|
|
5
|
+
export interface IterationCostRollupDto {
|
|
6
|
+
readonly iterationId: string;
|
|
7
|
+
/** Sum of cost in minor units (per `cost.currency_scale`) keyed by ISO currency code. */
|
|
8
|
+
readonly estimatedCostMinorByCurrency: Readonly<Record<string, number>>;
|
|
9
|
+
/** Currency scale (denominator) per currency, when present on the metric points. */
|
|
10
|
+
readonly estimatedCostCurrencyScaleByCurrency: Readonly<Record<string, number>>;
|
|
11
|
+
}
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
import { Query } from "../bus/Query";
|
|
2
|
+
import type { IterationCostRollupDto } from "../contracts/IterationCostContracts";
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Query that returns a per-iteration cost rollup for a single run.
|
|
6
|
+
*
|
|
7
|
+
* The rollup is keyed by `iterationId` and contains the sum of
|
|
8
|
+
* `codemation.cost.estimated` metric points grouped by ISO currency code.
|
|
9
|
+
*/
|
|
10
|
+
export class GetIterationCostQuery extends Query<ReadonlyArray<IterationCostRollupDto>> {
|
|
11
|
+
constructor(public readonly runId: string) {
|
|
12
|
+
super();
|
|
13
|
+
}
|
|
14
|
+
}
|
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
import { CostTrackingTelemetryAttributeNames, CostTrackingTelemetryMetricNames, inject } from "@codemation/core";
|
|
2
|
+
import { ApplicationTokens } from "../../applicationTokens";
|
|
3
|
+
import type { TelemetryMetricPointRecord, TelemetryMetricPointStore } from "../../domain/telemetry/TelemetryContracts";
|
|
4
|
+
import { HandlesQuery } from "../../infrastructure/di/HandlesQueryRegistry";
|
|
5
|
+
import { QueryHandler } from "../bus/QueryHandler";
|
|
6
|
+
import type { IterationCostRollupDto } from "../contracts/IterationCostContracts";
|
|
7
|
+
import { GetIterationCostQuery } from "./GetIterationCostQuery";
|
|
8
|
+
|
|
9
|
+
interface MutableCostBucket {
|
|
10
|
+
readonly iterationId: string;
|
|
11
|
+
readonly estimatedCostMinorByCurrency: Record<string, number>;
|
|
12
|
+
readonly estimatedCostCurrencyScaleByCurrency: Record<string, number>;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
@HandlesQuery.for(GetIterationCostQuery)
|
|
16
|
+
export class GetIterationCostQueryHandler extends QueryHandler<
|
|
17
|
+
GetIterationCostQuery,
|
|
18
|
+
ReadonlyArray<IterationCostRollupDto>
|
|
19
|
+
> {
|
|
20
|
+
constructor(
|
|
21
|
+
@inject(ApplicationTokens.TelemetryMetricPointStore)
|
|
22
|
+
private readonly metricPointStore: TelemetryMetricPointStore,
|
|
23
|
+
) {
|
|
24
|
+
super();
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
async execute(query: GetIterationCostQuery): Promise<ReadonlyArray<IterationCostRollupDto>> {
|
|
28
|
+
const points = await this.metricPointStore.list({
|
|
29
|
+
runId: query.runId,
|
|
30
|
+
metricNames: [CostTrackingTelemetryMetricNames.estimatedCost],
|
|
31
|
+
});
|
|
32
|
+
if (points.length === 0) {
|
|
33
|
+
return [];
|
|
34
|
+
}
|
|
35
|
+
const buckets = new Map<string, MutableCostBucket>();
|
|
36
|
+
for (const point of points) {
|
|
37
|
+
this.accumulate(point, buckets);
|
|
38
|
+
}
|
|
39
|
+
return [...buckets.values()].map((bucket) => ({
|
|
40
|
+
iterationId: bucket.iterationId,
|
|
41
|
+
estimatedCostMinorByCurrency: { ...bucket.estimatedCostMinorByCurrency },
|
|
42
|
+
estimatedCostCurrencyScaleByCurrency: { ...bucket.estimatedCostCurrencyScaleByCurrency },
|
|
43
|
+
}));
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
private accumulate(point: TelemetryMetricPointRecord, buckets: Map<string, MutableCostBucket>): void {
|
|
47
|
+
const iterationId = point.iterationId;
|
|
48
|
+
if (!iterationId || iterationId.length === 0) {
|
|
49
|
+
return;
|
|
50
|
+
}
|
|
51
|
+
const currency = this.readCurrency(point);
|
|
52
|
+
if (!currency) {
|
|
53
|
+
return;
|
|
54
|
+
}
|
|
55
|
+
const currencyScale = this.readCurrencyScale(point);
|
|
56
|
+
const bucket = this.bucketFor(iterationId, buckets);
|
|
57
|
+
bucket.estimatedCostMinorByCurrency[currency] = (bucket.estimatedCostMinorByCurrency[currency] ?? 0) + point.value;
|
|
58
|
+
if (typeof currencyScale === "number") {
|
|
59
|
+
bucket.estimatedCostCurrencyScaleByCurrency[currency] = currencyScale;
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
private bucketFor(iterationId: string, buckets: Map<string, MutableCostBucket>): MutableCostBucket {
|
|
64
|
+
const existing = buckets.get(iterationId);
|
|
65
|
+
if (existing) {
|
|
66
|
+
return existing;
|
|
67
|
+
}
|
|
68
|
+
const bucket: MutableCostBucket = {
|
|
69
|
+
iterationId,
|
|
70
|
+
estimatedCostMinorByCurrency: {},
|
|
71
|
+
estimatedCostCurrencyScaleByCurrency: {},
|
|
72
|
+
};
|
|
73
|
+
buckets.set(iterationId, bucket);
|
|
74
|
+
return bucket;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
private readCurrency(point: TelemetryMetricPointRecord): string | undefined {
|
|
78
|
+
const fromAttribute = point.dimensions?.[CostTrackingTelemetryAttributeNames.currency];
|
|
79
|
+
if (typeof fromAttribute === "string" && fromAttribute.length > 0) {
|
|
80
|
+
return fromAttribute;
|
|
81
|
+
}
|
|
82
|
+
if (typeof point.unit === "string" && point.unit.length > 0) {
|
|
83
|
+
return point.unit;
|
|
84
|
+
}
|
|
85
|
+
return undefined;
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
private readCurrencyScale(point: TelemetryMetricPointRecord): number | undefined {
|
|
89
|
+
const value = point.dimensions?.[CostTrackingTelemetryAttributeNames.currencyScale];
|
|
90
|
+
return typeof value === "number" && Number.isFinite(value) ? value : undefined;
|
|
91
|
+
}
|
|
92
|
+
}
|
|
@@ -1,10 +1,14 @@
|
|
|
1
|
-
import type { WorkflowRunDetailDto } from "@codemation/core";
|
|
1
|
+
import type { RunIterationDto, WorkflowRunDetailDto } from "@codemation/core";
|
|
2
2
|
import { inject } from "@codemation/core";
|
|
3
3
|
import { ApplicationTokens } from "../../applicationTokens";
|
|
4
4
|
import type { WorkflowRunRepository } from "../../domain/runs/WorkflowRunRepository";
|
|
5
5
|
import { HandlesQuery } from "../../infrastructure/di/HandlesQueryRegistry";
|
|
6
6
|
import { QueryHandler } from "../bus/QueryHandler";
|
|
7
|
+
import type { IterationCostRollupDto } from "../contracts/IterationCostContracts";
|
|
8
|
+
import { GetIterationCostQuery } from "./GetIterationCostQuery";
|
|
9
|
+
import { GetIterationCostQueryHandler } from "./GetIterationCostQueryHandler";
|
|
7
10
|
import { GetWorkflowRunDetailQuery } from "./GetWorkflowRunDetailQuery";
|
|
11
|
+
import { RunIterationProjectionFactory } from "./RunIterationProjectionFactory";
|
|
8
12
|
|
|
9
13
|
@HandlesQuery.for(GetWorkflowRunDetailQuery)
|
|
10
14
|
export class GetWorkflowRunDetailQueryHandler extends QueryHandler<
|
|
@@ -14,11 +18,49 @@ export class GetWorkflowRunDetailQueryHandler extends QueryHandler<
|
|
|
14
18
|
constructor(
|
|
15
19
|
@inject(ApplicationTokens.WorkflowRunRepository)
|
|
16
20
|
private readonly workflowRunRepository: WorkflowRunRepository,
|
|
21
|
+
@inject(RunIterationProjectionFactory)
|
|
22
|
+
private readonly runIterationProjectionFactory: RunIterationProjectionFactory,
|
|
23
|
+
@inject(GetIterationCostQueryHandler)
|
|
24
|
+
private readonly getIterationCostQueryHandler: GetIterationCostQueryHandler,
|
|
17
25
|
) {
|
|
18
26
|
super();
|
|
19
27
|
}
|
|
20
28
|
|
|
21
29
|
async execute(query: GetWorkflowRunDetailQuery): Promise<WorkflowRunDetailDto | undefined> {
|
|
22
|
-
|
|
30
|
+
const detail = await this.workflowRunRepository.loadRunDetail?.(query.runId);
|
|
31
|
+
if (!detail) {
|
|
32
|
+
return undefined;
|
|
33
|
+
}
|
|
34
|
+
const baseIterations = this.runIterationProjectionFactory.project(detail.executionInstances);
|
|
35
|
+
const iterations = await this.joinIterationCosts(query.runId, baseIterations);
|
|
36
|
+
return { ...detail, iterations };
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
private async joinIterationCosts(
|
|
40
|
+
runId: string,
|
|
41
|
+
iterations: ReadonlyArray<RunIterationDto>,
|
|
42
|
+
): Promise<ReadonlyArray<RunIterationDto>> {
|
|
43
|
+
if (iterations.length === 0) {
|
|
44
|
+
return iterations;
|
|
45
|
+
}
|
|
46
|
+
const rollups = await this.getIterationCostQueryHandler.execute(new GetIterationCostQuery(runId));
|
|
47
|
+
if (rollups.length === 0) {
|
|
48
|
+
return iterations;
|
|
49
|
+
}
|
|
50
|
+
const rollupsByIterationId = new Map<string, IterationCostRollupDto>();
|
|
51
|
+
for (const rollup of rollups) {
|
|
52
|
+
rollupsByIterationId.set(rollup.iterationId, rollup);
|
|
53
|
+
}
|
|
54
|
+
return iterations.map((iteration) => {
|
|
55
|
+
const rollup = rollupsByIterationId.get(iteration.iterationId);
|
|
56
|
+
if (!rollup) {
|
|
57
|
+
return iteration;
|
|
58
|
+
}
|
|
59
|
+
return {
|
|
60
|
+
...iteration,
|
|
61
|
+
estimatedCostMinorByCurrency: rollup.estimatedCostMinorByCurrency,
|
|
62
|
+
estimatedCostCurrencyScaleByCurrency: rollup.estimatedCostCurrencyScaleByCurrency,
|
|
63
|
+
};
|
|
64
|
+
});
|
|
23
65
|
}
|
|
24
66
|
}
|
|
@@ -0,0 +1,123 @@
|
|
|
1
|
+
import {
|
|
2
|
+
injectable,
|
|
3
|
+
type ExecutionInstanceDto,
|
|
4
|
+
type NodeExecutionStatus,
|
|
5
|
+
type RunIterationDto,
|
|
6
|
+
} from "@codemation/core";
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Builds the per-iteration projection from a run's connection invocations.
|
|
10
|
+
*
|
|
11
|
+
* One iteration represents a single item being processed by an agent within an activation. All
|
|
12
|
+
* invocations (LLM rounds, tool calls) emitted while handling that item share the same iterationId
|
|
13
|
+
* and project into one {@link RunIterationDto}.
|
|
14
|
+
*
|
|
15
|
+
* Old runs (persisted before iteration ids existed) fall back to grouping by the agent
|
|
16
|
+
* activationId so the UI still sees coherent groups instead of a flat list.
|
|
17
|
+
*/
|
|
18
|
+
@injectable()
|
|
19
|
+
export class RunIterationProjectionFactory {
|
|
20
|
+
project(executionInstances: ReadonlyArray<ExecutionInstanceDto>): ReadonlyArray<RunIterationDto> {
|
|
21
|
+
const invocations = executionInstances.filter((row) => row.kind === "connectionInvocation");
|
|
22
|
+
if (invocations.length === 0) {
|
|
23
|
+
return [];
|
|
24
|
+
}
|
|
25
|
+
const grouped = new Map<string, ExecutionInstanceDto[]>();
|
|
26
|
+
for (const invocation of invocations) {
|
|
27
|
+
const key = this.iterationKey(invocation);
|
|
28
|
+
if (!key) {
|
|
29
|
+
continue;
|
|
30
|
+
}
|
|
31
|
+
const bucket = grouped.get(key);
|
|
32
|
+
if (bucket) {
|
|
33
|
+
bucket.push(invocation);
|
|
34
|
+
} else {
|
|
35
|
+
grouped.set(key, [invocation]);
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
const iterations: RunIterationDto[] = [];
|
|
39
|
+
for (const [key, group] of grouped.entries()) {
|
|
40
|
+
iterations.push(this.toIteration(key, group));
|
|
41
|
+
}
|
|
42
|
+
iterations.sort((left, right) => this.compareIterations(left, right));
|
|
43
|
+
return iterations;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
private iterationKey(invocation: ExecutionInstanceDto): string | undefined {
|
|
47
|
+
if (invocation.iterationId) {
|
|
48
|
+
return invocation.iterationId;
|
|
49
|
+
}
|
|
50
|
+
if (invocation.activationId) {
|
|
51
|
+
return `legacy::${invocation.workflowNodeId}::${invocation.activationId}::${invocation.itemIndex ?? 0}`;
|
|
52
|
+
}
|
|
53
|
+
return undefined;
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
private toIteration(iterationKey: string, group: ReadonlyArray<ExecutionInstanceDto>): RunIterationDto {
|
|
57
|
+
const sorted = [...group].sort((a, b) => this.compareInvocations(a, b));
|
|
58
|
+
const first = sorted[0]!;
|
|
59
|
+
const status = this.aggregateStatus(sorted);
|
|
60
|
+
const iterationId = first.iterationId ?? iterationKey;
|
|
61
|
+
return {
|
|
62
|
+
iterationId,
|
|
63
|
+
agentNodeId: first.workflowNodeId,
|
|
64
|
+
activationId: first.activationId ?? "synthetic",
|
|
65
|
+
itemIndex: first.itemIndex ?? 0,
|
|
66
|
+
status,
|
|
67
|
+
startedAt: this.minIso(sorted.map((row) => row.startedAt)),
|
|
68
|
+
finishedAt: status === "running" ? undefined : this.maxIso(sorted.map((row) => row.finishedAt)),
|
|
69
|
+
invocationIds: sorted.map((row) => row.instanceId),
|
|
70
|
+
parentInvocationId: first.parentInvocationId,
|
|
71
|
+
};
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
private aggregateStatus(group: ReadonlyArray<ExecutionInstanceDto>): NodeExecutionStatus {
|
|
75
|
+
if (group.some((row) => row.status === "failed")) {
|
|
76
|
+
return "failed";
|
|
77
|
+
}
|
|
78
|
+
if (group.every((row) => row.status === "completed")) {
|
|
79
|
+
return "completed";
|
|
80
|
+
}
|
|
81
|
+
return "running";
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
private minIso(values: ReadonlyArray<string | undefined>): string | undefined {
|
|
85
|
+
let min: string | undefined;
|
|
86
|
+
for (const value of values) {
|
|
87
|
+
if (!value) continue;
|
|
88
|
+
if (!min || value < min) {
|
|
89
|
+
min = value;
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
return min;
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
private maxIso(values: ReadonlyArray<string | undefined>): string | undefined {
|
|
96
|
+
let max: string | undefined;
|
|
97
|
+
for (const value of values) {
|
|
98
|
+
if (!value) continue;
|
|
99
|
+
if (!max || value > max) {
|
|
100
|
+
max = value;
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
return max;
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
private compareInvocations(left: ExecutionInstanceDto, right: ExecutionInstanceDto): number {
|
|
107
|
+
const leftStart = left.startedAt ?? left.queuedAt ?? "";
|
|
108
|
+
const rightStart = right.startedAt ?? right.queuedAt ?? "";
|
|
109
|
+
if (leftStart !== rightStart) {
|
|
110
|
+
return leftStart.localeCompare(rightStart);
|
|
111
|
+
}
|
|
112
|
+
return left.runIndex - right.runIndex;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
private compareIterations(left: RunIterationDto, right: RunIterationDto): number {
|
|
116
|
+
if (left.itemIndex !== right.itemIndex) {
|
|
117
|
+
return left.itemIndex - right.itemIndex;
|
|
118
|
+
}
|
|
119
|
+
const leftStart = left.startedAt ?? "";
|
|
120
|
+
const rightStart = right.startedAt ?? "";
|
|
121
|
+
return leftStart.localeCompare(rightStart);
|
|
122
|
+
}
|
|
123
|
+
}
|
|
@@ -5,6 +5,7 @@ export { GetTelemetryDashboardRunsQueryHandler } from "./GetTelemetryDashboardRu
|
|
|
5
5
|
export { GetTelemetryRunTraceQueryHandler } from "./GetTelemetryRunTraceQueryHandler";
|
|
6
6
|
export { GetTelemetryDashboardSummaryQueryHandler } from "./GetTelemetryDashboardSummaryQueryHandler";
|
|
7
7
|
export { GetTelemetryDashboardTimeseriesQueryHandler } from "./GetTelemetryDashboardTimeseriesQueryHandler";
|
|
8
|
+
export { GetIterationCostQueryHandler } from "./GetIterationCostQueryHandler";
|
|
8
9
|
export { GetWorkflowRunDetailQueryHandler } from "./GetWorkflowRunDetailQueryHandler";
|
|
9
10
|
export { GetWorkflowDebuggerOverlayQueryHandler } from "./GetWorkflowDebuggerOverlayQueryHandler";
|
|
10
11
|
export { GetWorkflowDetailQueryHandler } from "./GetWorkflowDetailQueryHandler";
|