mirror of
https://gitee.com/wanwujie/sub2api
synced 2026-05-04 21:20:51 +08:00
对称参照 Claude BetaPolicy 的 fast-mode 过滤实现,新增针对 OpenAI 上游
service_tier 字段(priority / flex,含客户端 "fast" → "priority" 归一化)的
pass / filter / block 三态策略,覆盖全部 OpenAI 入口 + admin 配置入口。
后端核心
- 新增 SettingKeyOpenAIFastPolicySettings、OpenAIFastPolicyRule、
OpenAIFastPolicySettings 配置模型,含规则的 service_tier × action × scope
× 模型白名单 × fallback action 维度。
- SettingService.Get/SetOpenAIFastPolicySettings;缺失时返回内置默认策略
(所有模型的 priority 走 filter,whitelist 为空,fallback=pass)。设计
依据:service_tier=fast 是用户级开关,与 model 字段正交,默认锁定特定
model slug 会留下"用 gpt-4 + fast 透传 priority 上游"的绕过路径。JSON
解析失败不再静默 fallback,slog.Warn 记录脏数据,便于运维定位。
- service_tier 归一化(trim + ToLower + fast→priority + 白名单 priority/flex)
与策略评估(evaluateOpenAIFastPolicy)作为唯一真实来源,HTTP / WS 共用。
抽出纯函数 evaluateOpenAIFastPolicyWithSettings,配合 ctx-bound settings
快照(withOpenAIFastPolicyContext / openAIFastPolicySettingsFromContext),
WS 长会话入口预取一次后所有帧复用,避免每帧打到 settingService。
HTTP 入口(4 个)
- Chat Completions、Anthropic 兼容(Messages,含 BetaFastMode→priority 二次
命中)、原生 Responses、Passthrough Responses 全部接入
applyOpenAIFastPolicyToBody,filter 走 sjson 顶层删除 service_tier,block
返回 403 forbidden_error JSON。
- 4 入口统一使用 upstream 视角的 model(GetMappedModel +
normalizeOpenAIModelForUpstream + Codex OAuth normalize 后的 slug),
避免 chat/messages/native /responses/passthrough 因为 model 维度不同
造成 whitelist 命中差异。
- 在 pass 路径也把客户端 "fast" 别名归一化为 "priority" 写回 body,
否则 native /responses 与 passthrough 入口会把 "fast" 原样透传给上游
导致 400/拒绝(chat-completions 入口的 normalizeResponsesBodyServiceTier
此前已具备同等行为)。
WebSocket 入口
- 新增 applyOpenAIFastPolicyToWSResponseCreate:严格匹配
type="response.create",仅处理顶层 service_tier;filter 用 sjson 删字段,
block 返回 typed *OpenAIFastBlockedError。
- ingress 路径在 parseClientPayload 内调用,block 命中先 Write Realtime
风格 error event 再返回 OpenAIWSClientCloseError(StatusPolicyViolation
=1008),依赖底层 WebSocket Conn.Write 的同步 flush 保证 error 先于
close。
- passthrough 路径在 RunEntry 前对 firstClientMessage 应用策略,并通过
openAIWSPolicyEnforcingFrameConn 包装 ReadFrame 对每个 client→upstream
帧执行策略;后续帧无 model 字段时回退到 capturedSessionModel。
filter 闭包内同时侦测 session.update / session.created 帧的 session.model
字段刷新 capturedSessionModel,封堵"首帧 model=gpt-4o(pass)→
session.update 改为 gpt-5.5 → 不带 model 的 response.create fallback
到 gpt-4o"的 mid-session 绕过路径。
- passthrough billing:requestServiceTier 在策略 filter 之后再从
firstClientMessage 提取,filter 命中时 OpenAIForwardResult.ServiceTier
上报 nil(default tier),与 HTTP 入口(reqBody 来自 post-filter map)
/ WS ingress(payload 来自 post-filter bytes)的语义一致。
- 错误事件 schema:{event_id: "evt_<32hex>", type: "error",
error: {type: "forbidden_error", code: "policy_violation", message}},
与 OpenAI codex 客户端 error event 解析兼容。
Admin / Frontend
- dto.SystemSettings / UpdateSettingsRequest 新增
openai_fast_policy_settings 字段(omitempty),bulk GET/PUT 接入。
- Settings 页 Gateway 页签新增 Fast/Flex Policy 表单卡片:
service_tier × action × scope × 模型白名单 × fallback action 全字段配置。
- 前端守门:openaiFastPolicyLoaded 标志仅在 GET 真带回字段时才允许回写,
避免 rollout/错误把默认规则覆盖成空;saveSettings 回写循环 skip 该字段,
由专用刷新逻辑处理;仅 action=block 时发送 error_message,匹配后端
omitempty 行为。
测试
- HTTP 路径:openai_fast_policy_test.go 覆盖默认配置(whitelist=[],所有
模型 priority filter)/ block 自定义错误 / scope 区分 / filter 删字段 /
block 不改 body / block 短路上游 / Anthropic BetaFastMode 触发 OpenAI
fast policy 等场景。
- WebSocket 路径:openai_fast_policy_ws_test.go 覆盖
helper 单元(filter / fast→priority 归一化 / flex 透传 / block typed
error / 无 service_tier 字节不变 / 非 response.create 帧不动 / 空 type
帧不动 / event_id+code 字段断言 / 非字符串 service_tier 容错)+
pass 路径 fast 别名归一化回归 +
ingress 端到端(filter 后上游不含 service_tier / block 后客户端先收
error event 再收 close 1008 且上游 0 写)+
passthrough capturedSessionModel fallback 用例(whitelist 策略下首帧
建立、缺 model 命中 fallback、缺少 fallback 时的 leak 文档化)+
passthrough session.update / session.created 旋转 capturedSessionModel
的 mid-session 绕过回归 +
passthrough billing post-filter ServiceTier 与 idempotent filter 回归。
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
355 lines
18 KiB
Go
355 lines
18 KiB
Go
package dto
|
|
|
|
import (
|
|
"encoding/json"
|
|
"strings"
|
|
)
|
|
|
|
// CustomMenuItem represents a user-configured custom menu entry.
|
|
type CustomMenuItem struct {
|
|
ID string `json:"id"`
|
|
Label string `json:"label"`
|
|
IconSVG string `json:"icon_svg"`
|
|
URL string `json:"url"`
|
|
Visibility string `json:"visibility"` // "user" or "admin"
|
|
SortOrder int `json:"sort_order"`
|
|
}
|
|
|
|
// CustomEndpoint represents an admin-configured API endpoint for quick copy.
|
|
type CustomEndpoint struct {
|
|
Name string `json:"name"`
|
|
Endpoint string `json:"endpoint"`
|
|
Description string `json:"description"`
|
|
}
|
|
|
|
// SystemSettings represents the admin settings API response payload.
|
|
type SystemSettings struct {
|
|
RegistrationEnabled bool `json:"registration_enabled"`
|
|
EmailVerifyEnabled bool `json:"email_verify_enabled"`
|
|
RegistrationEmailSuffixWhitelist []string `json:"registration_email_suffix_whitelist"`
|
|
PromoCodeEnabled bool `json:"promo_code_enabled"`
|
|
PasswordResetEnabled bool `json:"password_reset_enabled"`
|
|
FrontendURL string `json:"frontend_url"`
|
|
InvitationCodeEnabled bool `json:"invitation_code_enabled"`
|
|
TotpEnabled bool `json:"totp_enabled"` // TOTP 双因素认证
|
|
TotpEncryptionKeyConfigured bool `json:"totp_encryption_key_configured"` // TOTP 加密密钥是否已配置
|
|
|
|
SMTPHost string `json:"smtp_host"`
|
|
SMTPPort int `json:"smtp_port"`
|
|
SMTPUsername string `json:"smtp_username"`
|
|
SMTPPasswordConfigured bool `json:"smtp_password_configured"`
|
|
SMTPFrom string `json:"smtp_from_email"`
|
|
SMTPFromName string `json:"smtp_from_name"`
|
|
SMTPUseTLS bool `json:"smtp_use_tls"`
|
|
|
|
TurnstileEnabled bool `json:"turnstile_enabled"`
|
|
TurnstileSiteKey string `json:"turnstile_site_key"`
|
|
TurnstileSecretKeyConfigured bool `json:"turnstile_secret_key_configured"`
|
|
|
|
LinuxDoConnectEnabled bool `json:"linuxdo_connect_enabled"`
|
|
LinuxDoConnectClientID string `json:"linuxdo_connect_client_id"`
|
|
LinuxDoConnectClientSecretConfigured bool `json:"linuxdo_connect_client_secret_configured"`
|
|
LinuxDoConnectRedirectURL string `json:"linuxdo_connect_redirect_url"`
|
|
|
|
WeChatConnectEnabled bool `json:"wechat_connect_enabled"`
|
|
WeChatConnectAppID string `json:"wechat_connect_app_id"`
|
|
WeChatConnectAppSecretConfigured bool `json:"wechat_connect_app_secret_configured"`
|
|
WeChatConnectOpenAppID string `json:"wechat_connect_open_app_id"`
|
|
WeChatConnectOpenAppSecretConfigured bool `json:"wechat_connect_open_app_secret_configured"`
|
|
WeChatConnectMPAppID string `json:"wechat_connect_mp_app_id"`
|
|
WeChatConnectMPAppSecretConfigured bool `json:"wechat_connect_mp_app_secret_configured"`
|
|
WeChatConnectMobileAppID string `json:"wechat_connect_mobile_app_id"`
|
|
WeChatConnectMobileAppSecretConfigured bool `json:"wechat_connect_mobile_app_secret_configured"`
|
|
WeChatConnectOpenEnabled bool `json:"wechat_connect_open_enabled"`
|
|
WeChatConnectMPEnabled bool `json:"wechat_connect_mp_enabled"`
|
|
WeChatConnectMobileEnabled bool `json:"wechat_connect_mobile_enabled"`
|
|
WeChatConnectMode string `json:"wechat_connect_mode"`
|
|
WeChatConnectScopes string `json:"wechat_connect_scopes"`
|
|
WeChatConnectRedirectURL string `json:"wechat_connect_redirect_url"`
|
|
WeChatConnectFrontendRedirectURL string `json:"wechat_connect_frontend_redirect_url"`
|
|
|
|
OIDCConnectEnabled bool `json:"oidc_connect_enabled"`
|
|
OIDCConnectProviderName string `json:"oidc_connect_provider_name"`
|
|
OIDCConnectClientID string `json:"oidc_connect_client_id"`
|
|
OIDCConnectClientSecretConfigured bool `json:"oidc_connect_client_secret_configured"`
|
|
OIDCConnectIssuerURL string `json:"oidc_connect_issuer_url"`
|
|
OIDCConnectDiscoveryURL string `json:"oidc_connect_discovery_url"`
|
|
OIDCConnectAuthorizeURL string `json:"oidc_connect_authorize_url"`
|
|
OIDCConnectTokenURL string `json:"oidc_connect_token_url"`
|
|
OIDCConnectUserInfoURL string `json:"oidc_connect_userinfo_url"`
|
|
OIDCConnectJWKSURL string `json:"oidc_connect_jwks_url"`
|
|
OIDCConnectScopes string `json:"oidc_connect_scopes"`
|
|
OIDCConnectRedirectURL string `json:"oidc_connect_redirect_url"`
|
|
OIDCConnectFrontendRedirectURL string `json:"oidc_connect_frontend_redirect_url"`
|
|
OIDCConnectTokenAuthMethod string `json:"oidc_connect_token_auth_method"`
|
|
OIDCConnectUsePKCE bool `json:"oidc_connect_use_pkce"`
|
|
OIDCConnectValidateIDToken bool `json:"oidc_connect_validate_id_token"`
|
|
OIDCConnectAllowedSigningAlgs string `json:"oidc_connect_allowed_signing_algs"`
|
|
OIDCConnectClockSkewSeconds int `json:"oidc_connect_clock_skew_seconds"`
|
|
OIDCConnectRequireEmailVerified bool `json:"oidc_connect_require_email_verified"`
|
|
OIDCConnectUserInfoEmailPath string `json:"oidc_connect_userinfo_email_path"`
|
|
OIDCConnectUserInfoIDPath string `json:"oidc_connect_userinfo_id_path"`
|
|
OIDCConnectUserInfoUsernamePath string `json:"oidc_connect_userinfo_username_path"`
|
|
|
|
SiteName string `json:"site_name"`
|
|
SiteLogo string `json:"site_logo"`
|
|
SiteSubtitle string `json:"site_subtitle"`
|
|
APIBaseURL string `json:"api_base_url"`
|
|
ContactInfo string `json:"contact_info"`
|
|
DocURL string `json:"doc_url"`
|
|
HomeContent string `json:"home_content"`
|
|
HideCcsImportButton bool `json:"hide_ccs_import_button"`
|
|
PurchaseSubscriptionEnabled bool `json:"purchase_subscription_enabled"`
|
|
PurchaseSubscriptionURL string `json:"purchase_subscription_url"`
|
|
TableDefaultPageSize int `json:"table_default_page_size"`
|
|
TablePageSizeOptions []int `json:"table_page_size_options"`
|
|
CustomMenuItems []CustomMenuItem `json:"custom_menu_items"`
|
|
CustomEndpoints []CustomEndpoint `json:"custom_endpoints"`
|
|
|
|
DefaultConcurrency int `json:"default_concurrency"`
|
|
DefaultBalance float64 `json:"default_balance"`
|
|
AffiliateRebateRate float64 `json:"affiliate_rebate_rate"`
|
|
AffiliateRebateFreezeHours int `json:"affiliate_rebate_freeze_hours"`
|
|
AffiliateRebateDurationDays int `json:"affiliate_rebate_duration_days"`
|
|
AffiliateRebatePerInviteeCap float64 `json:"affiliate_rebate_per_invitee_cap"`
|
|
DefaultUserRPMLimit int `json:"default_user_rpm_limit"`
|
|
DefaultSubscriptions []DefaultSubscriptionSetting `json:"default_subscriptions"`
|
|
|
|
// Model fallback configuration
|
|
EnableModelFallback bool `json:"enable_model_fallback"`
|
|
FallbackModelAnthropic string `json:"fallback_model_anthropic"`
|
|
FallbackModelOpenAI string `json:"fallback_model_openai"`
|
|
FallbackModelGemini string `json:"fallback_model_gemini"`
|
|
FallbackModelAntigravity string `json:"fallback_model_antigravity"`
|
|
|
|
// Identity patch configuration (Claude -> Gemini)
|
|
EnableIdentityPatch bool `json:"enable_identity_patch"`
|
|
IdentityPatchPrompt string `json:"identity_patch_prompt"`
|
|
|
|
// Ops monitoring (vNext)
|
|
OpsMonitoringEnabled bool `json:"ops_monitoring_enabled"`
|
|
OpsRealtimeMonitoringEnabled bool `json:"ops_realtime_monitoring_enabled"`
|
|
OpsQueryModeDefault string `json:"ops_query_mode_default"`
|
|
OpsMetricsIntervalSeconds int `json:"ops_metrics_interval_seconds"`
|
|
|
|
MinClaudeCodeVersion string `json:"min_claude_code_version"`
|
|
MaxClaudeCodeVersion string `json:"max_claude_code_version"`
|
|
|
|
// 分组隔离
|
|
AllowUngroupedKeyScheduling bool `json:"allow_ungrouped_key_scheduling"`
|
|
|
|
// Backend Mode
|
|
BackendModeEnabled bool `json:"backend_mode_enabled"`
|
|
|
|
// Gateway forwarding behavior
|
|
EnableFingerprintUnification bool `json:"enable_fingerprint_unification"`
|
|
EnableMetadataPassthrough bool `json:"enable_metadata_passthrough"`
|
|
EnableCCHSigning bool `json:"enable_cch_signing"`
|
|
|
|
// Web Search Emulation
|
|
WebSearchEmulationEnabled bool `json:"web_search_emulation_enabled"`
|
|
|
|
// Payment visible method routing
|
|
PaymentVisibleMethodAlipaySource string `json:"payment_visible_method_alipay_source"`
|
|
PaymentVisibleMethodWxpaySource string `json:"payment_visible_method_wxpay_source"`
|
|
PaymentVisibleMethodAlipayEnabled bool `json:"payment_visible_method_alipay_enabled"`
|
|
PaymentVisibleMethodWxpayEnabled bool `json:"payment_visible_method_wxpay_enabled"`
|
|
|
|
// OpenAI account scheduling
|
|
OpenAIAdvancedSchedulerEnabled bool `json:"openai_advanced_scheduler_enabled"`
|
|
|
|
// Payment configuration
|
|
PaymentEnabled bool `json:"payment_enabled"`
|
|
PaymentMinAmount float64 `json:"payment_min_amount"`
|
|
PaymentMaxAmount float64 `json:"payment_max_amount"`
|
|
PaymentDailyLimit float64 `json:"payment_daily_limit"`
|
|
PaymentOrderTimeoutMin int `json:"payment_order_timeout_minutes"`
|
|
PaymentMaxPendingOrders int `json:"payment_max_pending_orders"`
|
|
PaymentEnabledTypes []string `json:"payment_enabled_types"`
|
|
PaymentBalanceDisabled bool `json:"payment_balance_disabled"`
|
|
PaymentBalanceRechargeMultiplier float64 `json:"payment_balance_recharge_multiplier"`
|
|
PaymentRechargeFeeRate float64 `json:"payment_recharge_fee_rate"`
|
|
PaymentLoadBalanceStrat string `json:"payment_load_balance_strategy"`
|
|
PaymentProductNamePrefix string `json:"payment_product_name_prefix"`
|
|
PaymentProductNameSuffix string `json:"payment_product_name_suffix"`
|
|
PaymentHelpImageURL string `json:"payment_help_image_url"`
|
|
PaymentHelpText string `json:"payment_help_text"`
|
|
|
|
// Cancel rate limit
|
|
PaymentCancelRateLimitEnabled bool `json:"payment_cancel_rate_limit_enabled"`
|
|
PaymentCancelRateLimitMax int `json:"payment_cancel_rate_limit_max"`
|
|
PaymentCancelRateLimitWindow int `json:"payment_cancel_rate_limit_window"`
|
|
PaymentCancelRateLimitUnit string `json:"payment_cancel_rate_limit_unit"`
|
|
PaymentCancelRateLimitMode string `json:"payment_cancel_rate_limit_window_mode"`
|
|
|
|
// Balance low notification
|
|
BalanceLowNotifyEnabled bool `json:"balance_low_notify_enabled"`
|
|
BalanceLowNotifyThreshold float64 `json:"balance_low_notify_threshold"`
|
|
BalanceLowNotifyRechargeURL string `json:"balance_low_notify_recharge_url"`
|
|
AccountQuotaNotifyEnabled bool `json:"account_quota_notify_enabled"`
|
|
AccountQuotaNotifyEmails []NotifyEmailEntry `json:"account_quota_notify_emails"`
|
|
|
|
// Channel Monitor feature switch
|
|
ChannelMonitorEnabled bool `json:"channel_monitor_enabled"`
|
|
ChannelMonitorDefaultIntervalSeconds int `json:"channel_monitor_default_interval_seconds"`
|
|
|
|
// Available Channels feature switch (user-facing aggregate view)
|
|
AvailableChannelsEnabled bool `json:"available_channels_enabled"`
|
|
|
|
// Affiliate (邀请返利) feature switch
|
|
AffiliateEnabled bool `json:"affiliate_enabled"`
|
|
|
|
// OpenAI fast/flex policy
|
|
OpenAIFastPolicySettings *OpenAIFastPolicySettings `json:"openai_fast_policy_settings,omitempty"`
|
|
}
|
|
|
|
type DefaultSubscriptionSetting struct {
|
|
GroupID int64 `json:"group_id"`
|
|
ValidityDays int `json:"validity_days"`
|
|
}
|
|
|
|
type PublicSettings struct {
|
|
RegistrationEnabled bool `json:"registration_enabled"`
|
|
EmailVerifyEnabled bool `json:"email_verify_enabled"`
|
|
ForceEmailOnThirdPartySignup bool `json:"force_email_on_third_party_signup"`
|
|
RegistrationEmailSuffixWhitelist []string `json:"registration_email_suffix_whitelist"`
|
|
PromoCodeEnabled bool `json:"promo_code_enabled"`
|
|
PasswordResetEnabled bool `json:"password_reset_enabled"`
|
|
InvitationCodeEnabled bool `json:"invitation_code_enabled"`
|
|
TotpEnabled bool `json:"totp_enabled"` // TOTP 双因素认证
|
|
TurnstileEnabled bool `json:"turnstile_enabled"`
|
|
TurnstileSiteKey string `json:"turnstile_site_key"`
|
|
SiteName string `json:"site_name"`
|
|
SiteLogo string `json:"site_logo"`
|
|
SiteSubtitle string `json:"site_subtitle"`
|
|
APIBaseURL string `json:"api_base_url"`
|
|
ContactInfo string `json:"contact_info"`
|
|
DocURL string `json:"doc_url"`
|
|
HomeContent string `json:"home_content"`
|
|
HideCcsImportButton bool `json:"hide_ccs_import_button"`
|
|
PurchaseSubscriptionEnabled bool `json:"purchase_subscription_enabled"`
|
|
PurchaseSubscriptionURL string `json:"purchase_subscription_url"`
|
|
TableDefaultPageSize int `json:"table_default_page_size"`
|
|
TablePageSizeOptions []int `json:"table_page_size_options"`
|
|
CustomMenuItems []CustomMenuItem `json:"custom_menu_items"`
|
|
CustomEndpoints []CustomEndpoint `json:"custom_endpoints"`
|
|
LinuxDoOAuthEnabled bool `json:"linuxdo_oauth_enabled"`
|
|
WeChatOAuthEnabled bool `json:"wechat_oauth_enabled"`
|
|
WeChatOAuthOpenEnabled bool `json:"wechat_oauth_open_enabled"`
|
|
WeChatOAuthMPEnabled bool `json:"wechat_oauth_mp_enabled"`
|
|
WeChatOAuthMobileEnabled bool `json:"wechat_oauth_mobile_enabled"`
|
|
OIDCOAuthEnabled bool `json:"oidc_oauth_enabled"`
|
|
OIDCOAuthProviderName string `json:"oidc_oauth_provider_name"`
|
|
SoraClientEnabled bool `json:"sora_client_enabled"`
|
|
BackendModeEnabled bool `json:"backend_mode_enabled"`
|
|
PaymentEnabled bool `json:"payment_enabled"`
|
|
Version string `json:"version"`
|
|
BalanceLowNotifyEnabled bool `json:"balance_low_notify_enabled"`
|
|
AccountQuotaNotifyEnabled bool `json:"account_quota_notify_enabled"`
|
|
BalanceLowNotifyThreshold float64 `json:"balance_low_notify_threshold"`
|
|
BalanceLowNotifyRechargeURL string `json:"balance_low_notify_recharge_url"`
|
|
|
|
ChannelMonitorEnabled bool `json:"channel_monitor_enabled"`
|
|
ChannelMonitorDefaultIntervalSeconds int `json:"channel_monitor_default_interval_seconds"`
|
|
|
|
AvailableChannelsEnabled bool `json:"available_channels_enabled"`
|
|
|
|
AffiliateEnabled bool `json:"affiliate_enabled"`
|
|
}
|
|
|
|
// OverloadCooldownSettings 529过载冷却配置 DTO
|
|
type OverloadCooldownSettings struct {
|
|
Enabled bool `json:"enabled"`
|
|
CooldownMinutes int `json:"cooldown_minutes"`
|
|
}
|
|
|
|
// StreamTimeoutSettings 流超时处理配置 DTO
|
|
type StreamTimeoutSettings struct {
|
|
Enabled bool `json:"enabled"`
|
|
Action string `json:"action"`
|
|
TempUnschedMinutes int `json:"temp_unsched_minutes"`
|
|
ThresholdCount int `json:"threshold_count"`
|
|
ThresholdWindowMinutes int `json:"threshold_window_minutes"`
|
|
}
|
|
|
|
// RectifierSettings 请求整流器配置 DTO
|
|
type RectifierSettings struct {
|
|
Enabled bool `json:"enabled"`
|
|
ThinkingSignatureEnabled bool `json:"thinking_signature_enabled"`
|
|
ThinkingBudgetEnabled bool `json:"thinking_budget_enabled"`
|
|
APIKeySignatureEnabled bool `json:"apikey_signature_enabled"`
|
|
APIKeySignaturePatterns []string `json:"apikey_signature_patterns"`
|
|
}
|
|
|
|
// BetaPolicyRule Beta 策略规则 DTO
|
|
type BetaPolicyRule struct {
|
|
BetaToken string `json:"beta_token"`
|
|
Action string `json:"action"`
|
|
Scope string `json:"scope"`
|
|
ErrorMessage string `json:"error_message,omitempty"`
|
|
ModelWhitelist []string `json:"model_whitelist,omitempty"`
|
|
FallbackAction string `json:"fallback_action,omitempty"`
|
|
FallbackErrorMessage string `json:"fallback_error_message,omitempty"`
|
|
}
|
|
|
|
// BetaPolicySettings Beta 策略配置 DTO
|
|
type BetaPolicySettings struct {
|
|
Rules []BetaPolicyRule `json:"rules"`
|
|
}
|
|
|
|
// OpenAIFastPolicyRule OpenAI fast/flex 策略规则 DTO
|
|
type OpenAIFastPolicyRule struct {
|
|
ServiceTier string `json:"service_tier"`
|
|
Action string `json:"action"`
|
|
Scope string `json:"scope"`
|
|
ErrorMessage string `json:"error_message,omitempty"`
|
|
ModelWhitelist []string `json:"model_whitelist,omitempty"`
|
|
FallbackAction string `json:"fallback_action,omitempty"`
|
|
FallbackErrorMessage string `json:"fallback_error_message,omitempty"`
|
|
}
|
|
|
|
// OpenAIFastPolicySettings OpenAI fast 策略配置 DTO
|
|
type OpenAIFastPolicySettings struct {
|
|
Rules []OpenAIFastPolicyRule `json:"rules"`
|
|
}
|
|
|
|
// ParseCustomMenuItems parses a JSON string into a slice of CustomMenuItem.
|
|
// Returns empty slice on empty/invalid input.
|
|
func ParseCustomMenuItems(raw string) []CustomMenuItem {
|
|
raw = strings.TrimSpace(raw)
|
|
if raw == "" || raw == "[]" {
|
|
return []CustomMenuItem{}
|
|
}
|
|
var items []CustomMenuItem
|
|
if err := json.Unmarshal([]byte(raw), &items); err != nil {
|
|
return []CustomMenuItem{}
|
|
}
|
|
return items
|
|
}
|
|
|
|
// ParseUserVisibleMenuItems parses custom menu items and filters out admin-only entries.
|
|
func ParseUserVisibleMenuItems(raw string) []CustomMenuItem {
|
|
items := ParseCustomMenuItems(raw)
|
|
filtered := make([]CustomMenuItem, 0, len(items))
|
|
for _, item := range items {
|
|
if item.Visibility != "admin" {
|
|
filtered = append(filtered, item)
|
|
}
|
|
}
|
|
return filtered
|
|
}
|
|
|
|
// ParseCustomEndpoints parses a JSON string into a slice of CustomEndpoint.
|
|
// Returns empty slice on empty/invalid input.
|
|
func ParseCustomEndpoints(raw string) []CustomEndpoint {
|
|
raw = strings.TrimSpace(raw)
|
|
if raw == "" || raw == "[]" {
|
|
return []CustomEndpoint{}
|
|
}
|
|
var items []CustomEndpoint
|
|
if err := json.Unmarshal([]byte(raw), &items); err != nil {
|
|
return []CustomEndpoint{}
|
|
}
|
|
return items
|
|
}
|