1. 16 Mar, 2026 2 commits
    • erio's avatar
      feat(dashboard): add per-user drill-down for group, model, and endpoint distributions · 4b41e898
      erio authored
      Click on a group name, model name, or endpoint name in the distribution
      tables to expand and show per-user usage breakdown (requests, tokens,
      actual cost, standard cost).
      
      Backend: new GET /admin/dashboard/user-breakdown API with group_id,
      model, endpoint, endpoint_type filters.
      Frontend: clickable rows with expand/collapse sub-table in all three
      distribution charts.
      4b41e898
    • erio's avatar
      fix(billing): add window expiration check to Redis rate limit Lua script · 67c05062
      erio authored
      The updateRateLimitUsageScript Lua script previously performed
      unconditional HINCRBYFLOAT on all usage counters without checking
      whether the rate limit window had expired. This caused usage to
      accumulate across window boundaries in Redis while the DB correctly
      reset on expiration, leading to incorrect 429 rate limiting that
      could persist for up to 24 hours.
      
      The Lua script now checks each window timestamp before incrementing:
      - If the window has expired, usage is reset to the current cost and
        the window timestamp is updated (matching DB-side semantics)
      - If the window is still valid, usage is accumulated normally
      
      This also resolves the async race condition where stale HINCRBYFLOAT
      tasks from the worker queue could pollute a freshly rebuilt cache
      after invalidation, since the script now self-corrects expired windows.
      
      Closes #1049
      67c05062
  2. 15 Mar, 2026 3 commits
  3. 14 Mar, 2026 4 commits
  4. 13 Mar, 2026 1 commit
    • wucm667's avatar
      feat: 账号配额支持固定时间重置模式 · 5b850059
      wucm667 authored
      - 后端新增 rolling/fixed 两种配额重置模式,支持日配额和周配额
      - fixed 模式下可配置重置时刻(小时)、重置星期几(周配额)及时区(IANA)
      - 在 account_repo.go 中使用 SQL 表达式适配两种模式的过期判断与重置时间推进
      - 新增 ComputeQuotaResetAt / ValidateQuotaResetConfig 等辅助函数
      - DTO 层新增相关字段并在 mappers 中完整映射
      - 前端 QuotaLimitCard 新增 rolling/fixed 切换 UI、时区选择器
      - CreateAccountModal / EditAccountModal 透传新配置字段
      - i18n(zh/en)同步新增相关翻译词条
      5b850059
  5. 12 Mar, 2026 9 commits
  6. 11 Mar, 2026 6 commits
  7. 09 Mar, 2026 3 commits
  8. 07 Mar, 2026 8 commits
  9. 05 Mar, 2026 4 commits
    • erio's avatar
    • erio's avatar
      feat: add quota limit for API key accounts · 05527b13
      erio authored
      - Add configurable spending limit (quota_limit) for apikey-type accounts
      - Atomic quota accumulation via PostgreSQL JSONB operations on TotalCost
      - Scheduler filters out over-quota accounts with outbox-triggered snapshot refresh
      - Display quota usage ($used / $limit) in account capacity column
      - Add "Reset Quota" action in account menu to reset usage to zero
      - Editing account settings preserves quota_used (no accidental reset)
      - Covers all 3 billing paths: Anthropic, Gemini, OpenAI RecordUsage
      
      chore: bump version to 0.1.90.4
      05527b13
    • erio's avatar
      fix: include upstream error details in usage API error response · 9117c2a4
      erio authored
      When FetchUsageWithOptions receives a non-200 response from the
      Anthropic API (e.g. 429 Rate Limited, 401 Unauthorized), the error
      was wrapped with fmt.Errorf which infraerrors.FromError cannot
      recognize, causing a generic "internal error" message with no details.
      
      Replace fmt.Errorf with infraerrors.New(500, "UPSTREAM_ERROR", msg)
      so the upstream error details (status code + body) are included in
      the 500 response message. The HTTP status remains 500 to avoid
      interfering with frontend auth routing (e.g. 401 would trigger
      JWT expiry redirect).
      9117c2a4
    • shaw's avatar
      33bae6f4