-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathopenairesponses.go
More file actions
185 lines (162 loc) · 4.91 KB
/
openairesponses.go
File metadata and controls
185 lines (162 loc) · 4.91 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
package iteragent
import (
"bytes"
"context"
"encoding/json"
"fmt"
"io"
"net/http"
"strings"
"time"
)
type OpenAIResponsesConfig struct {
APIKey string
BaseURL string
Model string
MaxTokens int
Temperature float32
ThinkingLevel ThinkingLevel
}
type OpenAIResponsesProvider struct {
config OpenAIResponsesConfig
client *http.Client
}
func NewOpenAIResponses(config OpenAIResponsesConfig) Provider {
return &OpenAIResponsesProvider{
config: config,
client: &http.Client{Timeout: 120 * time.Second},
}
}
func (p *OpenAIResponsesProvider) Name() string {
return fmt.Sprintf("openai_responses(%s)", p.config.Model)
}
type responsesRequest struct {
Model string `json:"model"`
Input []map[string]interface{} `json:"input"`
MaxTokens int `json:"max_tokens,omitempty"`
Temperature float32 `json:"temperature,omitempty"`
Store bool `json:"store,omitempty"`
ResponseFormat string `json:"response_format,omitempty"`
Tools []json.RawMessage `json:"tools,omitempty"`
ParallelToolCalls bool `json:"parallel_tool_calls,omitempty"`
}
func messagesToResponsesFormat(messages []Message) []map[string]interface{} {
result := make([]map[string]interface{}, 0, len(messages))
for _, msg := range messages {
if msg.Role == "system" {
result = append(result, map[string]interface{}{
"role": "system",
"type": "message",
"content": msg.Content,
})
} else if msg.Role == "user" {
result = append(result, map[string]interface{}{
"role": "user",
"type": "message",
"content": msg.Content,
})
} else if msg.Role == "assistant" {
result = append(result, map[string]interface{}{
"role": "assistant",
"type": "message",
"content": msg.Content,
})
}
}
return result
}
func (p *OpenAIResponsesProvider) Complete(ctx context.Context, messages []Message, opts ...CompletionOptions) (string, error) {
baseURL := p.config.BaseURL
if baseURL == "" {
baseURL = "https://api.openai.com/v1"
}
url := baseURL + "/responses"
body := responsesRequest{
Model: p.config.Model,
Input: messagesToResponsesFormat(messages),
MaxTokens: p.config.MaxTokens,
Temperature: p.config.Temperature,
Store: false,
}
jsonBody, err := json.Marshal(body)
if err != nil {
return "", fmt.Errorf("marshal request: %w", err)
}
req, err := http.NewRequestWithContext(ctx, "POST", url, bytes.NewReader(jsonBody))
if err != nil {
return "", fmt.Errorf("create request: %w", err)
}
req.Header.Set("Content-Type", "application/json")
req.Header.Set("Authorization", "Bearer "+p.config.APIKey)
resp, err := p.client.Do(req)
if err != nil {
return "", fmt.Errorf("request failed: %w", err)
}
defer resp.Body.Close()
respBody, err := io.ReadAll(resp.Body)
if err != nil {
return "", fmt.Errorf("read response: %w", err)
}
if resp.StatusCode != http.StatusOK {
return "", fmt.Errorf("OpenAI Responses API error (%d): %s", resp.StatusCode, string(respBody))
}
var response struct {
Output []struct {
Type string `json:"type"`
Content string `json:"content,omitempty"`
Thoughts string `json:"thoughts,omitempty"`
} `json:"output"`
}
if err := json.Unmarshal(respBody, &response); err != nil {
return "", fmt.Errorf("parse response: %w", err)
}
var result strings.Builder
for _, output := range response.Output {
if output.Type == "message" {
result.WriteString(output.Content)
}
}
return result.String(), nil
}
// CompleteStream implements Provider for the OpenAI Responses API.
// It uses the response.content_part.delta SSE event to deliver text tokens incrementally.
func (p *OpenAIResponsesProvider) CompleteStream(ctx context.Context, messages []Message, opt CompletionOptions, onToken func(string)) (string, error) {
baseURL := p.config.BaseURL
if baseURL == "" {
baseURL = "https://api.openai.com/v1"
}
url := baseURL + "/responses"
maxTokens := p.config.MaxTokens
if opt.MaxTokens > 0 {
maxTokens = opt.MaxTokens
}
temperature := p.config.Temperature
if opt.Temperature > 0 {
temperature = opt.Temperature
}
body := responsesRequest{
Model: p.config.Model,
Input: messagesToResponsesFormat(messages),
MaxTokens: maxTokens,
Temperature: temperature,
Store: false,
}
jsonBody, err := json.Marshal(body)
if err != nil {
return "", fmt.Errorf("marshal request: %w", err)
}
var full strings.Builder
sseClient := NewSSEClient()
err = sseClient.Stream(ctx, url, map[string]string{"Authorization": "Bearer " + p.config.APIKey}, jsonBody, func(e SSEEvent) {
if token, ok := ParseOpenAIResponsesSSE(e); ok {
full.WriteString(token)
if onToken != nil {
onToken(token)
}
}
})
if err != nil {
return "", fmt.Errorf("openai responses stream: %w", err)
}
return full.String(), nil
}