feat(rtk): add Kiro format support for tool result compression (#1194)

- Add compressKiroFormat() to handle conversationState.history and currentMessage
- Compress toolResults[].content[].text in Kiro's AWS CodeWhisperer format
- Preserve error tool results (status === 'error')
- Add 7 comprehensive tests covering all edge cases
- Verified with real usage: 13.6% savings on npm install output
This commit is contained in:
Zanuar Tri Romadon 2026-05-17 15:13:37 +07:00 committed by GitHub
parent 6d383224a4
commit e03b28138a
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
2 changed files with 327 additions and 0 deletions

View file

@ -8,6 +8,12 @@ import { safeApply } from "./applyFilter.js";
export function compressMessages(body, enabled) {
if (!enabled) return null;
if (!body) return null;
// Kiro format: conversationState.history + conversationState.currentMessage
if (body.conversationState) {
return compressKiroFormat(body, enabled);
}
// Support both OpenAI/Claude "messages" and OpenAI Responses "input"
const items = Array.isArray(body.messages) ? body.messages
: Array.isArray(body.input) ? body.input
@ -81,6 +87,36 @@ export function compressMessages(body, enabled) {
return stats;
}
// Compress Kiro format: conversationState.history[].userInputMessage.userInputMessageContext.toolResults[].content[].text
function compressKiroFormat(body, enabled) {
const stats = { bytesBefore: 0, bytesAfter: 0, hits: [] };
try {
const state = body.conversationState;
const allMessages = [...(Array.isArray(state?.history) ? state.history : [])];
if (state?.currentMessage) allMessages.push(state.currentMessage);
for (const msg of allMessages) {
const toolResults = msg?.userInputMessage?.userInputMessageContext?.toolResults;
if (!Array.isArray(toolResults)) continue;
for (const tr of toolResults) {
if (tr.status === "error") continue; // preserve error traces
if (!Array.isArray(tr.content)) continue;
for (const part of tr.content) {
if (part && typeof part.text === "string") {
part.text = compressText(part.text, stats, "kiro-tool-result");
}
}
}
}
} catch (e) {
console.warn("[RTK] compressKiroFormat error:", e.message);
return null;
}
return stats;
}
function compressText(text, stats, shape) {
const bytesIn = text.length;
stats.bytesBefore += bytesIn;

291
tests/unit/rtkKiro.test.js Normal file
View file

@ -0,0 +1,291 @@
// Tests for Kiro format RTK support
// Verifies that RTK compression works with Kiro's conversationState format
import { describe, it, expect } from "vitest";
import { compressMessages } from "../../open-sse/rtk/index.js";
describe("Kiro format RTK support", () => {
it("compresses tool results in Kiro conversationState.currentMessage", () => {
const kiroBody = {
conversationState: {
chatTriggerType: "MANUAL",
conversationId: "test-123",
currentMessage: {
userInputMessage: {
content: "Install express",
modelId: "claude-sonnet-4.5",
userInputMessageContext: {
toolResults: [
{
toolUseId: "tool_1",
status: "success",
content: [
{
text: [
"npm warn deprecated har-validator@5.1.5: this library is no longer supported",
"npm warn deprecated uuid@3.4.0: uuid@10 and below is no longer supported",
"npm warn deprecated request@2.88.2: request has been deprecated",
"npm warn deprecated inflight@1.0.6: This module is not supported",
"npm warn deprecated glob@7.2.3: Glob versions prior to v9 are no longer supported",
"npm warn deprecated rimraf@2.7.1: Rimraf versions prior to v4 are no longer supported",
"",
"added 47 packages, and audited 48 packages in 13s",
"",
"3 packages are looking for funding",
" run `npm fund` for details",
"",
"4 vulnerabilities (2 moderate, 2 critical)",
"",
"Some issues need review, and may require choosing",
"a different dependency.",
"",
"Run `npm audit` for details."
].join("\n")
}
]
}
]
}
}
},
history: []
}
};
const stats = compressMessages(kiroBody, true);
expect(stats).not.toBeNull();
expect(stats.bytesBefore).toBeGreaterThan(500);
expect(stats.bytesAfter).toBeLessThan(stats.bytesBefore);
expect(stats.hits.length).toBe(1);
expect(stats.hits[0].filter).toBe("build-output");
expect(stats.hits[0].shape).toBe("kiro-tool-result");
// Verify compression happened
const savedBytes = stats.bytesBefore - stats.bytesAfter;
expect(savedBytes).toBeGreaterThan(0);
const savedPercent = (savedBytes / stats.bytesBefore) * 100;
expect(savedPercent).toBeGreaterThan(10); // At least 10% savings
});
it("compresses tool results in Kiro conversationState.history", () => {
// Need >500 bytes for compression to trigger
const compilingLines = [];
for (let i = 1; i <= 20; i++) {
compilingLines.push(` Compiling package-${i} v1.0.${i}`);
}
compilingLines.push(" Finished `dev` profile [unoptimized + debuginfo] target(s) in 12.34s");
const kiroBody = {
conversationState: {
chatTriggerType: "MANUAL",
conversationId: "test-456",
currentMessage: {
userInputMessage: {
content: "What happened?",
modelId: "claude-sonnet-4.5"
}
},
history: [
{
userInputMessage: {
content: "Run cargo build",
modelId: "claude-sonnet-4.5",
userInputMessageContext: {
toolResults: [
{
toolUseId: "tool_2",
status: "success",
content: [
{
text: compilingLines.join("\n")
}
]
}
]
}
}
}
]
}
};
const stats = compressMessages(kiroBody, true);
expect(stats).not.toBeNull();
expect(stats.hits.length).toBe(1);
expect(stats.hits[0].filter).toBe("build-output");
expect(stats.bytesAfter).toBeLessThan(stats.bytesBefore);
});
it("handles multiple tool results across history and currentMessage", () => {
// Need >500 bytes for each tool result to trigger compression
const deprecations1 = [];
const deprecations2 = [];
for (let i = 1; i <= 10; i++) {
deprecations1.push(`npm warn deprecated package-${i}@1.0.0: This version is deprecated`);
deprecations2.push(`npm warn deprecated lib-${i}@2.0.0: This library is no longer supported`);
}
deprecations1.push("added 50 packages in 5s");
deprecations2.push("added 1 package in 2s");
const kiroBody = {
conversationState: {
chatTriggerType: "MANUAL",
conversationId: "test-789",
currentMessage: {
userInputMessage: {
content: "Install lodash",
modelId: "claude-sonnet-4.5",
userInputMessageContext: {
toolResults: [
{
toolUseId: "tool_3",
status: "success",
content: [
{
text: deprecations2.join("\n")
}
]
}
]
}
}
},
history: [
{
userInputMessage: {
content: "Install express",
modelId: "claude-sonnet-4.5",
userInputMessageContext: {
toolResults: [
{
toolUseId: "tool_4",
status: "success",
content: [
{
text: deprecations1.join("\n")
}
]
}
]
}
}
}
]
}
};
const stats = compressMessages(kiroBody, true);
expect(stats).not.toBeNull();
expect(stats.hits.length).toBe(2); // Both tool results compressed
expect(stats.hits.every(h => h.filter === "build-output")).toBe(true);
});
it("preserves error tool results without compression", () => {
const kiroBody = {
conversationState: {
chatTriggerType: "MANUAL",
conversationId: "test-error",
currentMessage: {
userInputMessage: {
content: "Install invalid-package",
modelId: "claude-sonnet-4.5",
userInputMessageContext: {
toolResults: [
{
toolUseId: "tool_5",
status: "error",
content: [
{
text: "npm error code E404\nnpm error 404 Not Found - GET https://registry.npmjs.org/invalid-package"
}
]
}
]
}
}
},
history: []
}
};
const originalText = kiroBody.conversationState.currentMessage.userInputMessage.userInputMessageContext.toolResults[0].content[0].text;
const stats = compressMessages(kiroBody, true);
expect(stats).not.toBeNull();
expect(stats.hits.length).toBe(0); // Error not compressed
// Verify error text unchanged
const afterText = kiroBody.conversationState.currentMessage.userInputMessage.userInputMessageContext.toolResults[0].content[0].text;
expect(afterText).toBe(originalText);
});
it("returns null when RTK is disabled", () => {
const kiroBody = {
conversationState: {
chatTriggerType: "MANUAL",
conversationId: "test-disabled",
currentMessage: {
userInputMessage: {
content: "Test",
modelId: "claude-sonnet-4.5",
userInputMessageContext: {
toolResults: [
{
toolUseId: "tool_6",
status: "success",
content: [{ text: "npm install express\nadded 50 packages" }]
}
]
}
}
},
history: []
}
};
const stats = compressMessages(kiroBody, false); // RTK disabled
expect(stats).toBeNull();
});
it("handles Kiro body with no tool results gracefully", () => {
const kiroBody = {
conversationState: {
chatTriggerType: "MANUAL",
conversationId: "test-no-tools",
currentMessage: {
userInputMessage: {
content: "Hello",
modelId: "claude-sonnet-4.5"
}
},
history: []
}
};
const stats = compressMessages(kiroBody, true);
expect(stats).not.toBeNull();
expect(stats.hits.length).toBe(0);
expect(stats.bytesBefore).toBe(0);
expect(stats.bytesAfter).toBe(0);
});
it("handles malformed Kiro body without crashing", () => {
const malformedBodies = [
{ conversationState: null },
{ conversationState: {} },
{ conversationState: { history: null, currentMessage: null } },
{ conversationState: { history: "not-an-array" } }
];
for (const body of malformedBodies) {
const stats = compressMessages(body, true);
// Malformed bodies may return null (caught by try-catch) or empty stats
if (stats !== null) {
expect(stats.hits.length).toBe(0);
}
}
});
});