zot/packages/agent/modelsync.go
patriceckhart fa7d8d8be5 refactor: split source into packages/{provider,core,tui,agent}
Single Go module, four top-level packages under packages/. Import
paths become github.com/patriceckhart/zot/packages/<name>; downstream
consumers can depend on individual packages without pulling the rest.

Layout:
  packages/provider/     LLM clients + catalog
  packages/provider/auth/ credential store + OAuth + login server
  packages/core/         agent loop, sessions, cost
  packages/tui/          terminal toolkit + chat view
  packages/agent/        CLI wiring, system prompt
    extensions/ extproto/ modes/ tools/ skills/ swarm/
    sdk/  (was pkg/zotcore, package renamed zotcore -> sdk)
    ext/  (was pkg/zotext, package renamed zotext -> ext)

internal/ and pkg/ removed. The internal/assets logo moved into
packages/provider/auth/assets.

Public Go SDK identifiers renamed:
  pkg/zotcore (package zotcore) -> packages/agent/sdk (package sdk)
  pkg/zotext  (package zotext)  -> packages/agent/ext (package ext)

This breaks Go-based extensions and embedders; the JSON wire protocol
for extensions and RPC is unchanged, so non-Go extensions, already-
built extension binaries, and zot rpc consumers are unaffected.

Docs, examples, and the built-in write-zot-extension skill updated
for the new paths and identifiers. Shadow-bug fixes in code samples
(ext := ext.New -> e := ext.New).
2026-05-27 09:07:15 +02:00

162 lines
5.2 KiB
Go

package agent
import (
"context"
"fmt"
"os"
"path/filepath"
"time"
"github.com/patriceckhart/zot/packages/provider"
)
// ModelCachePath returns the on-disk location of the merged model cache.
func ModelCachePath() string {
return filepath.Join(ZotHome(), "models-cache.json")
}
// UserModelsPath returns the path to the user's models.json override.
func UserModelsPath() string {
return filepath.Join(ZotHome(), "models.json")
}
// LoadCachedModels loads the cache file and applies it to the provider
// package so FindModel / ModelsForProvider see live ids immediately.
// Safe to call before any credentials are known.
func LoadCachedModels() {
c, err := provider.LoadCache(ModelCachePath())
if err != nil {
return
}
if len(c.Models) > 0 {
provider.SetLiveModels(c.Models)
}
}
// LoadUserModels reads $ZOT_HOME/models.json and merges any user-defined
// models into the active catalog. User models take highest precedence.
// Any validation issues (bad provider id, empty model id, malformed
// JSON, negative widths) are surfaced as one warning per line on stderr;
// the well-formed entries from the rest of the file are still loaded.
func LoadUserModels() {
models, warnings := provider.LoadUserModelsWithWarnings(UserModelsPath())
for _, w := range warnings {
fmt.Fprintln(os.Stderr, "zot:", w)
}
provider.SetUserModels(models)
}
// ValidateAndRepairConfig checks the persisted config.json's
// (Provider, Model) pair against the active catalog and repairs any
// mismatch in-place (and on disk) before any UI renders. Three failure
// modes are handled:
//
// - cfg.Provider is empty or unknown -> reset to "anthropic".
// - cfg.Model is empty -> set to the provider's default.
// - cfg.Model belongs to a different provider than cfg.Provider
// (e.g. provider=anthropic + model=kimi-for-coding from a stale
// half-applied switch) -> reset model to the provider's default.
//
// Silent on success; one stderr line per repair. Errors loading or
// saving the file are non-fatal — the caller continues with defaults.
func ValidateAndRepairConfig() {
cfg, err := LoadConfig()
if err != nil {
fmt.Fprintf(os.Stderr, "zot: config.json: %v (using defaults)\n", err)
return
}
changed := false
if cfg.Provider != "" && !isKnownProvider(cfg.Provider) {
fmt.Fprintf(os.Stderr, "zot: config.json: unknown provider %q reset to \"anthropic\"\n", cfg.Provider)
cfg.Provider = "anthropic"
cfg.Model = ""
changed = true
}
if cfg.Provider != "" && cfg.Model != "" {
if _, err := provider.FindModel(cfg.Provider, cfg.Model); err != nil {
if m, err := provider.FindModel("", cfg.Model); err == nil {
fix := defaultModelForProvider(cfg.Provider)
fmt.Fprintf(os.Stderr,
"zot: config.json: model %q belongs to provider %q (config has provider=%q); switched model to %q\n",
cfg.Model, m.Provider, cfg.Provider, fix)
cfg.Model = fix
changed = true
} else if cfg.Provider != "ollama" {
// Model id not in any catalog. Reset to provider's default.
fix := defaultModelForProvider(cfg.Provider)
fmt.Fprintf(os.Stderr,
"zot: config.json: model %q not found in the active catalog; switched to %q\n",
cfg.Model, fix)
cfg.Model = fix
changed = true
}
}
}
if changed {
if err := SaveConfig(cfg); err != nil {
fmt.Fprintf(os.Stderr, "zot: config.json: failed to persist repair: %v\n", err)
}
}
}
// RefreshModelsAsync kicks a background discovery for every provider
// we have credentials for. Refreshed results are merged into the
// active catalog and persisted to the on-disk cache.
//
// Silent on error: discovery is a nice-to-have. Callers can still use
// the baked-in catalog if this fails.
func RefreshModelsAsync() {
go refreshModels()
}
func refreshModels() {
cached, _ := provider.LoadCache(ModelCachePath())
if cached.IsFresh() {
return
}
ctx, cancel := context.WithTimeout(context.Background(), 20*time.Second)
defer cancel()
var all []provider.Model
if cred, method, err := ResolveCredential("anthropic", ""); err == nil && method == "apikey" {
// /v1/models on Anthropic is API-key only; OAuth tokens can
// also list models via the bearer header, but we skip OAuth
// here to avoid surprise rate-limit hits on subscription keys.
if live, err := provider.DiscoverAnthropic(ctx, cred, ""); err == nil {
all = append(all, live...)
}
}
if cred, method, err := ResolveCredential("openai", ""); err == nil && method == "apikey" {
if live, err := provider.DiscoverOpenAI(ctx, cred, ""); err == nil {
all = append(all, live...)
}
}
if cred, method, err := ResolveCredential("kimi", ""); err == nil && method == "apikey" {
if live, err := provider.DiscoverOpenAI(ctx, cred, "https://api.kimi.com/coding/v1"); err == nil {
for i := range live {
live[i].Provider = "kimi"
live[i].Source = "live"
}
all = append(all, live...)
}
}
if cred, method, err := ResolveCredential("google", ""); err == nil && method == "apikey" {
if live, err := provider.DiscoverGoogle(ctx, cred, ""); err == nil {
all = append(all, live...)
}
}
if len(all) == 0 {
return
}
provider.SetLiveModels(all)
_ = provider.SaveCache(ModelCachePath(), provider.ModelCache{
FetchedAt: time.Now().UTC(),
Models: all,
})
}