mirror of
https://github.com/photoprism/photoprism.git
synced 2025-12-12 08:44:04 +01:00
164 lines
4.8 KiB
Go
164 lines
4.8 KiB
Go
package vision
|
|
|
|
import (
|
|
"encoding/json"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/photoprism/photoprism/internal/ai/vision/ollama"
|
|
"github.com/photoprism/photoprism/pkg/http/header"
|
|
"github.com/photoprism/photoprism/pkg/http/scheme"
|
|
)
|
|
|
|
func TestNewApiRequest(t *testing.T) {
|
|
t.Run("Data", func(t *testing.T) {
|
|
thumbnails := Files{examplesPath + "/chameleon_lime.jpg"}
|
|
result, err := NewApiRequestImages(thumbnails, scheme.Data)
|
|
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, result)
|
|
// t.Logf("request: %#v", result)
|
|
|
|
if result != nil {
|
|
json, jsonErr := result.JSON()
|
|
assert.NoError(t, jsonErr)
|
|
assert.NotEmpty(t, json)
|
|
// t.Logf("json: %s", json)
|
|
}
|
|
})
|
|
t.Run("Https", func(t *testing.T) {
|
|
thumbnails := Files{examplesPath + "/chameleon_lime.jpg"}
|
|
result, err := NewApiRequestImages(thumbnails, scheme.Https)
|
|
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, result)
|
|
// t.Logf("request: %#v", result)
|
|
if result != nil {
|
|
json, jsonErr := result.JSON()
|
|
assert.NoError(t, jsonErr)
|
|
assert.NotEmpty(t, json)
|
|
t.Logf("json: %s", json)
|
|
}
|
|
})
|
|
}
|
|
|
|
func TestPerformApiRequestOllama(t *testing.T) {
|
|
t.Run("Labels", func(t *testing.T) {
|
|
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
var req ApiRequest
|
|
assert.NoError(t, json.NewDecoder(r.Body).Decode(&req))
|
|
assert.Equal(t, FormatJSON, req.Format)
|
|
assert.NoError(t, json.NewEncoder(w).Encode(ollama.Response{
|
|
Model: "qwen2.5vl:latest",
|
|
Response: `{"labels":[{"name":"test","confidence":0.9,"topicality":0.8}]}`,
|
|
}))
|
|
}))
|
|
defer server.Close()
|
|
|
|
apiRequest := &ApiRequest{
|
|
Id: "test",
|
|
Model: "qwen2.5vl:latest",
|
|
Format: FormatJSON,
|
|
Images: []string{"data:image/jpeg;base64,AA=="},
|
|
ResponseFormat: ApiFormatOllama,
|
|
}
|
|
|
|
resp, err := PerformApiRequest(apiRequest, server.URL, http.MethodPost, "")
|
|
assert.NoError(t, err)
|
|
assert.Len(t, resp.Result.Labels, 1)
|
|
assert.Equal(t, "Test", resp.Result.Labels[0].Name)
|
|
assert.Nil(t, resp.Result.Caption)
|
|
})
|
|
t.Run("LabelsWithCodeFence", func(t *testing.T) {
|
|
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
assert.NoError(t, json.NewEncoder(w).Encode(ollama.Response{
|
|
Model: "gemma3:latest",
|
|
Response: "```json\n{\"labels\":[{\"name\":\"lingerie\",\"confidence\":0.81,\"topicality\":0.73}]}\n```\nThe model provided additional commentary.",
|
|
}))
|
|
}))
|
|
defer server.Close()
|
|
|
|
apiRequest := &ApiRequest{
|
|
Id: "fenced",
|
|
Model: "gemma3:latest",
|
|
Format: FormatJSON,
|
|
Images: []string{"data:image/jpeg;base64,AA=="},
|
|
ResponseFormat: ApiFormatOllama,
|
|
}
|
|
|
|
resp, err := PerformApiRequest(apiRequest, server.URL, http.MethodPost, "")
|
|
assert.NoError(t, err)
|
|
if assert.Len(t, resp.Result.Labels, 1) {
|
|
assert.Equal(t, "Lingerie", resp.Result.Labels[0].Name)
|
|
}
|
|
})
|
|
t.Run("CaptionFallback", func(t *testing.T) {
|
|
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
assert.NoError(t, json.NewEncoder(w).Encode(ollama.Response{
|
|
Model: "qwen2.5vl:latest",
|
|
Response: "plain text",
|
|
}))
|
|
}))
|
|
defer server.Close()
|
|
|
|
apiRequest := &ApiRequest{
|
|
Id: "test2",
|
|
Model: "qwen2.5vl:latest",
|
|
Format: FormatJSON,
|
|
Images: []string{"data:image/jpeg;base64,AA=="},
|
|
ResponseFormat: ApiFormatOllama,
|
|
}
|
|
|
|
resp, err := PerformApiRequest(apiRequest, server.URL, http.MethodPost, "")
|
|
assert.NoError(t, err)
|
|
assert.Len(t, resp.Result.Labels, 0)
|
|
if assert.NotNil(t, resp.Result.Caption) {
|
|
assert.Equal(t, "plain text", resp.Result.Caption.Text)
|
|
}
|
|
})
|
|
}
|
|
|
|
func TestPerformApiRequestOpenAIHeaders(t *testing.T) {
|
|
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
assert.Equal(t, "org-123", r.Header.Get(header.OpenAIOrg))
|
|
assert.Equal(t, "proj-abc", r.Header.Get(header.OpenAIProject))
|
|
|
|
response := map[string]any{
|
|
"id": "resp_123",
|
|
"model": "gpt-5-mini",
|
|
"output": []any{
|
|
map[string]any{
|
|
"role": "assistant",
|
|
"content": []any{
|
|
map[string]any{
|
|
"type": "output_text",
|
|
"text": "A scenic mountain view.",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
assert.NoError(t, json.NewEncoder(w).Encode(response))
|
|
}))
|
|
defer server.Close()
|
|
|
|
req := &ApiRequest{
|
|
Id: "headers",
|
|
Model: "gpt-5-mini",
|
|
Images: []string{"data:image/jpeg;base64,AA=="},
|
|
ResponseFormat: ApiFormatOpenAI,
|
|
Org: "org-123",
|
|
Project: "proj-abc",
|
|
}
|
|
|
|
resp, err := PerformApiRequest(req, server.URL, http.MethodPost, "")
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, resp)
|
|
assert.NotNil(t, resp.Result.Caption)
|
|
assert.Equal(t, "A scenic mountain view.", resp.Result.Caption.Text)
|
|
}
|