WIP: add v1/responses support

Only supporting the stateless part of the API.

Closes: #9659
This commit is contained in:
Devon Rifkin
2025-12-05 14:14:49 -08:00
parent cc9555aff0
commit af4b13599d
5 changed files with 2040 additions and 22 deletions

View File

@@ -433,3 +433,111 @@ func ChatMiddleware() gin.HandlerFunc {
c.Next()
}
}
type ResponsesWriter struct {
BaseWriter
converter *openai.ResponsesStreamConverter
model string
stream bool
responseID string
itemID string
}
func (w *ResponsesWriter) writeEvent(eventType string, data any) error {
d, err := json.Marshal(data)
if err != nil {
return err
}
_, err = w.ResponseWriter.Write([]byte(fmt.Sprintf("event: %s\ndata: %s\n\n", eventType, d)))
if err != nil {
return err
}
if f, ok := w.ResponseWriter.(http.Flusher); ok {
f.Flush()
}
return nil
}
func (w *ResponsesWriter) writeResponse(data []byte) (int, error) {
var chatResponse api.ChatResponse
if err := json.Unmarshal(data, &chatResponse); err != nil {
return 0, err
}
if w.stream {
w.ResponseWriter.Header().Set("Content-Type", "text/event-stream")
events := w.converter.Process(chatResponse)
for _, event := range events {
if err := w.writeEvent(event.Event, event.Data); err != nil {
return 0, err
}
}
return len(data), nil
}
// Non-streaming response
w.ResponseWriter.Header().Set("Content-Type", "application/json")
response := openai.ToResponse(w.model, w.responseID, w.itemID, chatResponse)
return len(data), json.NewEncoder(w.ResponseWriter).Encode(response)
}
func (w *ResponsesWriter) Write(data []byte) (int, error) {
code := w.ResponseWriter.Status()
if code != http.StatusOK {
return w.writeError(data)
}
return w.writeResponse(data)
}
func ResponsesMiddleware() gin.HandlerFunc {
return func(c *gin.Context) {
var req openai.ResponsesRequest
if err := c.ShouldBindJSON(&req); err != nil {
c.AbortWithStatusJSON(http.StatusBadRequest, openai.NewError(http.StatusBadRequest, err.Error()))
return
}
chatReq, err := openai.FromResponsesRequest(req)
if err != nil {
c.AbortWithStatusJSON(http.StatusBadRequest, openai.NewError(http.StatusBadRequest, err.Error()))
return
}
// Check if client requested streaming (defaults to false)
streamRequested := req.Stream != nil && *req.Stream
// Pass streaming preference to the underlying chat request
chatReq.Stream = &streamRequested
var b bytes.Buffer
if err := json.NewEncoder(&b).Encode(chatReq); err != nil {
c.AbortWithStatusJSON(http.StatusInternalServerError, openai.NewError(http.StatusInternalServerError, err.Error()))
return
}
c.Request.Body = io.NopCloser(&b)
responseID := fmt.Sprintf("resp_%d", rand.Intn(999999))
itemID := fmt.Sprintf("msg_%d", rand.Intn(999999))
w := &ResponsesWriter{
BaseWriter: BaseWriter{ResponseWriter: c.Writer},
converter: openai.NewResponsesStreamConverter(responseID, itemID, req.Model),
model: req.Model,
stream: streamRequested,
responseID: responseID,
itemID: itemID,
}
// Set headers based on streaming mode
if streamRequested {
c.Writer.Header().Set("Content-Type", "text/event-stream")
c.Writer.Header().Set("Cache-Control", "no-cache")
c.Writer.Header().Set("Connection", "keep-alive")
}
c.Writer = w
c.Next()
}
}

View File

@@ -487,29 +487,9 @@ func FromChatRequest(r ChatCompletionRequest) (*api.ChatRequest, error) {
}
}
types := []string{"jpeg", "jpg", "png", "webp"}
valid := false
// support blank mime type to match api/chat taking just unadorned base64
if strings.HasPrefix(url, "data:;base64,") {
url = strings.TrimPrefix(url, "data:;base64,")
valid = true
}
for _, t := range types {
prefix := "data:image/" + t + ";base64,"
if strings.HasPrefix(url, prefix) {
url = strings.TrimPrefix(url, prefix)
valid = true
break
}
}
if !valid {
return nil, errors.New("invalid image input")
}
img, err := base64.StdEncoding.DecodeString(url)
img, err := decodeImageURL(url)
if err != nil {
return nil, errors.New("invalid message format")
return nil, err
}
messages = append(messages, api.Message{Role: msg.Role, Images: []api.ImageData{img}})
@@ -648,6 +628,35 @@ func nameFromToolCallID(messages []Message, toolCallID string) string {
return ""
}
// decodeImageURL decodes a base64 data URI into raw image bytes.
func decodeImageURL(url string) (api.ImageData, error) {
types := []string{"jpeg", "jpg", "png", "webp"}
// Support blank mime type to match /api/chat's behavior of taking just unadorned base64
if strings.HasPrefix(url, "data:;base64,") {
url = strings.TrimPrefix(url, "data:;base64,")
} else {
valid := false
for _, t := range types {
prefix := "data:image/" + t + ";base64,"
if strings.HasPrefix(url, prefix) {
url = strings.TrimPrefix(url, prefix)
valid = true
break
}
}
if !valid {
return nil, errors.New("invalid image input")
}
}
img, err := base64.StdEncoding.DecodeString(url)
if err != nil {
return nil, errors.New("invalid image input")
}
return img, nil
}
// FromCompletionToolCall converts OpenAI ToolCall format to api.ToolCall
func FromCompletionToolCall(toolCalls []ToolCall) ([]api.ToolCall, error) {
apiToolCalls := make([]api.ToolCall, len(toolCalls))

900
openai/responses.go Normal file
View File

@@ -0,0 +1,900 @@
package openai
import (
"encoding/json"
"fmt"
"math/rand"
"github.com/ollama/ollama/api"
)
// ResponsesContent is a discriminated union for input content types.
// Concrete types: ResponsesTextContent, ResponsesImageContent
type ResponsesContent interface {
responsesContent() // unexported marker method
}
type ResponsesTextContent struct {
Type string `json:"type"` // always "input_text"
Text string `json:"text"`
}
func (ResponsesTextContent) responsesContent() {}
type ResponsesImageContent struct {
Type string `json:"type"` // always "input_image"
// TODO(drifkin): is this really required? that seems verbose and a default is specified in the docs
Detail string `json:"detail"` // required
FileID string `json:"file_id,omitempty"` // optional
ImageURL string `json:"image_url,omitempty"` // optional
}
func (ResponsesImageContent) responsesContent() {}
type ResponsesInputMessage struct {
Type string `json:"type"` // always "message"
Role string `json:"role"` // one of `user`, `system`, `developer`
Content []ResponsesContent `json:"content,omitempty"`
}
func (m *ResponsesInputMessage) UnmarshalJSON(data []byte) error {
var aux struct {
Type string `json:"type"`
Role string `json:"role"`
Content []json.RawMessage `json:"content"`
}
if err := json.Unmarshal(data, &aux); err != nil {
return err
}
m.Type = aux.Type
m.Role = aux.Role
if len(aux.Content) == 0 {
return nil
}
m.Content = make([]ResponsesContent, 0, len(aux.Content))
for i, raw := range aux.Content {
// Peek at the type field to determine which concrete type to use
var typeField struct {
Type string `json:"type"`
}
if err := json.Unmarshal(raw, &typeField); err != nil {
return fmt.Errorf("content[%d]: %w", i, err)
}
switch typeField.Type {
case "input_text":
var content ResponsesTextContent
if err := json.Unmarshal(raw, &content); err != nil {
return fmt.Errorf("content[%d]: %w", i, err)
}
m.Content = append(m.Content, content)
case "input_image":
var content ResponsesImageContent
if err := json.Unmarshal(raw, &content); err != nil {
return fmt.Errorf("content[%d]: %w", i, err)
}
m.Content = append(m.Content, content)
default:
return fmt.Errorf("content[%d]: unknown content type: %s", i, typeField.Type)
}
}
return nil
}
type ResponsesOutputMessage struct{}
// ResponsesInputItem is a discriminated union for input items.
// Concrete types: ResponsesInputMessage (more to come)
type ResponsesInputItem interface {
responsesInputItem() // unexported marker method
}
func (ResponsesInputMessage) responsesInputItem() {}
// ResponsesFunctionCall represents an assistant's function call in conversation history.
type ResponsesFunctionCall struct {
ID string `json:"id,omitempty"` // item ID
Type string `json:"type"` // always "function_call"
CallID string `json:"call_id"` // the tool call ID
Name string `json:"name"` // function name
Arguments string `json:"arguments"` // JSON arguments string
}
func (ResponsesFunctionCall) responsesInputItem() {}
// ResponsesFunctionCallOutput represents a function call result from the client.
type ResponsesFunctionCallOutput struct {
Type string `json:"type"` // always "function_call_output"
CallID string `json:"call_id"` // links to the original function call
Output string `json:"output"` // the function result
}
func (ResponsesFunctionCallOutput) responsesInputItem() {}
// ResponsesReasoningInput represents a reasoning item passed back as input.
// This is used when the client sends previous reasoning back for context.
type ResponsesReasoningInput struct {
ID string `json:"id,omitempty"`
Type string `json:"type"` // always "reasoning"
Summary []ResponsesReasoningSummary `json:"summary,omitempty"`
EncryptedContent string `json:"encrypted_content,omitempty"`
}
func (ResponsesReasoningInput) responsesInputItem() {}
// unmarshalResponsesInputItem unmarshals a single input item from JSON.
func unmarshalResponsesInputItem(data []byte) (ResponsesInputItem, error) {
var typeField struct {
Type string `json:"type"`
}
if err := json.Unmarshal(data, &typeField); err != nil {
return nil, err
}
switch typeField.Type {
case "message":
var msg ResponsesInputMessage
if err := json.Unmarshal(data, &msg); err != nil {
return nil, err
}
return msg, nil
case "function_call":
var fc ResponsesFunctionCall
if err := json.Unmarshal(data, &fc); err != nil {
return nil, err
}
return fc, nil
case "function_call_output":
var output ResponsesFunctionCallOutput
if err := json.Unmarshal(data, &output); err != nil {
return nil, err
}
return output, nil
case "reasoning":
var reasoning ResponsesReasoningInput
if err := json.Unmarshal(data, &reasoning); err != nil {
return nil, err
}
return reasoning, nil
default:
return nil, fmt.Errorf("unknown input item type: %s", typeField.Type)
}
}
// ResponsesInput can be either:
// - a string (equivalent to a text input with the user role)
// - an array of input items (see ResponsesInputItem)
type ResponsesInput struct {
Text string // set if input was a plain string
Items []ResponsesInputItem // set if input was an array
}
func (r *ResponsesInput) UnmarshalJSON(data []byte) error {
// Try string first
var s string
if err := json.Unmarshal(data, &s); err == nil {
r.Text = s
return nil
}
// Otherwise, try array of input items
var rawItems []json.RawMessage
if err := json.Unmarshal(data, &rawItems); err != nil {
return fmt.Errorf("input must be a string or array: %w", err)
}
r.Items = make([]ResponsesInputItem, 0, len(rawItems))
for i, raw := range rawItems {
item, err := unmarshalResponsesInputItem(raw)
if err != nil {
return fmt.Errorf("input[%d]: %w", i, err)
}
r.Items = append(r.Items, item)
}
return nil
}
type ResponsesReasoning struct {
// originally: optional, default is per-model
Effort string `json:"effort,omitempty"`
// originally: deprecated, use `summary` instead. One of `auto`, `concise`, `detailed`
GenerateSummary string `json:"generate_summary,omitempty"`
// originally: optional, one of `auto`, `concise`, `detailed`
Summary string `json:"summary,omitempty"`
}
// ResponsesTool represents a tool in the Responses API format.
// Note: This differs from api.Tool which nests fields under "function".
type ResponsesTool struct {
Type string `json:"type"` // "function"
Name string `json:"name"`
Description string `json:"description,omitempty"`
Strict bool `json:"strict,omitempty"`
Parameters map[string]any `json:"parameters,omitempty"`
}
type ResponsesRequest struct {
Model string `json:"model"`
// originally: optional, default is false
// for us: not supported
Background bool `json:"background"`
// originally: optional `string | {id: string}`
// for us: not supported
Conversation json.RawMessage `json:"conversation"`
// originally: string[]
// for us: ignored
Include []string `json:"include"`
Input ResponsesInput `json:"input"`
Reasoning ResponsesReasoning `json:"reasoning"`
// optional, default is 1.0
Temperature *float64 `json:"temperature"`
// optional, default is 1.0
TopP *float64 `json:"top_p"`
// optional, default is `"disabled"`
Truncation *string `json:"truncation"`
Tools []ResponsesTool `json:"tools,omitempty"`
// optional, default is false
Stream *bool `json:"stream,omitempty"`
}
// FromResponsesRequest converts a ResponsesRequest to api.ChatRequest
func FromResponsesRequest(r ResponsesRequest) (*api.ChatRequest, error) {
var messages []api.Message
// Handle simple string input
if r.Input.Text != "" {
messages = append(messages, api.Message{
Role: "user",
Content: r.Input.Text,
})
}
// Handle array of input items
// Track pending reasoning to merge with the next assistant message
var pendingThinking string
for _, item := range r.Input.Items {
switch v := item.(type) {
case ResponsesReasoningInput:
// Store thinking to merge with the next assistant message
pendingThinking = v.EncryptedContent
case ResponsesInputMessage:
msg, err := convertInputMessage(v)
if err != nil {
return nil, err
}
// If this is an assistant message, attach pending thinking
if msg.Role == "assistant" && pendingThinking != "" {
msg.Thinking = pendingThinking
pendingThinking = ""
}
messages = append(messages, msg)
case ResponsesFunctionCall:
// Convert function call to assistant message with tool calls
var args api.ToolCallFunctionArguments
if v.Arguments != "" {
if err := json.Unmarshal([]byte(v.Arguments), &args); err != nil {
return nil, fmt.Errorf("failed to parse function call arguments: %w", err)
}
}
msg := api.Message{
Role: "assistant",
ToolCalls: []api.ToolCall{{
ID: v.CallID,
Function: api.ToolCallFunction{
Name: v.Name,
Arguments: args,
},
}},
}
// Attach pending thinking
if pendingThinking != "" {
msg.Thinking = pendingThinking
pendingThinking = ""
}
messages = append(messages, msg)
case ResponsesFunctionCallOutput:
messages = append(messages, api.Message{
Role: "tool",
Content: v.Output,
ToolCallID: v.CallID,
})
}
}
// If there's trailing reasoning without a following message, emit it
if pendingThinking != "" {
messages = append(messages, api.Message{
Role: "assistant",
Thinking: pendingThinking,
})
}
options := make(map[string]any)
if r.Temperature != nil {
options["temperature"] = *r.Temperature
} else {
options["temperature"] = 1.0
}
if r.TopP != nil {
options["top_p"] = *r.TopP
} else {
options["top_p"] = 1.0
}
// Convert tools from Responses API format to api.Tool format
var tools []api.Tool
for _, t := range r.Tools {
tool, err := convertTool(t)
if err != nil {
return nil, err
}
tools = append(tools, tool)
}
return &api.ChatRequest{
Model: r.Model,
Messages: messages,
Options: options,
Tools: tools,
}, nil
}
func convertTool(t ResponsesTool) (api.Tool, error) {
// Convert parameters from map[string]any to api.ToolFunctionParameters
var params api.ToolFunctionParameters
if t.Parameters != nil {
// Marshal and unmarshal to convert
b, err := json.Marshal(t.Parameters)
if err != nil {
return api.Tool{}, fmt.Errorf("failed to marshal tool parameters: %w", err)
}
if err := json.Unmarshal(b, &params); err != nil {
return api.Tool{}, fmt.Errorf("failed to unmarshal tool parameters: %w", err)
}
}
return api.Tool{
Type: t.Type,
Function: api.ToolFunction{
Name: t.Name,
Description: t.Description,
Parameters: params,
},
}, nil
}
func convertInputMessage(m ResponsesInputMessage) (api.Message, error) {
var content string
var images []api.ImageData
for _, c := range m.Content {
switch v := c.(type) {
case ResponsesTextContent:
content += v.Text
case ResponsesImageContent:
if v.ImageURL == "" {
continue // Skip if no URL (FileID not supported)
}
img, err := decodeImageURL(v.ImageURL)
if err != nil {
return api.Message{}, err
}
images = append(images, img)
}
}
return api.Message{
Role: m.Role,
Content: content,
Images: images,
}, nil
}
// Response types for the Responses API
type ResponsesResponse struct {
ID string `json:"id"`
Object string `json:"object"`
CreatedAt int64 `json:"created_at"`
Status string `json:"status"`
Model string `json:"model"`
Output []ResponsesOutputItem `json:"output"`
Usage *ResponsesUsage `json:"usage,omitempty"`
}
type ResponsesOutputItem struct {
ID string `json:"id"`
Type string `json:"type"` // "message", "function_call", or "reasoning"
Status string `json:"status,omitempty"`
Role string `json:"role,omitempty"` // for message
Content []ResponsesOutputContent `json:"content,omitempty"` // for message
CallID string `json:"call_id,omitempty"` // for function_call
Name string `json:"name,omitempty"` // for function_call
Arguments string `json:"arguments,omitempty"` // for function_call
// Reasoning fields
Summary []ResponsesReasoningSummary `json:"summary,omitempty"` // for reasoning
EncryptedContent string `json:"encrypted_content,omitempty"` // for reasoning
}
type ResponsesReasoningSummary struct {
Type string `json:"type"` // "summary_text"
Text string `json:"text"`
}
type ResponsesOutputContent struct {
Type string `json:"type"` // "output_text"
Text string `json:"text"`
}
type ResponsesUsage struct {
InputTokens int `json:"input_tokens"`
OutputTokens int `json:"output_tokens"`
TotalTokens int `json:"total_tokens"`
}
// ToResponse converts an api.ChatResponse to a Responses API response
func ToResponse(model, responseID, itemID string, chatResponse api.ChatResponse) ResponsesResponse {
var output []ResponsesOutputItem
// Add reasoning item if thinking is present
if chatResponse.Message.Thinking != "" {
output = append(output, ResponsesOutputItem{
ID: fmt.Sprintf("rs_%s", responseID),
Type: "reasoning",
Summary: []ResponsesReasoningSummary{
{
Type: "summary_text",
Text: chatResponse.Message.Thinking,
},
},
EncryptedContent: chatResponse.Message.Thinking, // Plain text for now
})
}
if len(chatResponse.Message.ToolCalls) > 0 {
toolCalls := ToToolCalls(chatResponse.Message.ToolCalls)
for i, tc := range toolCalls {
output = append(output, ResponsesOutputItem{
ID: fmt.Sprintf("fc_%s_%d", responseID, i),
Type: "function_call",
CallID: tc.ID,
Name: tc.Function.Name,
Arguments: tc.Function.Arguments,
})
}
} else {
output = append(output, ResponsesOutputItem{
ID: itemID,
Type: "message",
Status: "completed",
Role: "assistant",
Content: []ResponsesOutputContent{
{
Type: "output_text",
Text: chatResponse.Message.Content,
},
},
})
}
return ResponsesResponse{
ID: responseID,
Object: "response",
CreatedAt: chatResponse.CreatedAt.Unix(),
Status: "completed",
Model: model,
Output: output,
Usage: &ResponsesUsage{
InputTokens: chatResponse.PromptEvalCount,
OutputTokens: chatResponse.EvalCount,
TotalTokens: chatResponse.PromptEvalCount + chatResponse.EvalCount,
},
}
}
// Streaming events: <https://platform.openai.com/docs/api-reference/responses-streaming>
// ResponsesStreamEvent represents a single Server-Sent Event for the Responses API.
type ResponsesStreamEvent struct {
Event string // The event type (e.g., "response.created")
Data any // The event payload (will be JSON-marshaled)
}
// ResponsesStreamConverter converts api.ChatResponse objects to Responses API
// streaming events. It maintains state across multiple calls to handle the
// streaming event sequence correctly.
type ResponsesStreamConverter struct {
// Configuration (immutable after creation)
responseID string
itemID string
model string
// State tracking (mutated across Process calls)
firstWrite bool
outputIndex int
contentIndex int
contentStarted bool
toolCallsSent bool
accumulatedText string
// Reasoning/thinking state
accumulatedThinking string
reasoningItemID string
reasoningStarted bool
reasoningDone bool
}
// NewResponsesStreamConverter creates a new converter with the given configuration.
func NewResponsesStreamConverter(responseID, itemID, model string) *ResponsesStreamConverter {
return &ResponsesStreamConverter{
responseID: responseID,
itemID: itemID,
model: model,
firstWrite: true,
}
}
// Process takes a ChatResponse and returns the events that should be emitted.
// Events are returned in order. The caller is responsible for serializing
// and sending these events.
func (c *ResponsesStreamConverter) Process(r api.ChatResponse) []ResponsesStreamEvent {
var events []ResponsesStreamEvent
hasToolCalls := len(r.Message.ToolCalls) > 0
hasThinking := r.Message.Thinking != ""
// First chunk - emit initial events
if c.firstWrite {
c.firstWrite = false
events = append(events, c.createResponseCreatedEvent())
events = append(events, c.createResponseInProgressEvent())
}
// Handle reasoning/thinking (before other content)
if hasThinking {
events = append(events, c.processThinking(r.Message.Thinking)...)
}
// Handle tool calls
if hasToolCalls {
events = append(events, c.processToolCalls(r.Message.ToolCalls)...)
c.toolCallsSent = true
}
// Handle text content (only if no tool calls)
if !hasToolCalls && !c.toolCallsSent && r.Message.Content != "" {
events = append(events, c.processTextContent(r.Message.Content)...)
}
// Done - emit closing events
if r.Done {
events = append(events, c.processCompletion(r)...)
}
return events
}
func (c *ResponsesStreamConverter) createResponseCreatedEvent() ResponsesStreamEvent {
return ResponsesStreamEvent{
Event: "response.created",
Data: map[string]any{
"type": "response.created",
"response": map[string]any{
"id": c.responseID,
"object": "response",
"status": "in_progress",
},
},
}
}
func (c *ResponsesStreamConverter) createResponseInProgressEvent() ResponsesStreamEvent {
return ResponsesStreamEvent{
Event: "response.in_progress",
Data: map[string]any{
"type": "response.in_progress",
"response": map[string]any{
"id": c.responseID,
"object": "response",
"status": "in_progress",
},
},
}
}
func (c *ResponsesStreamConverter) processThinking(thinking string) []ResponsesStreamEvent {
var events []ResponsesStreamEvent
// Start reasoning item if not started
if !c.reasoningStarted {
c.reasoningStarted = true
c.reasoningItemID = fmt.Sprintf("rs_%d", rand.Intn(999999))
events = append(events, ResponsesStreamEvent{
Event: "response.output_item.added",
Data: map[string]any{
"type": "response.output_item.added",
"output_index": c.outputIndex,
"item": map[string]any{
"id": c.reasoningItemID,
"type": "reasoning",
"summary": []any{},
},
},
})
}
// Accumulate thinking
c.accumulatedThinking += thinking
// Emit delta
events = append(events, ResponsesStreamEvent{
Event: "response.reasoning_summary_text.delta",
Data: map[string]any{
"type": "response.reasoning_summary_text.delta",
"item_id": c.reasoningItemID,
"output_index": c.outputIndex,
"delta": thinking,
},
})
return events
}
func (c *ResponsesStreamConverter) finishReasoning() []ResponsesStreamEvent {
if !c.reasoningStarted || c.reasoningDone {
return nil
}
c.reasoningDone = true
events := []ResponsesStreamEvent{
{
Event: "response.reasoning_summary_text.done",
Data: map[string]any{
"type": "response.reasoning_summary_text.done",
"item_id": c.reasoningItemID,
"output_index": c.outputIndex,
"text": c.accumulatedThinking,
},
},
{
Event: "response.output_item.done",
Data: map[string]any{
"type": "response.output_item.done",
"output_index": c.outputIndex,
"item": map[string]any{
"id": c.reasoningItemID,
"type": "reasoning",
"summary": []map[string]any{{"type": "summary_text", "text": c.accumulatedThinking}},
"encrypted_content": c.accumulatedThinking, // Plain text for now
},
},
},
}
c.outputIndex++
return events
}
func (c *ResponsesStreamConverter) processToolCalls(toolCalls []api.ToolCall) []ResponsesStreamEvent {
var events []ResponsesStreamEvent
// Finish reasoning first if it was started
events = append(events, c.finishReasoning()...)
converted := ToToolCalls(toolCalls)
for i, tc := range converted {
fcItemID := fmt.Sprintf("fc_%d_%d", rand.Intn(999999), i)
// response.output_item.added for function call
events = append(events, ResponsesStreamEvent{
Event: "response.output_item.added",
Data: map[string]any{
"type": "response.output_item.added",
"output_index": c.outputIndex + i,
"item": map[string]any{
"id": fcItemID,
"type": "function_call",
"call_id": tc.ID,
"name": tc.Function.Name,
"arguments": "",
},
},
})
// response.function_call_arguments.delta
if tc.Function.Arguments != "" {
events = append(events, ResponsesStreamEvent{
Event: "response.function_call_arguments.delta",
Data: map[string]any{
"type": "response.function_call_arguments.delta",
"item_id": fcItemID,
"output_index": c.outputIndex + i,
"delta": tc.Function.Arguments,
},
})
}
// response.function_call_arguments.done
events = append(events, ResponsesStreamEvent{
Event: "response.function_call_arguments.done",
Data: map[string]any{
"type": "response.function_call_arguments.done",
"item_id": fcItemID,
"output_index": c.outputIndex + i,
"arguments": tc.Function.Arguments,
},
})
// response.output_item.done for function call
events = append(events, ResponsesStreamEvent{
Event: "response.output_item.done",
Data: map[string]any{
"type": "response.output_item.done",
"output_index": c.outputIndex + i,
"item": map[string]any{
"id": fcItemID,
"type": "function_call",
"call_id": tc.ID,
"name": tc.Function.Name,
"arguments": tc.Function.Arguments,
},
},
})
}
return events
}
func (c *ResponsesStreamConverter) processTextContent(content string) []ResponsesStreamEvent {
var events []ResponsesStreamEvent
// Finish reasoning first if it was started
events = append(events, c.finishReasoning()...)
// Emit output item and content part for first text content
if !c.contentStarted {
c.contentStarted = true
// response.output_item.added
events = append(events, ResponsesStreamEvent{
Event: "response.output_item.added",
Data: map[string]any{
"type": "response.output_item.added",
"output_index": c.outputIndex,
"item": map[string]any{
"id": c.itemID,
"type": "message",
"role": "assistant",
"content": []any{},
},
},
})
// response.content_part.added
events = append(events, ResponsesStreamEvent{
Event: "response.content_part.added",
Data: map[string]any{
"type": "response.content_part.added",
"item_id": c.itemID,
"output_index": c.outputIndex,
"content_index": c.contentIndex,
"part": map[string]any{
"type": "output_text",
"text": "",
},
},
})
}
// Accumulate text
c.accumulatedText += content
// Emit content delta
events = append(events, ResponsesStreamEvent{
Event: "response.output_text.delta",
Data: map[string]any{
"type": "response.output_text.delta",
"item_id": c.itemID,
"output_index": c.outputIndex,
"content_index": 0,
"delta": content,
},
})
return events
}
func (c *ResponsesStreamConverter) processCompletion(r api.ChatResponse) []ResponsesStreamEvent {
var events []ResponsesStreamEvent
// Finish reasoning if not done
events = append(events, c.finishReasoning()...)
// Emit text completion events if we had text content
if !c.toolCallsSent && c.contentStarted {
// response.output_text.done
events = append(events, ResponsesStreamEvent{
Event: "response.output_text.done",
Data: map[string]any{
"type": "response.output_text.done",
"item_id": c.itemID,
"output_index": c.outputIndex,
"content_index": 0,
"text": c.accumulatedText,
},
})
// response.content_part.done
events = append(events, ResponsesStreamEvent{
Event: "response.content_part.done",
Data: map[string]any{
"type": "response.content_part.done",
"item_id": c.itemID,
"output_index": c.outputIndex,
"content_index": 0,
"part": map[string]any{
"type": "output_text",
"text": c.accumulatedText,
},
},
})
// response.output_item.done
events = append(events, ResponsesStreamEvent{
Event: "response.output_item.done",
Data: map[string]any{
"type": "response.output_item.done",
"output_index": c.outputIndex,
"item": map[string]any{
"id": c.itemID,
"type": "message",
"role": "assistant",
},
},
})
}
// response.completed
events = append(events, ResponsesStreamEvent{
Event: "response.completed",
Data: map[string]any{
"type": "response.completed",
"response": map[string]any{
"id": c.responseID,
"object": "response",
"status": "completed",
"usage": map[string]any{
"input_tokens": r.PromptEvalCount,
"output_tokens": r.EvalCount,
"total_tokens": r.PromptEvalCount + r.EvalCount,
},
},
},
})
return events
}

999
openai/responses_test.go Normal file
View File

@@ -0,0 +1,999 @@
package openai
import (
"encoding/json"
"testing"
"time"
"github.com/ollama/ollama/api"
)
func TestResponsesInputMessage_UnmarshalJSON(t *testing.T) {
tests := []struct {
name string
json string
want ResponsesInputMessage
wantErr bool
}{
{
name: "text content",
json: `{"type": "message", "role": "user", "content": [{"type": "input_text", "text": "hello"}]}`,
want: ResponsesInputMessage{
Type: "message",
Role: "user",
Content: []ResponsesContent{ResponsesTextContent{Type: "input_text", Text: "hello"}},
},
},
{
name: "image content",
json: `{"type": "message", "role": "user", "content": [{"type": "input_image", "detail": "auto", "image_url": "https://example.com/img.png"}]}`,
want: ResponsesInputMessage{
Type: "message",
Role: "user",
Content: []ResponsesContent{ResponsesImageContent{
Type: "input_image",
Detail: "auto",
ImageURL: "https://example.com/img.png",
}},
},
},
{
name: "multiple content items",
json: `{"type": "message", "role": "user", "content": [{"type": "input_text", "text": "hello"}, {"type": "input_text", "text": "world"}]}`,
want: ResponsesInputMessage{
Type: "message",
Role: "user",
Content: []ResponsesContent{
ResponsesTextContent{Type: "input_text", Text: "hello"},
ResponsesTextContent{Type: "input_text", Text: "world"},
},
},
},
{
name: "unknown content type",
json: `{"type": "message", "role": "user", "content": [{"type": "unknown"}]}`,
wantErr: true,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
var got ResponsesInputMessage
err := json.Unmarshal([]byte(tt.json), &got)
if tt.wantErr {
if err == nil {
t.Error("expected error, got nil")
}
return
}
if err != nil {
t.Fatalf("unexpected error: %v", err)
}
if got.Type != tt.want.Type {
t.Errorf("Type = %q, want %q", got.Type, tt.want.Type)
}
if got.Role != tt.want.Role {
t.Errorf("Role = %q, want %q", got.Role, tt.want.Role)
}
if len(got.Content) != len(tt.want.Content) {
t.Fatalf("len(Content) = %d, want %d", len(got.Content), len(tt.want.Content))
}
for i := range tt.want.Content {
switch wantContent := tt.want.Content[i].(type) {
case ResponsesTextContent:
gotContent, ok := got.Content[i].(ResponsesTextContent)
if !ok {
t.Fatalf("Content[%d] type = %T, want ResponsesTextContent", i, got.Content[i])
}
if gotContent != wantContent {
t.Errorf("Content[%d] = %+v, want %+v", i, gotContent, wantContent)
}
case ResponsesImageContent:
gotContent, ok := got.Content[i].(ResponsesImageContent)
if !ok {
t.Fatalf("Content[%d] type = %T, want ResponsesImageContent", i, got.Content[i])
}
if gotContent != wantContent {
t.Errorf("Content[%d] = %+v, want %+v", i, gotContent, wantContent)
}
}
}
})
}
}
func TestResponsesInput_UnmarshalJSON(t *testing.T) {
tests := []struct {
name string
json string
wantText string
wantItems int
wantErr bool
}{
{
name: "plain string",
json: `"hello world"`,
wantText: "hello world",
},
{
name: "array with one message",
json: `[{"type": "message", "role": "user", "content": [{"type": "input_text", "text": "hello"}]}]`,
wantItems: 1,
},
{
name: "array with multiple messages",
json: `[{"type": "message", "role": "system", "content": [{"type": "input_text", "text": "you are helpful"}]}, {"type": "message", "role": "user", "content": [{"type": "input_text", "text": "hello"}]}]`,
wantItems: 2,
},
{
name: "invalid input",
json: `123`,
wantErr: true,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
var got ResponsesInput
err := json.Unmarshal([]byte(tt.json), &got)
if tt.wantErr {
if err == nil {
t.Error("expected error, got nil")
}
return
}
if err != nil {
t.Fatalf("unexpected error: %v", err)
}
if got.Text != tt.wantText {
t.Errorf("Text = %q, want %q", got.Text, tt.wantText)
}
if len(got.Items) != tt.wantItems {
t.Errorf("len(Items) = %d, want %d", len(got.Items), tt.wantItems)
}
})
}
}
func TestUnmarshalResponsesInputItem(t *testing.T) {
t.Run("message item", func(t *testing.T) {
got, err := unmarshalResponsesInputItem([]byte(`{"type": "message", "role": "user", "content": [{"type": "input_text", "text": "hello"}]}`))
if err != nil {
t.Fatalf("unexpected error: %v", err)
}
msg, ok := got.(ResponsesInputMessage)
if !ok {
t.Fatalf("got type %T, want ResponsesInputMessage", got)
}
if msg.Role != "user" {
t.Errorf("Role = %q, want %q", msg.Role, "user")
}
})
t.Run("function_call item", func(t *testing.T) {
got, err := unmarshalResponsesInputItem([]byte(`{"type": "function_call", "call_id": "call_abc123", "name": "get_weather", "arguments": "{\"city\":\"Paris\"}"}`))
if err != nil {
t.Fatalf("unexpected error: %v", err)
}
fc, ok := got.(ResponsesFunctionCall)
if !ok {
t.Fatalf("got type %T, want ResponsesFunctionCall", got)
}
if fc.Type != "function_call" {
t.Errorf("Type = %q, want %q", fc.Type, "function_call")
}
if fc.CallID != "call_abc123" {
t.Errorf("CallID = %q, want %q", fc.CallID, "call_abc123")
}
if fc.Name != "get_weather" {
t.Errorf("Name = %q, want %q", fc.Name, "get_weather")
}
})
t.Run("function_call_output item", func(t *testing.T) {
got, err := unmarshalResponsesInputItem([]byte(`{"type": "function_call_output", "call_id": "call_abc123", "output": "the result"}`))
if err != nil {
t.Fatalf("unexpected error: %v", err)
}
output, ok := got.(ResponsesFunctionCallOutput)
if !ok {
t.Fatalf("got type %T, want ResponsesFunctionCallOutput", got)
}
if output.Type != "function_call_output" {
t.Errorf("Type = %q, want %q", output.Type, "function_call_output")
}
if output.CallID != "call_abc123" {
t.Errorf("CallID = %q, want %q", output.CallID, "call_abc123")
}
if output.Output != "the result" {
t.Errorf("Output = %q, want %q", output.Output, "the result")
}
})
t.Run("unknown item type", func(t *testing.T) {
_, err := unmarshalResponsesInputItem([]byte(`{"type": "unknown_type"}`))
if err == nil {
t.Error("expected error, got nil")
}
})
}
func TestResponsesRequest_UnmarshalJSON(t *testing.T) {
tests := []struct {
name string
json string
check func(t *testing.T, req ResponsesRequest)
wantErr bool
}{
{
name: "simple string input",
json: `{"model": "gpt-4", "input": "hello"}`,
check: func(t *testing.T, req ResponsesRequest) {
if req.Model != "gpt-4" {
t.Errorf("Model = %q, want %q", req.Model, "gpt-4")
}
if req.Input.Text != "hello" {
t.Errorf("Input.Text = %q, want %q", req.Input.Text, "hello")
}
},
},
{
name: "array input with messages",
json: `{"model": "gpt-4", "input": [{"type": "message", "role": "user", "content": [{"type": "input_text", "text": "hello"}]}]}`,
check: func(t *testing.T, req ResponsesRequest) {
if len(req.Input.Items) != 1 {
t.Fatalf("len(Input.Items) = %d, want 1", len(req.Input.Items))
}
msg, ok := req.Input.Items[0].(ResponsesInputMessage)
if !ok {
t.Fatalf("Input.Items[0] type = %T, want ResponsesInputMessage", req.Input.Items[0])
}
if msg.Role != "user" {
t.Errorf("Role = %q, want %q", msg.Role, "user")
}
},
},
{
name: "with temperature",
json: `{"model": "gpt-4", "input": "hello", "temperature": 0.5}`,
check: func(t *testing.T, req ResponsesRequest) {
if req.Temperature == nil || *req.Temperature != 0.5 {
t.Errorf("Temperature = %v, want 0.5", req.Temperature)
}
},
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
var got ResponsesRequest
err := json.Unmarshal([]byte(tt.json), &got)
if tt.wantErr {
if err == nil {
t.Error("expected error, got nil")
}
return
}
if err != nil {
t.Fatalf("unexpected error: %v", err)
}
if tt.check != nil {
tt.check(t, got)
}
})
}
}
func TestFromResponsesRequest_Tools(t *testing.T) {
reqJSON := `{
"model": "gpt-4",
"input": "hello",
"tools": [
{
"type": "function",
"name": "shell",
"description": "Runs a shell command",
"strict": false,
"parameters": {
"type": "object",
"properties": {
"command": {
"type": "array",
"items": {"type": "string"},
"description": "The command to execute"
}
},
"required": ["command"]
}
}
]
}`
var req ResponsesRequest
if err := json.Unmarshal([]byte(reqJSON), &req); err != nil {
t.Fatalf("failed to unmarshal request: %v", err)
}
// Check that tools were parsed
if len(req.Tools) != 1 {
t.Fatalf("expected 1 tool, got %d", len(req.Tools))
}
if req.Tools[0].Name != "shell" {
t.Errorf("expected tool name 'shell', got %q", req.Tools[0].Name)
}
// Convert and check
chatReq, err := FromResponsesRequest(req)
if err != nil {
t.Fatalf("failed to convert request: %v", err)
}
if len(chatReq.Tools) != 1 {
t.Fatalf("expected 1 converted tool, got %d", len(chatReq.Tools))
}
tool := chatReq.Tools[0]
if tool.Type != "function" {
t.Errorf("expected tool type 'function', got %q", tool.Type)
}
if tool.Function.Name != "shell" {
t.Errorf("expected function name 'shell', got %q", tool.Function.Name)
}
if tool.Function.Description != "Runs a shell command" {
t.Errorf("expected function description 'Runs a shell command', got %q", tool.Function.Description)
}
if tool.Function.Parameters.Type != "object" {
t.Errorf("expected parameters type 'object', got %q", tool.Function.Parameters.Type)
}
if len(tool.Function.Parameters.Required) != 1 || tool.Function.Parameters.Required[0] != "command" {
t.Errorf("expected required ['command'], got %v", tool.Function.Parameters.Required)
}
}
func TestFromResponsesRequest_FunctionCallOutput(t *testing.T) {
// Test a complete tool call round-trip:
// 1. User message asking about weather
// 2. Assistant's function call (from previous response)
// 3. Function call output (the tool result)
reqJSON := `{
"model": "gpt-4",
"input": [
{"type": "message", "role": "user", "content": [{"type": "input_text", "text": "what is the weather?"}]},
{"type": "function_call", "call_id": "call_abc123", "name": "get_weather", "arguments": "{\"city\":\"Paris\"}"},
{"type": "function_call_output", "call_id": "call_abc123", "output": "sunny, 72F"}
]
}`
var req ResponsesRequest
if err := json.Unmarshal([]byte(reqJSON), &req); err != nil {
t.Fatalf("failed to unmarshal request: %v", err)
}
// Check that input items were parsed
if len(req.Input.Items) != 3 {
t.Fatalf("expected 3 input items, got %d", len(req.Input.Items))
}
// Verify the function_call item
fc, ok := req.Input.Items[1].(ResponsesFunctionCall)
if !ok {
t.Fatalf("Input.Items[1] type = %T, want ResponsesFunctionCall", req.Input.Items[1])
}
if fc.Name != "get_weather" {
t.Errorf("Name = %q, want %q", fc.Name, "get_weather")
}
// Verify the function_call_output item
fcOutput, ok := req.Input.Items[2].(ResponsesFunctionCallOutput)
if !ok {
t.Fatalf("Input.Items[2] type = %T, want ResponsesFunctionCallOutput", req.Input.Items[2])
}
if fcOutput.CallID != "call_abc123" {
t.Errorf("CallID = %q, want %q", fcOutput.CallID, "call_abc123")
}
// Convert and check
chatReq, err := FromResponsesRequest(req)
if err != nil {
t.Fatalf("failed to convert request: %v", err)
}
if len(chatReq.Messages) != 3 {
t.Fatalf("expected 3 messages, got %d", len(chatReq.Messages))
}
// Check the user message
userMsg := chatReq.Messages[0]
if userMsg.Role != "user" {
t.Errorf("expected role 'user', got %q", userMsg.Role)
}
// Check the assistant message with tool call
assistantMsg := chatReq.Messages[1]
if assistantMsg.Role != "assistant" {
t.Errorf("expected role 'assistant', got %q", assistantMsg.Role)
}
if len(assistantMsg.ToolCalls) != 1 {
t.Fatalf("expected 1 tool call, got %d", len(assistantMsg.ToolCalls))
}
if assistantMsg.ToolCalls[0].ID != "call_abc123" {
t.Errorf("expected tool call ID 'call_abc123', got %q", assistantMsg.ToolCalls[0].ID)
}
if assistantMsg.ToolCalls[0].Function.Name != "get_weather" {
t.Errorf("expected function name 'get_weather', got %q", assistantMsg.ToolCalls[0].Function.Name)
}
// Check the tool response message
toolMsg := chatReq.Messages[2]
if toolMsg.Role != "tool" {
t.Errorf("expected role 'tool', got %q", toolMsg.Role)
}
if toolMsg.Content != "sunny, 72F" {
t.Errorf("expected content 'sunny, 72F', got %q", toolMsg.Content)
}
if toolMsg.ToolCallID != "call_abc123" {
t.Errorf("expected ToolCallID 'call_abc123', got %q", toolMsg.ToolCallID)
}
}
func TestDecodeImageURL(t *testing.T) {
// Valid PNG base64 (1x1 red pixel)
validPNG := ""
t.Run("valid png", func(t *testing.T) {
img, err := decodeImageURL(validPNG)
if err != nil {
t.Fatalf("unexpected error: %v", err)
}
if len(img) == 0 {
t.Error("expected non-empty image data")
}
})
t.Run("valid jpeg", func(t *testing.T) {
// Just test the prefix validation with minimal base64
_, err := decodeImageURL("")
if err != nil {
t.Fatalf("unexpected error: %v", err)
}
})
t.Run("blank mime type", func(t *testing.T) {
_, err := decodeImageURL("data:;base64,dGVzdA==")
if err != nil {
t.Fatalf("unexpected error: %v", err)
}
})
t.Run("invalid mime type", func(t *testing.T) {
_, err := decodeImageURL("")
if err == nil {
t.Error("expected error for unsupported mime type")
}
})
t.Run("invalid base64", func(t *testing.T) {
_, err := decodeImageURL("-valid-base64!")
if err == nil {
t.Error("expected error for invalid base64")
}
})
t.Run("not a data url", func(t *testing.T) {
_, err := decodeImageURL("https://example.com/image.png")
if err == nil {
t.Error("expected error for non-data URL")
}
})
}
func TestFromResponsesRequest_Images(t *testing.T) {
// 1x1 red PNG pixel
pngBase64 := "iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVR42mP8z8DwHwAFBQIAX8jx0gAAAABJRU5ErkJggg=="
reqJSON := `{
"model": "llava",
"input": [
{"type": "message", "role": "user", "content": [
{"type": "input_text", "text": "What is in this image?"},
{"type": "input_image", "detail": "auto", "image_url": "data:image/png;base64,` + pngBase64 + `"}
]}
]
}`
var req ResponsesRequest
if err := json.Unmarshal([]byte(reqJSON), &req); err != nil {
t.Fatalf("failed to unmarshal request: %v", err)
}
chatReq, err := FromResponsesRequest(req)
if err != nil {
t.Fatalf("failed to convert request: %v", err)
}
if len(chatReq.Messages) != 1 {
t.Fatalf("expected 1 message, got %d", len(chatReq.Messages))
}
msg := chatReq.Messages[0]
if msg.Role != "user" {
t.Errorf("expected role 'user', got %q", msg.Role)
}
if msg.Content != "What is in this image?" {
t.Errorf("expected content 'What is in this image?', got %q", msg.Content)
}
if len(msg.Images) != 1 {
t.Fatalf("expected 1 image, got %d", len(msg.Images))
}
if len(msg.Images[0]) == 0 {
t.Error("expected non-empty image data")
}
}
func TestResponsesStreamConverter_TextOnly(t *testing.T) {
converter := NewResponsesStreamConverter("resp_123", "msg_456", "gpt-4")
// First chunk with content
events := converter.Process(api.ChatResponse{
Message: api.Message{
Content: "Hello",
},
})
// Should have: response.created, response.in_progress, output_item.added, content_part.added, output_text.delta
if len(events) != 5 {
t.Fatalf("expected 5 events, got %d", len(events))
}
if events[0].Event != "response.created" {
t.Errorf("events[0].Event = %q, want %q", events[0].Event, "response.created")
}
if events[1].Event != "response.in_progress" {
t.Errorf("events[1].Event = %q, want %q", events[1].Event, "response.in_progress")
}
if events[2].Event != "response.output_item.added" {
t.Errorf("events[2].Event = %q, want %q", events[2].Event, "response.output_item.added")
}
if events[3].Event != "response.content_part.added" {
t.Errorf("events[3].Event = %q, want %q", events[3].Event, "response.content_part.added")
}
if events[4].Event != "response.output_text.delta" {
t.Errorf("events[4].Event = %q, want %q", events[4].Event, "response.output_text.delta")
}
// Second chunk with more content
events = converter.Process(api.ChatResponse{
Message: api.Message{
Content: " World",
},
})
// Should only have output_text.delta (no more created/in_progress/added)
if len(events) != 1 {
t.Fatalf("expected 1 event, got %d", len(events))
}
if events[0].Event != "response.output_text.delta" {
t.Errorf("events[0].Event = %q, want %q", events[0].Event, "response.output_text.delta")
}
// Final chunk
events = converter.Process(api.ChatResponse{
Message: api.Message{},
Done: true,
})
// Should have: output_text.done, content_part.done, output_item.done, response.completed
if len(events) != 4 {
t.Fatalf("expected 4 events, got %d", len(events))
}
if events[0].Event != "response.output_text.done" {
t.Errorf("events[0].Event = %q, want %q", events[0].Event, "response.output_text.done")
}
// Check that accumulated text is present
data := events[0].Data.(map[string]any)
if data["text"] != "Hello World" {
t.Errorf("accumulated text = %q, want %q", data["text"], "Hello World")
}
}
func TestResponsesStreamConverter_ToolCalls(t *testing.T) {
converter := NewResponsesStreamConverter("resp_123", "msg_456", "gpt-4")
events := converter.Process(api.ChatResponse{
Message: api.Message{
ToolCalls: []api.ToolCall{
{
ID: "call_abc",
Function: api.ToolCallFunction{
Name: "get_weather",
Arguments: api.ToolCallFunctionArguments{"city": "Paris"},
},
},
},
},
})
// Should have: created, in_progress, output_item.added, arguments.delta, arguments.done, output_item.done
if len(events) != 6 {
t.Fatalf("expected 6 events, got %d", len(events))
}
if events[2].Event != "response.output_item.added" {
t.Errorf("events[2].Event = %q, want %q", events[2].Event, "response.output_item.added")
}
if events[3].Event != "response.function_call_arguments.delta" {
t.Errorf("events[3].Event = %q, want %q", events[3].Event, "response.function_call_arguments.delta")
}
if events[4].Event != "response.function_call_arguments.done" {
t.Errorf("events[4].Event = %q, want %q", events[4].Event, "response.function_call_arguments.done")
}
if events[5].Event != "response.output_item.done" {
t.Errorf("events[5].Event = %q, want %q", events[5].Event, "response.output_item.done")
}
}
func TestResponsesStreamConverter_Reasoning(t *testing.T) {
converter := NewResponsesStreamConverter("resp_123", "msg_456", "gpt-4")
// First chunk with thinking
events := converter.Process(api.ChatResponse{
Message: api.Message{
Thinking: "Let me think...",
},
})
// Should have: created, in_progress, output_item.added (reasoning), reasoning_summary_text.delta
if len(events) != 4 {
t.Fatalf("expected 4 events, got %d", len(events))
}
if events[2].Event != "response.output_item.added" {
t.Errorf("events[2].Event = %q, want %q", events[2].Event, "response.output_item.added")
}
// Check it's a reasoning item
data := events[2].Data.(map[string]any)
item := data["item"].(map[string]any)
if item["type"] != "reasoning" {
t.Errorf("item type = %q, want %q", item["type"], "reasoning")
}
if events[3].Event != "response.reasoning_summary_text.delta" {
t.Errorf("events[3].Event = %q, want %q", events[3].Event, "response.reasoning_summary_text.delta")
}
// Second chunk with text content (reasoning should close first)
events = converter.Process(api.ChatResponse{
Message: api.Message{
Content: "The answer is 42",
},
})
// Should have: reasoning_summary_text.done, output_item.done (reasoning), output_item.added (message), content_part.added, output_text.delta
if len(events) != 5 {
t.Fatalf("expected 5 events, got %d", len(events))
}
if events[0].Event != "response.reasoning_summary_text.done" {
t.Errorf("events[0].Event = %q, want %q", events[0].Event, "response.reasoning_summary_text.done")
}
if events[1].Event != "response.output_item.done" {
t.Errorf("events[1].Event = %q, want %q", events[1].Event, "response.output_item.done")
}
// Check the reasoning done item has encrypted_content
doneData := events[1].Data.(map[string]any)
doneItem := doneData["item"].(map[string]any)
if doneItem["encrypted_content"] != "Let me think..." {
t.Errorf("encrypted_content = %q, want %q", doneItem["encrypted_content"], "Let me think...")
}
}
func TestFromResponsesRequest_ReasoningMerge(t *testing.T) {
t.Run("reasoning merged with following message", func(t *testing.T) {
reqJSON := `{
"model": "qwen3",
"input": [
{"type": "message", "role": "user", "content": [{"type": "input_text", "text": "solve 2+2"}]},
{"type": "reasoning", "id": "rs_123", "encrypted_content": "Let me think about this math problem...", "summary": [{"type": "summary_text", "text": "Thinking about math"}]},
{"type": "message", "role": "assistant", "content": [{"type": "input_text", "text": "The answer is 4"}]}
]
}`
var req ResponsesRequest
if err := json.Unmarshal([]byte(reqJSON), &req); err != nil {
t.Fatalf("failed to unmarshal request: %v", err)
}
chatReq, err := FromResponsesRequest(req)
if err != nil {
t.Fatalf("failed to convert request: %v", err)
}
// Should have 2 messages: user and assistant (with thinking merged)
if len(chatReq.Messages) != 2 {
t.Fatalf("expected 2 messages, got %d", len(chatReq.Messages))
}
// Check user message
if chatReq.Messages[0].Role != "user" {
t.Errorf("Messages[0].Role = %q, want %q", chatReq.Messages[0].Role, "user")
}
// Check assistant message has both content and thinking
assistantMsg := chatReq.Messages[1]
if assistantMsg.Role != "assistant" {
t.Errorf("Messages[1].Role = %q, want %q", assistantMsg.Role, "assistant")
}
if assistantMsg.Content != "The answer is 4" {
t.Errorf("Messages[1].Content = %q, want %q", assistantMsg.Content, "The answer is 4")
}
if assistantMsg.Thinking != "Let me think about this math problem..." {
t.Errorf("Messages[1].Thinking = %q, want %q", assistantMsg.Thinking, "Let me think about this math problem...")
}
})
t.Run("reasoning merged with following function call", func(t *testing.T) {
reqJSON := `{
"model": "qwen3",
"input": [
{"type": "message", "role": "user", "content": [{"type": "input_text", "text": "what is the weather?"}]},
{"type": "reasoning", "id": "rs_123", "encrypted_content": "I need to call a tool for this...", "summary": []},
{"type": "function_call", "call_id": "call_abc", "name": "get_weather", "arguments": "{\"city\":\"Paris\"}"}
]
}`
var req ResponsesRequest
if err := json.Unmarshal([]byte(reqJSON), &req); err != nil {
t.Fatalf("failed to unmarshal request: %v", err)
}
chatReq, err := FromResponsesRequest(req)
if err != nil {
t.Fatalf("failed to convert request: %v", err)
}
// Should have 2 messages: user and assistant (with thinking + tool call)
if len(chatReq.Messages) != 2 {
t.Fatalf("expected 2 messages, got %d", len(chatReq.Messages))
}
// Check assistant message has both tool call and thinking
assistantMsg := chatReq.Messages[1]
if assistantMsg.Role != "assistant" {
t.Errorf("Messages[1].Role = %q, want %q", assistantMsg.Role, "assistant")
}
if assistantMsg.Thinking != "I need to call a tool for this..." {
t.Errorf("Messages[1].Thinking = %q, want %q", assistantMsg.Thinking, "I need to call a tool for this...")
}
if len(assistantMsg.ToolCalls) != 1 {
t.Fatalf("expected 1 tool call, got %d", len(assistantMsg.ToolCalls))
}
if assistantMsg.ToolCalls[0].Function.Name != "get_weather" {
t.Errorf("ToolCalls[0].Function.Name = %q, want %q", assistantMsg.ToolCalls[0].Function.Name, "get_weather")
}
})
t.Run("multi-turn conversation with reasoning", func(t *testing.T) {
// Simulates: user asks -> model thinks + responds -> user follows up
reqJSON := `{
"model": "qwen3",
"input": [
{"type": "message", "role": "user", "content": [{"type": "input_text", "text": "What is 2+2?"}]},
{"type": "reasoning", "id": "rs_001", "encrypted_content": "This is a simple arithmetic problem. 2+2=4.", "summary": [{"type": "summary_text", "text": "Calculating 2+2"}]},
{"type": "message", "role": "assistant", "content": [{"type": "input_text", "text": "The answer is 4."}]},
{"type": "message", "role": "user", "content": [{"type": "input_text", "text": "Now multiply that by 3"}]}
]
}`
var req ResponsesRequest
if err := json.Unmarshal([]byte(reqJSON), &req); err != nil {
t.Fatalf("failed to unmarshal request: %v", err)
}
chatReq, err := FromResponsesRequest(req)
if err != nil {
t.Fatalf("failed to convert request: %v", err)
}
// Should have 3 messages:
// 1. user: "What is 2+2?"
// 2. assistant: thinking + "The answer is 4."
// 3. user: "Now multiply that by 3"
if len(chatReq.Messages) != 3 {
t.Fatalf("expected 3 messages, got %d", len(chatReq.Messages))
}
// Check first user message
if chatReq.Messages[0].Role != "user" || chatReq.Messages[0].Content != "What is 2+2?" {
t.Errorf("Messages[0] = {Role: %q, Content: %q}, want {Role: \"user\", Content: \"What is 2+2?\"}",
chatReq.Messages[0].Role, chatReq.Messages[0].Content)
}
// Check assistant message has merged thinking + content
if chatReq.Messages[1].Role != "assistant" {
t.Errorf("Messages[1].Role = %q, want \"assistant\"", chatReq.Messages[1].Role)
}
if chatReq.Messages[1].Content != "The answer is 4." {
t.Errorf("Messages[1].Content = %q, want \"The answer is 4.\"", chatReq.Messages[1].Content)
}
if chatReq.Messages[1].Thinking != "This is a simple arithmetic problem. 2+2=4." {
t.Errorf("Messages[1].Thinking = %q, want \"This is a simple arithmetic problem. 2+2=4.\"",
chatReq.Messages[1].Thinking)
}
// Check second user message
if chatReq.Messages[2].Role != "user" || chatReq.Messages[2].Content != "Now multiply that by 3" {
t.Errorf("Messages[2] = {Role: %q, Content: %q}, want {Role: \"user\", Content: \"Now multiply that by 3\"}",
chatReq.Messages[2].Role, chatReq.Messages[2].Content)
}
})
t.Run("multi-turn with tool calls and reasoning", func(t *testing.T) {
// Simulates: user asks -> model thinks + calls tool -> tool responds -> model thinks + responds -> user follows up
reqJSON := `{
"model": "qwen3",
"input": [
{"type": "message", "role": "user", "content": [{"type": "input_text", "text": "What is the weather in Paris?"}]},
{"type": "reasoning", "id": "rs_001", "encrypted_content": "I need to call the weather API for Paris.", "summary": []},
{"type": "function_call", "call_id": "call_abc", "name": "get_weather", "arguments": "{\"city\":\"Paris\"}"},
{"type": "function_call_output", "call_id": "call_abc", "output": "Sunny, 72°F"},
{"type": "reasoning", "id": "rs_002", "encrypted_content": "The weather API returned sunny and 72°F. I should format this nicely.", "summary": []},
{"type": "message", "role": "assistant", "content": [{"type": "input_text", "text": "It's sunny and 72°F in Paris!"}]},
{"type": "message", "role": "user", "content": [{"type": "input_text", "text": "What about London?"}]}
]
}`
var req ResponsesRequest
if err := json.Unmarshal([]byte(reqJSON), &req); err != nil {
t.Fatalf("failed to unmarshal request: %v", err)
}
chatReq, err := FromResponsesRequest(req)
if err != nil {
t.Fatalf("failed to convert request: %v", err)
}
// Should have 5 messages:
// 1. user: "What is the weather in Paris?"
// 2. assistant: thinking + tool call
// 3. tool: "Sunny, 72°F"
// 4. assistant: thinking + "It's sunny and 72°F in Paris!"
// 5. user: "What about London?"
if len(chatReq.Messages) != 5 {
t.Fatalf("expected 5 messages, got %d", len(chatReq.Messages))
}
// Message 1: user
if chatReq.Messages[0].Role != "user" {
t.Errorf("Messages[0].Role = %q, want \"user\"", chatReq.Messages[0].Role)
}
// Message 2: assistant with thinking + tool call
if chatReq.Messages[1].Role != "assistant" {
t.Errorf("Messages[1].Role = %q, want \"assistant\"", chatReq.Messages[1].Role)
}
if chatReq.Messages[1].Thinking != "I need to call the weather API for Paris." {
t.Errorf("Messages[1].Thinking = %q, want \"I need to call the weather API for Paris.\"", chatReq.Messages[1].Thinking)
}
if len(chatReq.Messages[1].ToolCalls) != 1 || chatReq.Messages[1].ToolCalls[0].Function.Name != "get_weather" {
t.Errorf("Messages[1].ToolCalls not as expected")
}
// Message 3: tool response
if chatReq.Messages[2].Role != "tool" || chatReq.Messages[2].Content != "Sunny, 72°F" {
t.Errorf("Messages[2] = {Role: %q, Content: %q}, want {Role: \"tool\", Content: \"Sunny, 72°F\"}",
chatReq.Messages[2].Role, chatReq.Messages[2].Content)
}
// Message 4: assistant with thinking + content
if chatReq.Messages[3].Role != "assistant" {
t.Errorf("Messages[3].Role = %q, want \"assistant\"", chatReq.Messages[3].Role)
}
if chatReq.Messages[3].Thinking != "The weather API returned sunny and 72°F. I should format this nicely." {
t.Errorf("Messages[3].Thinking = %q, want correct thinking", chatReq.Messages[3].Thinking)
}
if chatReq.Messages[3].Content != "It's sunny and 72°F in Paris!" {
t.Errorf("Messages[3].Content = %q, want \"It's sunny and 72°F in Paris!\"", chatReq.Messages[3].Content)
}
// Message 5: user follow-up
if chatReq.Messages[4].Role != "user" || chatReq.Messages[4].Content != "What about London?" {
t.Errorf("Messages[4] = {Role: %q, Content: %q}, want {Role: \"user\", Content: \"What about London?\"}",
chatReq.Messages[4].Role, chatReq.Messages[4].Content)
}
})
t.Run("trailing reasoning creates separate message", func(t *testing.T) {
reqJSON := `{
"model": "qwen3",
"input": [
{"type": "message", "role": "user", "content": [{"type": "input_text", "text": "think about this"}]},
{"type": "reasoning", "id": "rs_123", "encrypted_content": "Still thinking...", "summary": []}
]
}`
var req ResponsesRequest
if err := json.Unmarshal([]byte(reqJSON), &req); err != nil {
t.Fatalf("failed to unmarshal request: %v", err)
}
chatReq, err := FromResponsesRequest(req)
if err != nil {
t.Fatalf("failed to convert request: %v", err)
}
// Should have 2 messages: user and assistant (thinking only)
if len(chatReq.Messages) != 2 {
t.Fatalf("expected 2 messages, got %d", len(chatReq.Messages))
}
// Check assistant message has only thinking
assistantMsg := chatReq.Messages[1]
if assistantMsg.Role != "assistant" {
t.Errorf("Messages[1].Role = %q, want %q", assistantMsg.Role, "assistant")
}
if assistantMsg.Thinking != "Still thinking..." {
t.Errorf("Messages[1].Thinking = %q, want %q", assistantMsg.Thinking, "Still thinking...")
}
if assistantMsg.Content != "" {
t.Errorf("Messages[1].Content = %q, want empty", assistantMsg.Content)
}
})
}
func TestToResponse_WithReasoning(t *testing.T) {
response := ToResponse("gpt-4", "resp_123", "msg_456", api.ChatResponse{
CreatedAt: time.Now(),
Message: api.Message{
Thinking: "Analyzing the question...",
Content: "The answer is 42",
},
Done: true,
})
// Should have 2 output items: reasoning + message
if len(response.Output) != 2 {
t.Fatalf("expected 2 output items, got %d", len(response.Output))
}
// First item should be reasoning
if response.Output[0].Type != "reasoning" {
t.Errorf("Output[0].Type = %q, want %q", response.Output[0].Type, "reasoning")
}
if len(response.Output[0].Summary) != 1 {
t.Fatalf("expected 1 summary item, got %d", len(response.Output[0].Summary))
}
if response.Output[0].Summary[0].Text != "Analyzing the question..." {
t.Errorf("Summary[0].Text = %q, want %q", response.Output[0].Summary[0].Text, "Analyzing the question...")
}
if response.Output[0].EncryptedContent != "Analyzing the question..." {
t.Errorf("EncryptedContent = %q, want %q", response.Output[0].EncryptedContent, "Analyzing the question...")
}
// Second item should be message
if response.Output[1].Type != "message" {
t.Errorf("Output[1].Type = %q, want %q", response.Output[1].Type, "message")
}
if response.Output[1].Content[0].Text != "The answer is 42" {
t.Errorf("Content[0].Text = %q, want %q", response.Output[1].Content[0].Text, "The answer is 42")
}
}

View File

@@ -1523,6 +1523,7 @@ func (s *Server) GenerateRoutes(rc *ollama.Registry) (http.Handler, error) {
r.POST("/v1/embeddings", middleware.EmbeddingsMiddleware(), s.EmbedHandler)
r.GET("/v1/models", middleware.ListMiddleware(), s.ListHandler)
r.GET("/v1/models/:model", middleware.RetrieveMiddleware(), s.ShowHandler)
r.POST("/v1/responses", middleware.ResponsesMiddleware(), s.ChatHandler)
if rc != nil {
// wrap old with new
@@ -2376,3 +2377,4 @@ func filterThinkTags(msgs []api.Message, m *Model) []api.Message {
}
return msgs
}