@kilnai/runtime 0.1.14 → 0.1.16
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 +5 -1
- 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/channels/whatsapp-api.d.ts +40 -1
- package/dist/channels/whatsapp-api.d.ts.map +1 -1
- package/dist/channels/whatsapp-api.js +32 -1
- package/dist/channels/whatsapp-api.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/gateway-routes.d.ts.map +1 -1
- package/dist/gateway/gateway-routes.js +23 -1
- package/dist/gateway/gateway-routes.js.map +1 -1
- package/dist/gateway/gateway-server.js +2 -2
- 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 +214 -0
- package/dist/gateway/handoff-routes.js.map +1 -0
- package/dist/gateway/message-pipeline.d.ts +49 -0
- package/dist/gateway/message-pipeline.d.ts.map +1 -0
- package/dist/gateway/message-pipeline.js +99 -0
- package/dist/gateway/message-pipeline.js.map +1 -0
- package/dist/gateway/mode-b-routes.d.ts.map +1 -1
- package/dist/gateway/mode-b-routes.js +29 -38
- package/dist/gateway/mode-b-routes.js.map +1 -1
- package/dist/gateway/outbound-routes.d.ts +9 -0
- package/dist/gateway/outbound-routes.d.ts.map +1 -0
- package/dist/gateway/outbound-routes.js +66 -0
- package/dist/gateway/outbound-routes.js.map +1 -0
- package/dist/gateway/tenant-admin-routes.js +3 -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/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/whatsapp-webhook-routes.d.ts.map +1 -1
- package/dist/gateway/whatsapp-webhook-routes.js +68 -15
- package/dist/gateway/whatsapp-webhook-routes.js.map +1 -1
- package/dist/gateway/ws-tenant-routes.d.ts.map +1 -1
- package/dist/gateway/ws-tenant-routes.js +35 -1
- package/dist/gateway/ws-tenant-routes.js.map +1 -1
- package/dist/index.d.ts +21 -2
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +13 -1
- package/dist/index.js.map +1 -1
- 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 +7 -0
- package/dist/session/mode-b-orchestrator.d.ts.map +1 -1
- package/dist/session/mode-b-orchestrator.js +65 -1
- package/dist/session/mode-b-orchestrator.js.map +1 -1
- package/dist/session/mode-b-session.d.ts +27 -0
- package/dist/session/mode-b-session.d.ts.map +1 -1
- package/dist/session/mode-b-session.js +65 -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 +16 -9
- package/dist/session/session-registry.d.ts.map +1 -1
- package/dist/session/session-registry.js +47 -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 +22 -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/package.json +7 -3
|
@@ -0,0 +1,214 @@
|
|
|
1
|
+
// Gateway: Handoff routes -- Hono sub-app for operator message injection and session mode transitions
|
|
2
|
+
// Enables human-in-the-loop workflows: escalate to human, send operator messages, release back to AI
|
|
3
|
+
import { Hono } from "hono";
|
|
4
|
+
import { KilnError, textParts } from "@kilnai/core";
|
|
5
|
+
import { sendWhatsAppMessage } from "../channels/whatsapp-api.js";
|
|
6
|
+
import { requireBearer } from "./auth-middleware.js";
|
|
7
|
+
import { TraceContext } from "./trace-context.js";
|
|
8
|
+
export function createHandoffRoutes(config) {
|
|
9
|
+
const app = new Hono();
|
|
10
|
+
if (config.adminToken) {
|
|
11
|
+
app.use("*", requireBearer(config.adminToken));
|
|
12
|
+
}
|
|
13
|
+
else {
|
|
14
|
+
console.warn("[handoff-routes] No adminToken configured -- handoff endpoints are unauthenticated");
|
|
15
|
+
}
|
|
16
|
+
// POST /handoff -- Initiate handoff (transition to queued or human_active)
|
|
17
|
+
app.post("/handoff", async (c) => {
|
|
18
|
+
const trace = new TraceContext();
|
|
19
|
+
let body;
|
|
20
|
+
try {
|
|
21
|
+
body = await c.req.json();
|
|
22
|
+
}
|
|
23
|
+
catch {
|
|
24
|
+
return c.json({ success: false, error: "Invalid JSON body" }, 400);
|
|
25
|
+
}
|
|
26
|
+
if (!body.tenantId || !body.userId || !body.targetMode) {
|
|
27
|
+
return c.json({ success: false, error: "Missing required fields: tenantId, userId, targetMode" }, 400);
|
|
28
|
+
}
|
|
29
|
+
trace.log("handoff", "Initiating handoff", { tenantId: body.tenantId, userId: body.userId, targetMode: body.targetMode });
|
|
30
|
+
const session = await config.sessionRegistry.get(config.appName, body.userId, body.tenantId);
|
|
31
|
+
if (!session) {
|
|
32
|
+
return c.json({ success: false, error: "Session not found" }, 404);
|
|
33
|
+
}
|
|
34
|
+
const previousMode = session.sessionMode;
|
|
35
|
+
try {
|
|
36
|
+
session.setSessionMode(body.targetMode);
|
|
37
|
+
await config.sessionRegistry.save(session);
|
|
38
|
+
}
|
|
39
|
+
catch (err) {
|
|
40
|
+
if (err instanceof KilnError && (err.code === "INVALID_SESSION_TRANSITION" || err.code === "CONCURRENT_SESSION_MODIFICATION")) {
|
|
41
|
+
return c.json({ success: false, error: err.message }, 409);
|
|
42
|
+
}
|
|
43
|
+
throw err;
|
|
44
|
+
}
|
|
45
|
+
if (config.eventEmitter && session.tenantId) {
|
|
46
|
+
config.eventEmitter.emit({
|
|
47
|
+
eventType: "HANDOFF_INITIATED",
|
|
48
|
+
tenantId: session.tenantId,
|
|
49
|
+
channel: "api",
|
|
50
|
+
externalUserId: body.userId,
|
|
51
|
+
sessionMode: session.sessionMode,
|
|
52
|
+
escalationReason: body.reason,
|
|
53
|
+
operatorId: body.operatorId,
|
|
54
|
+
traceId: trace.traceId,
|
|
55
|
+
timestamp: new Date().toISOString(),
|
|
56
|
+
});
|
|
57
|
+
}
|
|
58
|
+
trace.log("handoff", "Handoff complete", { sessionId: session.id, previousMode, newMode: session.sessionMode });
|
|
59
|
+
return c.json({
|
|
60
|
+
success: true,
|
|
61
|
+
sessionId: session.id,
|
|
62
|
+
previousMode,
|
|
63
|
+
newMode: session.sessionMode,
|
|
64
|
+
});
|
|
65
|
+
});
|
|
66
|
+
// POST /release -- Release session back to AI
|
|
67
|
+
app.post("/release", async (c) => {
|
|
68
|
+
const trace = new TraceContext();
|
|
69
|
+
let body;
|
|
70
|
+
try {
|
|
71
|
+
body = await c.req.json();
|
|
72
|
+
}
|
|
73
|
+
catch {
|
|
74
|
+
return c.json({ success: false, error: "Invalid JSON body" }, 400);
|
|
75
|
+
}
|
|
76
|
+
if (!body.tenantId || !body.userId) {
|
|
77
|
+
return c.json({ success: false, error: "Missing required fields: tenantId, userId" }, 400);
|
|
78
|
+
}
|
|
79
|
+
trace.log("handoff", "Releasing session to AI", { tenantId: body.tenantId, userId: body.userId });
|
|
80
|
+
const session = await config.sessionRegistry.get(config.appName, body.userId, body.tenantId);
|
|
81
|
+
if (!session) {
|
|
82
|
+
return c.json({ success: false, error: "Session not found" }, 404);
|
|
83
|
+
}
|
|
84
|
+
const previousMode = session.sessionMode;
|
|
85
|
+
try {
|
|
86
|
+
session.setSessionMode("ai_active");
|
|
87
|
+
if (body.contextSummary) {
|
|
88
|
+
session.addUserMessage(textParts("[Handoff context] " + body.contextSummary));
|
|
89
|
+
}
|
|
90
|
+
await config.sessionRegistry.save(session);
|
|
91
|
+
}
|
|
92
|
+
catch (err) {
|
|
93
|
+
if (err instanceof KilnError && (err.code === "INVALID_SESSION_TRANSITION" || err.code === "CONCURRENT_SESSION_MODIFICATION")) {
|
|
94
|
+
return c.json({ success: false, error: err.message }, 409);
|
|
95
|
+
}
|
|
96
|
+
throw err;
|
|
97
|
+
}
|
|
98
|
+
if (config.eventEmitter && session.tenantId) {
|
|
99
|
+
config.eventEmitter.emit({
|
|
100
|
+
eventType: "HANDOFF_RELEASED",
|
|
101
|
+
tenantId: session.tenantId,
|
|
102
|
+
channel: "api",
|
|
103
|
+
externalUserId: body.userId,
|
|
104
|
+
sessionMode: "ai_active",
|
|
105
|
+
summary: body.contextSummary,
|
|
106
|
+
traceId: trace.traceId,
|
|
107
|
+
timestamp: new Date().toISOString(),
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
trace.log("handoff", "Release complete", { sessionId: session.id, previousMode });
|
|
111
|
+
return c.json({
|
|
112
|
+
success: true,
|
|
113
|
+
sessionId: session.id,
|
|
114
|
+
previousMode,
|
|
115
|
+
newMode: "ai_active",
|
|
116
|
+
});
|
|
117
|
+
});
|
|
118
|
+
// POST /operator-message -- Send human-authored message to end user
|
|
119
|
+
app.post("/operator-message", async (c) => {
|
|
120
|
+
const trace = new TraceContext();
|
|
121
|
+
let body;
|
|
122
|
+
try {
|
|
123
|
+
body = await c.req.json();
|
|
124
|
+
}
|
|
125
|
+
catch {
|
|
126
|
+
return c.json({ success: false, error: "Invalid JSON body" }, 400);
|
|
127
|
+
}
|
|
128
|
+
if (!body.tenantId || !body.userId || !body.message || !body.channel) {
|
|
129
|
+
return c.json({ success: false, error: "Missing required fields: tenantId, userId, message, channel" }, 400);
|
|
130
|
+
}
|
|
131
|
+
trace.log("handoff", "Sending operator message", { tenantId: body.tenantId, userId: body.userId, channel: body.channel });
|
|
132
|
+
const session = await config.sessionRegistry.get(config.appName, body.userId, body.tenantId);
|
|
133
|
+
if (!session) {
|
|
134
|
+
return c.json({ success: false, error: "Session not found" }, 404);
|
|
135
|
+
}
|
|
136
|
+
if (session.sessionMode === "ai_active" || session.sessionMode === "resolved") {
|
|
137
|
+
return c.json({ success: false, error: `Cannot send operator message in ${session.sessionMode} mode` }, 409);
|
|
138
|
+
}
|
|
139
|
+
// Inject message into session history as assistant message
|
|
140
|
+
session.injectOperatorMessage(textParts(body.message));
|
|
141
|
+
await config.sessionRegistry.save(session);
|
|
142
|
+
// Deliver via channel
|
|
143
|
+
if (body.channel === "web") {
|
|
144
|
+
if (config.webChannel) {
|
|
145
|
+
const payload = JSON.stringify({
|
|
146
|
+
type: "output",
|
|
147
|
+
text: body.message,
|
|
148
|
+
parts: textParts(body.message),
|
|
149
|
+
target: "session",
|
|
150
|
+
userId: body.userId,
|
|
151
|
+
});
|
|
152
|
+
config.webChannel.sendToUser(body.userId, payload);
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
else if (body.channel === "whatsapp") {
|
|
156
|
+
const tenant = config.tenantRegistry.get(body.tenantId);
|
|
157
|
+
if (!tenant || !tenant.whatsappPhoneNumberId || !tenant.whatsappAccessToken) {
|
|
158
|
+
return c.json({ success: false, error: "Tenant has no WhatsApp credentials configured" }, 422);
|
|
159
|
+
}
|
|
160
|
+
const accessToken = tenant.whatsappAccessToken.startsWith("$")
|
|
161
|
+
? (process.env[tenant.whatsappAccessToken.slice(1)] ?? tenant.whatsappAccessToken)
|
|
162
|
+
: tenant.whatsappAccessToken;
|
|
163
|
+
try {
|
|
164
|
+
await sendWhatsAppMessage(tenant.whatsappPhoneNumberId, accessToken, body.userId, {
|
|
165
|
+
type: "text",
|
|
166
|
+
text: { body: body.message },
|
|
167
|
+
});
|
|
168
|
+
}
|
|
169
|
+
catch (err) {
|
|
170
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
171
|
+
return c.json({ success: false, error: `WhatsApp delivery failed: ${message}` }, 502);
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
if (config.eventEmitter && session.tenantId) {
|
|
175
|
+
config.eventEmitter.emit({
|
|
176
|
+
eventType: "OPERATOR_MESSAGE_SENT",
|
|
177
|
+
tenantId: session.tenantId,
|
|
178
|
+
channel: body.channel,
|
|
179
|
+
externalUserId: body.userId,
|
|
180
|
+
messageContent: body.message,
|
|
181
|
+
messageRole: "operator",
|
|
182
|
+
operatorId: body.operatorId,
|
|
183
|
+
traceId: trace.traceId,
|
|
184
|
+
timestamp: new Date().toISOString(),
|
|
185
|
+
});
|
|
186
|
+
}
|
|
187
|
+
trace.log("handoff", "Operator message delivered", { channel: body.channel });
|
|
188
|
+
return c.json({ success: true, delivered: true });
|
|
189
|
+
});
|
|
190
|
+
// GET /session-history -- Get full conversation history
|
|
191
|
+
app.get("/session-history", async (c) => {
|
|
192
|
+
const trace = new TraceContext();
|
|
193
|
+
const tenantId = c.req.query("tenantId");
|
|
194
|
+
const userId = c.req.query("userId");
|
|
195
|
+
if (!tenantId || !userId) {
|
|
196
|
+
return c.json({ success: false, error: "Missing required query params: tenantId, userId" }, 400);
|
|
197
|
+
}
|
|
198
|
+
trace.log("handoff", "Retrieving session history", { tenantId, userId });
|
|
199
|
+
const session = await config.sessionRegistry.get(config.appName, userId, tenantId);
|
|
200
|
+
if (!session) {
|
|
201
|
+
return c.json({ success: false, error: "Session not found" }, 404);
|
|
202
|
+
}
|
|
203
|
+
return c.json({
|
|
204
|
+
sessionId: session.id,
|
|
205
|
+
mode: session.sessionMode,
|
|
206
|
+
messageCount: session.messageCount,
|
|
207
|
+
history: session.conversationHistory,
|
|
208
|
+
createdAt: session.createdAt.toISOString(),
|
|
209
|
+
lastActivityAt: session.lastActivityAt.toISOString(),
|
|
210
|
+
});
|
|
211
|
+
});
|
|
212
|
+
return app;
|
|
213
|
+
}
|
|
214
|
+
//# sourceMappingURL=handoff-routes.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"handoff-routes.js","sourceRoot":"","sources":["../../src/gateway/handoff-routes.ts"],"names":[],"mappings":"AAAA,sGAAsG;AACtG,qGAAqG;AAErG,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,EAAE,SAAS,EAAE,SAAS,EAAE,MAAM,cAAc,CAAC;AAKpD,OAAO,EAAE,mBAAmB,EAAE,MAAM,6BAA6B,CAAC;AAClE,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;AAiClD,MAAM,UAAU,mBAAmB,CAAC,MAA2B;IAC7D,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;IAEvB,IAAI,MAAM,CAAC,UAAU,EAAE,CAAC;QACtB,GAAG,CAAC,GAAG,CAAC,GAAG,EAAE,aAAa,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC;IACjD,CAAC;SAAM,CAAC;QACN,OAAO,CAAC,IAAI,CAAC,oFAAoF,CAAC,CAAC;IACrG,CAAC;IAED,2EAA2E;IAC3E,GAAG,CAAC,IAAI,CAAC,UAAU,EAAE,KAAK,EAAE,CAAC,EAAE,EAAE;QAC/B,MAAM,KAAK,GAAG,IAAI,YAAY,EAAE,CAAC;QAEjC,IAAI,IAAoB,CAAC;QACzB,IAAI,CAAC;YACH,IAAI,GAAG,MAAM,CAAC,CAAC,GAAG,CAAC,IAAI,EAAkB,CAAC;QAC5C,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,mBAAmB,EAAE,EAAE,GAAG,CAAC,CAAC;QACrE,CAAC;QAED,IAAI,CAAC,IAAI,CAAC,QAAQ,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC;YACvD,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,uDAAuD,EAAE,EAAE,GAAG,CAAC,CAAC;QACzG,CAAC;QAED,KAAK,CAAC,GAAG,CAAC,SAAS,EAAE,oBAAoB,EAAE,EAAE,QAAQ,EAAE,IAAI,CAAC,QAAQ,EAAE,MAAM,EAAE,IAAI,CAAC,MAAM,EAAE,UAAU,EAAE,IAAI,CAAC,UAAU,EAAE,CAAC,CAAC;QAE1H,MAAM,OAAO,GAAG,MAAM,MAAM,CAAC,eAAe,CAAC,GAAG,CAAC,MAAM,CAAC,OAAO,EAAE,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;QAC7F,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,mBAAmB,EAAE,EAAE,GAAG,CAAC,CAAC;QACrE,CAAC;QAED,MAAM,YAAY,GAAG,OAAO,CAAC,WAAW,CAAC;QAEzC,IAAI,CAAC;YACH,OAAO,CAAC,cAAc,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;YACxC,MAAM,MAAM,CAAC,eAAe,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC7C,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,IAAI,GAAG,YAAY,SAAS,IAAI,CAAC,GAAG,CAAC,IAAI,KAAK,4BAA4B,IAAI,GAAG,CAAC,IAAI,KAAK,iCAAiC,CAAC,EAAE,CAAC;gBAC9H,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,GAAG,CAAC,OAAO,EAAE,EAAE,GAAG,CAAC,CAAC;YAC7D,CAAC;YACD,MAAM,GAAG,CAAC;QACZ,CAAC;QAED,IAAI,MAAM,CAAC,YAAY,IAAI,OAAO,CAAC,QAAQ,EAAE,CAAC;YAC5C,MAAM,CAAC,YAAY,CAAC,IAAI,CAAC;gBACvB,SAAS,EAAE,mBAAmB;gBAC9B,QAAQ,EAAE,OAAO,CAAC,QAAQ;gBAC1B,OAAO,EAAE,KAAK;gBACd,cAAc,EAAE,IAAI,CAAC,MAAM;gBAC3B,WAAW,EAAE,OAAO,CAAC,WAAW;gBAChC,gBAAgB,EAAE,IAAI,CAAC,MAAM;gBAC7B,UAAU,EAAE,IAAI,CAAC,UAAU;gBAC3B,OAAO,EAAE,KAAK,CAAC,OAAO;gBACtB,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;aACpC,CAAC,CAAC;QACL,CAAC;QAED,KAAK,CAAC,GAAG,CAAC,SAAS,EAAE,kBAAkB,EAAE,EAAE,SAAS,EAAE,OAAO,CAAC,EAAE,EAAE,YAAY,EAAE,OAAO,EAAE,OAAO,CAAC,WAAW,EAAE,CAAC,CAAC;QAEhH,OAAO,CAAC,CAAC,IAAI,CAAC;YACZ,OAAO,EAAE,IAAI;YACb,SAAS,EAAE,OAAO,CAAC,EAAE;YACrB,YAAY;YACZ,OAAO,EAAE,OAAO,CAAC,WAAW;SAC7B,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,8CAA8C;IAC9C,GAAG,CAAC,IAAI,CAAC,UAAU,EAAE,KAAK,EAAE,CAAC,EAAE,EAAE;QAC/B,MAAM,KAAK,GAAG,IAAI,YAAY,EAAE,CAAC;QAEjC,IAAI,IAAoB,CAAC;QACzB,IAAI,CAAC;YACH,IAAI,GAAG,MAAM,CAAC,CAAC,GAAG,CAAC,IAAI,EAAkB,CAAC;QAC5C,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,mBAAmB,EAAE,EAAE,GAAG,CAAC,CAAC;QACrE,CAAC;QAED,IAAI,CAAC,IAAI,CAAC,QAAQ,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;YACnC,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,2CAA2C,EAAE,EAAE,GAAG,CAAC,CAAC;QAC7F,CAAC;QAED,KAAK,CAAC,GAAG,CAAC,SAAS,EAAE,yBAAyB,EAAE,EAAE,QAAQ,EAAE,IAAI,CAAC,QAAQ,EAAE,MAAM,EAAE,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC;QAElG,MAAM,OAAO,GAAG,MAAM,MAAM,CAAC,eAAe,CAAC,GAAG,CAAC,MAAM,CAAC,OAAO,EAAE,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;QAC7F,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,mBAAmB,EAAE,EAAE,GAAG,CAAC,CAAC;QACrE,CAAC;QAED,MAAM,YAAY,GAAG,OAAO,CAAC,WAAW,CAAC;QAEzC,IAAI,CAAC;YACH,OAAO,CAAC,cAAc,CAAC,WAAW,CAAC,CAAC;YACpC,IAAI,IAAI,CAAC,cAAc,EAAE,CAAC;gBACxB,OAAO,CAAC,cAAc,CAAC,SAAS,CAAC,oBAAoB,GAAG,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC;YAChF,CAAC;YACD,MAAM,MAAM,CAAC,eAAe,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC7C,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,IAAI,GAAG,YAAY,SAAS,IAAI,CAAC,GAAG,CAAC,IAAI,KAAK,4BAA4B,IAAI,GAAG,CAAC,IAAI,KAAK,iCAAiC,CAAC,EAAE,CAAC;gBAC9H,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,GAAG,CAAC,OAAO,EAAE,EAAE,GAAG,CAAC,CAAC;YAC7D,CAAC;YACD,MAAM,GAAG,CAAC;QACZ,CAAC;QAED,IAAI,MAAM,CAAC,YAAY,IAAI,OAAO,CAAC,QAAQ,EAAE,CAAC;YAC5C,MAAM,CAAC,YAAY,CAAC,IAAI,CAAC;gBACvB,SAAS,EAAE,kBAAkB;gBAC7B,QAAQ,EAAE,OAAO,CAAC,QAAQ;gBAC1B,OAAO,EAAE,KAAK;gBACd,cAAc,EAAE,IAAI,CAAC,MAAM;gBAC3B,WAAW,EAAE,WAAW;gBACxB,OAAO,EAAE,IAAI,CAAC,cAAc;gBAC5B,OAAO,EAAE,KAAK,CAAC,OAAO;gBACtB,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;aACpC,CAAC,CAAC;QACL,CAAC;QAED,KAAK,CAAC,GAAG,CAAC,SAAS,EAAE,kBAAkB,EAAE,EAAE,SAAS,EAAE,OAAO,CAAC,EAAE,EAAE,YAAY,EAAE,CAAC,CAAC;QAElF,OAAO,CAAC,CAAC,IAAI,CAAC;YACZ,OAAO,EAAE,IAAI;YACb,SAAS,EAAE,OAAO,CAAC,EAAE;YACrB,YAAY;YACZ,OAAO,EAAE,WAAW;SACrB,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,oEAAoE;IACpE,GAAG,CAAC,IAAI,CAAC,mBAAmB,EAAE,KAAK,EAAE,CAAC,EAAE,EAAE;QACxC,MAAM,KAAK,GAAG,IAAI,YAAY,EAAE,CAAC;QAEjC,IAAI,IAA4B,CAAC;QACjC,IAAI,CAAC;YACH,IAAI,GAAG,MAAM,CAAC,CAAC,GAAG,CAAC,IAAI,EAA0B,CAAC;QACpD,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,mBAAmB,EAAE,EAAE,GAAG,CAAC,CAAC;QACrE,CAAC;QAED,IAAI,CAAC,IAAI,CAAC,QAAQ,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,OAAO,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC;YACrE,OAAO,CAAC,CAAC,IAAI,CACX,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,6DAA6D,EAAE,EACxF,GAAG,CACJ,CAAC;QACJ,CAAC;QAED,KAAK,CAAC,GAAG,CAAC,SAAS,EAAE,0BAA0B,EAAE,EAAE,QAAQ,EAAE,IAAI,CAAC,QAAQ,EAAE,MAAM,EAAE,IAAI,CAAC,MAAM,EAAE,OAAO,EAAE,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC;QAE1H,MAAM,OAAO,GAAG,MAAM,MAAM,CAAC,eAAe,CAAC,GAAG,CAAC,MAAM,CAAC,OAAO,EAAE,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;QAC7F,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,mBAAmB,EAAE,EAAE,GAAG,CAAC,CAAC;QACrE,CAAC;QAED,IAAI,OAAO,CAAC,WAAW,KAAK,WAAW,IAAI,OAAO,CAAC,WAAW,KAAK,UAAU,EAAE,CAAC;YAC9E,OAAO,CAAC,CAAC,IAAI,CACX,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,mCAAmC,OAAO,CAAC,WAAW,OAAO,EAAE,EACxF,GAAG,CACJ,CAAC;QACJ,CAAC;QAED,2DAA2D;QAC3D,OAAO,CAAC,qBAAqB,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC;QACvD,MAAM,MAAM,CAAC,eAAe,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAE3C,sBAAsB;QACtB,IAAI,IAAI,CAAC,OAAO,KAAK,KAAK,EAAE,CAAC;YAC3B,IAAI,MAAM,CAAC,UAAU,EAAE,CAAC;gBACtB,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC;oBAC7B,IAAI,EAAE,QAAQ;oBACd,IAAI,EAAE,IAAI,CAAC,OAAO;oBAClB,KAAK,EAAE,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC;oBAC9B,MAAM,EAAE,SAAS;oBACjB,MAAM,EAAE,IAAI,CAAC,MAAM;iBACpB,CAAC,CAAC;gBACH,MAAM,CAAC,UAAU,CAAC,UAAU,CAAC,IAAI,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;YACrD,CAAC;QACH,CAAC;aAAM,IAAI,IAAI,CAAC,OAAO,KAAK,UAAU,EAAE,CAAC;YACvC,MAAM,MAAM,GAAG,MAAM,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;YACxD,IAAI,CAAC,MAAM,IAAI,CAAC,MAAM,CAAC,qBAAqB,IAAI,CAAC,MAAM,CAAC,mBAAmB,EAAE,CAAC;gBAC5E,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,+CAA+C,EAAE,EAAE,GAAG,CAAC,CAAC;YACjG,CAAC;YAED,MAAM,WAAW,GAAG,MAAM,CAAC,mBAAmB,CAAC,UAAU,CAAC,GAAG,CAAC;gBAC5D,CAAC,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,mBAAmB,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,MAAM,CAAC,mBAAmB,CAAC;gBAClF,CAAC,CAAC,MAAM,CAAC,mBAAmB,CAAC;YAE/B,IAAI,CAAC;gBACH,MAAM,mBAAmB,CAAC,MAAM,CAAC,qBAAqB,EAAE,WAAW,EAAE,IAAI,CAAC,MAAM,EAAE;oBAChF,IAAI,EAAE,MAAM;oBACZ,IAAI,EAAE,EAAE,IAAI,EAAE,IAAI,CAAC,OAAO,EAAE;iBAC7B,CAAC,CAAC;YACL,CAAC;YAAC,OAAO,GAAG,EAAE,CAAC;gBACb,MAAM,OAAO,GAAG,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;gBACjE,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,6BAA6B,OAAO,EAAE,EAAE,EAAE,GAAG,CAAC,CAAC;YACxF,CAAC;QACH,CAAC;QAED,IAAI,MAAM,CAAC,YAAY,IAAI,OAAO,CAAC,QAAQ,EAAE,CAAC;YAC5C,MAAM,CAAC,YAAY,CAAC,IAAI,CAAC;gBACvB,SAAS,EAAE,uBAAuB;gBAClC,QAAQ,EAAE,OAAO,CAAC,QAAQ;gBAC1B,OAAO,EAAE,IAAI,CAAC,OAAO;gBACrB,cAAc,EAAE,IAAI,CAAC,MAAM;gBAC3B,cAAc,EAAE,IAAI,CAAC,OAAO;gBAC5B,WAAW,EAAE,UAAU;gBACvB,UAAU,EAAE,IAAI,CAAC,UAAU;gBAC3B,OAAO,EAAE,KAAK,CAAC,OAAO;gBACtB,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;aACpC,CAAC,CAAC;QACL,CAAC;QAED,KAAK,CAAC,GAAG,CAAC,SAAS,EAAE,4BAA4B,EAAE,EAAE,OAAO,EAAE,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC;QAE9E,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IACpD,CAAC,CAAC,CAAC;IAEH,wDAAwD;IACxD,GAAG,CAAC,GAAG,CAAC,kBAAkB,EAAE,KAAK,EAAE,CAAC,EAAE,EAAE;QACtC,MAAM,KAAK,GAAG,IAAI,YAAY,EAAE,CAAC;QACjC,MAAM,QAAQ,GAAG,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QACzC,MAAM,MAAM,GAAG,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;QAErC,IAAI,CAAC,QAAQ,IAAI,CAAC,MAAM,EAAE,CAAC;YACzB,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,iDAAiD,EAAE,EAAE,GAAG,CAAC,CAAC;QACnG,CAAC;QAED,KAAK,CAAC,GAAG,CAAC,SAAS,EAAE,4BAA4B,EAAE,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAC,CAAC;QAEzE,MAAM,OAAO,GAAG,MAAM,MAAM,CAAC,eAAe,CAAC,GAAG,CAAC,MAAM,CAAC,OAAO,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAC;QACnF,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,mBAAmB,EAAE,EAAE,GAAG,CAAC,CAAC;QACrE,CAAC;QAED,OAAO,CAAC,CAAC,IAAI,CAAC;YACZ,SAAS,EAAE,OAAO,CAAC,EAAE;YACrB,IAAI,EAAE,OAAO,CAAC,WAAW;YACzB,YAAY,EAAE,OAAO,CAAC,YAAY;YAClC,OAAO,EAAE,OAAO,CAAC,mBAAmB;YACpC,SAAS,EAAE,OAAO,CAAC,SAAS,CAAC,WAAW,EAAE;YAC1C,cAAc,EAAE,OAAO,CAAC,cAAc,CAAC,WAAW,EAAE;SACrD,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,OAAO,GAAG,CAAC;AACb,CAAC"}
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
import type { ContentPart } from "@kilnai/core";
|
|
2
|
+
import type { ModeBOrchestrator } from "../session/mode-b-orchestrator.js";
|
|
3
|
+
import type { SessionRegistry } from "../session/session-registry.js";
|
|
4
|
+
import type { BillingConfig } from "./budget-middleware.js";
|
|
5
|
+
import type { ConversationEventEmitter } from "./conversation-event-emitter.js";
|
|
6
|
+
import type { SessionMode } from "../session/session-mode.js";
|
|
7
|
+
import type { EscalationSignal } from "../session/escalation-detector.js";
|
|
8
|
+
export interface InboundMessageContext {
|
|
9
|
+
readonly orchestrator: ModeBOrchestrator;
|
|
10
|
+
readonly sessionRegistry: SessionRegistry;
|
|
11
|
+
readonly appName: string;
|
|
12
|
+
readonly tenantId?: string;
|
|
13
|
+
readonly userId: string;
|
|
14
|
+
readonly systemPrompt: string;
|
|
15
|
+
readonly userParts: readonly ContentPart[];
|
|
16
|
+
readonly billing?: BillingConfig;
|
|
17
|
+
readonly eventEmitter?: ConversationEventEmitter;
|
|
18
|
+
readonly channel: string;
|
|
19
|
+
readonly idleTimeoutMs?: number;
|
|
20
|
+
readonly recalledMemory?: string;
|
|
21
|
+
readonly callBuiltinTools?: ReadonlyMap<string, (input: Record<string, unknown>) => Promise<unknown>>;
|
|
22
|
+
readonly traceId?: string;
|
|
23
|
+
}
|
|
24
|
+
export interface InboundMessageResult {
|
|
25
|
+
readonly parts: readonly ContentPart[];
|
|
26
|
+
readonly inputTokens: number;
|
|
27
|
+
readonly outputTokens: number;
|
|
28
|
+
readonly cacheReadTokens: number;
|
|
29
|
+
readonly cacheWriteTokens: number;
|
|
30
|
+
readonly queued: boolean;
|
|
31
|
+
readonly sessionId: string;
|
|
32
|
+
readonly sessionMode: SessionMode;
|
|
33
|
+
readonly escalation?: EscalationSignal;
|
|
34
|
+
readonly contextSummary?: string;
|
|
35
|
+
readonly traceId: string;
|
|
36
|
+
}
|
|
37
|
+
export interface BudgetDeniedResult {
|
|
38
|
+
readonly budgetExhausted: true;
|
|
39
|
+
readonly message: string;
|
|
40
|
+
}
|
|
41
|
+
export type ProcessResult = {
|
|
42
|
+
ok: true;
|
|
43
|
+
result: InboundMessageResult;
|
|
44
|
+
} | {
|
|
45
|
+
ok: false;
|
|
46
|
+
budgetDenied: BudgetDeniedResult;
|
|
47
|
+
};
|
|
48
|
+
export declare function processInboundMessage(ctx: InboundMessageContext): Promise<ProcessResult>;
|
|
49
|
+
//# sourceMappingURL=message-pipeline.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"message-pipeline.d.ts","sourceRoot":"","sources":["../../src/gateway/message-pipeline.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,cAAc,CAAC;AAChD,OAAO,KAAK,EAAE,iBAAiB,EAAqB,MAAM,mCAAmC,CAAC;AAC9F,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,gCAAgC,CAAC;AACtE,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AAE5D,OAAO,KAAK,EAAE,wBAAwB,EAAE,MAAM,iCAAiC,CAAC;AAChF,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,4BAA4B,CAAC;AAC9D,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,mCAAmC,CAAC;AAG1E,MAAM,WAAW,qBAAqB;IACpC,QAAQ,CAAC,YAAY,EAAE,iBAAiB,CAAC;IACzC,QAAQ,CAAC,eAAe,EAAE,eAAe,CAAC;IAC1C,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC;IACzB,QAAQ,CAAC,QAAQ,CAAC,EAAE,MAAM,CAAC;IAC3B,QAAQ,CAAC,MAAM,EAAE,MAAM,CAAC;IACxB,QAAQ,CAAC,YAAY,EAAE,MAAM,CAAC;IAC9B,QAAQ,CAAC,SAAS,EAAE,SAAS,WAAW,EAAE,CAAC;IAC3C,QAAQ,CAAC,OAAO,CAAC,EAAE,aAAa,CAAC;IACjC,QAAQ,CAAC,YAAY,CAAC,EAAE,wBAAwB,CAAC;IACjD,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC;IACzB,QAAQ,CAAC,aAAa,CAAC,EAAE,MAAM,CAAC;IAChC,QAAQ,CAAC,cAAc,CAAC,EAAE,MAAM,CAAC;IACjC,QAAQ,CAAC,gBAAgB,CAAC,EAAE,WAAW,CAAC,MAAM,EAAE,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,KAAK,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC;IACtG,QAAQ,CAAC,OAAO,CAAC,EAAE,MAAM,CAAC;CAC3B;AAED,MAAM,WAAW,oBAAoB;IACnC,QAAQ,CAAC,KAAK,EAAE,SAAS,WAAW,EAAE,CAAC;IACvC,QAAQ,CAAC,WAAW,EAAE,MAAM,CAAC;IAC7B,QAAQ,CAAC,YAAY,EAAE,MAAM,CAAC;IAC9B,QAAQ,CAAC,eAAe,EAAE,MAAM,CAAC;IACjC,QAAQ,CAAC,gBAAgB,EAAE,MAAM,CAAC;IAClC,QAAQ,CAAC,MAAM,EAAE,OAAO,CAAC;IACzB,QAAQ,CAAC,SAAS,EAAE,MAAM,CAAC;IAC3B,QAAQ,CAAC,WAAW,EAAE,WAAW,CAAC;IAClC,QAAQ,CAAC,UAAU,CAAC,EAAE,gBAAgB,CAAC;IACvC,QAAQ,CAAC,cAAc,CAAC,EAAE,MAAM,CAAC;IACjC,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC;CAC1B;AAED,MAAM,WAAW,kBAAkB;IACjC,QAAQ,CAAC,eAAe,EAAE,IAAI,CAAC;IAC/B,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC;CAC1B;AAED,MAAM,MAAM,aAAa,GACrB;IAAE,EAAE,EAAE,IAAI,CAAC;IAAC,MAAM,EAAE,oBAAoB,CAAA;CAAE,GAC1C;IAAE,EAAE,EAAE,KAAK,CAAC;IAAC,YAAY,EAAE,kBAAkB,CAAA;CAAE,CAAC;AAEpD,wBAAsB,qBAAqB,CAAC,GAAG,EAAE,qBAAqB,GAAG,OAAO,CAAC,aAAa,CAAC,CA8G9F"}
|
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
import { checkBudget, reportUsage } from "./budget-middleware.js";
|
|
2
|
+
import { TraceContext } from "./trace-context.js";
|
|
3
|
+
export async function processInboundMessage(ctx) {
|
|
4
|
+
const trace = new TraceContext(ctx.traceId);
|
|
5
|
+
trace.log("pipeline", "Processing inbound message", { appName: ctx.appName, userId: ctx.userId, channel: ctx.channel });
|
|
6
|
+
// Budget check
|
|
7
|
+
if (ctx.billing) {
|
|
8
|
+
const budgetResult = await checkBudget(ctx.billing, ctx.tenantId ?? ctx.userId);
|
|
9
|
+
if (!budgetResult.allowed) {
|
|
10
|
+
trace.log("pipeline", "Budget denied");
|
|
11
|
+
return {
|
|
12
|
+
ok: false,
|
|
13
|
+
budgetDenied: {
|
|
14
|
+
budgetExhausted: true,
|
|
15
|
+
message: ctx.billing.overBudgetMessage ?? "Budget exhausted.",
|
|
16
|
+
},
|
|
17
|
+
};
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
// Get or create session
|
|
21
|
+
const session = await ctx.sessionRegistry.getOrCreate({
|
|
22
|
+
appName: ctx.appName,
|
|
23
|
+
tenantId: ctx.tenantId,
|
|
24
|
+
userId: ctx.userId,
|
|
25
|
+
systemPrompt: ctx.systemPrompt,
|
|
26
|
+
idleTimeoutMs: ctx.idleTimeoutMs,
|
|
27
|
+
});
|
|
28
|
+
trace.log("pipeline", "Session ready", { sessionId: session.id, sessionMode: session.sessionMode });
|
|
29
|
+
// Process message
|
|
30
|
+
const result = await ctx.orchestrator.processMessage(session, ctx.userParts, ctx.recalledMemory, ctx.callBuiltinTools);
|
|
31
|
+
// Persist mutated session (required for non-reference stores like Redis)
|
|
32
|
+
await ctx.sessionRegistry.save(session);
|
|
33
|
+
// Report usage (fire-and-forget)
|
|
34
|
+
if (ctx.billing) {
|
|
35
|
+
reportUsage(ctx.billing, {
|
|
36
|
+
tenantId: ctx.tenantId ?? ctx.userId,
|
|
37
|
+
messages: 1,
|
|
38
|
+
tokens: result.inputTokens + result.outputTokens,
|
|
39
|
+
model: ctx.orchestrator.model ?? "unknown",
|
|
40
|
+
});
|
|
41
|
+
}
|
|
42
|
+
// Emit events (fire-and-forget)
|
|
43
|
+
if (ctx.eventEmitter && ctx.tenantId) {
|
|
44
|
+
ctx.eventEmitter.emit({
|
|
45
|
+
eventType: "MESSAGE_RECEIVED",
|
|
46
|
+
tenantId: ctx.tenantId,
|
|
47
|
+
channel: ctx.channel,
|
|
48
|
+
externalUserId: ctx.userId,
|
|
49
|
+
traceId: trace.traceId,
|
|
50
|
+
timestamp: new Date().toISOString(),
|
|
51
|
+
});
|
|
52
|
+
// Emit HANDOFF_MESSAGE_QUEUED when message was queued (session not ai_active)
|
|
53
|
+
if (result.queued) {
|
|
54
|
+
ctx.eventEmitter.emit({
|
|
55
|
+
eventType: "HANDOFF_MESSAGE_QUEUED",
|
|
56
|
+
tenantId: ctx.tenantId,
|
|
57
|
+
channel: ctx.channel,
|
|
58
|
+
externalUserId: ctx.userId,
|
|
59
|
+
sessionMode: session.sessionMode,
|
|
60
|
+
traceId: trace.traceId,
|
|
61
|
+
timestamp: new Date().toISOString(),
|
|
62
|
+
});
|
|
63
|
+
}
|
|
64
|
+
// Emit ESCALATION_DETECTED when escalation signal is present
|
|
65
|
+
if (result.escalation) {
|
|
66
|
+
trace.warn("pipeline", "Escalation detected", { reason: result.escalation.reason });
|
|
67
|
+
ctx.eventEmitter.emit({
|
|
68
|
+
eventType: "ESCALATION_DETECTED",
|
|
69
|
+
tenantId: ctx.tenantId,
|
|
70
|
+
channel: ctx.channel,
|
|
71
|
+
externalUserId: ctx.userId,
|
|
72
|
+
escalationReason: result.escalation.reason,
|
|
73
|
+
escalationDetail: result.escalation.detail,
|
|
74
|
+
summary: result.contextSummary,
|
|
75
|
+
sessionMode: session.sessionMode,
|
|
76
|
+
traceId: trace.traceId,
|
|
77
|
+
timestamp: new Date().toISOString(),
|
|
78
|
+
});
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
trace.log("pipeline", "Message processed", { queued: result.queued, tokens: result.inputTokens + result.outputTokens });
|
|
82
|
+
return {
|
|
83
|
+
ok: true,
|
|
84
|
+
result: {
|
|
85
|
+
parts: result.parts,
|
|
86
|
+
inputTokens: result.inputTokens,
|
|
87
|
+
outputTokens: result.outputTokens,
|
|
88
|
+
cacheReadTokens: result.cacheReadTokens,
|
|
89
|
+
cacheWriteTokens: result.cacheWriteTokens,
|
|
90
|
+
queued: result.queued,
|
|
91
|
+
sessionId: session.id,
|
|
92
|
+
sessionMode: session.sessionMode,
|
|
93
|
+
escalation: result.escalation,
|
|
94
|
+
contextSummary: result.contextSummary,
|
|
95
|
+
traceId: trace.traceId,
|
|
96
|
+
},
|
|
97
|
+
};
|
|
98
|
+
}
|
|
99
|
+
//# sourceMappingURL=message-pipeline.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"message-pipeline.js","sourceRoot":"","sources":["../../src/gateway/message-pipeline.ts"],"names":[],"mappings":"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"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"mode-b-routes.d.ts","sourceRoot":"","sources":["../../src/gateway/mode-b-routes.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAG5B,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,mCAAmC,CAAC;AAC3E,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,gCAAgC,CAAC;AAEtE,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;
|
|
1
|
+
{"version":3,"file":"mode-b-routes.d.ts","sourceRoot":"","sources":["../../src/gateway/mode-b-routes.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAG5B,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,mCAAmC,CAAC;AAC3E,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,gCAAgC,CAAC;AAEtE,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AAI5D,6CAA6C;AAC7C,MAAM,WAAW,eAAe;IAC9B,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC;IACzB,QAAQ,CAAC,YAAY,EAAE,iBAAiB,CAAC;IACzC,QAAQ,CAAC,eAAe,EAAE,eAAe,CAAC;IAC1C,QAAQ,CAAC,OAAO,CAAC,EAAE,aAAa,CAAC;IACjC,QAAQ,CAAC,YAAY,EAAE,MAAM,CAAC;IAC9B,QAAQ,CAAC,MAAM,CAAC,EAAE,MAAM,CAAC;CAC1B;AAUD,wBAAgB,iBAAiB,CAAC,OAAO,EAAE,eAAe,GAAG,IAAI,CA2FhE"}
|
|
@@ -2,8 +2,9 @@
|
|
|
2
2
|
// Handles message processing, session listing, and session removal
|
|
3
3
|
import { Hono } from "hono";
|
|
4
4
|
import { textParts, extractText } from "@kilnai/core";
|
|
5
|
-
import {
|
|
5
|
+
import { checkTier } from "./budget-middleware.js";
|
|
6
6
|
import { requireApiKey } from "./auth-middleware.js";
|
|
7
|
+
import { processInboundMessage } from "./message-pipeline.js";
|
|
7
8
|
export function createModeBRoutes(runtime) {
|
|
8
9
|
const app = new Hono();
|
|
9
10
|
if (runtime.apiKey) {
|
|
@@ -27,60 +28,50 @@ export function createModeBRoutes(runtime) {
|
|
|
27
28
|
if (!body.userId || typeof body.userId !== "string") {
|
|
28
29
|
return c.json({ error: "userId is required" }, 400);
|
|
29
30
|
}
|
|
30
|
-
//
|
|
31
|
-
if (runtime.billing) {
|
|
32
|
-
const
|
|
33
|
-
if (!
|
|
31
|
+
// Tier enforcement (Mode B specific -- not in the pipeline)
|
|
32
|
+
if (runtime.billing?.tiers && body.plan) {
|
|
33
|
+
const tierResult = checkTier(runtime.billing, body.plan, "fast");
|
|
34
|
+
if (!tierResult.allowed) {
|
|
34
35
|
return c.json({
|
|
35
|
-
content:
|
|
36
|
-
|
|
36
|
+
content: `Tier "fast" is not available on your "${body.plan}" plan. Allowed tiers: ${tierResult.allowedTiers.join(", ")}`,
|
|
37
|
+
tierRestricted: true,
|
|
37
38
|
});
|
|
38
39
|
}
|
|
39
|
-
// Tier enforcement
|
|
40
|
-
if (runtime.billing.tiers && body.plan) {
|
|
41
|
-
const tierResult = checkTier(runtime.billing, body.plan, "fast");
|
|
42
|
-
if (!tierResult.allowed) {
|
|
43
|
-
return c.json({
|
|
44
|
-
content: `Tier "fast" is not available on your "${body.plan}" plan. Allowed tiers: ${tierResult.allowedTiers.join(", ")}`,
|
|
45
|
-
tierRestricted: true,
|
|
46
|
-
});
|
|
47
|
-
}
|
|
48
|
-
}
|
|
49
40
|
}
|
|
50
|
-
|
|
51
|
-
const session = runtime.sessionRegistry.getOrCreate({
|
|
52
|
-
appName: runtime.appName,
|
|
53
|
-
userId: body.userId,
|
|
54
|
-
systemPrompt: runtime.systemPrompt,
|
|
55
|
-
});
|
|
56
|
-
// Process message
|
|
57
|
-
let result;
|
|
41
|
+
let processResult;
|
|
58
42
|
try {
|
|
59
|
-
|
|
43
|
+
processResult = await processInboundMessage({
|
|
44
|
+
orchestrator: runtime.orchestrator,
|
|
45
|
+
sessionRegistry: runtime.sessionRegistry,
|
|
46
|
+
appName: runtime.appName,
|
|
47
|
+
userId: body.userId,
|
|
48
|
+
systemPrompt: runtime.systemPrompt,
|
|
49
|
+
userParts,
|
|
50
|
+
billing: runtime.billing,
|
|
51
|
+
channel: "api",
|
|
52
|
+
});
|
|
60
53
|
}
|
|
61
54
|
catch (err) {
|
|
62
55
|
console.error(`[${runtime.appName}] processMessage error:`, err);
|
|
63
56
|
return c.json({ error: String(err) }, 500);
|
|
64
57
|
}
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
messages: 1,
|
|
70
|
-
tokens: result.inputTokens + result.outputTokens,
|
|
71
|
-
model: runtime.orchestrator.model ?? "unknown",
|
|
58
|
+
if (!processResult.ok) {
|
|
59
|
+
return c.json({
|
|
60
|
+
content: processResult.budgetDenied.message,
|
|
61
|
+
budgetExhausted: true,
|
|
72
62
|
});
|
|
73
63
|
}
|
|
64
|
+
const result = processResult.result;
|
|
74
65
|
return c.json({
|
|
75
66
|
content: extractText(result.parts),
|
|
76
67
|
parts: result.parts,
|
|
77
68
|
inputTokens: result.inputTokens,
|
|
78
69
|
outputTokens: result.outputTokens,
|
|
79
|
-
sessionId:
|
|
70
|
+
sessionId: result.sessionId,
|
|
80
71
|
});
|
|
81
72
|
});
|
|
82
|
-
app.get("/sessions", (c) => {
|
|
83
|
-
const sessions = runtime.sessionRegistry.activeSessions();
|
|
73
|
+
app.get("/sessions", async (c) => {
|
|
74
|
+
const sessions = await runtime.sessionRegistry.activeSessions();
|
|
84
75
|
return c.json({
|
|
85
76
|
sessions: sessions.map((s) => ({
|
|
86
77
|
id: s.id,
|
|
@@ -91,9 +82,9 @@ export function createModeBRoutes(runtime) {
|
|
|
91
82
|
})),
|
|
92
83
|
});
|
|
93
84
|
});
|
|
94
|
-
app.delete("/sessions/:userId", (c) => {
|
|
85
|
+
app.delete("/sessions/:userId", async (c) => {
|
|
95
86
|
const userId = c.req.param("userId");
|
|
96
|
-
const removed = runtime.sessionRegistry.remove(runtime.appName, userId);
|
|
87
|
+
const removed = await runtime.sessionRegistry.remove(runtime.appName, userId);
|
|
97
88
|
return c.json({ removed });
|
|
98
89
|
});
|
|
99
90
|
return app;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"mode-b-routes.js","sourceRoot":"","sources":["../../src/gateway/mode-b-routes.ts"],"names":[],"mappings":"AAAA,mEAAmE;AACnE,mEAAmE;AAEnE,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAE5B,OAAO,EAAE,SAAS,EAAE,WAAW,EAAE,MAAM,cAAc,CAAC;AAGtD,OAAO,EAAE,
|
|
1
|
+
{"version":3,"file":"mode-b-routes.js","sourceRoot":"","sources":["../../src/gateway/mode-b-routes.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { Hono } from "hono";
|
|
2
|
+
import type { TenantRegistry } from "../tenant/tenant-registry.js";
|
|
3
|
+
export interface OutboundRoutesConfig {
|
|
4
|
+
readonly tenantRegistry: TenantRegistry;
|
|
5
|
+
readonly appName: string;
|
|
6
|
+
readonly adminToken?: string;
|
|
7
|
+
}
|
|
8
|
+
export declare function createOutboundRoutes(config: OutboundRoutesConfig): Hono;
|
|
9
|
+
//# sourceMappingURL=outbound-routes.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"outbound-routes.d.ts","sourceRoot":"","sources":["../../src/gateway/outbound-routes.ts"],"names":[],"mappings":"AAIA,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,8BAA8B,CAAC;AAKnE,MAAM,WAAW,oBAAoB;IACnC,QAAQ,CAAC,cAAc,EAAE,cAAc,CAAC;IACxC,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC;IACzB,QAAQ,CAAC,UAAU,CAAC,EAAE,MAAM,CAAC;CAC9B;AAeD,wBAAgB,oBAAoB,CAAC,MAAM,EAAE,oBAAoB,GAAG,IAAI,CA8EvE"}
|
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
// Gateway: Outbound send routes -- Hono sub-app for business-initiated messages
|
|
2
|
+
// Generic mechanism: product backends call this to send messages through any channel.
|
|
3
|
+
// Kilvo (and future products) own the policy (when to send, to whom, compliance).
|
|
4
|
+
import { Hono } from "hono";
|
|
5
|
+
import { sendWhatsAppMessage, sendWhatsAppTemplate } from "../channels/whatsapp-api.js";
|
|
6
|
+
import { requireBearer } from "./auth-middleware.js";
|
|
7
|
+
export function createOutboundRoutes(config) {
|
|
8
|
+
const app = new Hono();
|
|
9
|
+
if (config.adminToken) {
|
|
10
|
+
app.use("*", requireBearer(config.adminToken));
|
|
11
|
+
}
|
|
12
|
+
app.post("/send", async (c) => {
|
|
13
|
+
let body;
|
|
14
|
+
try {
|
|
15
|
+
body = await c.req.json();
|
|
16
|
+
}
|
|
17
|
+
catch {
|
|
18
|
+
return c.json({ success: false, error: "Invalid JSON body" }, 400);
|
|
19
|
+
}
|
|
20
|
+
if (!body.tenantId || !body.channel || !body.to || !body.type) {
|
|
21
|
+
return c.json({ success: false, error: "Missing required fields: tenantId, channel, to, type" }, 400);
|
|
22
|
+
}
|
|
23
|
+
if (body.channel !== "whatsapp") {
|
|
24
|
+
return c.json({ success: false, error: `Unsupported channel: ${body.channel}` }, 400);
|
|
25
|
+
}
|
|
26
|
+
const tenant = config.tenantRegistry.get(body.tenantId);
|
|
27
|
+
if (!tenant || tenant.appName !== config.appName) {
|
|
28
|
+
return c.json({ success: false, error: "Tenant not found" }, 404);
|
|
29
|
+
}
|
|
30
|
+
if (!tenant.whatsappPhoneNumberId || !tenant.whatsappAccessToken) {
|
|
31
|
+
return c.json({ success: false, error: "Tenant has no WhatsApp credentials configured" }, 422);
|
|
32
|
+
}
|
|
33
|
+
const accessToken = tenant.whatsappAccessToken.startsWith("$")
|
|
34
|
+
? (process.env[tenant.whatsappAccessToken.slice(1)] ?? tenant.whatsappAccessToken)
|
|
35
|
+
: tenant.whatsappAccessToken;
|
|
36
|
+
try {
|
|
37
|
+
let result;
|
|
38
|
+
if (body.type === "template") {
|
|
39
|
+
if (!body.template?.name || !body.template?.language) {
|
|
40
|
+
return c.json({ success: false, error: "Template sends require template.name and template.language" }, 400);
|
|
41
|
+
}
|
|
42
|
+
result = await sendWhatsAppTemplate(tenant.whatsappPhoneNumberId, accessToken, body.to, body.template.name, body.template.language, body.template.components);
|
|
43
|
+
}
|
|
44
|
+
else {
|
|
45
|
+
if (!body.text) {
|
|
46
|
+
return c.json({ success: false, error: "Text sends require text field" }, 400);
|
|
47
|
+
}
|
|
48
|
+
const res = await sendWhatsAppMessage(tenant.whatsappPhoneNumberId, accessToken, body.to, { type: "text", text: { body: body.text } });
|
|
49
|
+
const json = (await res.json());
|
|
50
|
+
const messageId = json.messages?.[0]?.id;
|
|
51
|
+
if (!messageId) {
|
|
52
|
+
throw new Error("WhatsApp API returned no message ID");
|
|
53
|
+
}
|
|
54
|
+
result = { whatsappMessageId: messageId };
|
|
55
|
+
}
|
|
56
|
+
return c.json({ success: true, whatsappMessageId: result.whatsappMessageId });
|
|
57
|
+
}
|
|
58
|
+
catch (err) {
|
|
59
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
60
|
+
console.error(`[outbound] Send failed tenant=${body.tenantId} to=${body.to}: ${message}`);
|
|
61
|
+
return c.json({ success: false, error: message }, 502);
|
|
62
|
+
}
|
|
63
|
+
});
|
|
64
|
+
return app;
|
|
65
|
+
}
|
|
66
|
+
//# sourceMappingURL=outbound-routes.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"outbound-routes.js","sourceRoot":"","sources":["../../src/gateway/outbound-routes.ts"],"names":[],"mappings":"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"}
|