| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565 |
- package message
- import (
- "encoding/base64"
- "errors"
- "fmt"
- "slices"
- "strings"
- "time"
- "charm.land/catwalk/pkg/catwalk"
- "charm.land/fantasy"
- "charm.land/fantasy/providers/anthropic"
- "charm.land/fantasy/providers/google"
- "charm.land/fantasy/providers/openai"
- )
- type MessageRole string
- const (
- Assistant MessageRole = "assistant"
- User MessageRole = "user"
- System MessageRole = "system"
- Tool MessageRole = "tool"
- )
- type FinishReason string
- const (
- FinishReasonEndTurn FinishReason = "end_turn"
- FinishReasonMaxTokens FinishReason = "max_tokens"
- FinishReasonToolUse FinishReason = "tool_use"
- FinishReasonCanceled FinishReason = "canceled"
- FinishReasonError FinishReason = "error"
- FinishReasonPermissionDenied FinishReason = "permission_denied"
- // Should never happen
- FinishReasonUnknown FinishReason = "unknown"
- )
- type ContentPart interface {
- isPart()
- }
- type ReasoningContent struct {
- Thinking string `json:"thinking"`
- Signature string `json:"signature"`
- ThoughtSignature string `json:"thought_signature"` // Used for google
- ToolID string `json:"tool_id"` // Used for openrouter google models
- ResponsesData *openai.ResponsesReasoningMetadata `json:"responses_data"`
- StartedAt int64 `json:"started_at,omitempty"`
- FinishedAt int64 `json:"finished_at,omitempty"`
- }
- func (tc ReasoningContent) String() string {
- return tc.Thinking
- }
- func (ReasoningContent) isPart() {}
- type TextContent struct {
- Text string `json:"text"`
- }
- func (tc TextContent) String() string {
- return tc.Text
- }
- func (TextContent) isPart() {}
- type ImageURLContent struct {
- URL string `json:"url"`
- Detail string `json:"detail,omitempty"`
- }
- func (iuc ImageURLContent) String() string {
- return iuc.URL
- }
- func (ImageURLContent) isPart() {}
- type BinaryContent struct {
- Path string
- MIMEType string
- Data []byte
- }
- func (bc BinaryContent) String(p catwalk.InferenceProvider) string {
- base64Encoded := base64.StdEncoding.EncodeToString(bc.Data)
- if p == catwalk.InferenceProviderOpenAI {
- return "data:" + bc.MIMEType + ";base64," + base64Encoded
- }
- return base64Encoded
- }
- func (BinaryContent) isPart() {}
- type ToolCall struct {
- ID string `json:"id"`
- Name string `json:"name"`
- Input string `json:"input"`
- ProviderExecuted bool `json:"provider_executed"`
- Finished bool `json:"finished"`
- }
- func (ToolCall) isPart() {}
- type ToolResult struct {
- ToolCallID string `json:"tool_call_id"`
- Name string `json:"name"`
- Content string `json:"content"`
- Data string `json:"data"`
- MIMEType string `json:"mime_type"`
- Metadata string `json:"metadata"`
- IsError bool `json:"is_error"`
- }
- func (ToolResult) isPart() {}
- type Finish struct {
- Reason FinishReason `json:"reason"`
- Time int64 `json:"time"`
- Message string `json:"message,omitempty"`
- Details string `json:"details,omitempty"`
- }
- func (Finish) isPart() {}
- type Message struct {
- ID string
- Role MessageRole
- SessionID string
- Parts []ContentPart
- Model string
- Provider string
- CreatedAt int64
- UpdatedAt int64
- IsSummaryMessage bool
- }
- func (m *Message) Content() TextContent {
- for _, part := range m.Parts {
- if c, ok := part.(TextContent); ok {
- return c
- }
- }
- return TextContent{}
- }
- func (m *Message) ReasoningContent() ReasoningContent {
- for _, part := range m.Parts {
- if c, ok := part.(ReasoningContent); ok {
- return c
- }
- }
- return ReasoningContent{}
- }
- func (m *Message) ImageURLContent() []ImageURLContent {
- imageURLContents := make([]ImageURLContent, 0)
- for _, part := range m.Parts {
- if c, ok := part.(ImageURLContent); ok {
- imageURLContents = append(imageURLContents, c)
- }
- }
- return imageURLContents
- }
- func (m *Message) BinaryContent() []BinaryContent {
- binaryContents := make([]BinaryContent, 0)
- for _, part := range m.Parts {
- if c, ok := part.(BinaryContent); ok {
- binaryContents = append(binaryContents, c)
- }
- }
- return binaryContents
- }
- func (m *Message) ToolCalls() []ToolCall {
- toolCalls := make([]ToolCall, 0)
- for _, part := range m.Parts {
- if c, ok := part.(ToolCall); ok {
- toolCalls = append(toolCalls, c)
- }
- }
- return toolCalls
- }
- func (m *Message) ToolResults() []ToolResult {
- toolResults := make([]ToolResult, 0)
- for _, part := range m.Parts {
- if c, ok := part.(ToolResult); ok {
- toolResults = append(toolResults, c)
- }
- }
- return toolResults
- }
- func (m *Message) IsFinished() bool {
- for _, part := range m.Parts {
- if _, ok := part.(Finish); ok {
- return true
- }
- }
- return false
- }
- func (m *Message) FinishPart() *Finish {
- for _, part := range m.Parts {
- if c, ok := part.(Finish); ok {
- return &c
- }
- }
- return nil
- }
- func (m *Message) FinishReason() FinishReason {
- for _, part := range m.Parts {
- if c, ok := part.(Finish); ok {
- return c.Reason
- }
- }
- return ""
- }
- func (m *Message) IsThinking() bool {
- if m.ReasoningContent().Thinking != "" && m.Content().Text == "" && !m.IsFinished() {
- return true
- }
- return false
- }
- func (m *Message) AppendContent(delta string) {
- found := false
- for i, part := range m.Parts {
- if c, ok := part.(TextContent); ok {
- m.Parts[i] = TextContent{Text: c.Text + delta}
- found = true
- }
- }
- if !found {
- m.Parts = append(m.Parts, TextContent{Text: delta})
- }
- }
- func (m *Message) AppendReasoningContent(delta string) {
- found := false
- for i, part := range m.Parts {
- if c, ok := part.(ReasoningContent); ok {
- m.Parts[i] = ReasoningContent{
- Thinking: c.Thinking + delta,
- Signature: c.Signature,
- StartedAt: c.StartedAt,
- FinishedAt: c.FinishedAt,
- }
- found = true
- }
- }
- if !found {
- m.Parts = append(m.Parts, ReasoningContent{
- Thinking: delta,
- StartedAt: time.Now().Unix(),
- })
- }
- }
- func (m *Message) AppendThoughtSignature(signature string, toolCallID string) {
- for i, part := range m.Parts {
- if c, ok := part.(ReasoningContent); ok {
- m.Parts[i] = ReasoningContent{
- Thinking: c.Thinking,
- ThoughtSignature: c.ThoughtSignature + signature,
- ToolID: toolCallID,
- Signature: c.Signature,
- StartedAt: c.StartedAt,
- FinishedAt: c.FinishedAt,
- }
- return
- }
- }
- m.Parts = append(m.Parts, ReasoningContent{ThoughtSignature: signature})
- }
- func (m *Message) AppendReasoningSignature(signature string) {
- for i, part := range m.Parts {
- if c, ok := part.(ReasoningContent); ok {
- m.Parts[i] = ReasoningContent{
- Thinking: c.Thinking,
- Signature: c.Signature + signature,
- StartedAt: c.StartedAt,
- FinishedAt: c.FinishedAt,
- }
- return
- }
- }
- m.Parts = append(m.Parts, ReasoningContent{Signature: signature})
- }
- func (m *Message) SetReasoningResponsesData(data *openai.ResponsesReasoningMetadata) {
- for i, part := range m.Parts {
- if c, ok := part.(ReasoningContent); ok {
- m.Parts[i] = ReasoningContent{
- Thinking: c.Thinking,
- ResponsesData: data,
- StartedAt: c.StartedAt,
- FinishedAt: c.FinishedAt,
- }
- return
- }
- }
- }
- func (m *Message) FinishThinking() {
- for i, part := range m.Parts {
- if c, ok := part.(ReasoningContent); ok {
- if c.FinishedAt == 0 {
- m.Parts[i] = ReasoningContent{
- Thinking: c.Thinking,
- Signature: c.Signature,
- StartedAt: c.StartedAt,
- FinishedAt: time.Now().Unix(),
- }
- }
- return
- }
- }
- }
- func (m *Message) ThinkingDuration() time.Duration {
- reasoning := m.ReasoningContent()
- if reasoning.StartedAt == 0 {
- return 0
- }
- endTime := reasoning.FinishedAt
- if endTime == 0 {
- endTime = time.Now().Unix()
- }
- return time.Duration(endTime-reasoning.StartedAt) * time.Second
- }
- func (m *Message) FinishToolCall(toolCallID string) {
- for i, part := range m.Parts {
- if c, ok := part.(ToolCall); ok {
- if c.ID == toolCallID {
- m.Parts[i] = ToolCall{
- ID: c.ID,
- Name: c.Name,
- Input: c.Input,
- Finished: true,
- }
- return
- }
- }
- }
- }
- func (m *Message) AppendToolCallInput(toolCallID string, inputDelta string) {
- for i, part := range m.Parts {
- if c, ok := part.(ToolCall); ok {
- if c.ID == toolCallID {
- m.Parts[i] = ToolCall{
- ID: c.ID,
- Name: c.Name,
- Input: c.Input + inputDelta,
- Finished: c.Finished,
- }
- return
- }
- }
- }
- }
- func (m *Message) AddToolCall(tc ToolCall) {
- for i, part := range m.Parts {
- if c, ok := part.(ToolCall); ok {
- if c.ID == tc.ID {
- m.Parts[i] = tc
- return
- }
- }
- }
- m.Parts = append(m.Parts, tc)
- }
- func (m *Message) SetToolCalls(tc []ToolCall) {
- // remove any existing tool call part it could have multiple
- parts := make([]ContentPart, 0)
- for _, part := range m.Parts {
- if _, ok := part.(ToolCall); ok {
- continue
- }
- parts = append(parts, part)
- }
- m.Parts = parts
- for _, toolCall := range tc {
- m.Parts = append(m.Parts, toolCall)
- }
- }
- func (m *Message) AddToolResult(tr ToolResult) {
- m.Parts = append(m.Parts, tr)
- }
- func (m *Message) SetToolResults(tr []ToolResult) {
- for _, toolResult := range tr {
- m.Parts = append(m.Parts, toolResult)
- }
- }
- // Clone returns a deep copy of the message with an independent Parts slice.
- // This prevents race conditions when the message is modified concurrently.
- func (m *Message) Clone() Message {
- clone := *m
- clone.Parts = make([]ContentPart, len(m.Parts))
- copy(clone.Parts, m.Parts)
- return clone
- }
- func (m *Message) AddFinish(reason FinishReason, message, details string) {
- // remove any existing finish part
- for i, part := range m.Parts {
- if _, ok := part.(Finish); ok {
- m.Parts = slices.Delete(m.Parts, i, i+1)
- break
- }
- }
- m.Parts = append(m.Parts, Finish{Reason: reason, Time: time.Now().Unix(), Message: message, Details: details})
- }
- func (m *Message) AddImageURL(url, detail string) {
- m.Parts = append(m.Parts, ImageURLContent{URL: url, Detail: detail})
- }
- func (m *Message) AddBinary(mimeType string, data []byte) {
- m.Parts = append(m.Parts, BinaryContent{MIMEType: mimeType, Data: data})
- }
- func PromptWithTextAttachments(prompt string, attachments []Attachment) string {
- var sb strings.Builder
- sb.WriteString(prompt)
- addedAttachments := false
- for _, content := range attachments {
- if !content.IsText() {
- continue
- }
- if !addedAttachments {
- sb.WriteString("\n<system_info>The files below have been attached by the user, consider them in your response</system_info>\n")
- addedAttachments = true
- }
- if content.FilePath != "" {
- fmt.Fprintf(&sb, "<file path='%s'>\n", content.FilePath)
- } else {
- sb.WriteString("<file>\n")
- }
- sb.WriteString("\n")
- sb.Write(content.Content)
- sb.WriteString("\n</file>\n")
- }
- return sb.String()
- }
- func (m *Message) ToAIMessage() []fantasy.Message {
- var messages []fantasy.Message
- switch m.Role {
- case User:
- var parts []fantasy.MessagePart
- text := strings.TrimSpace(m.Content().Text)
- var textAttachments []Attachment
- for _, content := range m.BinaryContent() {
- if !strings.HasPrefix(content.MIMEType, "text/") {
- continue
- }
- textAttachments = append(textAttachments, Attachment{
- FilePath: content.Path,
- MimeType: content.MIMEType,
- Content: content.Data,
- })
- }
- text = PromptWithTextAttachments(text, textAttachments)
- if text != "" {
- parts = append(parts, fantasy.TextPart{Text: text})
- }
- for _, content := range m.BinaryContent() {
- // skip text attachements
- if strings.HasPrefix(content.MIMEType, "text/") {
- continue
- }
- parts = append(parts, fantasy.FilePart{
- Filename: content.Path,
- Data: content.Data,
- MediaType: content.MIMEType,
- })
- }
- messages = append(messages, fantasy.Message{
- Role: fantasy.MessageRoleUser,
- Content: parts,
- })
- case Assistant:
- var parts []fantasy.MessagePart
- text := strings.TrimSpace(m.Content().Text)
- if text != "" {
- parts = append(parts, fantasy.TextPart{Text: text})
- }
- reasoning := m.ReasoningContent()
- if reasoning.Thinking != "" {
- reasoningPart := fantasy.ReasoningPart{Text: reasoning.Thinking, ProviderOptions: fantasy.ProviderOptions{}}
- if reasoning.Signature != "" {
- reasoningPart.ProviderOptions[anthropic.Name] = &anthropic.ReasoningOptionMetadata{
- Signature: reasoning.Signature,
- }
- }
- if reasoning.ResponsesData != nil {
- reasoningPart.ProviderOptions[openai.Name] = reasoning.ResponsesData
- }
- if reasoning.ThoughtSignature != "" {
- reasoningPart.ProviderOptions[google.Name] = &google.ReasoningMetadata{
- Signature: reasoning.ThoughtSignature,
- ToolID: reasoning.ToolID,
- }
- }
- parts = append(parts, reasoningPart)
- }
- for _, call := range m.ToolCalls() {
- parts = append(parts, fantasy.ToolCallPart{
- ToolCallID: call.ID,
- ToolName: call.Name,
- Input: call.Input,
- ProviderExecuted: call.ProviderExecuted,
- })
- }
- messages = append(messages, fantasy.Message{
- Role: fantasy.MessageRoleAssistant,
- Content: parts,
- })
- case Tool:
- var parts []fantasy.MessagePart
- for _, result := range m.ToolResults() {
- var content fantasy.ToolResultOutputContent
- if result.IsError {
- content = fantasy.ToolResultOutputContentError{
- Error: errors.New(result.Content),
- }
- } else if result.Data != "" {
- content = fantasy.ToolResultOutputContentMedia{
- Data: result.Data,
- MediaType: result.MIMEType,
- }
- } else {
- content = fantasy.ToolResultOutputContentText{
- Text: result.Content,
- }
- }
- parts = append(parts, fantasy.ToolResultPart{
- ToolCallID: result.ToolCallID,
- Output: content,
- })
- }
- messages = append(messages, fantasy.Message{
- Role: fantasy.MessageRoleTool,
- Content: parts,
- })
- }
- return messages
- }
|