1. 31 Dec, 2025 14 commits
    • IanShaw027's avatar
    • IanShaw027's avatar
      test(gateway): 补充账号调度优化的单元测试 · fe31495a
      IanShaw027 authored
      - 添加 GetAccountsLoadBatch 批量负载查询测试
      - 添加 CleanupExpiredAccountSlots 过期槽位清理测试
      - 添加 SelectAccountWithLoadAwareness 负载感知选择测试
      - 测试覆盖降级行为、账号排除、错误处理等场景
      fe31495a
    • IanShaw027's avatar
      feat(gateway): 实现负载感知的账号调度优化 · 592d2d09
      IanShaw027 authored
      - 新增调度配置:粘性会话排队、兜底排队、负载计算、槽位清理
      - 实现账号级等待队列和批量负载查询(Redis Lua 脚本)
      - 三层选择策略:粘性会话优先 → 负载感知选择 → 兜底排队
      - 后台定期清理过期槽位,防止资源泄漏
      - 集成到所有网关处理器(Claude/Gemini/OpenAI)
      592d2d09
    • NepetaLemon's avatar
      refactor: 移除 infrastructure 目录 (#108) · 2270a54f
      NepetaLemon authored
      * refactor: 迁移初始化 db 和 redis 到 repository
      
      * refactor: 迁移 errors 到 pkg
      2270a54f
    • shaw's avatar
      fix(billing): 修复限额为0时消费记录失败的问题 · c5b792ad
      shaw authored
      - 添加 normalizeLimit 函数,将 0 或负数限额规范化为 nil(无限制)
      - 简化 IncrementUsage,移除冗余的配额检查逻辑
        - 配额检查已在请求前由中间件和网关完成
        - 消费记录应无条件执行,确保数据完整性
      - 删除测试限额超出行为的无效集成测试
      c5b792ad
    • shaw's avatar
      style: fix gofmt formatting in test file · aac7dd6b
      shaw authored
      Remove redundant alignment whitespace before comments.
      aac7dd6b
    • yangjianbo's avatar
      fix(设置): 修复站点设置保存失败的问题 · 6f6dc303
      yangjianbo authored
      问题:
      1. Setting.value 字段设置了 NotEmpty() 约束,导致保存空字符串值时验证失败
      2. 数据库 settings 表缺少 key 字段的唯一约束,导致 ON CONFLICT 语句执行失败
      
      修复:
      - 移除 ent/schema/setting.go 中 value 字段的 NotEmpty() 约束
      - 新增迁移 015_fix_settings_unique_constraint.sql 添加缺失的唯一约束
      - 添加3个回归测试确保空值保存功能正常
      
      🤖 Generated with [Claude Code](https://claude.com/claude-code
      
      )
      Co-Authored-By: default avatarClaude Opus 4.5 <noreply@anthropic.com>
      6f6dc303
    • yangjianbo's avatar
      fix(仓储): 修复并发缓存前缀与软删除更新 · d77d0544
      yangjianbo authored
      补齐 Redis ZSET 前缀处理,确保并发释放计数正确
      
      删除时改用 Client().Mutate 走更新逻辑,保留软删除记录
      
      测试: make test-integration
      d77d0544
    • yangjianbo's avatar
      fix(lint): 修复 golangci-lint 报告的代码问题 · 682f546c
      yangjianbo authored
      - errcheck: 修复类型断言未检查返回值的问题
        - pool.go: 添加 sync.Map 类型断言安全检查
        - req_client_pool.go: 添加 sync.Map 类型断言安全检查
        - concurrency_cache_benchmark_test.go: 显式忽略断言返回值
        - gateway_service.go: 显式忽略 WriteString 返回值
      
      - gofmt: 修复代码格式问题
        - redis.go: 注释对齐
        - api_key_repo.go: 结构体字段对齐
        - concurrency_cache.go: 字段对齐
        - http_upstream.go: 注释对齐
      
      - unused: 删除未使用的代码
        - user_repo.go: 删除未使用的 sql 字段
        - usage_service.go: 删除未使用的 calculateStats 函数
      
      🤖 Generated with [Claude Code](https://claude.com/claude-code
      
      )
      Co-Authored-By: default avatarClaude Opus 4.5 <noreply@anthropic.com>
      682f546c
    • yangjianbo's avatar
      fix(数据层): 修复数据完整性与仓储一致性问题 · 5906f9ab
      yangjianbo authored
      ## 数据完整性修复 (fix-critical-data-integrity)
      - 添加 error_translate.go 统一错误转换层
      - 修复 nil 输入和 NotFound 错误处理
      - 增强仓储层错误一致性
      
      ## 仓储一致性修复 (fix-high-repository-consistency)
      - Group schema 添加 default_validity_days 字段
      - Account schema 添加 proxy edge 关联
      - 新增 UsageLog ent schema 定义
      - 修复 UpdateBalance/UpdateConcurrency 受影响行数校验
      
      ## 数据卫生修复 (fix-medium-data-hygiene)
      - UserSubscription 添加软删除支持 (SoftDeleteMixin)
      - RedeemCode/Setting 添加硬删除策略文档
      - account_groups/user_allowed_groups 的 created_at 声明 timestamptz
      - 停止写入 legacy users.allowed_groups 列
      - 新增迁移: 011-014 (索引优化、软删除、孤立数据审计、列清理)
      
      ## 测试补充
      - 添加 UserSubscription 软删除测试
      - 添加迁移回归测试
      - 添加 NotFound 错误测试
      
      🤖 Generated with [Claude Code](https://claude.com/claude-code
      
      )
      Co-Authored-By: default avatarClaude Opus 4.5 <noreply@anthropic.com>
      5906f9ab
    • yangjianbo's avatar
      fix(网关): 防止连接池缓存失控 · 820bb16c
      yangjianbo authored
      超限且无可淘汰条目时拒绝新建
      
      规范化代理地址并更新失败时的访问时间
      
      补充连接池上限与代理规范化测试
      820bb16c
    • yangjianbo's avatar
      perf(网关): 实现上游账号连接池隔离 · d1c98896
      yangjianbo authored
      新增隔离策略与连接池缓存回收
      
      连接池大小跟随账号并发并处理代理切换
      
      同步配置默认值与示例并补充测试
      d1c98896
    • yangjianbo's avatar
      fix(服务): 修复system判定、统计时区与缓存日志 · 3d7f8e4b
      yangjianbo authored
      - system 字段存在即视为显式提供,避免 null 触发默认注入
      - 日统计分组显式使用应用时区,缺失时从 TZ 回退到 UTC
      - 缓存写入队列丢弃日志节流汇总,关键任务同步回退
      
      测试: go test ./internal/service -run TestBillingCacheServiceQueueHighLoad
      3d7f8e4b
    • yangjianbo's avatar
      perf(后端): 完成性能优化与连接池配置 · 7efa8b54
      yangjianbo authored
      新增 DB/Redis 连接池配置与校验,并补充单测
      
      网关请求体大小限制与 413 处理
      
      HTTP/req 客户端池化并调整上游连接池默认值
      
      并发槽位改为 ZSET+Lua 与指数退避
      
      用量统计改 SQL 聚合并新增索引迁移
      
      计费缓存写入改工作池并补测试/基准
      
      测试: 在 backend/ 下运行 go test ./...
      7efa8b54
  2. 30 Dec, 2025 6 commits
    • shaw's avatar
      fix(仓储): 修复 BatchUpdateLastUsed 时间戳类型不匹配 · 4319cf7f
      shaw authored
      在原生 SQL 的 CASE WHEN 语句中,PostgreSQL 无法自动推断占位符参数类型,
      导致 time.Time 被当作 text 类型处理,与 last_used_at 列的 timestamptz 类型不匹配。
      
      添加显式类型转换 ::timestamptz 解决此问题。
      4319cf7f
    • yangjianbo's avatar
      fix(仓储): 规范 rows.Close 错误回传 · 8cb2d3b3
      yangjianbo authored
      统一 usage_log_repo 查询的 Close 错误处理,避免\n成功路径吞掉关闭失败
      
      scanSingleRow 使用 errors.Join 合并 Close 错误,\n保留 ErrNoRows 可判定
      
      测试: make -C backend test-unit
      8cb2d3b3
    • shaw's avatar
      style: 修复 gofmt 格式化问题 · 3d296d88
      shaw authored
      格式化以下测试文件以符合 Go 代码风格规范:
      - fixtures_integration_test.go
      - user_repo_integration_test.go
      - api_key_service_delete_test.go
      3d296d88
    • yangjianbo's avatar
      fix(仓储): 修复查询关闭错误并迁移集成测试 · aacbc98a
      yangjianbo authored
      修复 rows.Close 失败时的错误返回逻辑
      迁移网关路由集成测试到 ent 事务基建
      补齐仓储接口变更对应的测试桩方法
      新增 backend/Makefile 统一测试命令
      测试: GOTOOLCHAIN=go1.24.11 go test ./...
      测试: golangci-lint run ./... --timeout=5m
      测试: make test-integration
      aacbc98a
    • yangjianbo's avatar
      fix(仓库): 使用 ent 实现账号调度查询 · b9a753cd
      yangjianbo authored
      替换 gorm 查询并复用分组过滤逻辑,避免编译错误
      b9a753cd
    • shaw's avatar
      fix: 修复默认分组初始化导致启动失败的问题 · fb883f00
      shaw authored
      - 标准版不再创建默认分组,简易模式保持创建
      - 简易模式下删除默认分组后重启自动恢复(而非报唯一键冲突)
      - AutoMigrate 函数增加 runMode 参数以区分运行模式
      fb883f00
  3. 29 Dec, 2025 10 commits
  4. 28 Dec, 2025 5 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
    • 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
    • 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
  5. 27 Dec, 2025 5 commits