forked from sashabaranov/go-openai
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: support batches api (sashabaranov#746)
* feat: support batches api * update batch_test.go * fix golangci-lint check * fix golangci-lint check * fix tests coverage * fix tests coverage * fix tests coverage * fix tests coverage * fix tests coverage * fix tests coverage * fix tests coverage * fix: create batch api * update batch_test.go * feat: add `CreateBatchWithUploadFile` * feat: add `UploadBatchFile` * optimize variable and type naming * expose `BatchLineItem` interface * update batches const
- Loading branch information
Showing
4 changed files
with
655 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,275 @@ | ||
package openai | ||
|
||
import ( | ||
"bytes" | ||
"context" | ||
"encoding/json" | ||
"errors" | ||
"fmt" | ||
"net/http" | ||
"net/url" | ||
) | ||
|
||
const batchesSuffix = "/batches" | ||
|
||
type BatchEndpoint string | ||
|
||
const ( | ||
BatchEndpointChatCompletions BatchEndpoint = "/v1/chat/completions" | ||
BatchEndpointCompletions BatchEndpoint = "/v1/completions" | ||
BatchEndpointEmbeddings BatchEndpoint = "/v1/embeddings" | ||
) | ||
|
||
type BatchLineItem interface { | ||
MarshalBatchLineItem() []byte | ||
} | ||
|
||
type BatchChatCompletionRequest struct { | ||
CustomID string `json:"custom_id"` | ||
Body ChatCompletionRequest `json:"body"` | ||
Method string `json:"method"` | ||
URL BatchEndpoint `json:"url"` | ||
} | ||
|
||
func (r BatchChatCompletionRequest) MarshalBatchLineItem() []byte { | ||
marshal, _ := json.Marshal(r) | ||
return marshal | ||
} | ||
|
||
type BatchCompletionRequest struct { | ||
CustomID string `json:"custom_id"` | ||
Body CompletionRequest `json:"body"` | ||
Method string `json:"method"` | ||
URL BatchEndpoint `json:"url"` | ||
} | ||
|
||
func (r BatchCompletionRequest) MarshalBatchLineItem() []byte { | ||
marshal, _ := json.Marshal(r) | ||
return marshal | ||
} | ||
|
||
type BatchEmbeddingRequest struct { | ||
CustomID string `json:"custom_id"` | ||
Body EmbeddingRequest `json:"body"` | ||
Method string `json:"method"` | ||
URL BatchEndpoint `json:"url"` | ||
} | ||
|
||
func (r BatchEmbeddingRequest) MarshalBatchLineItem() []byte { | ||
marshal, _ := json.Marshal(r) | ||
return marshal | ||
} | ||
|
||
type Batch struct { | ||
ID string `json:"id"` | ||
Object string `json:"object"` | ||
Endpoint BatchEndpoint `json:"endpoint"` | ||
Errors *struct { | ||
Object string `json:"object,omitempty"` | ||
Data struct { | ||
Code string `json:"code,omitempty"` | ||
Message string `json:"message,omitempty"` | ||
Param *string `json:"param,omitempty"` | ||
Line *int `json:"line,omitempty"` | ||
} `json:"data"` | ||
} `json:"errors"` | ||
InputFileID string `json:"input_file_id"` | ||
CompletionWindow string `json:"completion_window"` | ||
Status string `json:"status"` | ||
OutputFileID *string `json:"output_file_id"` | ||
ErrorFileID *string `json:"error_file_id"` | ||
CreatedAt int `json:"created_at"` | ||
InProgressAt *int `json:"in_progress_at"` | ||
ExpiresAt *int `json:"expires_at"` | ||
FinalizingAt *int `json:"finalizing_at"` | ||
CompletedAt *int `json:"completed_at"` | ||
FailedAt *int `json:"failed_at"` | ||
ExpiredAt *int `json:"expired_at"` | ||
CancellingAt *int `json:"cancelling_at"` | ||
CancelledAt *int `json:"cancelled_at"` | ||
RequestCounts BatchRequestCounts `json:"request_counts"` | ||
Metadata map[string]any `json:"metadata"` | ||
} | ||
|
||
type BatchRequestCounts struct { | ||
Total int `json:"total"` | ||
Completed int `json:"completed"` | ||
Failed int `json:"failed"` | ||
} | ||
|
||
type CreateBatchRequest struct { | ||
InputFileID string `json:"input_file_id"` | ||
Endpoint BatchEndpoint `json:"endpoint"` | ||
CompletionWindow string `json:"completion_window"` | ||
Metadata map[string]any `json:"metadata"` | ||
} | ||
|
||
type BatchResponse struct { | ||
httpHeader | ||
Batch | ||
} | ||
|
||
var ErrUploadBatchFileFailed = errors.New("upload batch file failed") | ||
|
||
// CreateBatch — API call to Create batch. | ||
func (c *Client) CreateBatch( | ||
ctx context.Context, | ||
request CreateBatchRequest, | ||
) (response BatchResponse, err error) { | ||
if request.CompletionWindow == "" { | ||
request.CompletionWindow = "24h" | ||
} | ||
|
||
req, err := c.newRequest(ctx, http.MethodPost, c.fullURL(batchesSuffix), withBody(request)) | ||
if err != nil { | ||
return | ||
} | ||
|
||
err = c.sendRequest(req, &response) | ||
return | ||
} | ||
|
||
type UploadBatchFileRequest struct { | ||
FileName string | ||
Lines []BatchLineItem | ||
} | ||
|
||
func (r *UploadBatchFileRequest) MarshalJSONL() []byte { | ||
buff := bytes.Buffer{} | ||
for i, line := range r.Lines { | ||
if i != 0 { | ||
buff.Write([]byte("\n")) | ||
} | ||
buff.Write(line.MarshalBatchLineItem()) | ||
} | ||
return buff.Bytes() | ||
} | ||
|
||
func (r *UploadBatchFileRequest) AddChatCompletion(customerID string, body ChatCompletionRequest) { | ||
r.Lines = append(r.Lines, BatchChatCompletionRequest{ | ||
CustomID: customerID, | ||
Body: body, | ||
Method: "POST", | ||
URL: BatchEndpointChatCompletions, | ||
}) | ||
} | ||
|
||
func (r *UploadBatchFileRequest) AddCompletion(customerID string, body CompletionRequest) { | ||
r.Lines = append(r.Lines, BatchCompletionRequest{ | ||
CustomID: customerID, | ||
Body: body, | ||
Method: "POST", | ||
URL: BatchEndpointCompletions, | ||
}) | ||
} | ||
|
||
func (r *UploadBatchFileRequest) AddEmbedding(customerID string, body EmbeddingRequest) { | ||
r.Lines = append(r.Lines, BatchEmbeddingRequest{ | ||
CustomID: customerID, | ||
Body: body, | ||
Method: "POST", | ||
URL: BatchEndpointEmbeddings, | ||
}) | ||
} | ||
|
||
// UploadBatchFile — upload batch file. | ||
func (c *Client) UploadBatchFile(ctx context.Context, request UploadBatchFileRequest) (File, error) { | ||
if request.FileName == "" { | ||
request.FileName = "@batchinput.jsonl" | ||
} | ||
return c.CreateFileBytes(ctx, FileBytesRequest{ | ||
Name: request.FileName, | ||
Bytes: request.MarshalJSONL(), | ||
Purpose: PurposeBatch, | ||
}) | ||
} | ||
|
||
type CreateBatchWithUploadFileRequest struct { | ||
Endpoint BatchEndpoint `json:"endpoint"` | ||
CompletionWindow string `json:"completion_window"` | ||
Metadata map[string]any `json:"metadata"` | ||
UploadBatchFileRequest | ||
} | ||
|
||
// CreateBatchWithUploadFile — API call to Create batch with upload file. | ||
func (c *Client) CreateBatchWithUploadFile( | ||
ctx context.Context, | ||
request CreateBatchWithUploadFileRequest, | ||
) (response BatchResponse, err error) { | ||
var file File | ||
file, err = c.UploadBatchFile(ctx, UploadBatchFileRequest{ | ||
FileName: request.FileName, | ||
Lines: request.Lines, | ||
}) | ||
if err != nil { | ||
err = errors.Join(ErrUploadBatchFileFailed, err) | ||
return | ||
} | ||
return c.CreateBatch(ctx, CreateBatchRequest{ | ||
InputFileID: file.ID, | ||
Endpoint: request.Endpoint, | ||
CompletionWindow: request.CompletionWindow, | ||
Metadata: request.Metadata, | ||
}) | ||
} | ||
|
||
// RetrieveBatch — API call to Retrieve batch. | ||
func (c *Client) RetrieveBatch( | ||
ctx context.Context, | ||
batchID string, | ||
) (response BatchResponse, err error) { | ||
urlSuffix := fmt.Sprintf("%s/%s", batchesSuffix, batchID) | ||
req, err := c.newRequest(ctx, http.MethodGet, c.fullURL(urlSuffix)) | ||
if err != nil { | ||
return | ||
} | ||
err = c.sendRequest(req, &response) | ||
return | ||
} | ||
|
||
// CancelBatch — API call to Cancel batch. | ||
func (c *Client) CancelBatch( | ||
ctx context.Context, | ||
batchID string, | ||
) (response BatchResponse, err error) { | ||
urlSuffix := fmt.Sprintf("%s/%s/cancel", batchesSuffix, batchID) | ||
req, err := c.newRequest(ctx, http.MethodPost, c.fullURL(urlSuffix)) | ||
if err != nil { | ||
return | ||
} | ||
err = c.sendRequest(req, &response) | ||
return | ||
} | ||
|
||
type ListBatchResponse struct { | ||
httpHeader | ||
Object string `json:"object"` | ||
Data []Batch `json:"data"` | ||
FirstID string `json:"first_id"` | ||
LastID string `json:"last_id"` | ||
HasMore bool `json:"has_more"` | ||
} | ||
|
||
// ListBatch API call to List batch. | ||
func (c *Client) ListBatch(ctx context.Context, after *string, limit *int) (response ListBatchResponse, err error) { | ||
urlValues := url.Values{} | ||
if limit != nil { | ||
urlValues.Add("limit", fmt.Sprintf("%d", *limit)) | ||
} | ||
if after != nil { | ||
urlValues.Add("after", *after) | ||
} | ||
encodedValues := "" | ||
if len(urlValues) > 0 { | ||
encodedValues = "?" + urlValues.Encode() | ||
} | ||
|
||
urlSuffix := fmt.Sprintf("%s%s", batchesSuffix, encodedValues) | ||
req, err := c.newRequest(ctx, http.MethodGet, c.fullURL(urlSuffix)) | ||
if err != nil { | ||
return | ||
} | ||
|
||
err = c.sendRequest(req, &response) | ||
return | ||
} |
Oops, something went wrong.