1. 30 Dec, 2025 1 commit
  2. 29 Dec, 2025 13 commits
    • Junming Chen's avatar
      19d0ee13
    • song's avatar
      fix(antigravity): 修复 429 限流处理逻辑 · caa8c47b
      song authored
      - 只有 5 次重试全部失败后才标记账户限流
      - 使用 Gemini 格式解析 429 响应中的重试时间
      - Claude 模型无重试时间时默认 1 分钟,Gemini 默认 5 分钟
      - 添加生图模型映射 gemini-3-pro-image-preview
      caa8c47b
    • song's avatar
      fix: gofmt · 42e2c506
      song authored
      42e2c506
    • song's avatar
      style: interface{} → any (gofmt rewrite rule) · bc75edd8
      song authored
      bc75edd8
    • song's avatar
    • song's avatar
      fix: 删除未使用的代码并修复格式 · 026740b5
      song authored
      - 删除 client.go 中未使用的 proxyURL 字段
      - 删除 AntigravityGatewayService 中未使用的字段和方法
      - 修复 gofmt 格式问题
      026740b5
    • song's avatar
      fix: 修复 golangci-lint 检查错误 · 21a04332
      song authored
      - SA1029: 创建 ctxkey 包定义类型安全的 context key
      - ST1005: 错误字符串首字母改小写
      - errcheck: 显式忽略 bytes.Buffer.Write 返回值
      - 修复单元测试中 GatewayService 缺少 cfg 字段的问题
      21a04332
    • song's avatar
      234e98f1
    • song's avatar
      feat(antigravity): 添加专用路由,支持仅使用 antigravity 账户 · b31bfd53
      song authored
      添加 /antigravity/v1/* 和 /antigravity/v1beta/* 路由:
      - 通过 ForcePlatform 中间件强制使用 antigravity 平台
      - 跳过混合调度逻辑,仅调度 antigravity 账户
      - 支持按分组优先查找,找不到时回退查询全部 antigravity 账户
      
      修复 context key 类型不匹配问题:
      - middleware 和 service 统一使用字符串常量 "ctx_force_platform"
      - 解决 Go context.Value() 类型+值匹配导致的读取失败
      
      其他改动:
      - 嵌入式前端中间件白名单添加 /antigravity/ 路径
      - e2e 测试 Gemini 端点 URL 添加 endpointPrefix 支持
      b31bfd53
    • IanShaw027's avatar
      feat(frontend): 优化弹窗组件架构和用户体验 · 23412965
      IanShaw027 authored
      - 使用 BaseDialog 替代旧版 Modal 组件
      - 添加平滑过渡动画和更好的可访问性支持
      - 新增 ExportProgressDialog 导出进度弹窗
      - 优化所有账号管理和使用记录相关弹窗
      - 更新国际化文案,改进用户交互体验
      - 精简依赖,减少 package.json 体积
      23412965
    • song's avatar
      feat(antigravity): 添加混合调度可选功能 · 1ad29032
      song authored
      - 后端:账户模型添加 IsMixedSchedulingEnabled() 方法,读取 extra.mixed_scheduling
      - 后端:gateway_service 和 gemini_messages_compat_service 支持混合调度逻辑
      - 后端:分组创建支持指定 platform 参数
      - 前端:账户创建/编辑弹窗添加混合调度开关(仅 antigravity 账户显示)
      - 前端:混合调度开关添加问号图标和 tooltip 说明
      - 前端:GroupSelector 支持根据 mixedScheduling 属性过滤分组
      - 前端:分组创建支持选择 platform
      - 测试:e2e 测试添加 ENDPOINT_PREFIX 环境变量支持混合/隔离模式测试
      - 测试:删除过时的 Claude signature 测试用例
      1ad29032
    • shaw's avatar
      fix: 修复因移除 SimpleMode 导致的测试编译错误 · 32b4b139
      shaw authored
      - 移除 api_contract_test.go 中的 SettingKeySimpleMode 引用
      - 移除期望响应中的 simple_mode 字段
      - 修复 NewSettingHandler 调用参数数量
      32b4b139
    • shaw's avatar
      refactor: 移除旧版数据库配置的简易模式实现 · 31fef105
      shaw authored
      移除与 PR #66 冲突的旧版简易模式实现(commit 7d4b7dee)。
      新版简易模式通过 run_mode 配置文件/环境变量控制,无需数据库设置。
      
      后端变更:
      - 移除 SettingKeySimpleMode 常量
      - 移除 SystemSettings/PublicSettings 中的 SimpleMode 字段
      - 移除 setting_handler 中的简易模式切换逻辑
      - 移除 userService 依赖(不再需要自动设置管理员并发数)
      
      前端变更:
      - 移除 appStore.simpleMode 状态
      - 移除设置页面的"使用模式"设置区块
      - 移除 GroupsView 中的简易模式相关逻辑
      - 移除相关国际化文案
      31fef105
  3. 28 Dec, 2025 23 commits
    • IanShaw027's avatar
      test: 修复分组测试以适配默认分组 · 9e9811cb
      IanShaw027 authored
      由于简易模式会自动创建3个默认分组(anthropic-default, openai-default, gemini-default),
      需要更新测试用例的预期数量:
      - TestList: 期望5个分组(3个默认 + 2个测试)
      - TestListActive: 期望4个活跃分组(3个默认 + 1个测试)
      - TestListActiveByPlatform: 期望2个Anthropic分组(1个默认 + 1个测试)
      - TestListWithFilters_Platform: 期望2个OpenAI分组(1个默认 + 1个测试)
      9e9811cb
    • IanShaw027's avatar
      feat(全栈): 实现简易模式核心功能 · ecfad788
      IanShaw027 authored
      **功能概述**:
      实现简易模式(Simple Mode),为个人用户和小团队提供简化的使用体验,隐藏复杂的分组、订阅、配额等概念。
      
      **后端改动**:
      1. 配置系统
         - 新增 run_mode 配置项(standard/simple)
         - 支持环境变量 RUN_MODE
         - 默认值为 standard
      
      2. 数据库初始化
         - 自动创建3个默认分组:anthropic-default、openai-default、gemini-default
         - 默认分组配置:无并发限制、active状态、非独占
         - 幂等性保证:重复启动不会重复创建
      
      3. 账号管理
         - 创建账号时自动绑定对应平台的默认分组
         - 如果未指定分组,自动查找并绑定默认分组
      
      **前端改动**:
      1. 状态管理
         - authStore 新增 isSimpleMode 计算属性
         - 从后端API获取并同步运行模式
      
      2. UI隐藏
         - 侧边栏:隐藏分组管理、订阅管理、兑换码菜单
         - 账号管理页面:隐藏分组列
         - 创建/编辑账号对话框:隐藏分组选择器
      
      3. 路由守卫
         - 限制访问分组、订阅、兑换码相关页面
         - 访问受限页面时自动重定向到仪表板
      
      **配置示例**:
      ```yaml
      run_mode: simple
      
      run_mode: standard
      ```
      
      **影响范围**:
      - 后端:配置、数据库迁移、账号服务
      - 前端:认证状态、路由、UI组件
      - 部署:配置文件示例
      
      **兼容性**:
      - 简易模式和标准模式可无缝切换
      - 不需要数据迁移
      - 现有数据不受影响
      ecfad788
    • song's avatar
      cf1d0f23
    • song's avatar
      test: 添加 Claude signature 场景 e2e 测试 · 995adaee
      song authored
      - 新增 TestClaudeMessagesWithClaudeSignature 测试
      - 验证历史 thinking block 带有 Claude signature 时的处理
      - 修复配额刷新服务的次要问题
      995adaee
    • shaw's avatar
      fix(usage): 分离 API 响应和窗口统计缓存,修复 5h 窗口未激活时的显示 bug · 30b95cf5
      shaw authored
      问题:
      1. WindowStats 与 API 响应一起缓存 10 分钟,导致费用数据更新延迟
      2. 当 5h 窗口未激活(ResetsAt 为空)时,FiveHour 为 nil,导致所有窗口的 WindowStats 都无法显示
      
      修复:
      - 分离缓存:API 响应缓存 10 分钟,窗口统计独立缓存 1 分钟
      - RemainingSeconds 每次请求时实时计算
      - FiveHour 对象始终创建(即使 ResetsAt 为空)
      - addWindowStats 增强防护,支持 FiveHour 为 nil 时仍处理其他窗口
      30b95cf5
    • shaw's avatar
      fix(test): 测试用例添加 simple_mode 字段 · 25b8a226
      shaw authored
      API 响应新增 simple_mode 字段,同步更新测试期望值
      25b8a226
    • shaw's avatar
      fix: 修复 NewSettingHandler 参数不足导致的编译错误 · 0084da9c
      shaw authored
      - 测试文件添加第三个参数 userService(nil)
      - Handler 添加 userService 空指针检查,防止测试环境 panic
      0084da9c
    • song's avatar
      feat(antigravity): 添加配额窗口显示功能 · 08ce6de4
      song authored
      后端:
      - 新增 AntigravityQuotaRefresher 定时刷新配额
      - Client 添加 FetchAvailableModels 方法获取模型配额
      - 配额数据存入 account.extra.quota 字段
      
      前端:
      - AccountUsageCell 支持显示 Antigravity 账户配额
      - UsageProgressBar 新增 amber 颜色
      - 显示 G3P/G3F/G3I/C4.5 四个配额进度条
      08ce6de4
    • shaw's avatar
      feat: 添加简单模式功能 · 7d4b7dee
      shaw authored
      新增简单模式设置,适合个人使用场景:
      - 隐藏多用户管理相关菜单(用户管理、兑换码等)
      - 自动关闭用户注册功能
      - 管理员并发数自动设为无限制(99999)
      - 侧边栏根据模式动态调整菜单项
      
      同时优化分组页面的"专属分组"功能,添加帮助提示说明使用场景
      7d4b7dee
    • song's avatar
      build: e2e 测试添加 build tag 避免 CI 运行 · b6b73943
      song authored
      - 添加 //go:build e2e tag,CI 不会自动运行这些测试
      - Makefile 添加 test-e2e 目标用于本地手动运行
      b6b73943
    • song's avatar
      fix(gateway): Antigravity 账户 count_tokens 返回估算值 · ad15d997
      song authored
      Antigravity 不支持 count_tokens 转发,直接返回估算值,
      与 Antigravity-Manager 和 proxycast 实现保持一致。
      
      修复 count_tokens 请求选择到 Antigravity 账户时导致 401 的问题。
      ad15d997
    • song's avatar
      test: 更新 thinking signature 测试用例 · ff57c860
      song authored
      将测试从无效signature改为无signature场景:
      - 无效 signature 应该被上游拒绝(预期行为)
      - Gemini 模型接受没有 signature 的 thinking block
      ff57c860
    • song's avatar
      fix(antigravity): 只有 Gemini 模型支持 dummy thought signature · 635d7e77
      song authored
      参考 Antigravity-Manager 的实现:
      - 添加 allowDummyThought 参数,只有 gemini-* 模型才启用
      - Claude 模型通过 Vertex API 需要有效的 thought signatures
      - thinking block 保留原有 signature
      - tool_use 只在 Gemini 模型时才使用 dummy signature
      635d7e77
    • song's avatar
      fix(antigravity): 与 proxycast 保持一致的 thought_signature 处理 · ba9eb684
      song authored
      - function_call 无条件添加 dummy thought_signature(与 proxycast 一致)
      - thinking block 在 thinking 模式下统一使用 dummy signature 替换历史无效 signature
      - 添加测试用例:TestClaudeMessagesWithInvalidThinkingSignature
      ba9eb684
    • song's avatar
      fix(antigravity): 修复 Gemini 3 thought_signature 和 schema 验证问题 · 9594c9c8
      song authored
      - 添加 dummyThoughtSignature 常量,在 thinking 模式下为无 signature 的 tool_use 自动添加
      - 增强 cleanJSONSchema:过滤 required 中不存在的属性,确保 type/properties 字段存在
      - 扩展 excludedSchemaKeys:增加 $id, $ref, strict, const, examples 等不支持的字段
      - 修复 429 重试逻辑:仅在所有重试失败后才标记账户为 rate_limited
      - 添加 e2e 集成测试:TestClaudeMessagesWithThinkingAndTools
      9594c9c8
    • song's avatar
    • song's avatar
    • song's avatar
      6648e650
    • IanShaw027's avatar
      refactor(frontend): UI/UX改进和组件优化 · 506cb21c
      IanShaw027 authored
      - DataTable组件操作列自适应
      - 优化各种Modal弹窗
      - 统一API调用方式(AbortSignal)
      - 添加全局订阅状态管理
      - 优化各管理视图的交互和布局
      - 修复国际化翻译问题
      506cb21c
    • yangjianbo's avatar
      fix: 代码的核心问题是判错条件用错了层级: · fd51ff69
      yangjianbo authored
        - apiKeyService.GetByKey(...) 返回的“找不到 API key”在这个项目里通常会被翻译成业务错误(比如
          service.ErrApiKeyNotFound 这类 ApplicationError),而不是直接把 gorm.ErrRecordNotFound 透传到中
          间件层。
        - 因此你在中间件里用 errors.Is(err, gorm.ErrRecordNotFound) 去判断“无效 key”,很容易匹配不到(尤其
          是:后面加 Redis 缓存、换存储实现、或测试里用 stub repo 时,根本不会出现 gorm 的错误)。
        - 匹配不到时就会走到 500 Failed to validate API key,导致无效 API key 被错误地当成服务端故障返回
          500(应该是 401)。
      
        修复思路:中间件不要依赖 gorm 的错误,改成判断业务层错误,例如:
      
        if errors.Is(err, service.ErrApiKeyNotFound) {
            abortWithGoogleError(c, 401, "Invalid API key")
            return
        }
      
        如果你把 GetByKey 的“not found”统一封装成业务错误,这样才不会被底层实现(gorm/redis/mock)影响。
      fd51ff69
    • shaw's avatar
      fix: 防止订阅过期时间超出 JSON 序列化范围 · fbdff4f3
      shaw authored
      问题:当分配订阅天数过大时,expires_at 年份可能超过 9999,
      导致 time.Time JSON 序列化失败(RFC 3339 要求年份 <= 9999),
      使后台无法显示和删除异常数据。
      
      修复:
      - handler 层添加 validity_days 最大值验证(max=36500,即100年)
      - service 层添加 MaxValidityDays 和 MaxExpiresAt 双重保护
      - 启动时自动修复已存在的异常数据(expires_at > 2099年)
      fbdff4f3
    • shaw's avatar
      cd9d31f5
    • noreply's avatar
      feat: Schedule batch update for account last_used_at · cbfce49a
      noreply authored
      Implement deferred batch update mechanism to reduce database load:
      
      - Add DeferredService for batching account last_used_at updates
      - Add TimingWheelService for efficient recurring task scheduling
      - Integrate with GatewayService and OpenAIGatewayService
      - Implement BatchUpdateLastUsed repository method using CASE...WHEN SQL
      - Fix golangci-lint error: Replace interface{} with any
      
      Benefits:
      - Reduces database writes by batching updates (10-second intervals)
      - Improves request throughput by deferring non-critical updates
      - Maintains accurate account usage tracking for scheduling
      cbfce49a
  4. 27 Dec, 2025 3 commits