feat: add tier detection package
Probes Anthropic and LiteLLM endpoints to detect the current operating tier (Full / LANOnly / Airplane) so downstream code can gate model selection and managed-agent availability without manual configuration. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
64
internal/tier/tier.go
Normal file
64
internal/tier/tier.go
Normal file
@@ -0,0 +1,64 @@
|
|||||||
|
// internal/tier/tier.go
|
||||||
|
package tier
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"net/http"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Tier represents the current operating capability level.
|
||||||
|
type Tier int
|
||||||
|
|
||||||
|
const (
|
||||||
|
Full Tier = 1 // internet + Anthropic API reachable
|
||||||
|
LANOnly Tier = 2 // LiteLLM on LAN reachable, no internet
|
||||||
|
Airplane Tier = 3 // no network
|
||||||
|
)
|
||||||
|
|
||||||
|
// Info describes the current operating tier.
|
||||||
|
type Info struct {
|
||||||
|
Tier Tier `json:"tier"`
|
||||||
|
Label string `json:"label"`
|
||||||
|
AvailableModels []string `json:"available_models"`
|
||||||
|
ManagedAgents bool `json:"managed_agents"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// Detect probes the Anthropic endpoint and LiteLLM and returns the current tier.
|
||||||
|
// Each probe has a 2-second timeout.
|
||||||
|
func Detect(ctx context.Context, anthropicProbe, liteLLMBaseURL string) Info {
|
||||||
|
client := &http.Client{Timeout: 2 * time.Second}
|
||||||
|
|
||||||
|
if probe(ctx, client, anthropicProbe) {
|
||||||
|
return Info{
|
||||||
|
Tier: Full,
|
||||||
|
Label: "full-online",
|
||||||
|
ManagedAgents: true,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if probe(ctx, client, liteLLMBaseURL) {
|
||||||
|
return Info{
|
||||||
|
Tier: LANOnly,
|
||||||
|
Label: "lan-only",
|
||||||
|
ManagedAgents: false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return Info{
|
||||||
|
Tier: Airplane,
|
||||||
|
Label: "airplane",
|
||||||
|
ManagedAgents: false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func probe(ctx context.Context, client *http.Client, url string) bool {
|
||||||
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, url, nil)
|
||||||
|
if err != nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
resp, err := client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
resp.Body.Close()
|
||||||
|
return true
|
||||||
|
}
|
||||||
48
internal/tier/tier_test.go
Normal file
48
internal/tier/tier_test.go
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
// internal/tier/tier_test.go
|
||||||
|
package tier_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/mathiasbq/supervisor/internal/tier"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestDetect_Tier1_WhenBothReachable(t *testing.T) {
|
||||||
|
anthropic := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
}))
|
||||||
|
defer anthropic.Close()
|
||||||
|
|
||||||
|
litellm := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
}))
|
||||||
|
defer litellm.Close()
|
||||||
|
|
||||||
|
info := tier.Detect(context.Background(), anthropic.URL, litellm.URL)
|
||||||
|
assert.Equal(t, tier.Full, info.Tier)
|
||||||
|
assert.Equal(t, "full-online", info.Label)
|
||||||
|
assert.True(t, info.ManagedAgents)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestDetect_Tier2_WhenOnlyLiteLLMReachable(t *testing.T) {
|
||||||
|
litellm := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
}))
|
||||||
|
defer litellm.Close()
|
||||||
|
|
||||||
|
info := tier.Detect(context.Background(), "http://127.0.0.1:1", litellm.URL)
|
||||||
|
assert.Equal(t, tier.LANOnly, info.Tier)
|
||||||
|
assert.Equal(t, "lan-only", info.Label)
|
||||||
|
assert.False(t, info.ManagedAgents)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestDetect_Tier3_WhenNeitherReachable(t *testing.T) {
|
||||||
|
info := tier.Detect(context.Background(), "http://127.0.0.1:1", "http://127.0.0.1:2")
|
||||||
|
assert.Equal(t, tier.Airplane, info.Tier)
|
||||||
|
assert.Equal(t, "airplane", info.Label)
|
||||||
|
assert.False(t, info.ManagedAgents)
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user