mirror of
https://github.com/trustgraph-ai/trustgraph.git
synced 2026-07-01 09:29:38 +02:00
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>
This commit is contained in:
parent
28747e1a92
commit
5ed3f0e2d8
6 changed files with 236 additions and 1 deletions
|
|
@ -8,6 +8,7 @@ import type { PubSubBackend } from "../backend/types.js";
|
||||||
import type { Spec } from "../spec/types.js";
|
import type { Spec } from "../spec/types.js";
|
||||||
import type { Producer } from "../messaging/producer.js";
|
import type { Producer } from "../messaging/producer.js";
|
||||||
import type { Consumer } from "../messaging/consumer.js";
|
import type { Consumer } from "../messaging/consumer.js";
|
||||||
|
import type { RequestResponse } from "../messaging/request-response.js";
|
||||||
|
|
||||||
export interface FlowDefinition {
|
export interface FlowDefinition {
|
||||||
/** Topic overrides keyed by spec name */
|
/** Topic overrides keyed by spec name */
|
||||||
|
|
@ -19,6 +20,7 @@ export interface FlowDefinition {
|
||||||
export class Flow {
|
export class Flow {
|
||||||
private producers = new Map<string, Producer<unknown>>();
|
private producers = new Map<string, Producer<unknown>>();
|
||||||
private consumers = new Map<string, Consumer<unknown>>();
|
private consumers = new Map<string, Consumer<unknown>>();
|
||||||
|
private requestors = new Map<string, RequestResponse<unknown, unknown>>();
|
||||||
private parameters = new Map<string, unknown>();
|
private parameters = new Map<string, unknown>();
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
|
|
@ -49,6 +51,9 @@ export class Flow {
|
||||||
for (const producer of this.producers.values()) {
|
for (const producer of this.producers.values()) {
|
||||||
await producer.stop();
|
await producer.stop();
|
||||||
}
|
}
|
||||||
|
for (const rr of this.requestors.values()) {
|
||||||
|
await rr.stop();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
registerProducer(name: string, producer: Producer<unknown>): void {
|
registerProducer(name: string, producer: Producer<unknown>): void {
|
||||||
|
|
@ -59,6 +64,10 @@ export class Flow {
|
||||||
this.consumers.set(name, consumer);
|
this.consumers.set(name, consumer);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
registerRequestor(name: string, rr: RequestResponse<unknown, unknown>): void {
|
||||||
|
this.requestors.set(name, rr);
|
||||||
|
}
|
||||||
|
|
||||||
setParameter(name: string, value: unknown): void {
|
setParameter(name: string, value: unknown): void {
|
||||||
this.parameters.set(name, value);
|
this.parameters.set(name, value);
|
||||||
}
|
}
|
||||||
|
|
@ -75,6 +84,12 @@ export class Flow {
|
||||||
return c as Consumer<T>;
|
return c as Consumer<T>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
requestor<TReq, TRes>(name: string): RequestResponse<TReq, TRes> {
|
||||||
|
const rr = this.requestors.get(name);
|
||||||
|
if (!rr) throw new Error(`Requestor "${name}" not found in flow "${this.name}"`);
|
||||||
|
return rr as RequestResponse<TReq, TRes>;
|
||||||
|
}
|
||||||
|
|
||||||
parameter<T>(name: string): T {
|
parameter<T>(name: string): T {
|
||||||
const v = this.parameters.get(name);
|
const v = this.parameters.get(name);
|
||||||
if (v === undefined) throw new Error(`Parameter "${name}" not found in flow "${this.name}"`);
|
if (v === undefined) throw new Error(`Parameter "${name}" not found in flow "${this.name}"`);
|
||||||
|
|
|
||||||
|
|
@ -70,10 +70,18 @@ export interface AgentRequest {
|
||||||
question: string;
|
question: string;
|
||||||
collection?: string;
|
collection?: string;
|
||||||
streaming?: boolean;
|
streaming?: boolean;
|
||||||
|
group?: string[];
|
||||||
|
state?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface AgentResponse {
|
export interface AgentResponse {
|
||||||
answer: string;
|
/** Streaming chunk type */
|
||||||
|
chunk_type?: "thought" | "observation" | "answer" | "error";
|
||||||
|
content?: string;
|
||||||
|
end_of_message?: boolean;
|
||||||
|
end_of_dialog?: boolean;
|
||||||
|
/** Legacy non-streaming fields */
|
||||||
|
answer?: string;
|
||||||
error?: TgError;
|
error?: TgError;
|
||||||
endOfStream?: boolean;
|
endOfStream?: boolean;
|
||||||
endOfSession?: boolean;
|
endOfSession?: boolean;
|
||||||
|
|
@ -133,3 +141,161 @@ export interface PromptResponse {
|
||||||
prompt: string;
|
prompt: string;
|
||||||
error?: TgError;
|
error?: TgError;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ---------- Pipeline types ----------
|
||||||
|
|
||||||
|
export interface PipelineMetadata {
|
||||||
|
id: string;
|
||||||
|
root: string;
|
||||||
|
user: string;
|
||||||
|
collection: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface TextDocument {
|
||||||
|
metadata: PipelineMetadata;
|
||||||
|
text: string;
|
||||||
|
documentId: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Chunk {
|
||||||
|
metadata: PipelineMetadata;
|
||||||
|
chunk: string;
|
||||||
|
documentId: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface EntityContext {
|
||||||
|
entity: Term;
|
||||||
|
context: string;
|
||||||
|
chunkId: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface EntityContexts {
|
||||||
|
metadata: PipelineMetadata;
|
||||||
|
entities: EntityContext[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Triples {
|
||||||
|
metadata: PipelineMetadata;
|
||||||
|
triples: Triple[];
|
||||||
|
}
|
||||||
|
|
||||||
|
// ---------- Document metadata ----------
|
||||||
|
|
||||||
|
export interface DocumentMetadata {
|
||||||
|
id: string;
|
||||||
|
time: number;
|
||||||
|
kind: string;
|
||||||
|
title: string;
|
||||||
|
comments: string;
|
||||||
|
user: string;
|
||||||
|
tags: string[];
|
||||||
|
parentId?: string;
|
||||||
|
documentType: string; // "source" | "page" | "chunk" | "extracted"
|
||||||
|
metadata?: Triple[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ProcessingMetadata {
|
||||||
|
id: string;
|
||||||
|
documentId: string;
|
||||||
|
time: number;
|
||||||
|
flow: string;
|
||||||
|
user: string;
|
||||||
|
collection: string;
|
||||||
|
tags: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
// ---------- Librarian ----------
|
||||||
|
|
||||||
|
export type LibrarianOperation =
|
||||||
|
| "add-document"
|
||||||
|
| "remove-document"
|
||||||
|
| "list-documents"
|
||||||
|
| "get-document-metadata"
|
||||||
|
| "get-document-content"
|
||||||
|
| "add-child-document"
|
||||||
|
| "list-children"
|
||||||
|
| "add-processing"
|
||||||
|
| "remove-processing"
|
||||||
|
| "list-processing";
|
||||||
|
|
||||||
|
export interface LibrarianRequest {
|
||||||
|
operation: LibrarianOperation;
|
||||||
|
documentId?: string;
|
||||||
|
processingId?: string;
|
||||||
|
documentMetadata?: DocumentMetadata;
|
||||||
|
processingMetadata?: ProcessingMetadata;
|
||||||
|
content?: string; // base64
|
||||||
|
user?: string;
|
||||||
|
collection?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface LibrarianResponse {
|
||||||
|
error?: TgError;
|
||||||
|
documentMetadata?: DocumentMetadata;
|
||||||
|
content?: string; // base64
|
||||||
|
documents?: DocumentMetadata[];
|
||||||
|
processing?: ProcessingMetadata[];
|
||||||
|
}
|
||||||
|
|
||||||
|
// ---------- Knowledge core ----------
|
||||||
|
|
||||||
|
export type KnowledgeOperation =
|
||||||
|
| "list-kg-cores"
|
||||||
|
| "get-kg-core"
|
||||||
|
| "delete-kg-core"
|
||||||
|
| "put-kg-core"
|
||||||
|
| "load-kg-core";
|
||||||
|
|
||||||
|
export interface KnowledgeRequest {
|
||||||
|
operation: KnowledgeOperation;
|
||||||
|
user?: string;
|
||||||
|
id?: string;
|
||||||
|
flow?: string;
|
||||||
|
collection?: string;
|
||||||
|
triples?: Triple[];
|
||||||
|
graphEmbeddings?: { entity: Term; vectors: number[][] }[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface KnowledgeResponse {
|
||||||
|
error?: TgError;
|
||||||
|
ids?: string[];
|
||||||
|
eos?: boolean;
|
||||||
|
triples?: Triple[];
|
||||||
|
graphEmbeddings?: { entity: Term; vectors: number[][] }[];
|
||||||
|
}
|
||||||
|
|
||||||
|
// ---------- Collection management ----------
|
||||||
|
|
||||||
|
export type CollectionOperation =
|
||||||
|
| "list-collections"
|
||||||
|
| "update-collection"
|
||||||
|
| "delete-collection";
|
||||||
|
|
||||||
|
export interface CollectionManagementRequest {
|
||||||
|
operation: CollectionOperation;
|
||||||
|
user?: string;
|
||||||
|
collection?: string;
|
||||||
|
name?: string;
|
||||||
|
description?: string;
|
||||||
|
tags?: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface CollectionManagementResponse {
|
||||||
|
error?: TgError;
|
||||||
|
collections?: { user: string; collection: string; name: string; description: string; tags: string[] }[];
|
||||||
|
}
|
||||||
|
|
||||||
|
// ---------- Flow management ----------
|
||||||
|
|
||||||
|
export type FlowOperation = "list" | "get" | "start" | "stop";
|
||||||
|
|
||||||
|
export interface FlowRequest {
|
||||||
|
operation: FlowOperation;
|
||||||
|
id?: string;
|
||||||
|
blueprint?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface FlowResponse {
|
||||||
|
error?: TgError;
|
||||||
|
flows?: { id: string; status: string; blueprint?: string }[];
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -59,4 +59,20 @@ export const topics = {
|
||||||
// Prompt
|
// Prompt
|
||||||
promptRequest: topic("prompt-request"),
|
promptRequest: topic("prompt-request"),
|
||||||
promptResponse: topic("prompt-response"),
|
promptResponse: topic("prompt-response"),
|
||||||
|
|
||||||
|
// Librarian (document management)
|
||||||
|
librarianRequest: topic("librarian-request"),
|
||||||
|
librarianResponse: topic("librarian-response"),
|
||||||
|
|
||||||
|
// Knowledge core management
|
||||||
|
knowledgeRequest: topic("knowledge-request"),
|
||||||
|
knowledgeResponse: topic("knowledge-response"),
|
||||||
|
|
||||||
|
// Collection management
|
||||||
|
collectionManagementRequest: topic("collection-management-request"),
|
||||||
|
collectionManagementResponse: topic("collection-management-response"),
|
||||||
|
|
||||||
|
// Flow management
|
||||||
|
flowRequest: topic("flow-request"),
|
||||||
|
flowResponse: topic("flow-response"),
|
||||||
} as const;
|
} as const;
|
||||||
|
|
|
||||||
|
|
@ -2,3 +2,4 @@ export type { Spec } from "./types.js";
|
||||||
export { ConsumerSpec } from "./consumer-spec.js";
|
export { ConsumerSpec } from "./consumer-spec.js";
|
||||||
export { ProducerSpec } from "./producer-spec.js";
|
export { ProducerSpec } from "./producer-spec.js";
|
||||||
export { ParameterSpec } from "./parameter-spec.js";
|
export { ParameterSpec } from "./parameter-spec.js";
|
||||||
|
export { RequestResponseSpec } from "./request-response-spec.js";
|
||||||
|
|
|
||||||
36
ts/packages/base/src/spec/request-response-spec.ts
Normal file
36
ts/packages/base/src/spec/request-response-spec.ts
Normal file
|
|
@ -0,0 +1,36 @@
|
||||||
|
/**
|
||||||
|
* Request/response specification — declares a request/response client for a flow.
|
||||||
|
*
|
||||||
|
* Enables FlowProcessor handlers to make request/response calls to other services
|
||||||
|
* (e.g., calling the prompt service or LLM from within a knowledge extraction handler).
|
||||||
|
*
|
||||||
|
* Python reference: trustgraph-base/trustgraph/base/prompt_client_spec.py
|
||||||
|
*/
|
||||||
|
|
||||||
|
import type { Spec } from "./types.js";
|
||||||
|
import type { PubSubBackend } from "../backend/types.js";
|
||||||
|
import type { Flow, FlowDefinition } from "../processor/flow.js";
|
||||||
|
import { RequestResponse } from "../messaging/request-response.js";
|
||||||
|
|
||||||
|
export class RequestResponseSpec<TReq, TRes> implements Spec {
|
||||||
|
constructor(
|
||||||
|
public readonly name: string,
|
||||||
|
private readonly requestTopicName: string,
|
||||||
|
private readonly responseTopicName: string,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
async add(flow: Flow, pubsub: PubSubBackend, definition: FlowDefinition): Promise<void> {
|
||||||
|
const requestTopic = definition.topics?.[this.requestTopicName] ?? this.requestTopicName;
|
||||||
|
const responseTopic = definition.topics?.[this.responseTopicName] ?? this.responseTopicName;
|
||||||
|
|
||||||
|
const rr = new RequestResponse<TReq, TRes>({
|
||||||
|
pubsub,
|
||||||
|
requestTopic,
|
||||||
|
responseTopic,
|
||||||
|
subscription: `${flow.processorId}-${flow.name}-${this.name}`,
|
||||||
|
});
|
||||||
|
await rr.start();
|
||||||
|
|
||||||
|
flow.registerRequestor(this.name, rr as RequestResponse<unknown, unknown>);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -152,6 +152,7 @@ export class DispatcherManager {
|
||||||
!!res.endOfSession ||
|
!!res.endOfSession ||
|
||||||
!!res.end_of_stream ||
|
!!res.end_of_stream ||
|
||||||
!!res.end_of_session ||
|
!!res.end_of_session ||
|
||||||
|
!!res.end_of_dialog ||
|
||||||
!!res.eos ||
|
!!res.eos ||
|
||||||
// error responses are always final
|
// error responses are always final
|
||||||
!!res.error
|
!!res.error
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue