mirror of
https://github.com/router-for-me/CLIProxyAPI.git
synced 2026-02-03 04:50:52 +08:00
Break out config diffing, hashing, and OpenAI compatibility utilities into a dedicated diff package, update watcher to consume them, and add comprehensive tests for diff logic and watcher behavior.
152 lines
4.1 KiB
Go
152 lines
4.1 KiB
Go
package diff
|
|
|
|
import (
|
|
"crypto/sha256"
|
|
"encoding/hex"
|
|
"fmt"
|
|
"sort"
|
|
"strings"
|
|
|
|
"github.com/router-for-me/CLIProxyAPI/v6/internal/config"
|
|
)
|
|
|
|
type excludedModelsSummary struct {
|
|
hash string
|
|
count int
|
|
}
|
|
|
|
// SummarizeExcludedModels normalizes and hashes an excluded-model list.
|
|
func SummarizeExcludedModels(list []string) excludedModelsSummary {
|
|
if len(list) == 0 {
|
|
return excludedModelsSummary{}
|
|
}
|
|
seen := make(map[string]struct{}, len(list))
|
|
normalized := make([]string, 0, len(list))
|
|
for _, entry := range list {
|
|
if trimmed := strings.ToLower(strings.TrimSpace(entry)); trimmed != "" {
|
|
if _, exists := seen[trimmed]; exists {
|
|
continue
|
|
}
|
|
seen[trimmed] = struct{}{}
|
|
normalized = append(normalized, trimmed)
|
|
}
|
|
}
|
|
sort.Strings(normalized)
|
|
return excludedModelsSummary{
|
|
hash: ComputeExcludedModelsHash(normalized),
|
|
count: len(normalized),
|
|
}
|
|
}
|
|
|
|
// SummarizeOAuthExcludedModels summarizes OAuth excluded models per provider.
|
|
func SummarizeOAuthExcludedModels(entries map[string][]string) map[string]excludedModelsSummary {
|
|
if len(entries) == 0 {
|
|
return nil
|
|
}
|
|
out := make(map[string]excludedModelsSummary, len(entries))
|
|
for k, v := range entries {
|
|
key := strings.ToLower(strings.TrimSpace(k))
|
|
if key == "" {
|
|
continue
|
|
}
|
|
out[key] = SummarizeExcludedModels(v)
|
|
}
|
|
return out
|
|
}
|
|
|
|
// DiffOAuthExcludedModelChanges compares OAuth excluded models maps.
|
|
func DiffOAuthExcludedModelChanges(oldMap, newMap map[string][]string) ([]string, []string) {
|
|
oldSummary := SummarizeOAuthExcludedModels(oldMap)
|
|
newSummary := SummarizeOAuthExcludedModels(newMap)
|
|
keys := make(map[string]struct{}, len(oldSummary)+len(newSummary))
|
|
for k := range oldSummary {
|
|
keys[k] = struct{}{}
|
|
}
|
|
for k := range newSummary {
|
|
keys[k] = struct{}{}
|
|
}
|
|
changes := make([]string, 0, len(keys))
|
|
affected := make([]string, 0, len(keys))
|
|
for key := range keys {
|
|
oldInfo, okOld := oldSummary[key]
|
|
newInfo, okNew := newSummary[key]
|
|
switch {
|
|
case okOld && !okNew:
|
|
changes = append(changes, fmt.Sprintf("oauth-excluded-models[%s]: removed", key))
|
|
affected = append(affected, key)
|
|
case !okOld && okNew:
|
|
changes = append(changes, fmt.Sprintf("oauth-excluded-models[%s]: added (%d entries)", key, newInfo.count))
|
|
affected = append(affected, key)
|
|
case okOld && okNew && oldInfo.hash != newInfo.hash:
|
|
changes = append(changes, fmt.Sprintf("oauth-excluded-models[%s]: updated (%d -> %d entries)", key, oldInfo.count, newInfo.count))
|
|
affected = append(affected, key)
|
|
}
|
|
}
|
|
sort.Strings(changes)
|
|
sort.Strings(affected)
|
|
return changes, affected
|
|
}
|
|
|
|
type ampModelMappingsSummary struct {
|
|
hash string
|
|
count int
|
|
}
|
|
|
|
// SummarizeAmpModelMappings hashes Amp model mappings for change detection.
|
|
func SummarizeAmpModelMappings(mappings []config.AmpModelMapping) ampModelMappingsSummary {
|
|
if len(mappings) == 0 {
|
|
return ampModelMappingsSummary{}
|
|
}
|
|
entries := make([]string, 0, len(mappings))
|
|
for _, mapping := range mappings {
|
|
from := strings.TrimSpace(mapping.From)
|
|
to := strings.TrimSpace(mapping.To)
|
|
if from == "" && to == "" {
|
|
continue
|
|
}
|
|
entries = append(entries, from+"->"+to)
|
|
}
|
|
if len(entries) == 0 {
|
|
return ampModelMappingsSummary{}
|
|
}
|
|
sort.Strings(entries)
|
|
sum := sha256.Sum256([]byte(strings.Join(entries, "|")))
|
|
return ampModelMappingsSummary{
|
|
hash: hex.EncodeToString(sum[:]),
|
|
count: len(entries),
|
|
}
|
|
}
|
|
|
|
type vertexModelsSummary struct {
|
|
hash string
|
|
count int
|
|
}
|
|
|
|
// SummarizeVertexModels hashes vertex-compatible models for change detection.
|
|
func SummarizeVertexModels(models []config.VertexCompatModel) vertexModelsSummary {
|
|
if len(models) == 0 {
|
|
return vertexModelsSummary{}
|
|
}
|
|
names := make([]string, 0, len(models))
|
|
for _, m := range models {
|
|
name := strings.TrimSpace(m.Name)
|
|
alias := strings.TrimSpace(m.Alias)
|
|
if name == "" && alias == "" {
|
|
continue
|
|
}
|
|
if alias != "" {
|
|
name = alias
|
|
}
|
|
names = append(names, name)
|
|
}
|
|
if len(names) == 0 {
|
|
return vertexModelsSummary{}
|
|
}
|
|
sort.Strings(names)
|
|
sum := sha256.Sum256([]byte(strings.Join(names, "|")))
|
|
return vertexModelsSummary{
|
|
hash: hex.EncodeToString(sum[:]),
|
|
count: len(names),
|
|
}
|
|
}
|