2024-07-02 18:50:56 +00:00
|
|
|
package openai
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2024-07-14 05:07:45 +00:00
|
|
|
"encoding/base64"
|
2024-07-02 18:50:56 +00:00
|
|
|
"encoding/json"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
2024-07-02 23:01:45 +00:00
|
|
|
"strings"
|
2024-07-02 18:50:56 +00:00
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/gin-gonic/gin"
|
|
|
|
"github.com/ollama/ollama/api"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
)
|
|
|
|
|
2024-07-14 05:07:45 +00:00
|
|
|
const prefix = `data:image/jpeg;base64,`
|
|
|
|
const image = `iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAQAAAC1HAwCAAAAC0lEQVR42mNk+A8AAQUBAScY42YAAAAASUVORK5CYII=`
|
|
|
|
const imageURL = prefix + image
|
|
|
|
|
2024-07-09 20:48:31 +00:00
|
|
|
func TestMiddlewareRequests(t *testing.T) {
|
2024-07-02 18:50:56 +00:00
|
|
|
type testCase struct {
|
|
|
|
Name string
|
|
|
|
Method string
|
|
|
|
Path string
|
|
|
|
Handler func() gin.HandlerFunc
|
|
|
|
Setup func(t *testing.T, req *http.Request)
|
2024-07-09 20:48:31 +00:00
|
|
|
Expected func(t *testing.T, req *http.Request)
|
2024-07-02 18:50:56 +00:00
|
|
|
}
|
|
|
|
|
2024-07-09 20:48:31 +00:00
|
|
|
var capturedRequest *http.Request
|
2024-07-02 18:50:56 +00:00
|
|
|
|
2024-07-09 20:48:31 +00:00
|
|
|
captureRequestMiddleware := func() gin.HandlerFunc {
|
|
|
|
return func(c *gin.Context) {
|
|
|
|
bodyBytes, _ := io.ReadAll(c.Request.Body)
|
|
|
|
c.Request.Body = io.NopCloser(bytes.NewReader(bodyBytes))
|
|
|
|
capturedRequest = c.Request
|
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
}
|
2024-07-02 18:50:56 +00:00
|
|
|
|
2024-07-09 20:48:31 +00:00
|
|
|
testCases := []testCase{
|
|
|
|
{
|
|
|
|
Name: "chat handler",
|
|
|
|
Method: http.MethodPost,
|
|
|
|
Path: "/api/chat",
|
|
|
|
Handler: ChatMiddleware,
|
2024-07-02 18:50:56 +00:00
|
|
|
Setup: func(t *testing.T, req *http.Request) {
|
|
|
|
body := ChatCompletionRequest{
|
|
|
|
Model: "test-model",
|
|
|
|
Messages: []Message{{Role: "user", Content: "Hello"}},
|
|
|
|
}
|
|
|
|
|
|
|
|
bodyBytes, _ := json.Marshal(body)
|
|
|
|
|
|
|
|
req.Body = io.NopCloser(bytes.NewReader(bodyBytes))
|
|
|
|
req.Header.Set("Content-Type", "application/json")
|
|
|
|
},
|
2024-07-09 20:48:31 +00:00
|
|
|
Expected: func(t *testing.T, req *http.Request) {
|
|
|
|
var chatReq api.ChatRequest
|
|
|
|
if err := json.NewDecoder(req.Body).Decode(&chatReq); err != nil {
|
2024-07-02 18:50:56 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2024-07-09 20:48:31 +00:00
|
|
|
if chatReq.Messages[0].Role != "user" {
|
|
|
|
t.Fatalf("expected 'user', got %s", chatReq.Messages[0].Role)
|
2024-07-02 18:50:56 +00:00
|
|
|
}
|
|
|
|
|
2024-07-09 20:48:31 +00:00
|
|
|
if chatReq.Messages[0].Content != "Hello" {
|
|
|
|
t.Fatalf("expected 'Hello', got %s", chatReq.Messages[0].Content)
|
2024-07-02 18:50:56 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
2024-07-02 23:01:45 +00:00
|
|
|
{
|
2024-07-09 20:48:31 +00:00
|
|
|
Name: "completions handler",
|
|
|
|
Method: http.MethodPost,
|
|
|
|
Path: "/api/generate",
|
|
|
|
Handler: CompletionsMiddleware,
|
2024-07-02 23:01:45 +00:00
|
|
|
Setup: func(t *testing.T, req *http.Request) {
|
2024-07-09 20:48:31 +00:00
|
|
|
temp := float32(0.8)
|
2024-07-02 23:01:45 +00:00
|
|
|
body := CompletionRequest{
|
2024-07-09 20:48:31 +00:00
|
|
|
Model: "test-model",
|
|
|
|
Prompt: "Hello",
|
|
|
|
Temperature: &temp,
|
2024-07-09 21:01:26 +00:00
|
|
|
Stop: []string{"\n", "stop"},
|
2024-07-02 23:01:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bodyBytes, _ := json.Marshal(body)
|
|
|
|
|
|
|
|
req.Body = io.NopCloser(bytes.NewReader(bodyBytes))
|
|
|
|
req.Header.Set("Content-Type", "application/json")
|
|
|
|
},
|
2024-07-09 20:48:31 +00:00
|
|
|
Expected: func(t *testing.T, req *http.Request) {
|
|
|
|
var genReq api.GenerateRequest
|
|
|
|
if err := json.NewDecoder(req.Body).Decode(&genReq); err != nil {
|
2024-07-02 23:01:45 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2024-07-09 20:48:31 +00:00
|
|
|
if genReq.Prompt != "Hello" {
|
|
|
|
t.Fatalf("expected 'Hello', got %s", genReq.Prompt)
|
2024-07-02 23:01:45 +00:00
|
|
|
}
|
|
|
|
|
2024-07-09 20:48:31 +00:00
|
|
|
if genReq.Options["temperature"] != 1.6 {
|
|
|
|
t.Fatalf("expected 1.6, got %f", genReq.Options["temperature"])
|
2024-07-02 23:01:45 +00:00
|
|
|
}
|
2024-07-09 21:01:26 +00:00
|
|
|
|
|
|
|
stopTokens, ok := genReq.Options["stop"].([]any)
|
|
|
|
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("expected stop tokens to be a list")
|
|
|
|
}
|
|
|
|
|
|
|
|
if stopTokens[0] != "\n" || stopTokens[1] != "stop" {
|
|
|
|
t.Fatalf("expected ['\\n', 'stop'], got %v", stopTokens)
|
|
|
|
}
|
2024-07-02 23:01:45 +00:00
|
|
|
},
|
|
|
|
},
|
2024-07-14 05:07:45 +00:00
|
|
|
{
|
|
|
|
Name: "chat handler with image content",
|
|
|
|
Method: http.MethodPost,
|
|
|
|
Path: "/api/chat",
|
|
|
|
Handler: ChatMiddleware,
|
|
|
|
Setup: func(t *testing.T, req *http.Request) {
|
|
|
|
body := ChatCompletionRequest{
|
|
|
|
Model: "test-model",
|
|
|
|
Messages: []Message{
|
|
|
|
{
|
|
|
|
Role: "user", Content: []map[string]any{
|
|
|
|
{"type": "text", "text": "Hello"},
|
|
|
|
{"type": "image_url", "image_url": map[string]string{"url": imageURL}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
bodyBytes, _ := json.Marshal(body)
|
|
|
|
|
|
|
|
req.Body = io.NopCloser(bytes.NewReader(bodyBytes))
|
|
|
|
req.Header.Set("Content-Type", "application/json")
|
|
|
|
},
|
|
|
|
Expected: func(t *testing.T, req *http.Request) {
|
|
|
|
var chatReq api.ChatRequest
|
|
|
|
if err := json.NewDecoder(req.Body).Decode(&chatReq); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if chatReq.Messages[0].Role != "user" {
|
|
|
|
t.Fatalf("expected 'user', got %s", chatReq.Messages[0].Role)
|
|
|
|
}
|
|
|
|
|
|
|
|
if chatReq.Messages[0].Content != "Hello" {
|
|
|
|
t.Fatalf("expected 'Hello', got %s", chatReq.Messages[0].Content)
|
|
|
|
}
|
|
|
|
|
|
|
|
img, _ := base64.StdEncoding.DecodeString(imageURL[len(prefix):])
|
|
|
|
|
|
|
|
if !bytes.Equal(chatReq.Messages[0].Images[0], img) {
|
|
|
|
t.Fatalf("expected image encoding, got %s", chatReq.Messages[0].Images[0])
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
2024-07-16 20:36:08 +00:00
|
|
|
{
|
|
|
|
Name: "embed handler single input",
|
|
|
|
Method: http.MethodPost,
|
|
|
|
Path: "/api/embed",
|
|
|
|
Handler: EmbeddingsMiddleware,
|
|
|
|
Setup: func(t *testing.T, req *http.Request) {
|
|
|
|
body := EmbedRequest{
|
|
|
|
Input: "Hello",
|
|
|
|
Model: "test-model",
|
|
|
|
}
|
|
|
|
|
|
|
|
bodyBytes, _ := json.Marshal(body)
|
|
|
|
|
|
|
|
req.Body = io.NopCloser(bytes.NewReader(bodyBytes))
|
|
|
|
req.Header.Set("Content-Type", "application/json")
|
|
|
|
},
|
|
|
|
Expected: func(t *testing.T, req *http.Request) {
|
|
|
|
var embedReq api.EmbedRequest
|
|
|
|
if err := json.NewDecoder(req.Body).Decode(&embedReq); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if embedReq.Input != "Hello" {
|
|
|
|
t.Fatalf("expected 'Hello', got %s", embedReq.Input)
|
|
|
|
}
|
|
|
|
|
|
|
|
if embedReq.Model != "test-model" {
|
|
|
|
t.Fatalf("expected 'test-model', got %s", embedReq.Model)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Name: "embed handler batch input",
|
|
|
|
Method: http.MethodPost,
|
|
|
|
Path: "/api/embed",
|
|
|
|
Handler: EmbeddingsMiddleware,
|
|
|
|
Setup: func(t *testing.T, req *http.Request) {
|
|
|
|
body := EmbedRequest{
|
|
|
|
Input: []string{"Hello", "World"},
|
|
|
|
Model: "test-model",
|
|
|
|
}
|
|
|
|
|
|
|
|
bodyBytes, _ := json.Marshal(body)
|
|
|
|
|
|
|
|
req.Body = io.NopCloser(bytes.NewReader(bodyBytes))
|
|
|
|
req.Header.Set("Content-Type", "application/json")
|
|
|
|
},
|
|
|
|
Expected: func(t *testing.T, req *http.Request) {
|
|
|
|
var embedReq api.EmbedRequest
|
|
|
|
if err := json.NewDecoder(req.Body).Decode(&embedReq); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
input, ok := embedReq.Input.([]any)
|
|
|
|
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("expected input to be a list")
|
|
|
|
}
|
|
|
|
|
|
|
|
if input[0].(string) != "Hello" {
|
|
|
|
t.Fatalf("expected 'Hello', got %s", input[0])
|
|
|
|
}
|
|
|
|
|
|
|
|
if input[1].(string) != "World" {
|
|
|
|
t.Fatalf("expected 'World', got %s", input[1])
|
|
|
|
}
|
|
|
|
|
|
|
|
if embedReq.Model != "test-model" {
|
|
|
|
t.Fatalf("expected 'test-model', got %s", embedReq.Model)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
2024-07-09 20:48:31 +00:00
|
|
|
}
|
2024-07-02 23:01:45 +00:00
|
|
|
|
2024-07-09 20:48:31 +00:00
|
|
|
gin.SetMode(gin.TestMode)
|
|
|
|
router := gin.New()
|
2024-07-02 23:01:45 +00:00
|
|
|
|
2024-07-09 20:48:31 +00:00
|
|
|
endpoint := func(c *gin.Context) {
|
|
|
|
c.Status(http.StatusOK)
|
|
|
|
}
|
2024-07-02 23:01:45 +00:00
|
|
|
|
2024-07-09 20:48:31 +00:00
|
|
|
for _, tc := range testCases {
|
|
|
|
t.Run(tc.Name, func(t *testing.T) {
|
|
|
|
router = gin.New()
|
|
|
|
router.Use(captureRequestMiddleware())
|
|
|
|
router.Use(tc.Handler())
|
|
|
|
router.Handle(tc.Method, tc.Path, endpoint)
|
|
|
|
req, _ := http.NewRequest(tc.Method, tc.Path, nil)
|
2024-07-02 23:01:45 +00:00
|
|
|
|
2024-07-09 20:48:31 +00:00
|
|
|
if tc.Setup != nil {
|
|
|
|
tc.Setup(t, req)
|
|
|
|
}
|
2024-07-02 23:01:45 +00:00
|
|
|
|
2024-07-09 20:48:31 +00:00
|
|
|
resp := httptest.NewRecorder()
|
|
|
|
router.ServeHTTP(resp, req)
|
2024-07-02 23:01:45 +00:00
|
|
|
|
2024-07-09 20:48:31 +00:00
|
|
|
tc.Expected(t, capturedRequest)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2024-07-02 23:01:45 +00:00
|
|
|
|
2024-07-09 20:48:31 +00:00
|
|
|
func TestMiddlewareResponses(t *testing.T) {
|
|
|
|
type testCase struct {
|
|
|
|
Name string
|
|
|
|
Method string
|
|
|
|
Path string
|
|
|
|
TestPath string
|
|
|
|
Handler func() gin.HandlerFunc
|
|
|
|
Endpoint func(c *gin.Context)
|
|
|
|
Setup func(t *testing.T, req *http.Request)
|
|
|
|
Expected func(t *testing.T, resp *httptest.ResponseRecorder)
|
|
|
|
}
|
|
|
|
|
|
|
|
testCases := []testCase{
|
2024-07-02 23:01:45 +00:00
|
|
|
{
|
2024-07-09 20:48:31 +00:00
|
|
|
Name: "completions handler error forwarding",
|
2024-07-02 23:01:45 +00:00
|
|
|
Method: http.MethodPost,
|
|
|
|
Path: "/api/generate",
|
|
|
|
TestPath: "/api/generate",
|
|
|
|
Handler: CompletionsMiddleware,
|
|
|
|
Endpoint: func(c *gin.Context) {
|
|
|
|
c.JSON(http.StatusBadRequest, gin.H{"error": "invalid request"})
|
|
|
|
},
|
|
|
|
Setup: func(t *testing.T, req *http.Request) {
|
|
|
|
body := CompletionRequest{
|
|
|
|
Model: "test-model",
|
|
|
|
Prompt: "Hello",
|
|
|
|
}
|
|
|
|
|
|
|
|
bodyBytes, _ := json.Marshal(body)
|
|
|
|
|
|
|
|
req.Body = io.NopCloser(bytes.NewReader(bodyBytes))
|
|
|
|
req.Header.Set("Content-Type", "application/json")
|
|
|
|
},
|
|
|
|
Expected: func(t *testing.T, resp *httptest.ResponseRecorder) {
|
|
|
|
if resp.Code != http.StatusBadRequest {
|
|
|
|
t.Fatalf("expected 400, got %d", resp.Code)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !strings.Contains(resp.Body.String(), `"invalid request"`) {
|
|
|
|
t.Fatalf("error was not forwarded")
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
2024-07-02 18:50:56 +00:00
|
|
|
{
|
|
|
|
Name: "list handler",
|
|
|
|
Method: http.MethodGet,
|
|
|
|
Path: "/api/tags",
|
|
|
|
TestPath: "/api/tags",
|
|
|
|
Handler: ListMiddleware,
|
|
|
|
Endpoint: func(c *gin.Context) {
|
|
|
|
c.JSON(http.StatusOK, api.ListResponse{
|
|
|
|
Models: []api.ListModelResponse{
|
|
|
|
{
|
|
|
|
Name: "Test Model",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
},
|
|
|
|
Expected: func(t *testing.T, resp *httptest.ResponseRecorder) {
|
2024-07-02 23:01:45 +00:00
|
|
|
assert.Equal(t, http.StatusOK, resp.Code)
|
|
|
|
|
2024-07-02 18:50:56 +00:00
|
|
|
var listResp ListCompletion
|
|
|
|
if err := json.NewDecoder(resp.Body).Decode(&listResp); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if listResp.Object != "list" {
|
|
|
|
t.Fatalf("expected list, got %s", listResp.Object)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(listResp.Data) != 1 {
|
|
|
|
t.Fatalf("expected 1, got %d", len(listResp.Data))
|
|
|
|
}
|
|
|
|
|
|
|
|
if listResp.Data[0].Id != "Test Model" {
|
|
|
|
t.Fatalf("expected Test Model, got %s", listResp.Data[0].Id)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Name: "retrieve model",
|
|
|
|
Method: http.MethodGet,
|
|
|
|
Path: "/api/show/:model",
|
|
|
|
TestPath: "/api/show/test-model",
|
|
|
|
Handler: RetrieveMiddleware,
|
|
|
|
Endpoint: func(c *gin.Context) {
|
|
|
|
c.JSON(http.StatusOK, api.ShowResponse{
|
|
|
|
ModifiedAt: time.Date(2024, 6, 17, 13, 45, 0, 0, time.UTC),
|
|
|
|
})
|
|
|
|
},
|
|
|
|
Expected: func(t *testing.T, resp *httptest.ResponseRecorder) {
|
|
|
|
var retrieveResp Model
|
|
|
|
if err := json.NewDecoder(resp.Body).Decode(&retrieveResp); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if retrieveResp.Object != "model" {
|
|
|
|
t.Fatalf("Expected object to be model, got %s", retrieveResp.Object)
|
|
|
|
}
|
|
|
|
|
|
|
|
if retrieveResp.Id != "test-model" {
|
|
|
|
t.Fatalf("Expected id to be test-model, got %s", retrieveResp.Id)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
gin.SetMode(gin.TestMode)
|
|
|
|
router := gin.New()
|
|
|
|
|
|
|
|
for _, tc := range testCases {
|
|
|
|
t.Run(tc.Name, func(t *testing.T) {
|
|
|
|
router = gin.New()
|
|
|
|
router.Use(tc.Handler())
|
|
|
|
router.Handle(tc.Method, tc.Path, tc.Endpoint)
|
|
|
|
req, _ := http.NewRequest(tc.Method, tc.TestPath, nil)
|
|
|
|
|
|
|
|
if tc.Setup != nil {
|
|
|
|
tc.Setup(t, req)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp := httptest.NewRecorder()
|
|
|
|
router.ServeHTTP(resp, req)
|
|
|
|
|
|
|
|
tc.Expected(t, resp)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|