// Code generated by sqlc. DO NOT EDIT. // versions: // sqlc v1.31.1 // source: task_usage.sql package db import ( "context" "github.com/jackc/pgx/v5/pgtype" ) const getIssueUsageSummary = `-- name: GetIssueUsageSummary :one SELECT COALESCE(SUM(tu.input_tokens), 0)::bigint AS total_input_tokens, COALESCE(SUM(tu.output_tokens), 0)::bigint AS total_output_tokens, COALESCE(SUM(tu.cache_read_tokens), 0)::bigint AS total_cache_read_tokens, COALESCE(SUM(tu.cache_write_tokens), 0)::bigint AS total_cache_write_tokens, COUNT(DISTINCT tu.task_id)::int AS task_count FROM task_usage tu JOIN agent_task_queue atq ON atq.id = tu.task_id WHERE atq.issue_id = $1 ` type GetIssueUsageSummaryRow struct { TotalInputTokens int64 `json:"total_input_tokens"` TotalOutputTokens int64 `json:"total_output_tokens"` TotalCacheReadTokens int64 `json:"total_cache_read_tokens"` TotalCacheWriteTokens int64 `json:"total_cache_write_tokens"` TaskCount int32 `json:"task_count"` } func (q *Queries) GetIssueUsageSummary(ctx context.Context, issueID pgtype.UUID) (GetIssueUsageSummaryRow, error) { row := q.db.QueryRow(ctx, getIssueUsageSummary, issueID) var i GetIssueUsageSummaryRow err := row.Scan( &i.TotalInputTokens, &i.TotalOutputTokens, &i.TotalCacheReadTokens, &i.TotalCacheWriteTokens, &i.TaskCount, ) return i, err } const getTaskUsage = `-- name: GetTaskUsage :many SELECT id, task_id, provider, model, input_tokens, output_tokens, cache_read_tokens, cache_write_tokens, created_at, updated_at FROM task_usage WHERE task_id = $1 ORDER BY model ` func (q *Queries) GetTaskUsage(ctx context.Context, taskID pgtype.UUID) ([]TaskUsage, error) { rows, err := q.db.Query(ctx, getTaskUsage, taskID) if err != nil { return nil, err } defer rows.Close() items := []TaskUsage{} for rows.Next() { var i TaskUsage if err := rows.Scan( &i.ID, &i.TaskID, &i.Provider, &i.Model, &i.InputTokens, &i.OutputTokens, &i.CacheReadTokens, &i.CacheWriteTokens, &i.CreatedAt, &i.UpdatedAt, ); err != nil { return nil, err } items = append(items, i) } if err := rows.Err(); err != nil { return nil, err } return items, nil } const listDashboardAgentRunTime = `-- name: ListDashboardAgentRunTime :many SELECT atq.agent_id, COALESCE( SUM(EXTRACT(EPOCH FROM (atq.completed_at - atq.started_at)))::bigint, 0 )::bigint AS total_seconds, COUNT(*)::int AS task_count, COUNT(*) FILTER (WHERE atq.status = 'failed')::int AS failed_count FROM agent_task_queue atq JOIN agent a ON a.id = atq.agent_id LEFT JOIN issue i ON i.id = atq.issue_id WHERE a.workspace_id = $1 AND atq.status IN ('completed', 'failed') AND atq.started_at IS NOT NULL AND atq.completed_at IS NOT NULL AND atq.completed_at >= $2::timestamptz AND ($3::uuid IS NULL OR i.project_id = $3) GROUP BY atq.agent_id ORDER BY total_seconds DESC ` type ListDashboardAgentRunTimeParams struct { WorkspaceID pgtype.UUID `json:"workspace_id"` Since pgtype.Timestamptz `json:"since"` ProjectID pgtype.UUID `json:"project_id"` } type ListDashboardAgentRunTimeRow struct { AgentID pgtype.UUID `json:"agent_id"` TotalSeconds int64 `json:"total_seconds"` TaskCount int32 `json:"task_count"` FailedCount int32 `json:"failed_count"` } // Per-agent total task run time and task count for the workspace, optionally // scoped to a single project. Counts only terminal runs (completed or failed) // with both started_at and completed_at populated — queued/running tasks have // no finite duration. Anchored on completed_at so the window matches the // token cost window (which is anchored on tu.created_at, ~= completion time). // // No date bucketing, so no @tz — but @since is the viewer's local // start-of-day-(N) so the "last N days" window lines up with the per-agent // cost card; passed straight through without re-truncation. func (q *Queries) ListDashboardAgentRunTime(ctx context.Context, arg ListDashboardAgentRunTimeParams) ([]ListDashboardAgentRunTimeRow, error) { rows, err := q.db.Query(ctx, listDashboardAgentRunTime, arg.WorkspaceID, arg.Since, arg.ProjectID) if err != nil { return nil, err } defer rows.Close() items := []ListDashboardAgentRunTimeRow{} for rows.Next() { var i ListDashboardAgentRunTimeRow if err := rows.Scan( &i.AgentID, &i.TotalSeconds, &i.TaskCount, &i.FailedCount, ); err != nil { return nil, err } items = append(items, i) } if err := rows.Err(); err != nil { return nil, err } return items, nil } const listDashboardRunTimeDaily = `-- name: ListDashboardRunTimeDaily :many SELECT DATE(atq.completed_at AT TIME ZONE $2::text) AS date, COALESCE( SUM(EXTRACT(EPOCH FROM (atq.completed_at - atq.started_at)))::bigint, 0 )::bigint AS total_seconds, COUNT(*)::int AS task_count, COUNT(*) FILTER (WHERE atq.status = 'failed')::int AS failed_count FROM agent_task_queue atq JOIN agent a ON a.id = atq.agent_id LEFT JOIN issue i ON i.id = atq.issue_id WHERE a.workspace_id = $1 AND atq.status IN ('completed', 'failed') AND atq.started_at IS NOT NULL AND atq.completed_at IS NOT NULL AND atq.completed_at >= $3::timestamptz AND ($4::uuid IS NULL OR i.project_id = $4) GROUP BY DATE(atq.completed_at AT TIME ZONE $2::text) ORDER BY DATE(atq.completed_at AT TIME ZONE $2::text) DESC ` type ListDashboardRunTimeDailyParams struct { WorkspaceID pgtype.UUID `json:"workspace_id"` Tz string `json:"tz"` Since pgtype.Timestamptz `json:"since"` ProjectID pgtype.UUID `json:"project_id"` } type ListDashboardRunTimeDailyRow struct { Date pgtype.Date `json:"date"` TotalSeconds int64 `json:"total_seconds"` TaskCount int32 `json:"task_count"` FailedCount int32 `json:"failed_count"` } // Daily per-date run time + task counts for the workspace, optionally // scoped to a single project. Powers the workspace dashboard's "Time" // and "Tasks" metrics on the same toggle as Tokens / Cost. Bucketed by // completed_at (terminal time) sliced into calendar days under the // caller-supplied @tz — same Viewing-tz treatment as ListDashboardUsageDaily // so the Time / Tasks tabs cut their day boundary identically to the // Cost / Tokens tabs (a viewer east of UTC would otherwise see the four // tabs disagree on a "1d" window). Only terminal tasks (completed or // failed) with both started_at and completed_at populated contribute. // // @since is already the viewer's local start-of-day-(N) (parseSinceParamInTZ) // — passed straight through, NOT re-truncated; see ListDashboardUsageDaily. func (q *Queries) ListDashboardRunTimeDaily(ctx context.Context, arg ListDashboardRunTimeDailyParams) ([]ListDashboardRunTimeDailyRow, error) { rows, err := q.db.Query(ctx, listDashboardRunTimeDaily, arg.WorkspaceID, arg.Tz, arg.Since, arg.ProjectID, ) if err != nil { return nil, err } defer rows.Close() items := []ListDashboardRunTimeDailyRow{} for rows.Next() { var i ListDashboardRunTimeDailyRow if err := rows.Scan( &i.Date, &i.TotalSeconds, &i.TaskCount, &i.FailedCount, ); err != nil { return nil, err } items = append(items, i) } if err := rows.Err(); err != nil { return nil, err } return items, nil } const listDashboardUsageByAgent = `-- name: ListDashboardUsageByAgent :many SELECT agent_id, model, SUM(input_tokens)::bigint AS input_tokens, SUM(output_tokens)::bigint AS output_tokens, SUM(cache_read_tokens)::bigint AS cache_read_tokens, SUM(cache_write_tokens)::bigint AS cache_write_tokens, SUM(task_count)::int AS task_count FROM task_usage_hourly WHERE workspace_id = $1 AND bucket_hour >= $2::timestamptz AND ($3::uuid IS NULL OR project_id = $3) GROUP BY agent_id, model ORDER BY agent_id, model ` type ListDashboardUsageByAgentParams struct { WorkspaceID pgtype.UUID `json:"workspace_id"` Since pgtype.Timestamptz `json:"since"` ProjectID pgtype.UUID `json:"project_id"` } type ListDashboardUsageByAgentRow struct { AgentID pgtype.UUID `json:"agent_id"` Model string `json:"model"` InputTokens int64 `json:"input_tokens"` OutputTokens int64 `json:"output_tokens"` CacheReadTokens int64 `json:"cache_read_tokens"` CacheWriteTokens int64 `json:"cache_write_tokens"` TaskCount int32 `json:"task_count"` } // Per-(agent, model) token aggregates from `task_usage_hourly`. No // date grouping in the result, so this query takes no `@tz` — the // @since cutoff is a raw timestamptz the Go layer has already computed // in the viewer's tz. Model dimension is preserved so the client can // compute cost from its per-model pricing table; the client folds rows // by agent for the "by agent" list on the dashboard. // // task_count is summed across hourly buckets — one task that spans // multiple hours lands in multiple buckets, so this over-counts by // hour the same way the daily version over-counted by day. The // frontend prefers `ListDashboardAgentRunTime` for the user-facing // "tasks" column, so this stays informational only. func (q *Queries) ListDashboardUsageByAgent(ctx context.Context, arg ListDashboardUsageByAgentParams) ([]ListDashboardUsageByAgentRow, error) { rows, err := q.db.Query(ctx, listDashboardUsageByAgent, arg.WorkspaceID, arg.Since, arg.ProjectID) if err != nil { return nil, err } defer rows.Close() items := []ListDashboardUsageByAgentRow{} for rows.Next() { var i ListDashboardUsageByAgentRow if err := rows.Scan( &i.AgentID, &i.Model, &i.InputTokens, &i.OutputTokens, &i.CacheReadTokens, &i.CacheWriteTokens, &i.TaskCount, ); err != nil { return nil, err } items = append(items, i) } if err := rows.Err(); err != nil { return nil, err } return items, nil } const listDashboardUsageDaily = `-- name: ListDashboardUsageDaily :many SELECT DATE(bucket_hour AT TIME ZONE $2::text) AS date, model, SUM(input_tokens)::bigint AS input_tokens, SUM(output_tokens)::bigint AS output_tokens, SUM(cache_read_tokens)::bigint AS cache_read_tokens, SUM(cache_write_tokens)::bigint AS cache_write_tokens, SUM(task_count)::int AS task_count FROM task_usage_hourly WHERE workspace_id = $1 AND bucket_hour >= $3::timestamptz AND ($4::uuid IS NULL OR project_id = $4) GROUP BY DATE(bucket_hour AT TIME ZONE $2::text), model ORDER BY DATE(bucket_hour AT TIME ZONE $2::text) DESC, model ` type ListDashboardUsageDailyParams struct { WorkspaceID pgtype.UUID `json:"workspace_id"` Tz string `json:"tz"` Since pgtype.Timestamptz `json:"since"` ProjectID pgtype.UUID `json:"project_id"` } type ListDashboardUsageDailyRow struct { Date pgtype.Date `json:"date"` Model string `json:"model"` InputTokens int64 `json:"input_tokens"` OutputTokens int64 `json:"output_tokens"` CacheReadTokens int64 `json:"cache_read_tokens"` CacheWriteTokens int64 `json:"cache_write_tokens"` TaskCount int32 `json:"task_count"` } // Daily per-(date, model) token aggregates for the workspace, served // from the UTC-bucketed `task_usage_hourly` table and // sliced to calendar days under the caller-supplied @tz. Optionally // scoped to a single project via sqlc.narg('project_id'). Powers the // workspace dashboard's daily cost chart. // The viewer's tz is applied here at query time, so a viewer in // Asia/Shanghai gets their "today" cut at +08 and one in // America/Los_Angeles gets theirs at -08 against the same UTC rows. // // @since is already the viewer's local start-of-day-(N) as a UTC // instant (computed by parseSinceParamInTZ). It must NOT be re-truncated // with DATE_TRUNC here — DATE_TRUNC operates in the session tz and would // snap the cutoff back to UTC midnight, dragging in an extra partial // local day for any non-UTC viewer. func (q *Queries) ListDashboardUsageDaily(ctx context.Context, arg ListDashboardUsageDailyParams) ([]ListDashboardUsageDailyRow, error) { rows, err := q.db.Query(ctx, listDashboardUsageDaily, arg.WorkspaceID, arg.Tz, arg.Since, arg.ProjectID, ) if err != nil { return nil, err } defer rows.Close() items := []ListDashboardUsageDailyRow{} for rows.Next() { var i ListDashboardUsageDailyRow if err := rows.Scan( &i.Date, &i.Model, &i.InputTokens, &i.OutputTokens, &i.CacheReadTokens, &i.CacheWriteTokens, &i.TaskCount, ); err != nil { return nil, err } items = append(items, i) } if err := rows.Err(); err != nil { return nil, err } return items, nil } const upsertTaskUsage = `-- name: UpsertTaskUsage :exec INSERT INTO task_usage (task_id, provider, model, input_tokens, output_tokens, cache_read_tokens, cache_write_tokens, updated_at) VALUES ($1, $2, $3, $4, $5, $6, $7, now()) ON CONFLICT (task_id, provider, model) DO UPDATE SET input_tokens = EXCLUDED.input_tokens, output_tokens = EXCLUDED.output_tokens, cache_read_tokens = EXCLUDED.cache_read_tokens, cache_write_tokens = EXCLUDED.cache_write_tokens, updated_at = now() ` type UpsertTaskUsageParams struct { TaskID pgtype.UUID `json:"task_id"` Provider string `json:"provider"` Model string `json:"model"` InputTokens int64 `json:"input_tokens"` OutputTokens int64 `json:"output_tokens"` CacheReadTokens int64 `json:"cache_read_tokens"` CacheWriteTokens int64 `json:"cache_write_tokens"` } // Bumps `updated_at` on INSERT and on conflict so the hourly-rollup worker // detects the row as dirty and re-aggregates its bucket. // Without the conflict-side bump, a correction to historical token counts // would never propagate to the rollup. func (q *Queries) UpsertTaskUsage(ctx context.Context, arg UpsertTaskUsageParams) error { _, err := q.db.Exec(ctx, upsertTaskUsage, arg.TaskID, arg.Provider, arg.Model, arg.InputTokens, arg.OutputTokens, arg.CacheReadTokens, arg.CacheWriteTokens, ) return err }