2024-09-11 11:01:30 -07:00
|
|
|
package cmd
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2024-10-01 15:45:43 -07:00
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
2024-09-11 11:01:30 -07:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2024-10-01 15:45:43 -07:00
|
|
|
"strings"
|
2024-09-11 11:01:30 -07:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/google/go-cmp/cmp"
|
2024-10-01 15:45:43 -07:00
|
|
|
"github.com/spf13/cobra"
|
2024-09-11 11:01:30 -07:00
|
|
|
|
|
|
|
"github.com/ollama/ollama/api"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestShowInfo(t *testing.T) {
|
|
|
|
t.Run("bare details", func(t *testing.T) {
|
|
|
|
var b bytes.Buffer
|
|
|
|
if err := showInfo(&api.ShowResponse{
|
|
|
|
Details: api.ModelDetails{
|
|
|
|
Family: "test",
|
|
|
|
ParameterSize: "7B",
|
|
|
|
QuantizationLevel: "FP16",
|
|
|
|
},
|
|
|
|
}, &b); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expect := ` Model
|
|
|
|
architecture test
|
|
|
|
parameters 7B
|
|
|
|
quantization FP16
|
|
|
|
|
|
|
|
`
|
|
|
|
|
|
|
|
if diff := cmp.Diff(expect, b.String()); diff != "" {
|
|
|
|
t.Errorf("unexpected output (-want +got):\n%s", diff)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("bare model info", func(t *testing.T) {
|
|
|
|
var b bytes.Buffer
|
|
|
|
if err := showInfo(&api.ShowResponse{
|
|
|
|
ModelInfo: map[string]any{
|
|
|
|
"general.architecture": "test",
|
|
|
|
"general.parameter_count": float64(7_000_000_000),
|
|
|
|
"test.context_length": float64(0),
|
|
|
|
"test.embedding_length": float64(0),
|
|
|
|
},
|
|
|
|
Details: api.ModelDetails{
|
|
|
|
Family: "test",
|
|
|
|
ParameterSize: "7B",
|
|
|
|
QuantizationLevel: "FP16",
|
|
|
|
},
|
|
|
|
}, &b); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expect := ` Model
|
|
|
|
architecture test
|
|
|
|
parameters 7B
|
|
|
|
context length 0
|
|
|
|
embedding length 0
|
|
|
|
quantization FP16
|
|
|
|
|
|
|
|
`
|
|
|
|
if diff := cmp.Diff(expect, b.String()); diff != "" {
|
|
|
|
t.Errorf("unexpected output (-want +got):\n%s", diff)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("parameters", func(t *testing.T) {
|
|
|
|
var b bytes.Buffer
|
|
|
|
if err := showInfo(&api.ShowResponse{
|
|
|
|
Details: api.ModelDetails{
|
|
|
|
Family: "test",
|
|
|
|
ParameterSize: "7B",
|
|
|
|
QuantizationLevel: "FP16",
|
|
|
|
},
|
|
|
|
Parameters: `
|
|
|
|
stop never
|
|
|
|
stop gonna
|
|
|
|
stop give
|
|
|
|
stop you
|
|
|
|
stop up
|
|
|
|
temperature 99`,
|
|
|
|
}, &b); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expect := ` Model
|
|
|
|
architecture test
|
|
|
|
parameters 7B
|
|
|
|
quantization FP16
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
stop never
|
|
|
|
stop gonna
|
|
|
|
stop give
|
|
|
|
stop you
|
|
|
|
stop up
|
|
|
|
temperature 99
|
|
|
|
|
|
|
|
`
|
|
|
|
if diff := cmp.Diff(expect, b.String()); diff != "" {
|
|
|
|
t.Errorf("unexpected output (-want +got):\n%s", diff)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("project info", func(t *testing.T) {
|
|
|
|
var b bytes.Buffer
|
|
|
|
if err := showInfo(&api.ShowResponse{
|
|
|
|
Details: api.ModelDetails{
|
|
|
|
Family: "test",
|
|
|
|
ParameterSize: "7B",
|
|
|
|
QuantizationLevel: "FP16",
|
|
|
|
},
|
|
|
|
ProjectorInfo: map[string]any{
|
|
|
|
"general.architecture": "clip",
|
|
|
|
"general.parameter_count": float64(133_700_000),
|
|
|
|
"clip.vision.embedding_length": float64(0),
|
|
|
|
"clip.vision.projection_dim": float64(0),
|
|
|
|
},
|
|
|
|
}, &b); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expect := ` Model
|
|
|
|
architecture test
|
|
|
|
parameters 7B
|
|
|
|
quantization FP16
|
|
|
|
|
|
|
|
Projector
|
|
|
|
architecture clip
|
|
|
|
parameters 133.70M
|
|
|
|
embedding length 0
|
|
|
|
dimensions 0
|
|
|
|
|
|
|
|
`
|
|
|
|
if diff := cmp.Diff(expect, b.String()); diff != "" {
|
|
|
|
t.Errorf("unexpected output (-want +got):\n%s", diff)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("system", func(t *testing.T) {
|
|
|
|
var b bytes.Buffer
|
|
|
|
if err := showInfo(&api.ShowResponse{
|
|
|
|
Details: api.ModelDetails{
|
|
|
|
Family: "test",
|
|
|
|
ParameterSize: "7B",
|
|
|
|
QuantizationLevel: "FP16",
|
|
|
|
},
|
|
|
|
System: `You are a pirate!
|
|
|
|
Ahoy, matey!
|
|
|
|
Weigh anchor!
|
|
|
|
`,
|
|
|
|
}, &b); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expect := ` Model
|
|
|
|
architecture test
|
|
|
|
parameters 7B
|
|
|
|
quantization FP16
|
|
|
|
|
|
|
|
System
|
|
|
|
You are a pirate!
|
|
|
|
Ahoy, matey!
|
|
|
|
|
|
|
|
`
|
|
|
|
if diff := cmp.Diff(expect, b.String()); diff != "" {
|
|
|
|
t.Errorf("unexpected output (-want +got):\n%s", diff)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("license", func(t *testing.T) {
|
|
|
|
var b bytes.Buffer
|
|
|
|
license, err := os.ReadFile(filepath.Join("..", "LICENSE"))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := showInfo(&api.ShowResponse{
|
|
|
|
Details: api.ModelDetails{
|
|
|
|
Family: "test",
|
|
|
|
ParameterSize: "7B",
|
|
|
|
QuantizationLevel: "FP16",
|
|
|
|
},
|
|
|
|
License: string(license),
|
|
|
|
}, &b); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expect := ` Model
|
|
|
|
architecture test
|
|
|
|
parameters 7B
|
|
|
|
quantization FP16
|
|
|
|
|
|
|
|
License
|
|
|
|
MIT License
|
|
|
|
Copyright (c) Ollama
|
|
|
|
|
|
|
|
`
|
|
|
|
if diff := cmp.Diff(expect, b.String()); diff != "" {
|
|
|
|
t.Errorf("unexpected output (-want +got):\n%s", diff)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2024-10-01 15:45:43 -07:00
|
|
|
|
|
|
|
func TestDeleteHandler(t *testing.T) {
|
|
|
|
stopped := false
|
|
|
|
mockServer := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
if r.URL.Path == "/api/delete" && r.Method == http.MethodDelete {
|
|
|
|
var req api.DeleteRequest
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
|
|
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if req.Name == "test-model" {
|
|
|
|
w.WriteHeader(http.StatusOK)
|
|
|
|
} else {
|
|
|
|
w.WriteHeader(http.StatusNotFound)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if r.URL.Path == "/api/generate" && r.Method == http.MethodPost {
|
|
|
|
var req api.GenerateRequest
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
|
|
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if req.Model == "test-model" {
|
|
|
|
w.WriteHeader(http.StatusOK)
|
|
|
|
if err := json.NewEncoder(w).Encode(api.GenerateResponse{
|
|
|
|
Done: true,
|
|
|
|
}); err != nil {
|
|
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
|
|
}
|
|
|
|
stopped = true
|
|
|
|
return
|
|
|
|
} else {
|
|
|
|
w.WriteHeader(http.StatusNotFound)
|
|
|
|
if err := json.NewEncoder(w).Encode(api.GenerateResponse{
|
|
|
|
Done: false,
|
|
|
|
}); err != nil {
|
|
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}))
|
|
|
|
|
|
|
|
t.Setenv("OLLAMA_HOST", mockServer.URL)
|
|
|
|
t.Cleanup(mockServer.Close)
|
|
|
|
|
|
|
|
cmd := &cobra.Command{}
|
|
|
|
cmd.SetContext(context.TODO())
|
|
|
|
if err := DeleteHandler(cmd, []string{"test-model"}); err != nil {
|
|
|
|
t.Fatalf("DeleteHandler failed: %v", err)
|
|
|
|
}
|
|
|
|
if !stopped {
|
|
|
|
t.Fatal("Model was not stopped before deletion")
|
|
|
|
}
|
|
|
|
|
|
|
|
err := DeleteHandler(cmd, []string{"test-model-not-found"})
|
|
|
|
if err == nil || !strings.Contains(err.Error(), "unable to stop existing running model \"test-model-not-found\"") {
|
|
|
|
t.Fatalf("DeleteHandler failed: expected error about stopping non-existent model, got %v", err)
|
|
|
|
}
|
|
|
|
}
|