2026-04-05 21:09:33 -05:00
|
|
|
/**
|
|
|
|
|
* Runtime flow instance — created by FlowProcessor for each configured flow.
|
|
|
|
|
*
|
|
|
|
|
* Python reference: trustgraph-base/trustgraph/base/flow.py
|
|
|
|
|
*/
|
|
|
|
|
|
2026-05-12 08:06:58 -05:00
|
|
|
import { Effect, Exit, Scope } from "effect";
|
2026-04-05 21:09:33 -05:00
|
|
|
import type { PubSubBackend } from "../backend/types.js";
|
2026-05-12 08:06:58 -05:00
|
|
|
import { makePubSubService } from "../backend/pubsub.js";
|
|
|
|
|
import {
|
|
|
|
|
flowResourceNotFoundError,
|
|
|
|
|
type FlowResourceNotFoundError,
|
|
|
|
|
type PubSubError,
|
|
|
|
|
} from "../errors.js";
|
|
|
|
|
import {
|
|
|
|
|
ConsumerFactory,
|
|
|
|
|
ProducerFactory,
|
|
|
|
|
RequestResponseFactory,
|
|
|
|
|
type EffectConsumer,
|
|
|
|
|
type EffectProducer,
|
|
|
|
|
type EffectRequestOptions,
|
|
|
|
|
type EffectRequestResponse,
|
|
|
|
|
makeConsumerFactoryService,
|
|
|
|
|
makeProducerFactoryService,
|
|
|
|
|
makeRequestResponseFactoryService,
|
|
|
|
|
} from "../messaging/runtime.js";
|
|
|
|
|
import { loadMessagingRuntimeConfig } from "../runtime/messaging-config.js";
|
|
|
|
|
import type { Spec, SpecRuntimeRequirements } from "../spec/types.js";
|
2026-04-05 21:09:33 -05:00
|
|
|
|
|
|
|
|
export interface FlowDefinition {
|
|
|
|
|
/** Topic overrides keyed by spec name */
|
|
|
|
|
topics?: Record<string, string>;
|
|
|
|
|
/** Parameter values keyed by spec name */
|
|
|
|
|
parameters?: Record<string, unknown>;
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-12 08:06:58 -05:00
|
|
|
export interface FlowProducer<T> {
|
|
|
|
|
readonly send: (id: string, message: T) => Promise<void>;
|
|
|
|
|
readonly flush: () => Promise<void>;
|
|
|
|
|
readonly stop: () => Promise<void>;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export interface FlowConsumer {
|
|
|
|
|
readonly stop: () => Promise<void>;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export interface FlowRequestOptions<TRes> {
|
|
|
|
|
readonly timeoutMs?: number;
|
|
|
|
|
readonly recipient?: (response: TRes) => Promise<boolean>;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export interface FlowRequestor<TReq, TRes> {
|
|
|
|
|
readonly request: (
|
|
|
|
|
request: TReq,
|
|
|
|
|
options?: FlowRequestOptions<TRes>,
|
|
|
|
|
) => Promise<TRes>;
|
|
|
|
|
readonly stop: () => Promise<void>;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export class Flow<Requirements = never> {
|
|
|
|
|
private producers = new Map<string, EffectProducer<unknown>>();
|
|
|
|
|
private consumers = new Map<string, EffectConsumer>();
|
|
|
|
|
private requestors = new Map<string, EffectRequestResponse<unknown, unknown>>();
|
2026-04-05 21:09:33 -05:00
|
|
|
private parameters = new Map<string, unknown>();
|
2026-05-12 08:06:58 -05:00
|
|
|
private compatibilityScope: Scope.Closeable | null = null;
|
|
|
|
|
public readonly name: string;
|
|
|
|
|
public readonly processorId: string;
|
|
|
|
|
private readonly pubsub: PubSubBackend;
|
|
|
|
|
private readonly definition: FlowDefinition;
|
|
|
|
|
private readonly specifications: ReadonlyArray<Spec<Requirements>>;
|
2026-04-05 21:09:33 -05:00
|
|
|
|
|
|
|
|
constructor(
|
2026-05-12 08:06:58 -05:00
|
|
|
name: string,
|
|
|
|
|
processorId: string,
|
|
|
|
|
pubsub: PubSubBackend,
|
|
|
|
|
definition: FlowDefinition,
|
|
|
|
|
specifications: ReadonlyArray<Spec<Requirements>>,
|
|
|
|
|
) {
|
|
|
|
|
this.name = name;
|
|
|
|
|
this.processorId = processorId;
|
|
|
|
|
this.pubsub = pubsub;
|
|
|
|
|
this.definition = definition;
|
|
|
|
|
this.specifications = specifications;
|
|
|
|
|
}
|
2026-04-05 21:09:33 -05:00
|
|
|
|
2026-05-12 08:06:58 -05:00
|
|
|
startEffect(): Effect.Effect<void, PubSubError, SpecRuntimeRequirements | Requirements> {
|
|
|
|
|
const flow = this;
|
|
|
|
|
return Effect.gen(function* () {
|
|
|
|
|
for (const spec of flow.specifications) {
|
|
|
|
|
yield* spec.addEffect(flow, flow.definition);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
2026-04-05 21:09:33 -05:00
|
|
|
|
2026-05-12 08:06:58 -05:00
|
|
|
async start(): Promise<void> {
|
|
|
|
|
if (this.compatibilityScope !== null) {
|
|
|
|
|
await this.stop();
|
2026-04-05 21:09:33 -05:00
|
|
|
}
|
2026-05-12 08:06:58 -05:00
|
|
|
await this.runInCompatibilityScope(
|
|
|
|
|
this.startEffect() as Effect.Effect<void, PubSubError, SpecRuntimeRequirements>,
|
|
|
|
|
this.pubsub,
|
|
|
|
|
);
|
2026-04-05 21:09:33 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async stop(): Promise<void> {
|
2026-05-12 08:06:58 -05:00
|
|
|
const scope = this.compatibilityScope;
|
|
|
|
|
this.compatibilityScope = null;
|
|
|
|
|
if (scope !== null) {
|
|
|
|
|
await Effect.runPromise(Scope.close(scope, Exit.void));
|
feat: add schema foundation for document pipeline, agent, and deployment
Add missing topics (librarian, knowledge, collection-management, flow),
pipeline message types (TextDocument, Chunk, Triples, EntityContexts),
service message types (Librarian, Knowledge, Collection, Flow CRUD),
and update AgentResponse for streaming chunk format.
Add RequestResponseSpec enabling flow-scoped request/response calls
(needed by knowledge extraction and agent services). Add requestor
registry to Flow class with proper lifecycle management.
Add end_of_dialog to gateway's isComplete() check for agent streaming.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-06 00:11:29 -05:00
|
|
|
}
|
2026-05-12 08:06:58 -05:00
|
|
|
this.clearResources();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async runInCompatibilityScope<A, E>(
|
|
|
|
|
effect: Effect.Effect<A, E, SpecRuntimeRequirements>,
|
|
|
|
|
pubsub: PubSubBackend,
|
|
|
|
|
): Promise<A> {
|
|
|
|
|
const scope = await this.ensureCompatibilityScope();
|
|
|
|
|
const pubsubService = makePubSubService(pubsub);
|
|
|
|
|
const messagingConfig = await Effect.runPromise(loadMessagingRuntimeConfig());
|
|
|
|
|
return await Effect.runPromise(
|
|
|
|
|
effect.pipe(
|
|
|
|
|
Effect.provideService(ProducerFactory, ProducerFactory.of(makeProducerFactoryService(pubsubService))),
|
|
|
|
|
Effect.provideService(ConsumerFactory, ConsumerFactory.of(makeConsumerFactoryService(pubsubService, messagingConfig))),
|
|
|
|
|
Effect.provideService(
|
|
|
|
|
RequestResponseFactory,
|
|
|
|
|
RequestResponseFactory.of(makeRequestResponseFactoryService(pubsubService, messagingConfig)),
|
|
|
|
|
),
|
|
|
|
|
Scope.provide(scope),
|
|
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
clearResources(): void {
|
|
|
|
|
this.producers.clear();
|
|
|
|
|
this.consumers.clear();
|
|
|
|
|
this.requestors.clear();
|
|
|
|
|
this.parameters.clear();
|
2026-04-05 21:09:33 -05:00
|
|
|
}
|
|
|
|
|
|
2026-05-12 08:06:58 -05:00
|
|
|
registerProducer(name: string, producer: EffectProducer<unknown>): void {
|
2026-04-05 21:09:33 -05:00
|
|
|
this.producers.set(name, producer);
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-12 08:06:58 -05:00
|
|
|
registerConsumer(name: string, consumer: EffectConsumer): void {
|
2026-04-05 21:09:33 -05:00
|
|
|
this.consumers.set(name, consumer);
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-12 08:06:58 -05:00
|
|
|
registerRequestor(name: string, rr: EffectRequestResponse<unknown, unknown>): void {
|
feat: add schema foundation for document pipeline, agent, and deployment
Add missing topics (librarian, knowledge, collection-management, flow),
pipeline message types (TextDocument, Chunk, Triples, EntityContexts),
service message types (Librarian, Knowledge, Collection, Flow CRUD),
and update AgentResponse for streaming chunk format.
Add RequestResponseSpec enabling flow-scoped request/response calls
(needed by knowledge extraction and agent services). Add requestor
registry to Flow class with proper lifecycle management.
Add end_of_dialog to gateway's isComplete() check for agent streaming.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-06 00:11:29 -05:00
|
|
|
this.requestors.set(name, rr);
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-05 21:09:33 -05:00
|
|
|
setParameter(name: string, value: unknown): void {
|
|
|
|
|
this.parameters.set(name, value);
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-12 08:06:58 -05:00
|
|
|
producerEffect<T>(name: string): Effect.Effect<EffectProducer<T>, FlowResourceNotFoundError> {
|
2026-04-05 21:09:33 -05:00
|
|
|
const p = this.producers.get(name);
|
2026-05-12 08:06:58 -05:00
|
|
|
return p === undefined
|
|
|
|
|
? Effect.fail(flowResourceNotFoundError(this.name, "producer", name))
|
|
|
|
|
: Effect.succeed(p as EffectProducer<T>);
|
2026-04-05 21:09:33 -05:00
|
|
|
}
|
|
|
|
|
|
2026-05-12 08:06:58 -05:00
|
|
|
consumerEffect(name: string): Effect.Effect<EffectConsumer, FlowResourceNotFoundError> {
|
2026-04-05 21:09:33 -05:00
|
|
|
const c = this.consumers.get(name);
|
2026-05-12 08:06:58 -05:00
|
|
|
return c === undefined
|
|
|
|
|
? Effect.fail(flowResourceNotFoundError(this.name, "consumer", name))
|
|
|
|
|
: Effect.succeed(c);
|
2026-04-05 21:09:33 -05:00
|
|
|
}
|
|
|
|
|
|
2026-05-12 08:06:58 -05:00
|
|
|
requestorEffect<TReq, TRes>(
|
|
|
|
|
name: string,
|
|
|
|
|
): Effect.Effect<EffectRequestResponse<TReq, TRes>, FlowResourceNotFoundError> {
|
feat: add schema foundation for document pipeline, agent, and deployment
Add missing topics (librarian, knowledge, collection-management, flow),
pipeline message types (TextDocument, Chunk, Triples, EntityContexts),
service message types (Librarian, Knowledge, Collection, Flow CRUD),
and update AgentResponse for streaming chunk format.
Add RequestResponseSpec enabling flow-scoped request/response calls
(needed by knowledge extraction and agent services). Add requestor
registry to Flow class with proper lifecycle management.
Add end_of_dialog to gateway's isComplete() check for agent streaming.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-06 00:11:29 -05:00
|
|
|
const rr = this.requestors.get(name);
|
2026-05-12 08:06:58 -05:00
|
|
|
return rr === undefined
|
|
|
|
|
? Effect.fail(flowResourceNotFoundError(this.name, "requestor", name))
|
|
|
|
|
: Effect.succeed(rr as EffectRequestResponse<TReq, TRes>);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
parameterEffect<T>(name: string): Effect.Effect<T, FlowResourceNotFoundError> {
|
|
|
|
|
const v = this.parameters.get(name);
|
|
|
|
|
return v === undefined
|
|
|
|
|
? Effect.fail(flowResourceNotFoundError(this.name, "parameter", name))
|
|
|
|
|
: Effect.succeed(v as T);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
producer<T>(name: string): FlowProducer<T> {
|
|
|
|
|
const p = this.producers.get(name);
|
|
|
|
|
if (p === undefined) throw flowResourceNotFoundError(this.name, "producer", name);
|
|
|
|
|
return {
|
|
|
|
|
send: (id, message) => Effect.runPromise((p as EffectProducer<T>).send(id, message)),
|
|
|
|
|
flush: () => Effect.runPromise(p.flush),
|
|
|
|
|
stop: () => Effect.runPromise(p.flush.pipe(Effect.flatMap(() => p.close))),
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
consumer(name: string): FlowConsumer {
|
|
|
|
|
const c = this.consumers.get(name);
|
|
|
|
|
if (c === undefined) throw flowResourceNotFoundError(this.name, "consumer", name);
|
|
|
|
|
return {
|
|
|
|
|
stop: () => Effect.runPromise(c.stop),
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
requestor<TReq, TRes>(name: string): FlowRequestor<TReq, TRes> {
|
|
|
|
|
const rr = this.requestors.get(name);
|
|
|
|
|
if (rr === undefined) throw flowResourceNotFoundError(this.name, "requestor", name);
|
|
|
|
|
return {
|
|
|
|
|
request: (request, options) =>
|
|
|
|
|
Effect.runPromise(
|
|
|
|
|
(rr as EffectRequestResponse<TReq, TRes>).request(
|
|
|
|
|
request,
|
|
|
|
|
this.toEffectRequestOptions(options),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
stop: () => Effect.runPromise(rr.stop),
|
|
|
|
|
};
|
feat: add schema foundation for document pipeline, agent, and deployment
Add missing topics (librarian, knowledge, collection-management, flow),
pipeline message types (TextDocument, Chunk, Triples, EntityContexts),
service message types (Librarian, Knowledge, Collection, Flow CRUD),
and update AgentResponse for streaming chunk format.
Add RequestResponseSpec enabling flow-scoped request/response calls
(needed by knowledge extraction and agent services). Add requestor
registry to Flow class with proper lifecycle management.
Add end_of_dialog to gateway's isComplete() check for agent streaming.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-06 00:11:29 -05:00
|
|
|
}
|
|
|
|
|
|
2026-04-05 21:09:33 -05:00
|
|
|
parameter<T>(name: string): T {
|
|
|
|
|
const v = this.parameters.get(name);
|
2026-05-12 08:06:58 -05:00
|
|
|
if (v === undefined) throw flowResourceNotFoundError(this.name, "parameter", name);
|
2026-04-05 21:09:33 -05:00
|
|
|
return v as T;
|
|
|
|
|
}
|
2026-05-12 08:06:58 -05:00
|
|
|
|
|
|
|
|
private async ensureCompatibilityScope(): Promise<Scope.Closeable> {
|
|
|
|
|
if (this.compatibilityScope !== null) {
|
|
|
|
|
return this.compatibilityScope;
|
|
|
|
|
}
|
|
|
|
|
this.compatibilityScope = await Effect.runPromise(Scope.make());
|
|
|
|
|
return this.compatibilityScope;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private toEffectRequestOptions<TRes>(
|
|
|
|
|
options: FlowRequestOptions<TRes> | undefined,
|
|
|
|
|
): EffectRequestOptions<TRes> | undefined {
|
|
|
|
|
if (options === undefined) {
|
|
|
|
|
return undefined;
|
|
|
|
|
}
|
|
|
|
|
const recipient = options.recipient;
|
|
|
|
|
return {
|
|
|
|
|
...(options.timeoutMs === undefined ? {} : { timeoutMs: options.timeoutMs }),
|
|
|
|
|
...(recipient === undefined
|
|
|
|
|
? {}
|
|
|
|
|
: {
|
|
|
|
|
recipient: (response: TRes) => Effect.promise(() => recipient(response)),
|
|
|
|
|
}),
|
|
|
|
|
};
|
|
|
|
|
}
|
2026-04-05 21:09:33 -05:00
|
|
|
}
|