mirror of
https://github.com/router-for-me/CLIProxyAPI.git
synced 2026-02-03 13:00:52 +08:00
195 lines
6.4 KiB
Go
195 lines
6.4 KiB
Go
package diff
|
|
|
|
import (
|
|
"testing"
|
|
|
|
"github.com/router-for-me/CLIProxyAPI/v6/internal/config"
|
|
)
|
|
|
|
func TestComputeOpenAICompatModelsHash_Deterministic(t *testing.T) {
|
|
models := []config.OpenAICompatibilityModel{
|
|
{Name: "gpt-4", Alias: "gpt4"},
|
|
{Name: "gpt-3.5-turbo"},
|
|
}
|
|
hash1 := ComputeOpenAICompatModelsHash(models)
|
|
hash2 := ComputeOpenAICompatModelsHash(models)
|
|
if hash1 == "" {
|
|
t.Fatal("hash should not be empty")
|
|
}
|
|
if hash1 != hash2 {
|
|
t.Fatalf("hash should be deterministic, got %s vs %s", hash1, hash2)
|
|
}
|
|
changed := ComputeOpenAICompatModelsHash([]config.OpenAICompatibilityModel{{Name: "gpt-4"}, {Name: "gpt-4.1"}})
|
|
if hash1 == changed {
|
|
t.Fatal("hash should change when model list changes")
|
|
}
|
|
}
|
|
|
|
func TestComputeOpenAICompatModelsHash_NormalizesAndDedups(t *testing.T) {
|
|
a := []config.OpenAICompatibilityModel{
|
|
{Name: "gpt-4", Alias: "gpt4"},
|
|
{Name: " "},
|
|
{Name: "GPT-4", Alias: "GPT4"},
|
|
{Alias: "a1"},
|
|
}
|
|
b := []config.OpenAICompatibilityModel{
|
|
{Alias: "A1"},
|
|
{Name: "gpt-4", Alias: "gpt4"},
|
|
}
|
|
h1 := ComputeOpenAICompatModelsHash(a)
|
|
h2 := ComputeOpenAICompatModelsHash(b)
|
|
if h1 == "" || h2 == "" {
|
|
t.Fatal("expected non-empty hashes for non-empty model sets")
|
|
}
|
|
if h1 != h2 {
|
|
t.Fatalf("expected normalized hashes to match, got %s / %s", h1, h2)
|
|
}
|
|
}
|
|
|
|
func TestComputeVertexCompatModelsHash_DifferentInputs(t *testing.T) {
|
|
models := []config.VertexCompatModel{{Name: "gemini-pro", Alias: "pro"}}
|
|
hash1 := ComputeVertexCompatModelsHash(models)
|
|
hash2 := ComputeVertexCompatModelsHash([]config.VertexCompatModel{{Name: "gemini-1.5-pro", Alias: "pro"}})
|
|
if hash1 == "" || hash2 == "" {
|
|
t.Fatal("hashes should not be empty for non-empty models")
|
|
}
|
|
if hash1 == hash2 {
|
|
t.Fatal("hash should differ when model content differs")
|
|
}
|
|
}
|
|
|
|
func TestComputeVertexCompatModelsHash_IgnoresBlankAndOrder(t *testing.T) {
|
|
a := []config.VertexCompatModel{
|
|
{Name: "m1", Alias: "a1"},
|
|
{Name: " "},
|
|
{Name: "M1", Alias: "A1"},
|
|
}
|
|
b := []config.VertexCompatModel{
|
|
{Name: "m1", Alias: "a1"},
|
|
}
|
|
if h1, h2 := ComputeVertexCompatModelsHash(a), ComputeVertexCompatModelsHash(b); h1 == "" || h1 != h2 {
|
|
t.Fatalf("expected same hash ignoring blanks/dupes, got %q / %q", h1, h2)
|
|
}
|
|
}
|
|
|
|
func TestComputeClaudeModelsHash_Empty(t *testing.T) {
|
|
if got := ComputeClaudeModelsHash(nil); got != "" {
|
|
t.Fatalf("expected empty hash for nil models, got %q", got)
|
|
}
|
|
if got := ComputeClaudeModelsHash([]config.ClaudeModel{}); got != "" {
|
|
t.Fatalf("expected empty hash for empty slice, got %q", got)
|
|
}
|
|
}
|
|
|
|
func TestComputeCodexModelsHash_Empty(t *testing.T) {
|
|
if got := ComputeCodexModelsHash(nil); got != "" {
|
|
t.Fatalf("expected empty hash for nil models, got %q", got)
|
|
}
|
|
if got := ComputeCodexModelsHash([]config.CodexModel{}); got != "" {
|
|
t.Fatalf("expected empty hash for empty slice, got %q", got)
|
|
}
|
|
}
|
|
|
|
func TestComputeClaudeModelsHash_IgnoresBlankAndDedup(t *testing.T) {
|
|
a := []config.ClaudeModel{
|
|
{Name: "m1", Alias: "a1"},
|
|
{Name: " "},
|
|
{Name: "M1", Alias: "A1"},
|
|
}
|
|
b := []config.ClaudeModel{
|
|
{Name: "m1", Alias: "a1"},
|
|
}
|
|
if h1, h2 := ComputeClaudeModelsHash(a), ComputeClaudeModelsHash(b); h1 == "" || h1 != h2 {
|
|
t.Fatalf("expected same hash ignoring blanks/dupes, got %q / %q", h1, h2)
|
|
}
|
|
}
|
|
|
|
func TestComputeCodexModelsHash_IgnoresBlankAndDedup(t *testing.T) {
|
|
a := []config.CodexModel{
|
|
{Name: "m1", Alias: "a1"},
|
|
{Name: " "},
|
|
{Name: "M1", Alias: "A1"},
|
|
}
|
|
b := []config.CodexModel{
|
|
{Name: "m1", Alias: "a1"},
|
|
}
|
|
if h1, h2 := ComputeCodexModelsHash(a), ComputeCodexModelsHash(b); h1 == "" || h1 != h2 {
|
|
t.Fatalf("expected same hash ignoring blanks/dupes, got %q / %q", h1, h2)
|
|
}
|
|
}
|
|
|
|
func TestComputeExcludedModelsHash_Normalizes(t *testing.T) {
|
|
hash1 := ComputeExcludedModelsHash([]string{" A ", "b", "a"})
|
|
hash2 := ComputeExcludedModelsHash([]string{"a", " b", "A"})
|
|
if hash1 == "" || hash2 == "" {
|
|
t.Fatal("hash should not be empty for non-empty input")
|
|
}
|
|
if hash1 != hash2 {
|
|
t.Fatalf("hash should be order/space insensitive for same multiset, got %s vs %s", hash1, hash2)
|
|
}
|
|
hash3 := ComputeExcludedModelsHash([]string{"c"})
|
|
if hash1 == hash3 {
|
|
t.Fatal("hash should differ for different normalized sets")
|
|
}
|
|
}
|
|
|
|
func TestComputeOpenAICompatModelsHash_Empty(t *testing.T) {
|
|
if got := ComputeOpenAICompatModelsHash(nil); got != "" {
|
|
t.Fatalf("expected empty hash for nil input, got %q", got)
|
|
}
|
|
if got := ComputeOpenAICompatModelsHash([]config.OpenAICompatibilityModel{}); got != "" {
|
|
t.Fatalf("expected empty hash for empty slice, got %q", got)
|
|
}
|
|
if got := ComputeOpenAICompatModelsHash([]config.OpenAICompatibilityModel{{Name: " "}, {Alias: ""}}); got != "" {
|
|
t.Fatalf("expected empty hash for blank models, got %q", got)
|
|
}
|
|
}
|
|
|
|
func TestComputeVertexCompatModelsHash_Empty(t *testing.T) {
|
|
if got := ComputeVertexCompatModelsHash(nil); got != "" {
|
|
t.Fatalf("expected empty hash for nil input, got %q", got)
|
|
}
|
|
if got := ComputeVertexCompatModelsHash([]config.VertexCompatModel{}); got != "" {
|
|
t.Fatalf("expected empty hash for empty slice, got %q", got)
|
|
}
|
|
if got := ComputeVertexCompatModelsHash([]config.VertexCompatModel{{Name: " "}}); got != "" {
|
|
t.Fatalf("expected empty hash for blank models, got %q", got)
|
|
}
|
|
}
|
|
|
|
func TestComputeExcludedModelsHash_Empty(t *testing.T) {
|
|
if got := ComputeExcludedModelsHash(nil); got != "" {
|
|
t.Fatalf("expected empty hash for nil input, got %q", got)
|
|
}
|
|
if got := ComputeExcludedModelsHash([]string{}); got != "" {
|
|
t.Fatalf("expected empty hash for empty slice, got %q", got)
|
|
}
|
|
if got := ComputeExcludedModelsHash([]string{" ", ""}); got != "" {
|
|
t.Fatalf("expected empty hash for whitespace-only entries, got %q", got)
|
|
}
|
|
}
|
|
|
|
func TestComputeClaudeModelsHash_Deterministic(t *testing.T) {
|
|
models := []config.ClaudeModel{{Name: "a", Alias: "A"}, {Name: "b"}}
|
|
h1 := ComputeClaudeModelsHash(models)
|
|
h2 := ComputeClaudeModelsHash(models)
|
|
if h1 == "" || h1 != h2 {
|
|
t.Fatalf("expected deterministic hash, got %s / %s", h1, h2)
|
|
}
|
|
if h3 := ComputeClaudeModelsHash([]config.ClaudeModel{{Name: "a"}}); h3 == h1 {
|
|
t.Fatalf("expected different hash when models change, got %s", h3)
|
|
}
|
|
}
|
|
|
|
func TestComputeCodexModelsHash_Deterministic(t *testing.T) {
|
|
models := []config.CodexModel{{Name: "a", Alias: "A"}, {Name: "b"}}
|
|
h1 := ComputeCodexModelsHash(models)
|
|
h2 := ComputeCodexModelsHash(models)
|
|
if h1 == "" || h1 != h2 {
|
|
t.Fatalf("expected deterministic hash, got %s / %s", h1, h2)
|
|
}
|
|
if h3 := ComputeCodexModelsHash([]config.CodexModel{{Name: "a"}}); h3 == h1 {
|
|
t.Fatalf("expected different hash when models change, got %s", h3)
|
|
}
|
|
}
|