|
1 | 1 | import { describe, it, expect } from "@jest/globals"; |
2 | 2 | import type { ModelMessage, AssistantModelMessage, ToolModelMessage } from "ai"; |
3 | | -import { transformModelMessages, validateAnthropicCompliance } from "./modelMessageTransform"; |
| 3 | +import { |
| 4 | + transformModelMessages, |
| 5 | + validateAnthropicCompliance, |
| 6 | + addInterruptedSentinel, |
| 7 | +} from "./modelMessageTransform"; |
| 8 | +import type { CmuxMessage } from "@/types/message"; |
4 | 9 |
|
5 | 10 | describe("modelMessageTransform", () => { |
6 | 11 | describe("transformModelMessages", () => { |
@@ -396,6 +401,122 @@ describe("modelMessageTransform", () => { |
396 | 401 | }); |
397 | 402 | }); |
398 | 403 |
|
| 404 | + describe("addInterruptedSentinel", () => { |
| 405 | + it("should insert user message after partial assistant message", () => { |
| 406 | + const messages: CmuxMessage[] = [ |
| 407 | + { |
| 408 | + id: "user-1", |
| 409 | + role: "user", |
| 410 | + parts: [{ type: "text", text: "Hello" }], |
| 411 | + metadata: { timestamp: 1000 }, |
| 412 | + }, |
| 413 | + { |
| 414 | + id: "assistant-1", |
| 415 | + role: "assistant", |
| 416 | + parts: [{ type: "text", text: "Let me help..." }], |
| 417 | + metadata: { timestamp: 2000, partial: true }, |
| 418 | + }, |
| 419 | + ]; |
| 420 | + |
| 421 | + const result = addInterruptedSentinel(messages); |
| 422 | + |
| 423 | + // Should have 3 messages: user, assistant, [INTERRUPTED] user |
| 424 | + expect(result).toHaveLength(3); |
| 425 | + expect(result[0].id).toBe("user-1"); |
| 426 | + expect(result[1].id).toBe("assistant-1"); |
| 427 | + expect(result[2].id).toBe("interrupted-assistant-1"); |
| 428 | + expect(result[2].role).toBe("user"); |
| 429 | + expect(result[2].parts).toEqual([{ type: "text", text: "[INTERRUPTED]" }]); |
| 430 | + expect(result[2].metadata?.synthetic).toBe(true); |
| 431 | + expect(result[2].metadata?.timestamp).toBe(2000); |
| 432 | + }); |
| 433 | + |
| 434 | + it("should not insert sentinel for non-partial assistant messages", () => { |
| 435 | + const messages: CmuxMessage[] = [ |
| 436 | + { |
| 437 | + id: "user-1", |
| 438 | + role: "user", |
| 439 | + parts: [{ type: "text", text: "Hello" }], |
| 440 | + metadata: { timestamp: 1000 }, |
| 441 | + }, |
| 442 | + { |
| 443 | + id: "assistant-1", |
| 444 | + role: "assistant", |
| 445 | + parts: [{ type: "text", text: "Complete response" }], |
| 446 | + metadata: { timestamp: 2000, partial: false }, |
| 447 | + }, |
| 448 | + ]; |
| 449 | + |
| 450 | + const result = addInterruptedSentinel(messages); |
| 451 | + |
| 452 | + // Should remain unchanged (no sentinel) |
| 453 | + expect(result).toHaveLength(2); |
| 454 | + expect(result).toEqual(messages); |
| 455 | + }); |
| 456 | + |
| 457 | + it("should insert sentinel for reasoning-only partial messages", () => { |
| 458 | + const messages: CmuxMessage[] = [ |
| 459 | + { |
| 460 | + id: "user-1", |
| 461 | + role: "user", |
| 462 | + parts: [{ type: "text", text: "Calculate something" }], |
| 463 | + metadata: { timestamp: 1000 }, |
| 464 | + }, |
| 465 | + { |
| 466 | + id: "assistant-1", |
| 467 | + role: "assistant", |
| 468 | + parts: [{ type: "reasoning", text: "Let me think about this..." }], |
| 469 | + metadata: { timestamp: 2000, partial: true }, |
| 470 | + }, |
| 471 | + ]; |
| 472 | + |
| 473 | + const result = addInterruptedSentinel(messages); |
| 474 | + |
| 475 | + // Should have 3 messages: user, assistant (reasoning only), [INTERRUPTED] user |
| 476 | + expect(result).toHaveLength(3); |
| 477 | + expect(result[2].role).toBe("user"); |
| 478 | + expect(result[2].parts).toEqual([{ type: "text", text: "[INTERRUPTED]" }]); |
| 479 | + }); |
| 480 | + |
| 481 | + it("should handle multiple partial messages", () => { |
| 482 | + const messages: CmuxMessage[] = [ |
| 483 | + { |
| 484 | + id: "user-1", |
| 485 | + role: "user", |
| 486 | + parts: [{ type: "text", text: "First" }], |
| 487 | + metadata: { timestamp: 1000 }, |
| 488 | + }, |
| 489 | + { |
| 490 | + id: "assistant-1", |
| 491 | + role: "assistant", |
| 492 | + parts: [{ type: "text", text: "Response 1..." }], |
| 493 | + metadata: { timestamp: 2000, partial: true }, |
| 494 | + }, |
| 495 | + { |
| 496 | + id: "user-2", |
| 497 | + role: "user", |
| 498 | + parts: [{ type: "text", text: "Second" }], |
| 499 | + metadata: { timestamp: 3000 }, |
| 500 | + }, |
| 501 | + { |
| 502 | + id: "assistant-2", |
| 503 | + role: "assistant", |
| 504 | + parts: [{ type: "text", text: "Response 2..." }], |
| 505 | + metadata: { timestamp: 4000, partial: true }, |
| 506 | + }, |
| 507 | + ]; |
| 508 | + |
| 509 | + const result = addInterruptedSentinel(messages); |
| 510 | + |
| 511 | + // Should have 6 messages (4 original + 2 sentinels) |
| 512 | + expect(result).toHaveLength(6); |
| 513 | + expect(result[2].id).toBe("interrupted-assistant-1"); |
| 514 | + expect(result[2].role).toBe("user"); |
| 515 | + expect(result[5].id).toBe("interrupted-assistant-2"); |
| 516 | + expect(result[5].role).toBe("user"); |
| 517 | + }); |
| 518 | + }); |
| 519 | + |
399 | 520 | describe("reasoning part stripping for OpenAI", () => { |
400 | 521 | it("should strip reasoning parts for OpenAI provider", () => { |
401 | 522 | const messages: ModelMessage[] = [ |
|
0 commit comments