@kilnai/runtime 0.1.15 → 0.9.0
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/README.md +7 -3
- package/dist/channels/email-api.d.ts +29 -0
- package/dist/channels/email-api.d.ts.map +1 -0
- package/dist/channels/email-api.js +133 -0
- package/dist/channels/email-api.js.map +1 -0
- package/dist/channels/email-channel.d.ts +26 -0
- package/dist/channels/email-channel.d.ts.map +1 -0
- package/dist/channels/email-channel.js +59 -0
- package/dist/channels/email-channel.js.map +1 -0
- package/dist/channels/email-template.d.ts +10 -0
- package/dist/channels/email-template.d.ts.map +1 -0
- package/dist/channels/email-template.js +63 -0
- package/dist/channels/email-template.js.map +1 -0
- package/dist/channels/instagram-api.d.ts +29 -0
- package/dist/channels/instagram-api.d.ts.map +1 -0
- package/dist/channels/instagram-api.js +75 -0
- package/dist/channels/instagram-api.js.map +1 -0
- package/dist/channels/instagram-channel.d.ts +24 -0
- package/dist/channels/instagram-channel.d.ts.map +1 -0
- package/dist/channels/instagram-channel.js +55 -0
- package/dist/channels/instagram-channel.js.map +1 -0
- package/dist/channels/message-formatter.d.ts +17 -0
- package/dist/channels/message-formatter.d.ts.map +1 -1
- package/dist/channels/message-formatter.js +27 -0
- package/dist/channels/message-formatter.js.map +1 -1
- package/dist/channels/messenger-api.d.ts +27 -0
- package/dist/channels/messenger-api.d.ts.map +1 -0
- package/dist/channels/messenger-api.js +75 -0
- package/dist/channels/messenger-api.js.map +1 -0
- package/dist/channels/messenger-channel.d.ts +23 -0
- package/dist/channels/messenger-channel.d.ts.map +1 -0
- package/dist/channels/messenger-channel.js +55 -0
- package/dist/channels/messenger-channel.js.map +1 -0
- package/dist/channels/web-channel.d.ts +2 -0
- package/dist/channels/web-channel.d.ts.map +1 -1
- package/dist/channels/web-channel.js +9 -0
- package/dist/channels/web-channel.js.map +1 -1
- package/dist/enrichment/enrichment-runner.d.ts +17 -0
- package/dist/enrichment/enrichment-runner.d.ts.map +1 -0
- package/dist/enrichment/enrichment-runner.js +45 -0
- package/dist/enrichment/enrichment-runner.js.map +1 -0
- package/dist/enrichment/sqlite-enrichment-store.d.ts +14 -0
- package/dist/enrichment/sqlite-enrichment-store.d.ts.map +1 -0
- package/dist/enrichment/sqlite-enrichment-store.js +69 -0
- package/dist/enrichment/sqlite-enrichment-store.js.map +1 -0
- package/dist/gateway/audio-preprocessor.d.ts +11 -0
- package/dist/gateway/audio-preprocessor.d.ts.map +1 -0
- package/dist/gateway/audio-preprocessor.js +76 -0
- package/dist/gateway/audio-preprocessor.js.map +1 -0
- package/dist/gateway/auth-middleware.d.ts.map +1 -1
- package/dist/gateway/auth-middleware.js +10 -2
- package/dist/gateway/auth-middleware.js.map +1 -1
- package/dist/gateway/contact-memory-admin-routes.d.ts +9 -0
- package/dist/gateway/contact-memory-admin-routes.d.ts.map +1 -0
- package/dist/gateway/contact-memory-admin-routes.js +42 -0
- package/dist/gateway/contact-memory-admin-routes.js.map +1 -0
- package/dist/gateway/context-formatter.d.ts +5 -0
- package/dist/gateway/context-formatter.d.ts.map +1 -0
- package/dist/gateway/context-formatter.js +16 -0
- package/dist/gateway/context-formatter.js.map +1 -0
- package/dist/gateway/conversation-event-emitter.d.ts +4 -1
- package/dist/gateway/conversation-event-emitter.d.ts.map +1 -1
- package/dist/gateway/conversation-event-emitter.js +44 -13
- package/dist/gateway/conversation-event-emitter.js.map +1 -1
- package/dist/gateway/dev-routes.d.ts +1 -1
- package/dist/gateway/dev-routes.d.ts.map +1 -1
- package/dist/gateway/dev-routes.js +2 -2
- package/dist/gateway/dev-routes.js.map +1 -1
- package/dist/gateway/email-loop-guard.d.ts +10 -0
- package/dist/gateway/email-loop-guard.d.ts.map +1 -0
- package/dist/gateway/email-loop-guard.js +39 -0
- package/dist/gateway/email-loop-guard.js.map +1 -0
- package/dist/gateway/email-thread-store.d.ts +24 -0
- package/dist/gateway/email-thread-store.d.ts.map +1 -0
- package/dist/gateway/email-thread-store.js +34 -0
- package/dist/gateway/email-thread-store.js.map +1 -0
- package/dist/gateway/email-webhook-routes.d.ts +32 -0
- package/dist/gateway/email-webhook-routes.d.ts.map +1 -0
- package/dist/gateway/email-webhook-routes.js +411 -0
- package/dist/gateway/email-webhook-routes.js.map +1 -0
- package/dist/gateway/enrichment-admin-routes.d.ts +9 -0
- package/dist/gateway/enrichment-admin-routes.d.ts.map +1 -0
- package/dist/gateway/enrichment-admin-routes.js +41 -0
- package/dist/gateway/enrichment-admin-routes.js.map +1 -0
- package/dist/gateway/gateway-routes.d.ts +17 -1
- package/dist/gateway/gateway-routes.d.ts.map +1 -1
- package/dist/gateway/gateway-routes.js +55 -1
- package/dist/gateway/gateway-routes.js.map +1 -1
- package/dist/gateway/gateway-server.d.ts.map +1 -1
- package/dist/gateway/gateway-server.js +250 -14
- package/dist/gateway/gateway-server.js.map +1 -1
- package/dist/gateway/handoff-routes.d.ts +15 -0
- package/dist/gateway/handoff-routes.d.ts.map +1 -0
- package/dist/gateway/handoff-routes.js +253 -0
- package/dist/gateway/handoff-routes.js.map +1 -0
- package/dist/gateway/instagram-webhook-routes.d.ts +30 -0
- package/dist/gateway/instagram-webhook-routes.d.ts.map +1 -0
- package/dist/gateway/instagram-webhook-routes.js +394 -0
- package/dist/gateway/instagram-webhook-routes.js.map +1 -0
- package/dist/gateway/knowledge-admin-routes.d.ts +9 -0
- package/dist/gateway/knowledge-admin-routes.d.ts.map +1 -0
- package/dist/gateway/knowledge-admin-routes.js +84 -0
- package/dist/gateway/knowledge-admin-routes.js.map +1 -0
- package/dist/gateway/knowledge-factory.d.ts +19 -0
- package/dist/gateway/knowledge-factory.d.ts.map +1 -0
- package/dist/gateway/knowledge-factory.js +122 -0
- package/dist/gateway/knowledge-factory.js.map +1 -0
- package/dist/gateway/message-pipeline.d.ts +69 -0
- package/dist/gateway/message-pipeline.d.ts.map +1 -0
- package/dist/gateway/message-pipeline.js +185 -0
- package/dist/gateway/message-pipeline.js.map +1 -0
- package/dist/gateway/messenger-webhook-routes.d.ts +30 -0
- package/dist/gateway/messenger-webhook-routes.d.ts.map +1 -0
- package/dist/gateway/messenger-webhook-routes.js +394 -0
- package/dist/gateway/messenger-webhook-routes.js.map +1 -0
- package/dist/gateway/meta-webhook-foundation.d.ts +12 -0
- package/dist/gateway/meta-webhook-foundation.d.ts.map +1 -0
- package/dist/gateway/meta-webhook-foundation.js +25 -0
- package/dist/gateway/meta-webhook-foundation.js.map +1 -0
- package/dist/gateway/mode-b-routes.d.ts +8 -0
- package/dist/gateway/mode-b-routes.d.ts.map +1 -1
- package/dist/gateway/mode-b-routes.js +74 -36
- package/dist/gateway/mode-b-routes.js.map +1 -1
- package/dist/gateway/outbound-routes.d.ts.map +1 -1
- package/dist/gateway/outbound-routes.js +49 -21
- package/dist/gateway/outbound-routes.js.map +1 -1
- package/dist/gateway/routing-test-routes.d.ts +10 -0
- package/dist/gateway/routing-test-routes.d.ts.map +1 -0
- package/dist/gateway/routing-test-routes.js +76 -0
- package/dist/gateway/routing-test-routes.js.map +1 -0
- package/dist/gateway/sqlite-email-thread-store.d.ts +13 -0
- package/dist/gateway/sqlite-email-thread-store.d.ts.map +1 -0
- package/dist/gateway/sqlite-email-thread-store.js +66 -0
- package/dist/gateway/sqlite-email-thread-store.js.map +1 -0
- package/dist/gateway/stt-factory.d.ts +3 -0
- package/dist/gateway/stt-factory.d.ts.map +1 -0
- package/dist/gateway/stt-factory.js +21 -0
- package/dist/gateway/stt-factory.js.map +1 -0
- package/dist/gateway/tenant-admin-routes.d.ts.map +1 -1
- package/dist/gateway/tenant-admin-routes.js +17 -3
- package/dist/gateway/tenant-admin-routes.js.map +1 -1
- package/dist/gateway/tenant-routes.d.ts.map +1 -1
- package/dist/gateway/tenant-routes.js +18 -27
- package/dist/gateway/tenant-routes.js.map +1 -1
- package/dist/gateway/tenant-tool-factory.d.ts +10 -0
- package/dist/gateway/tenant-tool-factory.d.ts.map +1 -0
- package/dist/gateway/tenant-tool-factory.js +55 -0
- package/dist/gateway/tenant-tool-factory.js.map +1 -0
- package/dist/gateway/trace-context.d.ts +8 -0
- package/dist/gateway/trace-context.d.ts.map +1 -0
- package/dist/gateway/trace-context.js +17 -0
- package/dist/gateway/trace-context.js.map +1 -0
- package/dist/gateway/webhook-dedup.d.ts +13 -0
- package/dist/gateway/webhook-dedup.d.ts.map +1 -0
- package/dist/gateway/webhook-dedup.js +37 -0
- package/dist/gateway/webhook-dedup.js.map +1 -0
- package/dist/gateway/webhook-tool-executor.d.ts +17 -0
- package/dist/gateway/webhook-tool-executor.d.ts.map +1 -0
- package/dist/gateway/webhook-tool-executor.js +69 -0
- package/dist/gateway/webhook-tool-executor.js.map +1 -0
- package/dist/gateway/whatsapp-webhook-routes.d.ts +11 -0
- package/dist/gateway/whatsapp-webhook-routes.d.ts.map +1 -1
- package/dist/gateway/whatsapp-webhook-routes.js +205 -33
- package/dist/gateway/whatsapp-webhook-routes.js.map +1 -1
- package/dist/gateway/ws-tenant-routes.d.ts +9 -0
- package/dist/gateway/ws-tenant-routes.d.ts.map +1 -1
- package/dist/gateway/ws-tenant-routes.js +184 -7
- package/dist/gateway/ws-tenant-routes.js.map +1 -1
- package/dist/index.d.ts +63 -3
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +39 -1
- package/dist/index.js.map +1 -1
- package/dist/observability/composite-event-store.d.ts +13 -0
- package/dist/observability/composite-event-store.d.ts.map +1 -0
- package/dist/observability/composite-event-store.js +37 -0
- package/dist/observability/composite-event-store.js.map +1 -0
- package/dist/observability/prometheus-collector.d.ts +33 -0
- package/dist/observability/prometheus-collector.d.ts.map +1 -0
- package/dist/observability/prometheus-collector.js +170 -0
- package/dist/observability/prometheus-collector.js.map +1 -0
- package/dist/session/agent-handoff-summarizer.d.ts +11 -0
- package/dist/session/agent-handoff-summarizer.d.ts.map +1 -0
- package/dist/session/agent-handoff-summarizer.js +26 -0
- package/dist/session/agent-handoff-summarizer.js.map +1 -0
- package/dist/session/context-summarizer.d.ts +11 -0
- package/dist/session/context-summarizer.d.ts.map +1 -0
- package/dist/session/context-summarizer.js +23 -0
- package/dist/session/context-summarizer.js.map +1 -0
- package/dist/session/escalation-detector.d.ts +27 -0
- package/dist/session/escalation-detector.d.ts.map +1 -0
- package/dist/session/escalation-detector.js +75 -0
- package/dist/session/escalation-detector.js.map +1 -0
- package/dist/session/in-memory-session-store.d.ts +11 -0
- package/dist/session/in-memory-session-store.d.ts.map +1 -0
- package/dist/session/in-memory-session-store.js +26 -0
- package/dist/session/in-memory-session-store.js.map +1 -0
- package/dist/session/mode-b-orchestrator.d.ts +54 -1
- package/dist/session/mode-b-orchestrator.d.ts.map +1 -1
- package/dist/session/mode-b-orchestrator.js +349 -17
- package/dist/session/mode-b-orchestrator.js.map +1 -1
- package/dist/session/mode-b-session.d.ts +48 -1
- package/dist/session/mode-b-session.d.ts.map +1 -1
- package/dist/session/mode-b-session.js +105 -0
- package/dist/session/mode-b-session.js.map +1 -1
- package/dist/session/redis-session-store.d.ts +22 -0
- package/dist/session/redis-session-store.d.ts.map +1 -0
- package/dist/session/redis-session-store.js +44 -0
- package/dist/session/redis-session-store.js.map +1 -0
- package/dist/session/session-mode.d.ts +4 -0
- package/dist/session/session-mode.d.ts.map +1 -0
- package/dist/session/session-mode.js +20 -0
- package/dist/session/session-mode.js.map +1 -0
- package/dist/session/session-registry.d.ts +17 -9
- package/dist/session/session-registry.d.ts.map +1 -1
- package/dist/session/session-registry.js +80 -29
- package/dist/session/session-registry.js.map +1 -1
- package/dist/session/session-serializer.d.ts +4 -0
- package/dist/session/session-serializer.d.ts.map +1 -0
- package/dist/session/session-serializer.js +26 -0
- package/dist/session/session-serializer.js.map +1 -0
- package/dist/session/session-store.d.ts +9 -0
- package/dist/session/session-store.d.ts.map +1 -0
- package/dist/session/session-store.js +2 -0
- package/dist/session/session-store.js.map +1 -0
- package/dist/tenant/agent-resolver.d.ts +26 -0
- package/dist/tenant/agent-resolver.d.ts.map +1 -0
- package/dist/tenant/agent-resolver.js +232 -0
- package/dist/tenant/agent-resolver.js.map +1 -0
- package/dist/tenant/ping-pong-guard.d.ts +9 -0
- package/dist/tenant/ping-pong-guard.d.ts.map +1 -0
- package/dist/tenant/ping-pong-guard.js +30 -0
- package/dist/tenant/ping-pong-guard.js.map +1 -0
- package/dist/tenant/system-prompt-builder.d.ts +1 -1
- package/dist/tenant/system-prompt-builder.d.ts.map +1 -1
- package/dist/tenant/system-prompt-builder.js +13 -1
- package/dist/tenant/system-prompt-builder.js.map +1 -1
- package/dist/tenant/tenant-registry.d.ts +3 -0
- package/dist/tenant/tenant-registry.d.ts.map +1 -1
- package/dist/tenant/tenant-registry.js +83 -4
- package/dist/tenant/tenant-registry.js.map +1 -1
- package/dist/tenant/tenant-router.d.ts +27 -0
- package/dist/tenant/tenant-router.d.ts.map +1 -0
- package/dist/tenant/tenant-router.js +72 -0
- package/dist/tenant/tenant-router.js.map +1 -0
- package/dist/utils/hmac.d.ts +1 -0
- package/dist/utils/hmac.d.ts.map +1 -1
- package/dist/utils/hmac.js +3 -0
- package/dist/utils/hmac.js.map +1 -1
- package/package.json +7 -3
|
@@ -0,0 +1,232 @@
|
|
|
1
|
+
// Agent resolver: single integration point for multi-agent routing in all channel handlers.
|
|
2
|
+
// Resolves which agent handles a message, builds the agent-specific system prompt and tool context.
|
|
3
|
+
import { buildTenantSystemPrompt } from "./system-prompt-builder.js";
|
|
4
|
+
import { buildTenantToolContext } from "../gateway/tenant-tool-factory.js";
|
|
5
|
+
import { DefaultTenantRouter, EmbeddingTenantRouter } from "./tenant-router.js";
|
|
6
|
+
import { checkPingPong } from "./ping-pong-guard.js";
|
|
7
|
+
export function buildAgentSystemPrompt(basePrompt, agent) {
|
|
8
|
+
const parts = [basePrompt];
|
|
9
|
+
parts.push("");
|
|
10
|
+
parts.push("## Your Agent Identity");
|
|
11
|
+
parts.push(`You are "${agent.name}" — ${agent.role}.`);
|
|
12
|
+
parts.push(`Your primary goal: ${agent.goal}`);
|
|
13
|
+
if (agent.backstory) {
|
|
14
|
+
parts.push(`Background: ${agent.backstory}`);
|
|
15
|
+
}
|
|
16
|
+
if (agent.instructions) {
|
|
17
|
+
parts.push("");
|
|
18
|
+
parts.push("## Operating Instructions");
|
|
19
|
+
parts.push(agent.instructions);
|
|
20
|
+
}
|
|
21
|
+
return parts.join("\n");
|
|
22
|
+
}
|
|
23
|
+
export function resolveAgentContext(tenant, userParts, channel, session, existingBuiltins) {
|
|
24
|
+
const basePrompt = buildTenantSystemPrompt(tenant, channel);
|
|
25
|
+
// No agents or single agent without routing → single-agent mode
|
|
26
|
+
if (!tenant.agents || tenant.agents.length === 0) {
|
|
27
|
+
return {
|
|
28
|
+
systemPrompt: basePrompt,
|
|
29
|
+
tenantToolContext: buildTenantToolContext(tenant, existingBuiltins),
|
|
30
|
+
isHandoff: false,
|
|
31
|
+
};
|
|
32
|
+
}
|
|
33
|
+
if (tenant.agents.length === 1) {
|
|
34
|
+
const agent = tenant.agents[0];
|
|
35
|
+
const systemPrompt = buildAgentSystemPrompt(basePrompt, agent);
|
|
36
|
+
const toolCtx = buildAgentToolContext(tenant, agent, existingBuiltins);
|
|
37
|
+
return {
|
|
38
|
+
systemPrompt,
|
|
39
|
+
tenantToolContext: toolCtx,
|
|
40
|
+
activeAgentId: agent.id,
|
|
41
|
+
activeAgentName: agent.name,
|
|
42
|
+
isHandoff: false,
|
|
43
|
+
};
|
|
44
|
+
}
|
|
45
|
+
// Multi-agent: route to the correct agent
|
|
46
|
+
if (!tenant.routing) {
|
|
47
|
+
// No routing config → fall back to first agent
|
|
48
|
+
const agent = tenant.agents[0];
|
|
49
|
+
const systemPrompt = buildAgentSystemPrompt(basePrompt, agent);
|
|
50
|
+
const toolCtx = buildAgentToolContext(tenant, agent, existingBuiltins);
|
|
51
|
+
return {
|
|
52
|
+
systemPrompt,
|
|
53
|
+
tenantToolContext: toolCtx,
|
|
54
|
+
activeAgentId: agent.id,
|
|
55
|
+
activeAgentName: agent.name,
|
|
56
|
+
isHandoff: false,
|
|
57
|
+
};
|
|
58
|
+
}
|
|
59
|
+
const router = new DefaultTenantRouter(tenant.routing);
|
|
60
|
+
const routingResult = router.route(userParts);
|
|
61
|
+
// Ping-pong guard: prevent rapid agent switching loops
|
|
62
|
+
if (session && tenant.routing) {
|
|
63
|
+
const guardResult = checkPingPong(routingResult, session, tenant.routing);
|
|
64
|
+
if (guardResult.blocked) {
|
|
65
|
+
// Keep current agent
|
|
66
|
+
const currentAgent = tenant.agents.find((a) => a.id === session.activeAgentId)
|
|
67
|
+
?? tenant.agents.find((a) => a.isDefault) ?? tenant.agents[0];
|
|
68
|
+
const systemPrompt = buildAgentSystemPrompt(basePrompt, currentAgent);
|
|
69
|
+
const toolCtx = buildAgentToolContext(tenant, currentAgent, existingBuiltins);
|
|
70
|
+
return {
|
|
71
|
+
systemPrompt,
|
|
72
|
+
tenantToolContext: toolCtx,
|
|
73
|
+
activeAgentId: currentAgent.id,
|
|
74
|
+
activeAgentName: currentAgent.name,
|
|
75
|
+
routingResult,
|
|
76
|
+
previousAgentId: session.activeAgentId,
|
|
77
|
+
isHandoff: false,
|
|
78
|
+
pingPongBlocked: true,
|
|
79
|
+
pingPongReason: guardResult.reason,
|
|
80
|
+
};
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
// Find the selected agent
|
|
84
|
+
let selectedAgent = tenant.agents.find((a) => a.id === routingResult.agentId);
|
|
85
|
+
if (!selectedAgent) {
|
|
86
|
+
// Fail-open: use default agent or first agent
|
|
87
|
+
selectedAgent = tenant.agents.find((a) => a.isDefault) ?? tenant.agents[0];
|
|
88
|
+
}
|
|
89
|
+
const previousAgentId = session?.activeAgentId;
|
|
90
|
+
const isHandoff = previousAgentId !== undefined && previousAgentId !== selectedAgent.id;
|
|
91
|
+
const systemPrompt = buildAgentSystemPrompt(basePrompt, selectedAgent);
|
|
92
|
+
const toolCtx = buildAgentToolContext(tenant, selectedAgent, existingBuiltins);
|
|
93
|
+
return {
|
|
94
|
+
systemPrompt,
|
|
95
|
+
tenantToolContext: toolCtx,
|
|
96
|
+
activeAgentId: selectedAgent.id,
|
|
97
|
+
activeAgentName: selectedAgent.name,
|
|
98
|
+
routingResult,
|
|
99
|
+
previousAgentId,
|
|
100
|
+
isHandoff,
|
|
101
|
+
};
|
|
102
|
+
}
|
|
103
|
+
export async function resolveAgentContextAsync(tenant, userParts, session, deps, channel, existingBuiltins) {
|
|
104
|
+
// If agentRag is provided and tenant has multi-agent routing, use async Tier 2 routing
|
|
105
|
+
let result;
|
|
106
|
+
if (deps?.agentRag && tenant.agents && tenant.agents.length > 1 && tenant.routing) {
|
|
107
|
+
result = await resolveAgentContextWithEmbedding(tenant, userParts, channel, session, deps.agentRag, existingBuiltins);
|
|
108
|
+
}
|
|
109
|
+
else {
|
|
110
|
+
result = resolveAgentContext(tenant, userParts, channel, session, existingBuiltins);
|
|
111
|
+
}
|
|
112
|
+
// No handoff or blocked by guard or no summarizer → return as-is
|
|
113
|
+
if (!result.isHandoff || result.pingPongBlocked || !deps?.handoffSummarizer) {
|
|
114
|
+
return result;
|
|
115
|
+
}
|
|
116
|
+
// Resolve agent names for the brief
|
|
117
|
+
const fromAgent = tenant.agents?.find((a) => a.id === result.previousAgentId);
|
|
118
|
+
const toAgent = tenant.agents?.find((a) => a.id === result.activeAgentId);
|
|
119
|
+
const fromName = fromAgent?.name ?? result.previousAgentId ?? "unknown";
|
|
120
|
+
const toName = toAgent?.name ?? result.activeAgentId ?? "unknown";
|
|
121
|
+
// Emit handoff_requested event
|
|
122
|
+
if (deps.eventBus) {
|
|
123
|
+
const reqEvent = {
|
|
124
|
+
type: "handoff_requested",
|
|
125
|
+
fromAgent: fromName,
|
|
126
|
+
toAgent: toName,
|
|
127
|
+
reason: "agent_routing",
|
|
128
|
+
timestamp: new Date(),
|
|
129
|
+
sessionId: session.id,
|
|
130
|
+
};
|
|
131
|
+
deps.eventBus.emit(reqEvent);
|
|
132
|
+
}
|
|
133
|
+
let brief;
|
|
134
|
+
try {
|
|
135
|
+
brief = await deps.handoffSummarizer.summarize(session, fromName, toName);
|
|
136
|
+
}
|
|
137
|
+
catch {
|
|
138
|
+
// Fail-open: return original result without brief
|
|
139
|
+
return result;
|
|
140
|
+
}
|
|
141
|
+
// Emit handoff_completed event
|
|
142
|
+
if (deps.eventBus) {
|
|
143
|
+
const compEvent = {
|
|
144
|
+
type: "handoff_completed",
|
|
145
|
+
fromAgent: fromName,
|
|
146
|
+
toAgent: toName,
|
|
147
|
+
accepted: true,
|
|
148
|
+
timestamp: new Date(),
|
|
149
|
+
sessionId: session.id,
|
|
150
|
+
};
|
|
151
|
+
deps.eventBus.emit(compEvent);
|
|
152
|
+
}
|
|
153
|
+
const enrichedPrompt = brief ? `${result.systemPrompt}\n\n${brief}` : result.systemPrompt;
|
|
154
|
+
return {
|
|
155
|
+
...result,
|
|
156
|
+
systemPrompt: enrichedPrompt,
|
|
157
|
+
handoffBrief: brief || undefined,
|
|
158
|
+
};
|
|
159
|
+
}
|
|
160
|
+
async function resolveAgentContextWithEmbedding(tenant, userParts, channel, session, agentRag, existingBuiltins) {
|
|
161
|
+
const basePrompt = buildTenantSystemPrompt(tenant, channel);
|
|
162
|
+
const agents = tenant.agents;
|
|
163
|
+
const routing = tenant.routing;
|
|
164
|
+
const embeddingRouter = new EmbeddingTenantRouter(routing, agentRag, agents);
|
|
165
|
+
const routingResult = await embeddingRouter.routeAsync(userParts);
|
|
166
|
+
// Ping-pong guard
|
|
167
|
+
const guardResult = checkPingPong(routingResult, session, routing);
|
|
168
|
+
if (guardResult.blocked) {
|
|
169
|
+
const currentAgent = agents.find((a) => a.id === session.activeAgentId)
|
|
170
|
+
?? agents.find((a) => a.isDefault) ?? agents[0];
|
|
171
|
+
const systemPrompt = buildAgentSystemPrompt(basePrompt, currentAgent);
|
|
172
|
+
const toolCtx = buildAgentToolContext(tenant, currentAgent, existingBuiltins);
|
|
173
|
+
return {
|
|
174
|
+
systemPrompt,
|
|
175
|
+
tenantToolContext: toolCtx,
|
|
176
|
+
activeAgentId: currentAgent.id,
|
|
177
|
+
activeAgentName: currentAgent.name,
|
|
178
|
+
routingResult,
|
|
179
|
+
previousAgentId: session.activeAgentId,
|
|
180
|
+
isHandoff: false,
|
|
181
|
+
pingPongBlocked: true,
|
|
182
|
+
pingPongReason: guardResult.reason,
|
|
183
|
+
};
|
|
184
|
+
}
|
|
185
|
+
let selectedAgent = agents.find((a) => a.id === routingResult.agentId);
|
|
186
|
+
if (!selectedAgent) {
|
|
187
|
+
selectedAgent = agents.find((a) => a.isDefault) ?? agents[0];
|
|
188
|
+
}
|
|
189
|
+
const previousAgentId = session.activeAgentId;
|
|
190
|
+
const isHandoff = previousAgentId !== undefined && previousAgentId !== selectedAgent.id;
|
|
191
|
+
const systemPrompt = buildAgentSystemPrompt(basePrompt, selectedAgent);
|
|
192
|
+
const toolCtx = buildAgentToolContext(tenant, selectedAgent, existingBuiltins);
|
|
193
|
+
return {
|
|
194
|
+
systemPrompt,
|
|
195
|
+
tenantToolContext: toolCtx,
|
|
196
|
+
activeAgentId: selectedAgent.id,
|
|
197
|
+
activeAgentName: selectedAgent.name,
|
|
198
|
+
routingResult,
|
|
199
|
+
previousAgentId,
|
|
200
|
+
isHandoff,
|
|
201
|
+
};
|
|
202
|
+
}
|
|
203
|
+
function buildAgentToolContext(tenant, agent, existingBuiltins) {
|
|
204
|
+
// Build full tenant tool context first
|
|
205
|
+
const fullCtx = buildTenantToolContext(tenant, existingBuiltins);
|
|
206
|
+
// If agent has no tool restriction, inherit full tenant allowlist
|
|
207
|
+
if (!agent.tools || agent.tools.length === 0) {
|
|
208
|
+
return fullCtx;
|
|
209
|
+
}
|
|
210
|
+
// Intersect agent tools with tenant allowlist
|
|
211
|
+
const agentToolSet = new Set(agent.tools);
|
|
212
|
+
let intersectedAllowlist;
|
|
213
|
+
if (fullCtx.toolAllowlist) {
|
|
214
|
+
intersectedAllowlist = new Set();
|
|
215
|
+
for (const tool of agentToolSet) {
|
|
216
|
+
if (fullCtx.toolAllowlist.has(tool)) {
|
|
217
|
+
intersectedAllowlist.add(tool);
|
|
218
|
+
}
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
else {
|
|
222
|
+
intersectedAllowlist = agentToolSet;
|
|
223
|
+
}
|
|
224
|
+
return {
|
|
225
|
+
callBuiltinTools: fullCtx.callBuiltinTools,
|
|
226
|
+
toolDefinitions: fullCtx.toolDefinitions,
|
|
227
|
+
toolAllowlist: intersectedAllowlist,
|
|
228
|
+
rateLimiter: fullCtx.rateLimiter,
|
|
229
|
+
maxToolRounds: fullCtx.maxToolRounds,
|
|
230
|
+
};
|
|
231
|
+
}
|
|
232
|
+
//# sourceMappingURL=agent-resolver.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"agent-resolver.js","sourceRoot":"","sources":["../../src/tenant/agent-resolver.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import type { TenantRoutingConfig } from "@kilnai/core";
|
|
2
|
+
import type { ModeBSession } from "../session/mode-b-session.js";
|
|
3
|
+
import type { RoutingResult } from "./tenant-router.js";
|
|
4
|
+
export interface PingPongCheckResult {
|
|
5
|
+
readonly blocked: boolean;
|
|
6
|
+
readonly reason?: string;
|
|
7
|
+
}
|
|
8
|
+
export declare function checkPingPong(routingResult: RoutingResult, session: ModeBSession, config: TenantRoutingConfig): PingPongCheckResult;
|
|
9
|
+
//# sourceMappingURL=ping-pong-guard.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ping-pong-guard.d.ts","sourceRoot":"","sources":["../../src/tenant/ping-pong-guard.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,cAAc,CAAC;AACxD,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,8BAA8B,CAAC;AACjE,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,oBAAoB,CAAC;AAExD,MAAM,WAAW,mBAAmB;IAClC,QAAQ,CAAC,OAAO,EAAE,OAAO,CAAC;IAC1B,QAAQ,CAAC,MAAM,CAAC,EAAE,MAAM,CAAC;CAC1B;AAED,wBAAgB,aAAa,CAC3B,aAAa,EAAE,aAAa,EAC5B,OAAO,EAAE,YAAY,EACrB,MAAM,EAAE,mBAAmB,GAC1B,mBAAmB,CA+BrB"}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
// Ping-pong guard: prevents rapid agent switching loops in multi-agent routing.
|
|
2
|
+
// Pure stateless function -- no external deps.
|
|
3
|
+
export function checkPingPong(routingResult, session, config) {
|
|
4
|
+
// Same agent -- not a reroute
|
|
5
|
+
if (routingResult.agentId === session.activeAgentId) {
|
|
6
|
+
return { blocked: false };
|
|
7
|
+
}
|
|
8
|
+
// No active agent yet -- first routing, never block
|
|
9
|
+
if (!session.activeAgentId) {
|
|
10
|
+
return { blocked: false };
|
|
11
|
+
}
|
|
12
|
+
// Max handoffs exceeded
|
|
13
|
+
if (session.handoffCount >= (config.maxHandoffs ?? 3)) {
|
|
14
|
+
return { blocked: true, reason: "max_handoffs_exceeded" };
|
|
15
|
+
}
|
|
16
|
+
// Cooldown: not enough turns since last route change
|
|
17
|
+
const turnsSinceLastChange = session.conversationHistory.length - session.lastRouteChangeAt;
|
|
18
|
+
if (turnsSinceLastChange < (config.rerouteAfterTurns ?? 1)) {
|
|
19
|
+
return { blocked: true, reason: "cooldown_active" };
|
|
20
|
+
}
|
|
21
|
+
// Bidirectional pair: A→B→A blocked
|
|
22
|
+
const lastEntry = session.agentTurnHistory.length > 0
|
|
23
|
+
? session.agentTurnHistory[session.agentTurnHistory.length - 1]
|
|
24
|
+
: undefined;
|
|
25
|
+
if (lastEntry?.fromAgentId && lastEntry.fromAgentId === routingResult.agentId) {
|
|
26
|
+
return { blocked: true, reason: "bidirectional_pair" };
|
|
27
|
+
}
|
|
28
|
+
return { blocked: false };
|
|
29
|
+
}
|
|
30
|
+
//# sourceMappingURL=ping-pong-guard.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ping-pong-guard.js","sourceRoot":"","sources":["../../src/tenant/ping-pong-guard.ts"],"names":[],"mappings":"AAAA,gFAAgF;AAChF,+CAA+C;AAW/C,MAAM,UAAU,aAAa,CAC3B,aAA4B,EAC5B,OAAqB,EACrB,MAA2B;IAE3B,8BAA8B;IAC9B,IAAI,aAAa,CAAC,OAAO,KAAK,OAAO,CAAC,aAAa,EAAE,CAAC;QACpD,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;IAC5B,CAAC;IAED,oDAAoD;IACpD,IAAI,CAAC,OAAO,CAAC,aAAa,EAAE,CAAC;QAC3B,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;IAC5B,CAAC;IAED,wBAAwB;IACxB,IAAI,OAAO,CAAC,YAAY,IAAI,CAAC,MAAM,CAAC,WAAW,IAAI,CAAC,CAAC,EAAE,CAAC;QACtD,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,uBAAuB,EAAE,CAAC;IAC5D,CAAC;IAED,qDAAqD;IACrD,MAAM,oBAAoB,GAAG,OAAO,CAAC,mBAAmB,CAAC,MAAM,GAAG,OAAO,CAAC,iBAAiB,CAAC;IAC5F,IAAI,oBAAoB,GAAG,CAAC,MAAM,CAAC,iBAAiB,IAAI,CAAC,CAAC,EAAE,CAAC;QAC3D,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,iBAAiB,EAAE,CAAC;IACtD,CAAC;IAED,oCAAoC;IACpC,MAAM,SAAS,GAAG,OAAO,CAAC,gBAAgB,CAAC,MAAM,GAAG,CAAC;QACnD,CAAC,CAAC,OAAO,CAAC,gBAAgB,CAAC,OAAO,CAAC,gBAAgB,CAAC,MAAM,GAAG,CAAC,CAAC;QAC/D,CAAC,CAAC,SAAS,CAAC;IACd,IAAI,SAAS,EAAE,WAAW,IAAI,SAAS,CAAC,WAAW,KAAK,aAAa,CAAC,OAAO,EAAE,CAAC;QAC9E,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,oBAAoB,EAAE,CAAC;IACzD,CAAC;IAED,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;AAC5B,CAAC"}
|
|
@@ -1,3 +1,3 @@
|
|
|
1
1
|
import type { TenantConfig } from "@kilnai/core";
|
|
2
|
-
export declare function buildTenantSystemPrompt(tenant: TenantConfig, channel?: "web" | "whatsapp"): string;
|
|
2
|
+
export declare function buildTenantSystemPrompt(tenant: TenantConfig, channel?: "web" | "whatsapp" | "instagram" | "messenger" | "email"): string;
|
|
3
3
|
//# sourceMappingURL=system-prompt-builder.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"system-prompt-builder.d.ts","sourceRoot":"","sources":["../../src/tenant/system-prompt-builder.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,cAAc,CAAC;AAEjD,wBAAgB,uBAAuB,CAAC,MAAM,EAAE,YAAY,EAAE,OAAO,CAAC,EAAE,KAAK,GAAG,UAAU,GAAG,MAAM,
|
|
1
|
+
{"version":3,"file":"system-prompt-builder.d.ts","sourceRoot":"","sources":["../../src/tenant/system-prompt-builder.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,cAAc,CAAC;AAEjD,wBAAgB,uBAAuB,CAAC,MAAM,EAAE,YAAY,EAAE,OAAO,CAAC,EAAE,KAAK,GAAG,UAAU,GAAG,WAAW,GAAG,WAAW,GAAG,OAAO,GAAG,MAAM,CA6FxI"}
|
|
@@ -72,7 +72,19 @@ export function buildTenantSystemPrompt(tenant, channel) {
|
|
|
72
72
|
parts.push("");
|
|
73
73
|
parts.push(`Detect the customer's language from their messages and respond in the same language. If you cannot determine their language, default to ${lang}.`);
|
|
74
74
|
parts.push("Be concise. Do not fabricate information you do not know. Only respond about this business.");
|
|
75
|
-
if (channel
|
|
75
|
+
if (channel === "instagram") {
|
|
76
|
+
parts.push("");
|
|
77
|
+
parts.push("You are responding via Instagram DM. Keep messages under 1000 characters. Use plain text only -- no markdown, no bold, no headers, no bullet points. Be concise and direct.");
|
|
78
|
+
}
|
|
79
|
+
else if (channel === "messenger") {
|
|
80
|
+
parts.push("");
|
|
81
|
+
parts.push("You are responding via Facebook Messenger. Keep messages under 2000 characters. Use plain text only -- no markdown, no bold, no headers, no bullet points.");
|
|
82
|
+
}
|
|
83
|
+
else if (channel === "email") {
|
|
84
|
+
parts.push("");
|
|
85
|
+
parts.push("You are responding to an email. Structure your response with clear paragraphs. Use a professional tone appropriate for business correspondence. Do not include email headers (Subject, From, To) in your response -- only the body text. Provide complete, thorough answers.");
|
|
86
|
+
}
|
|
87
|
+
else if (channel !== "whatsapp") {
|
|
76
88
|
parts.push("");
|
|
77
89
|
parts.push("After your response, include 2-3 short follow-up questions the customer might ask next. Format them on a separate final line as: <<SUGG>>Question 1|Question 2|Question 3<</SUGG>>");
|
|
78
90
|
parts.push("Never mention or explain this format to the user.");
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"system-prompt-builder.js","sourceRoot":"","sources":["../../src/tenant/system-prompt-builder.ts"],"names":[],"mappings":"AAEA,MAAM,UAAU,uBAAuB,CAAC,MAAoB,EAAE,
|
|
1
|
+
{"version":3,"file":"system-prompt-builder.js","sourceRoot":"","sources":["../../src/tenant/system-prompt-builder.ts"],"names":[],"mappings":"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"}
|
|
@@ -29,6 +29,9 @@ export declare class TenantRegistry {
|
|
|
29
29
|
remove(tenantId: string): boolean;
|
|
30
30
|
resolveByPhone(phoneNumberId: string, appName: string): TenantConfig | undefined;
|
|
31
31
|
resolveByWidgetId(widgetId: string, appName: string): TenantConfig | undefined;
|
|
32
|
+
resolveByInstagramPageId(pageId: string, appName: string): TenantConfig | undefined;
|
|
33
|
+
resolveByMessengerPageId(pageId: string, appName: string): TenantConfig | undefined;
|
|
34
|
+
resolveByEmailAddress(emailAddress: string, appName: string): TenantConfig | undefined;
|
|
32
35
|
private persist;
|
|
33
36
|
}
|
|
34
37
|
//# sourceMappingURL=tenant-registry.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"tenant-registry.d.ts","sourceRoot":"","sources":["../../src/tenant/tenant-registry.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,SAAS,EAAE,MAAM,cAAc,CAAC;AACzC,OAAO,KAAK,EAAE,YAAY,EAAE,WAAW,EAAE,MAAM,cAAc,CAAC;AAO9D,qBAAa,mBAAoB,SAAQ,SAAS;gBACpC,QAAQ,EAAE,MAAM;CAO7B;AAED,qBAAa,2BAA4B,SAAQ,SAAS;IACxD,QAAQ,CAAC,MAAM,EAAE,SAAS;QAAE,KAAK,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,CAAA;KAAE,EAAE,CAAC;gBACnD,MAAM,EAAE,SAAS;QAAE,KAAK,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,CAAA;KAAE,EAAE;CAQlE;AAED,qBAAa,cAAc;IACzB,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAS;IACpC,OAAO,CAAC,QAAQ,CAAC,WAAW,CAA0B;IACtD,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAmC;gBAE/C,UAAU,EAAE,MAAM,EAAE,WAAW,CAAC,EAAE,WAAW;IAKzD,OAAO,CAAC,sBAAsB;
|
|
1
|
+
{"version":3,"file":"tenant-registry.d.ts","sourceRoot":"","sources":["../../src/tenant/tenant-registry.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,SAAS,EAAE,MAAM,cAAc,CAAC;AACzC,OAAO,KAAK,EAAE,YAAY,EAAE,WAAW,EAAE,MAAM,cAAc,CAAC;AAO9D,qBAAa,mBAAoB,SAAQ,SAAS;gBACpC,QAAQ,EAAE,MAAM;CAO7B;AAED,qBAAa,2BAA4B,SAAQ,SAAS;IACxD,QAAQ,CAAC,MAAM,EAAE,SAAS;QAAE,KAAK,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,CAAA;KAAE,EAAE,CAAC;gBACnD,MAAM,EAAE,SAAS;QAAE,KAAK,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,CAAA;KAAE,EAAE;CAQlE;AAED,qBAAa,cAAc;IACzB,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAS;IACpC,OAAO,CAAC,QAAQ,CAAC,WAAW,CAA0B;IACtD,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAmC;gBAE/C,UAAU,EAAE,MAAM,EAAE,WAAW,CAAC,EAAE,WAAW;IAKzD,OAAO,CAAC,sBAAsB;IAmC9B,OAAO,CAAC,cAAc;IAqCtB,OAAO,CAAC,aAAa;IAiBrB,IAAI,IAAI,IAAI;IAWZ,GAAG,CAAC,QAAQ,EAAE,MAAM,GAAG,YAAY,GAAG,SAAS;IAM/C,IAAI,CAAC,OAAO,CAAC,EAAE,MAAM,GAAG,SAAS,YAAY,EAAE;IAK/C,MAAM,CAAC,MAAM,EAAE,YAAY,GAAG,YAAY;IAgB1C,MAAM,CAAC,QAAQ,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,CAAC,IAAI,CAAC,YAAY,EAAE,UAAU,GAAG,SAAS,GAAG,WAAW,CAAC,CAAC,GAAG,YAAY;IA0BhH,MAAM,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO;IAUjC,cAAc,CAAC,aAAa,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,GAAG,YAAY,GAAG,SAAS;IAahF,iBAAiB,CAAC,QAAQ,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,GAAG,YAAY,GAAG,SAAS;IAa9E,wBAAwB,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,GAAG,YAAY,GAAG,SAAS;IAanF,wBAAwB,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,GAAG,YAAY,GAAG,SAAS;IAanF,qBAAqB,CAAC,YAAY,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,GAAG,YAAY,GAAG,SAAS;IActF,OAAO,CAAC,OAAO;CAKhB"}
|
|
@@ -2,7 +2,7 @@ import { mkdirSync, writeFileSync, readFileSync, readdirSync, existsSync, unlink
|
|
|
2
2
|
import { join } from "node:path";
|
|
3
3
|
import { KilnError } from "@kilnai/core";
|
|
4
4
|
import { validateTenantConfig } from "@kilnai/core";
|
|
5
|
-
const SENSITIVE_FIELDS = ["whatsappAccessToken", "whatsappVerifyToken"];
|
|
5
|
+
const SENSITIVE_FIELDS = ["whatsappAccessToken", "whatsappVerifyToken", "instagramAccessToken", "messengerAccessToken"];
|
|
6
6
|
const ENCRYPTED_PLACEHOLDER = "[encrypted]";
|
|
7
7
|
export class TenantNotFoundError extends KilnError {
|
|
8
8
|
constructor(tenantId) {
|
|
@@ -44,6 +44,24 @@ export class TenantRegistry {
|
|
|
44
44
|
result[field] = ENCRYPTED_PLACEHOLDER;
|
|
45
45
|
}
|
|
46
46
|
}
|
|
47
|
+
// Encrypt email transport API key
|
|
48
|
+
if (config.emailTransportConfig?.apiKey && config.emailTransportConfig.apiKey !== ENCRYPTED_PLACEHOLDER) {
|
|
49
|
+
const secretKey = `tenant:${config.tenantId}:emailTransportApiKey`;
|
|
50
|
+
this.secretStore.set(secretKey, config.emailTransportConfig.apiKey);
|
|
51
|
+
result.emailTransportConfig = { ...config.emailTransportConfig, apiKey: ENCRYPTED_PLACEHOLDER };
|
|
52
|
+
}
|
|
53
|
+
// Encrypt webhook tool secrets
|
|
54
|
+
if (config.webhookTools && config.webhookTools.length > 0) {
|
|
55
|
+
const encryptedTools = config.webhookTools.map(wt => {
|
|
56
|
+
if (wt.secret && wt.secret !== ENCRYPTED_PLACEHOLDER) {
|
|
57
|
+
const secretKey = `tenant:${config.tenantId}:webhookTool:${wt.name}`;
|
|
58
|
+
this.secretStore.set(secretKey, wt.secret);
|
|
59
|
+
return { ...wt, secret: ENCRYPTED_PLACEHOLDER };
|
|
60
|
+
}
|
|
61
|
+
return wt;
|
|
62
|
+
});
|
|
63
|
+
result.webhookTools = encryptedTools;
|
|
64
|
+
}
|
|
47
65
|
return result;
|
|
48
66
|
}
|
|
49
67
|
hydrateSecrets(config) {
|
|
@@ -59,14 +77,43 @@ export class TenantRegistry {
|
|
|
59
77
|
result[field] = decrypted;
|
|
60
78
|
}
|
|
61
79
|
}
|
|
80
|
+
// Hydrate email transport API key
|
|
81
|
+
if (config.emailTransportConfig?.apiKey === ENCRYPTED_PLACEHOLDER) {
|
|
82
|
+
const secretKey = `tenant:${config.tenantId}:emailTransportApiKey`;
|
|
83
|
+
const decrypted = this.secretStore.get(secretKey);
|
|
84
|
+
if (decrypted) {
|
|
85
|
+
result.emailTransportConfig = { ...config.emailTransportConfig, apiKey: decrypted };
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
// Hydrate webhook tool secrets
|
|
89
|
+
if (config.webhookTools && config.webhookTools.length > 0) {
|
|
90
|
+
const hydratedTools = config.webhookTools.map(wt => {
|
|
91
|
+
if (wt.secret === ENCRYPTED_PLACEHOLDER) {
|
|
92
|
+
const secretKey = `tenant:${config.tenantId}:webhookTool:${wt.name}`;
|
|
93
|
+
const decrypted = this.secretStore.get(secretKey);
|
|
94
|
+
if (decrypted)
|
|
95
|
+
return { ...wt, secret: decrypted };
|
|
96
|
+
}
|
|
97
|
+
return wt;
|
|
98
|
+
});
|
|
99
|
+
result.webhookTools = hydratedTools;
|
|
100
|
+
}
|
|
62
101
|
return result;
|
|
63
102
|
}
|
|
64
|
-
deleteSecrets(tenantId) {
|
|
103
|
+
deleteSecrets(tenantId, config) {
|
|
65
104
|
if (!this.secretStore)
|
|
66
105
|
return;
|
|
67
106
|
for (const field of SENSITIVE_FIELDS) {
|
|
68
107
|
this.secretStore.delete(`tenant:${tenantId}:${field}`);
|
|
69
108
|
}
|
|
109
|
+
// Delete email transport API key
|
|
110
|
+
this.secretStore.delete(`tenant:${tenantId}:emailTransportApiKey`);
|
|
111
|
+
// Delete webhook tool secrets if config is available
|
|
112
|
+
if (config?.webhookTools) {
|
|
113
|
+
for (const wt of config.webhookTools) {
|
|
114
|
+
this.secretStore.delete(`tenant:${tenantId}:webhookTool:${wt.name}`);
|
|
115
|
+
}
|
|
116
|
+
}
|
|
70
117
|
}
|
|
71
118
|
load() {
|
|
72
119
|
if (!existsSync(this.storageDir))
|
|
@@ -128,10 +175,11 @@ export class TenantRegistry {
|
|
|
128
175
|
return this.hydrateSecrets(encrypted);
|
|
129
176
|
}
|
|
130
177
|
remove(tenantId) {
|
|
131
|
-
|
|
178
|
+
const existing = this.tenants.get(tenantId);
|
|
179
|
+
if (!existing)
|
|
132
180
|
return false;
|
|
133
181
|
this.tenants.delete(tenantId);
|
|
134
|
-
this.deleteSecrets(tenantId);
|
|
182
|
+
this.deleteSecrets(tenantId, existing);
|
|
135
183
|
const filePath = join(this.storageDir, `${tenantId}.json`);
|
|
136
184
|
if (existsSync(filePath))
|
|
137
185
|
unlinkSync(filePath);
|
|
@@ -157,6 +205,37 @@ export class TenantRegistry {
|
|
|
157
205
|
}
|
|
158
206
|
return undefined;
|
|
159
207
|
}
|
|
208
|
+
resolveByInstagramPageId(pageId, appName) {
|
|
209
|
+
for (const tenant of this.tenants.values()) {
|
|
210
|
+
if (tenant.instagramPageId === pageId &&
|
|
211
|
+
tenant.appName === appName &&
|
|
212
|
+
tenant.enabled) {
|
|
213
|
+
return tenant;
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
return undefined;
|
|
217
|
+
}
|
|
218
|
+
resolveByMessengerPageId(pageId, appName) {
|
|
219
|
+
for (const tenant of this.tenants.values()) {
|
|
220
|
+
if (tenant.messengerPageId === pageId &&
|
|
221
|
+
tenant.appName === appName &&
|
|
222
|
+
tenant.enabled) {
|
|
223
|
+
return tenant;
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
return undefined;
|
|
227
|
+
}
|
|
228
|
+
resolveByEmailAddress(emailAddress, appName) {
|
|
229
|
+
const normalized = emailAddress.toLowerCase();
|
|
230
|
+
for (const tenant of this.tenants.values()) {
|
|
231
|
+
if (tenant.emailAddress?.toLowerCase() === normalized &&
|
|
232
|
+
tenant.appName === appName &&
|
|
233
|
+
tenant.enabled) {
|
|
234
|
+
return tenant;
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
return undefined;
|
|
238
|
+
}
|
|
160
239
|
persist(config) {
|
|
161
240
|
mkdirSync(this.storageDir, { recursive: true });
|
|
162
241
|
const filePath = join(this.storageDir, `${config.tenantId}.json`);
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"tenant-registry.js","sourceRoot":"","sources":["../../src/tenant/tenant-registry.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,aAAa,EAAE,YAAY,EAAE,WAAW,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACtG,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,SAAS,EAAE,MAAM,cAAc,CAAC;AAEzC,OAAO,EAAE,oBAAoB,EAAE,MAAM,cAAc,CAAC;AAEpD,MAAM,gBAAgB,GAAG,CAAC,qBAAqB,EAAE,qBAAqB,CAAU,CAAC;
|
|
1
|
+
{"version":3,"file":"tenant-registry.js","sourceRoot":"","sources":["../../src/tenant/tenant-registry.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import type { ContentPart, TenantRoutingConfig, TenantAgentConfig } from "@kilnai/core";
|
|
2
|
+
import type { AgentRAG } from "@kilnai/core";
|
|
3
|
+
export interface RoutingResult {
|
|
4
|
+
readonly agentId: string;
|
|
5
|
+
readonly tier: "rule" | "embedding" | "fallback";
|
|
6
|
+
readonly matchedPattern?: string;
|
|
7
|
+
readonly confidence?: number;
|
|
8
|
+
}
|
|
9
|
+
export interface TenantRouter {
|
|
10
|
+
route(userParts: readonly ContentPart[]): RoutingResult;
|
|
11
|
+
}
|
|
12
|
+
export declare class DefaultTenantRouter implements TenantRouter {
|
|
13
|
+
private readonly compiledRules;
|
|
14
|
+
private readonly fallbackAgentId;
|
|
15
|
+
constructor(config: TenantRoutingConfig);
|
|
16
|
+
route(userParts: readonly ContentPart[]): RoutingResult;
|
|
17
|
+
}
|
|
18
|
+
export declare class EmbeddingTenantRouter implements TenantRouter {
|
|
19
|
+
private readonly regexRouter;
|
|
20
|
+
private readonly agentRag;
|
|
21
|
+
private readonly agents;
|
|
22
|
+
private readonly embeddingThreshold;
|
|
23
|
+
constructor(config: TenantRoutingConfig, agentRag: AgentRAG, agents: readonly TenantAgentConfig[]);
|
|
24
|
+
route(userParts: readonly ContentPart[]): RoutingResult;
|
|
25
|
+
routeAsync(userParts: readonly ContentPart[]): Promise<RoutingResult>;
|
|
26
|
+
}
|
|
27
|
+
//# sourceMappingURL=tenant-router.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tenant-router.d.ts","sourceRoot":"","sources":["../../src/tenant/tenant-router.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,WAAW,EAAE,mBAAmB,EAAE,iBAAiB,EAAE,MAAM,cAAc,CAAC;AAExF,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,cAAc,CAAC;AAE7C,MAAM,WAAW,aAAa;IAC5B,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC;IACzB,QAAQ,CAAC,IAAI,EAAE,MAAM,GAAG,WAAW,GAAG,UAAU,CAAC;IACjD,QAAQ,CAAC,cAAc,CAAC,EAAE,MAAM,CAAC;IACjC,QAAQ,CAAC,UAAU,CAAC,EAAE,MAAM,CAAC;CAC9B;AAED,MAAM,WAAW,YAAY;IAC3B,KAAK,CAAC,SAAS,EAAE,SAAS,WAAW,EAAE,GAAG,aAAa,CAAC;CACzD;AAQD,qBAAa,mBAAoB,YAAW,YAAY;IACtD,OAAO,CAAC,QAAQ,CAAC,aAAa,CAA0B;IACxD,OAAO,CAAC,QAAQ,CAAC,eAAe,CAAS;gBAE7B,MAAM,EAAE,mBAAmB;IAkBvC,KAAK,CAAC,SAAS,EAAE,SAAS,WAAW,EAAE,GAAG,aAAa;CAkBxD;AAED,qBAAa,qBAAsB,YAAW,YAAY;IACxD,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAsB;IAClD,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAW;IACpC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAA+B;IACtD,OAAO,CAAC,QAAQ,CAAC,kBAAkB,CAAS;gBAEhC,MAAM,EAAE,mBAAmB,EAAE,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,SAAS,iBAAiB,EAAE;IAOjG,KAAK,CAAC,SAAS,EAAE,SAAS,WAAW,EAAE,GAAG,aAAa;IAIjD,UAAU,CAAC,SAAS,EAAE,SAAS,WAAW,EAAE,GAAG,OAAO,CAAC,aAAa,CAAC;CAgB5E"}
|
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
// Multi-agent routing: selects which agent handles a message based on regex rules + fallback
|
|
2
|
+
// Stateless, created per-call. Fail-open on broken regex rules.
|
|
3
|
+
import { extractText } from "@kilnai/core";
|
|
4
|
+
export class DefaultTenantRouter {
|
|
5
|
+
compiledRules;
|
|
6
|
+
fallbackAgentId;
|
|
7
|
+
constructor(config) {
|
|
8
|
+
this.fallbackAgentId = config.fallback;
|
|
9
|
+
const rules = [];
|
|
10
|
+
for (const rule of config.rules ?? []) {
|
|
11
|
+
try {
|
|
12
|
+
rules.push({
|
|
13
|
+
regex: new RegExp(rule.match, "i"),
|
|
14
|
+
agentId: rule.agent,
|
|
15
|
+
pattern: rule.match,
|
|
16
|
+
});
|
|
17
|
+
}
|
|
18
|
+
catch {
|
|
19
|
+
// Skip broken regex -- fail-open
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
this.compiledRules = rules;
|
|
23
|
+
}
|
|
24
|
+
route(userParts) {
|
|
25
|
+
const text = extractText(userParts);
|
|
26
|
+
for (const rule of this.compiledRules) {
|
|
27
|
+
if (rule.regex.test(text)) {
|
|
28
|
+
return {
|
|
29
|
+
agentId: rule.agentId,
|
|
30
|
+
tier: "rule",
|
|
31
|
+
matchedPattern: rule.pattern,
|
|
32
|
+
};
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
return {
|
|
36
|
+
agentId: this.fallbackAgentId,
|
|
37
|
+
tier: "fallback",
|
|
38
|
+
};
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
export class EmbeddingTenantRouter {
|
|
42
|
+
regexRouter;
|
|
43
|
+
agentRag;
|
|
44
|
+
agents;
|
|
45
|
+
embeddingThreshold;
|
|
46
|
+
constructor(config, agentRag, agents) {
|
|
47
|
+
this.regexRouter = new DefaultTenantRouter(config);
|
|
48
|
+
this.agentRag = agentRag;
|
|
49
|
+
this.agents = agents;
|
|
50
|
+
this.embeddingThreshold = config.embeddingThreshold ?? 0.75;
|
|
51
|
+
}
|
|
52
|
+
route(userParts) {
|
|
53
|
+
return this.regexRouter.route(userParts);
|
|
54
|
+
}
|
|
55
|
+
async routeAsync(userParts) {
|
|
56
|
+
const regexResult = this.regexRouter.route(userParts);
|
|
57
|
+
if (regexResult.tier === "rule")
|
|
58
|
+
return regexResult;
|
|
59
|
+
try {
|
|
60
|
+
const text = extractText(userParts);
|
|
61
|
+
const ragResult = await this.agentRag.selectAgent(text, this.agents);
|
|
62
|
+
if (ragResult && ragResult.score >= this.embeddingThreshold) {
|
|
63
|
+
return { agentId: ragResult.agentId, tier: "embedding", confidence: ragResult.score };
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
catch {
|
|
67
|
+
// Fail-open: fall through to fallback
|
|
68
|
+
}
|
|
69
|
+
return regexResult;
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
//# sourceMappingURL=tenant-router.js.map
|