mirror of
https://github.com/tiennm99/goclaw.git
synced 2026-06-09 22:12:07 +00:00
ed32e68c68
Part A — Channel quota limiter (managed mode): - DB-backed per-user/group request quotas with in-memory 60s TTL cache - Config merge priority: Groups > Channels > Providers > Default - Per-group quota override via channels.telegram.groups[chatID].quota - Migration 000009: index on channel_requests for quota queries - Hot-reload quota config via pub/sub (TopicConfigChanged) Part B — Per-run tool call budget: - Soft stop at configurable limit (default 25, per-agent override) - MaxToolCalls field on AgentDefaults + AgentSpec + LoopConfig - LLM gets one final call to summarize when budget exceeded Part C — Web UI + config page refactor: - QuotaSection with provider/channel dropdowns (useProviders, useChannelInstances) - Config page refactored to vertical sidebar tabs layout - Categories: General, Quota, Agents, Tools, Connections, Advanced, Raw Editor - Fixed config.patch RPC to serialize raw JSON + baseHash correctly - Config change pub/sub broadcast from handleApply/handlePatch Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
108 lines
3.7 KiB
Go
108 lines
3.7 KiB
Go
package cmd
|
|
|
|
import (
|
|
"fmt"
|
|
"log/slog"
|
|
"strings"
|
|
|
|
"github.com/nextlevelbuilder/goclaw/internal/channels"
|
|
)
|
|
|
|
// Matching TS pi-embedded-helpers/errors.ts error classification.
|
|
// Never expose raw JSON/API payloads to the user.
|
|
func formatAgentError(err error) string {
|
|
raw := err.Error()
|
|
lower := strings.ToLower(raw)
|
|
|
|
// 1. Timeout — must be checked BEFORE context overflow because
|
|
// "context deadline exceeded" contains both "context" and "exceeded",
|
|
// which would false-positive match the context overflow heuristic.
|
|
if containsAny(lower, "timeout", "timed out", "deadline exceeded") {
|
|
return "⚠️ Request timed out. Please try again."
|
|
}
|
|
|
|
// 2. Context overflow
|
|
if isContextOverflowError(lower) {
|
|
return "⚠️ Context overflow — message too large for this model. Try /new to start a fresh session."
|
|
}
|
|
|
|
// 3. Role ordering / message format errors (tool_use_id mismatch, roles must alternate, etc.)
|
|
if isMessageFormatError(lower) {
|
|
return "⚠️ Session history conflict — please try again. If this persists, use /new to start a fresh session."
|
|
}
|
|
|
|
// 4. Rate limit
|
|
if containsAny(lower, "rate limit", "rate_limit", "too many requests", "429", "quota exceeded", "resource_exhausted") {
|
|
return "⚠️ API rate limit reached. Please try again later."
|
|
}
|
|
|
|
// 5. Overloaded
|
|
if strings.Contains(lower, "overloaded") {
|
|
return "⚠️ The AI service is temporarily overloaded. Please try again in a moment."
|
|
}
|
|
|
|
// 6. Billing
|
|
if containsAny(lower, "billing", "insufficient credits", "credit balance", "payment required", "402") {
|
|
return "⚠️ API billing error — your API key may have run out of credits. Check your provider's billing dashboard."
|
|
}
|
|
|
|
// 7. Auth errors
|
|
if containsAny(lower, "invalid api key", "invalid_api_key", "unauthorized", "forbidden", "authentication", "401", "403", "access denied") {
|
|
return "⚠️ Authentication error. Please check your API key configuration."
|
|
}
|
|
|
|
// 8. Model config
|
|
if strings.Contains(lower, "not a valid model") {
|
|
return "⚠️ Model configuration error. Please check your config and restart."
|
|
}
|
|
|
|
// 9. Generic — log the full error but show only a safe message to user
|
|
slog.Warn("unclassified agent error", "error", raw)
|
|
return "⚠️ Sorry, something went wrong processing your message. Please try again."
|
|
}
|
|
|
|
// isContextOverflowError checks for context window/size overflow patterns.
|
|
func isContextOverflowError(lower string) bool {
|
|
return containsAny(lower,
|
|
"request_too_large",
|
|
"context length exceeded",
|
|
"maximum context length",
|
|
"prompt is too long",
|
|
"exceeds model context window",
|
|
"request exceeds the maximum size",
|
|
) || (strings.Contains(lower, "context") &&
|
|
containsAny(lower, "overflow", "too large", "too long", "limit", "exceeded"))
|
|
}
|
|
|
|
// isMessageFormatError checks for tool_use/tool_result mismatch, role ordering,
|
|
// and other message format errors that indicate corrupted session history.
|
|
func isMessageFormatError(lower string) bool {
|
|
return containsAny(lower,
|
|
"tool_use_id",
|
|
"tool_use.id",
|
|
"unexpected tool",
|
|
"roles must alternate",
|
|
"incorrect role information",
|
|
"invalid request format",
|
|
"tool_result block",
|
|
"tool_use block",
|
|
)
|
|
}
|
|
|
|
// containsAny returns true if s contains any of the given substrings.
|
|
func containsAny(s string, substrs ...string) bool {
|
|
for _, sub := range substrs {
|
|
if strings.Contains(s, sub) {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// formatQuotaExceeded formats a user-friendly quota exceeded message.
|
|
func formatQuotaExceeded(result channels.QuotaResult) string {
|
|
labels := map[string]string{"hour": "Hourly", "day": "Daily", "week": "Weekly"}
|
|
return fmt.Sprintf("⚠️ %s request limit reached (%d/%d). Please try again later.",
|
|
labels[result.Window], result.Used, result.Limit)
|
|
}
|