mirror of
https://github.com/memohai/Memoh.git
synced 2026-04-27 07:16:19 +09:00
c591af14b0
* feat: bot inbox * feat: unified header * fix: missing tool_call usage * feat: add group name in header
115 lines
2.7 KiB
Go
115 lines
2.7 KiB
Go
package flow
|
|
|
|
import (
|
|
"testing"
|
|
|
|
"github.com/memohai/memoh/internal/conversation"
|
|
)
|
|
|
|
func intPtr(v int) *int { return &v }
|
|
|
|
func TestTrimMessagesByTokens_DropsLeadingOrphanTool(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
messages := []messageWithUsage{
|
|
{
|
|
Message: conversation.ModelMessage{
|
|
Role: "user",
|
|
Content: conversation.NewTextContent("1111"),
|
|
},
|
|
},
|
|
{
|
|
Message: conversation.ModelMessage{
|
|
Role: "assistant",
|
|
ToolCalls: []conversation.ToolCall{
|
|
{
|
|
ID: "call-1",
|
|
Type: "function",
|
|
Function: conversation.ToolCallFunction{
|
|
Name: "calc",
|
|
Arguments: `{"x":1}`,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
UsageOutputTokens: intPtr(50),
|
|
},
|
|
{
|
|
Message: conversation.ModelMessage{
|
|
Role: "tool",
|
|
ToolCallID: "call-1",
|
|
Content: conversation.NewTextContent("2"),
|
|
},
|
|
},
|
|
{
|
|
Message: conversation.ModelMessage{
|
|
Role: "assistant",
|
|
Content: conversation.NewTextContent("done"),
|
|
},
|
|
UsageOutputTokens: intPtr(60),
|
|
},
|
|
}
|
|
|
|
// Budget 70: assistant(60) fits, adding assistant-tool-call(50) exceeds →
|
|
// cutoff lands on the tool message which must be skipped.
|
|
trimmed := trimMessagesByTokens(messages, 70)
|
|
if len(trimmed) == 0 {
|
|
t.Fatal("expected non-empty trimmed messages")
|
|
}
|
|
if trimmed[0].Role == "tool" {
|
|
t.Fatal("expected first trimmed message not to be tool")
|
|
}
|
|
}
|
|
|
|
func TestTrimMessagesByTokens_KeepsToolWhenPaired(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
messages := []messageWithUsage{
|
|
{
|
|
Message: conversation.ModelMessage{
|
|
Role: "assistant",
|
|
ToolCalls: []conversation.ToolCall{
|
|
{
|
|
ID: "call-1",
|
|
Type: "function",
|
|
Function: conversation.ToolCallFunction{
|
|
Name: "calc",
|
|
Arguments: `{"x":1}`,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
UsageOutputTokens: intPtr(10),
|
|
},
|
|
{
|
|
Message: conversation.ModelMessage{
|
|
Role: "tool",
|
|
ToolCallID: "call-1",
|
|
Content: conversation.NewTextContent("2"),
|
|
},
|
|
},
|
|
}
|
|
|
|
trimmed := trimMessagesByTokens(messages, 100)
|
|
if len(trimmed) != 2 {
|
|
t.Fatalf("expected 2 messages, got %d", len(trimmed))
|
|
}
|
|
if trimmed[0].Role != "assistant" || trimmed[1].Role != "tool" {
|
|
t.Fatalf("unexpected role order: %q -> %q", trimmed[0].Role, trimmed[1].Role)
|
|
}
|
|
}
|
|
|
|
func TestTrimMessagesByTokens_NoUsage_KeepsAll(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
messages := []messageWithUsage{
|
|
{Message: conversation.ModelMessage{Role: "user", Content: conversation.NewTextContent("hello")}},
|
|
{Message: conversation.ModelMessage{Role: "assistant", Content: conversation.NewTextContent("hi")}},
|
|
}
|
|
|
|
trimmed := trimMessagesByTokens(messages, 10)
|
|
if len(trimmed) != 2 {
|
|
t.Fatalf("messages without outputTokens should all be kept, got %d", len(trimmed))
|
|
}
|
|
}
|