2025-09-09 14:37:32 +05:30
|
|
|
from datetime import datetime, timezone
|
|
|
|
|
from typing import Optional
|
|
|
|
|
from zoneinfo import ZoneInfo
|
|
|
|
|
|
|
|
|
|
from dateutil.relativedelta import relativedelta
|
|
|
|
|
from sqlalchemy import Date, and_, cast, func, select
|
|
|
|
|
from sqlalchemy.dialects.postgresql import insert
|
|
|
|
|
from sqlalchemy.orm import joinedload
|
|
|
|
|
|
|
|
|
|
from api.db.base_client import BaseDBClient
|
|
|
|
|
from api.db.filters import apply_workflow_run_filters
|
|
|
|
|
from api.db.models import (
|
2026-06-09 16:10:26 +05:30
|
|
|
OrganizationConfigurationModel,
|
2025-09-09 14:37:32 +05:30
|
|
|
OrganizationModel,
|
|
|
|
|
OrganizationUsageCycleModel,
|
|
|
|
|
UserConfigurationModel,
|
|
|
|
|
UserModel,
|
|
|
|
|
WorkflowModel,
|
|
|
|
|
WorkflowRunModel,
|
|
|
|
|
)
|
feat: UI refresh and user onboarding (#430)
* docs: design spec for lead-gen surfaces (Credits & Billing, Hire-an-Expert, Top-up, Enterprise)
Add brainstorming spec for: sidebar OBSERVE→MANAGE rename + Credits & Billing
link + Hire-an-Expert footer button; new /billing page with extracted Dograh
Model Credits card + CTAs; Top-up / Hire-an-Expert / Enterprise intake modals
with inline math captcha; and a workflow-builder Hire-an-Expert nudge. Frontend
only; submissions fire PostHog events via a submitLead() seam for a future
MongoDB endpoint. Also gitignore .superpowers/ brainstorm mockups.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* docs: implementation plan for user-onboarding lead-gen surfaces
14 bite-sized tasks: PostHog events, shared helpers (field options,
work-email blocklist, submitLead seam, math captcha), three intake modals
(enterprise/hire/top-up), LeadFormsProvider context, AppLayout mount, sidebar
MANAGE rename + Credits & Billing link + footer Hire button, extracted
DograhCreditsCard, /billing page, credits removal from Agent Runs, builder
nudge, and a full verification/dogfood pass.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): register PostHog events for lead-gen surfaces
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): shared field options, work-email validation, and submit seam
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): inline math captcha field
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): enterprise intake modal
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): hire-an-expert modal with enterprise link
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): top-up modal with >20k volume-pricing gate
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): shared lead-forms context provider
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): mount LeadFormsProvider in app layout
Wrap the sidebar branch of AppLayout with LeadFormsProvider so the shared
lead modals are available to the sidebar, billing card, and builder nudge.
Includes eslint import-order auto-fixes in TopUpModal and LeadFormsContext.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): rename OBSERVE to MANAGE, add Credits & Billing link and Hire-an-Expert footer button
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): extract DograhCreditsCard with top-up + hire CTAs
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): add Credits & Billing page
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* refactor(lead-gen): move Dograh Model Credits card out of Agent Runs to /billing
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): delayed Hire-an-Expert nudge on the workflow builder
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* ci(ui): add lint:lead-flow guard script
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(ui): restructure lead forms, self-serve Buy Credits, dialog blur
Revised lead-capture surfaces and credits bar:
- Dialog overlay gains backdrop blur (bg-black/60 backdrop-blur-sm).
- Shared primitives: LeadModalShell (icon/eyebrow header, scrollable body,
sticky footer, trust-line slot), PhoneField (react-international-phone,
dark, E.164 out), FormTrustLine ("Average response: under 10 minutes...").
- HireExpertModal: Name, Company, Job title, agent goal, Phone (required),
monthly volume. EnterpriseModal: + work email (required logged-out),
conditional deployment (yes/no/maybe, source-gated), agent goal.
OnboardingModal: drop useCase. Phone mandatory except onboarding.
- Volume buckets match the backend qualifier (0-5k/5k-100k/100k+/not-sure).
- Delete TopUpModal; DograhCreditsCard now self-serve Buy Credits (amount
chips $5/$10/$25/$50/$100 + custom min $5 → startTopUp seam) + Hire an
Expert + dashed custom-pricing link opening Enterprise (billing_custom_pricing).
- PostHog events: drop topup_*, add buy_credits_clicked,
buy_credits_amount_selected, custom_pricing_clicked. LeadFormsContext
drops topup; LeadKind/LeadSource updated.
- Introduce a single --cta warm accent token (CTAs + focus rings only).
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(ui): split-screen auth + enterprise CTA + dark theme default
- AuthShell: dark two-column auth layout (brand/value panel with CSS-only
waveform motif + proof points + Bland-style enterprise CTA block on the
left, zinc-900 form card on the right; single-column on mobile).
- AuthEnterpriseCTA: "Talk to our team" → dograh.com/contact?intent=enterprise.
- stack-theme: dark StackTheme token overrides synced to globals.css.
- page.tsx: wrap StackHandler (non-fullPage) in AuthShell + StackTheme;
local-auth fallback preserved inside the shell. BackButton slimmed for the card.
- Dark locked as default: <html className="dark">, next-themes ThemeProvider
(defaultTheme="dark", enableSystem=false); inline no-FOUC script defaults dark.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* ui rezig, onboarding, billing, hire us & on prem cues
* ui changes
* chore: update comment
* chore: untrack docs/superpowers and gitignore it
* feat: refactor user configuration table
* feat(ui): 'check your email' confirmation on lead forms
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* added email and country in form submissions
* chore: update leads api
* fix: wrap dograh model config in card
---------
Co-authored-by: Pritesh <pritesh@dograh.com>
Co-authored-by: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
2026-06-17 19:49:33 +05:30
|
|
|
from api.enums import OrganizationConfigurationKey, UserConfigurationKey
|
2026-06-12 14:55:30 +05:30
|
|
|
from api.schemas.ai_model_configuration import EffectiveAIModelConfiguration
|
2026-06-16 15:19:49 +05:30
|
|
|
from api.utils.recording_artifacts import get_recording_storage_key
|
2025-09-09 14:37:32 +05:30
|
|
|
|
|
|
|
|
|
|
|
|
|
class OrganizationUsageClient(BaseDBClient):
|
2026-06-12 14:55:30 +05:30
|
|
|
"""Client for managing organization usage reporting aggregates."""
|
2025-09-09 14:37:32 +05:30
|
|
|
|
|
|
|
|
async def get_or_create_current_cycle(
|
|
|
|
|
self, organization_id: int, session=None
|
|
|
|
|
) -> OrganizationUsageCycleModel:
|
|
|
|
|
"""Get or create the current usage cycle for an organization.
|
|
|
|
|
|
|
|
|
|
Args:
|
|
|
|
|
organization_id: The organization ID
|
|
|
|
|
session: Optional session to use for the operation. If provided,
|
|
|
|
|
the caller is responsible for committing.
|
|
|
|
|
"""
|
|
|
|
|
if session is None:
|
|
|
|
|
async with self.async_session() as session:
|
|
|
|
|
return await self._get_or_create_current_cycle_impl(
|
|
|
|
|
organization_id, session, commit=True
|
|
|
|
|
)
|
|
|
|
|
else:
|
|
|
|
|
return await self._get_or_create_current_cycle_impl(
|
|
|
|
|
organization_id, session, commit=False
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
async def _get_or_create_current_cycle_impl(
|
|
|
|
|
self, organization_id: int, session, commit: bool
|
|
|
|
|
) -> OrganizationUsageCycleModel:
|
|
|
|
|
"""Internal implementation for get_or_create_current_cycle."""
|
2026-06-12 14:55:30 +05:30
|
|
|
period_start, period_end = self._calculate_current_period()
|
2025-09-09 14:37:32 +05:30
|
|
|
|
|
|
|
|
# Try to get existing cycle
|
|
|
|
|
cycle_result = await session.execute(
|
|
|
|
|
select(OrganizationUsageCycleModel).where(
|
|
|
|
|
and_(
|
|
|
|
|
OrganizationUsageCycleModel.organization_id == organization_id,
|
|
|
|
|
OrganizationUsageCycleModel.period_start == period_start,
|
|
|
|
|
OrganizationUsageCycleModel.period_end == period_end,
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
cycle = cycle_result.scalar_one_or_none()
|
|
|
|
|
|
|
|
|
|
if cycle:
|
|
|
|
|
return cycle
|
|
|
|
|
|
|
|
|
|
# Create new cycle if it doesn't exist
|
|
|
|
|
stmt = insert(OrganizationUsageCycleModel).values(
|
|
|
|
|
organization_id=organization_id,
|
|
|
|
|
period_start=period_start,
|
|
|
|
|
period_end=period_end,
|
2026-06-12 14:55:30 +05:30
|
|
|
# Deprecated non-null column retained for historical schema compatibility.
|
|
|
|
|
quota_dograh_tokens=0,
|
2025-09-09 14:37:32 +05:30
|
|
|
)
|
|
|
|
|
# Handle concurrent inserts gracefully
|
|
|
|
|
stmt = stmt.on_conflict_do_nothing(
|
|
|
|
|
index_elements=["organization_id", "period_start", "period_end"]
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
await session.execute(stmt)
|
|
|
|
|
|
|
|
|
|
if commit:
|
|
|
|
|
await session.commit()
|
|
|
|
|
|
|
|
|
|
# Fetch the created cycle
|
|
|
|
|
cycle_result = await session.execute(
|
|
|
|
|
select(OrganizationUsageCycleModel).where(
|
|
|
|
|
and_(
|
|
|
|
|
OrganizationUsageCycleModel.organization_id == organization_id,
|
|
|
|
|
OrganizationUsageCycleModel.period_start == period_start,
|
|
|
|
|
OrganizationUsageCycleModel.period_end == period_end,
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
return cycle_result.scalar_one()
|
|
|
|
|
|
|
|
|
|
async def get_current_usage(self, organization_id: int) -> dict:
|
2026-06-12 14:55:30 +05:30
|
|
|
"""Get current reporting-period usage information."""
|
2025-09-09 14:37:32 +05:30
|
|
|
async with self.async_session() as session:
|
|
|
|
|
org_result = await session.execute(
|
|
|
|
|
select(OrganizationModel).where(OrganizationModel.id == organization_id)
|
|
|
|
|
)
|
|
|
|
|
org = org_result.scalar_one()
|
|
|
|
|
|
|
|
|
|
# Get or create current cycle within the same session
|
|
|
|
|
cycle = await self._get_or_create_current_cycle_impl(
|
|
|
|
|
organization_id, session, commit=False
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
result = {
|
|
|
|
|
"period_start": cycle.period_start.isoformat(),
|
|
|
|
|
"period_end": cycle.period_end.isoformat(),
|
|
|
|
|
"used_dograh_tokens": cycle.used_dograh_tokens,
|
|
|
|
|
"total_duration_seconds": cycle.total_duration_seconds,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# Add USD fields if organization has pricing
|
|
|
|
|
if org.price_per_second_usd is not None:
|
|
|
|
|
result["used_amount_usd"] = cycle.used_amount_usd or 0
|
|
|
|
|
result["currency"] = "USD"
|
|
|
|
|
result["price_per_second_usd"] = org.price_per_second_usd
|
|
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
|
async def get_usage_history(
|
|
|
|
|
self,
|
|
|
|
|
organization_id: int,
|
|
|
|
|
start_date: Optional[datetime] = None,
|
|
|
|
|
end_date: Optional[datetime] = None,
|
|
|
|
|
limit: int = 50,
|
|
|
|
|
offset: int = 0,
|
|
|
|
|
filters: Optional[list[dict]] = None,
|
2026-05-08 14:48:53 +05:30
|
|
|
) -> tuple[list[dict], int, float, int]:
|
2025-09-09 14:37:32 +05:30
|
|
|
"""Get paginated workflow runs with usage for an organization."""
|
|
|
|
|
async with self.async_session() as session:
|
|
|
|
|
query = (
|
|
|
|
|
select(WorkflowRunModel)
|
|
|
|
|
.join(WorkflowModel, WorkflowRunModel.workflow_id == WorkflowModel.id)
|
|
|
|
|
.join(UserModel, WorkflowModel.user_id == UserModel.id)
|
|
|
|
|
.where(
|
|
|
|
|
UserModel.selected_organization_id == organization_id,
|
2026-06-12 14:55:30 +05:30
|
|
|
WorkflowRunModel.usage_info.isnot(None),
|
2025-09-09 14:37:32 +05:30
|
|
|
)
|
|
|
|
|
.order_by(WorkflowRunModel.created_at.desc())
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# Apply date filters if provided
|
|
|
|
|
if start_date:
|
|
|
|
|
query = query.where(WorkflowRunModel.created_at >= start_date)
|
|
|
|
|
if end_date:
|
|
|
|
|
query = query.where(WorkflowRunModel.created_at <= end_date)
|
|
|
|
|
|
|
|
|
|
# Only allow specific filters for usage history endpoint
|
|
|
|
|
# This ensures security and prevents unexpected filter attributes
|
2026-05-08 14:48:53 +05:30
|
|
|
allowed_filters = {
|
|
|
|
|
"duration",
|
|
|
|
|
"dispositionCode",
|
|
|
|
|
"callerNumber",
|
|
|
|
|
"calledNumber",
|
|
|
|
|
"runId",
|
|
|
|
|
"workflowId",
|
|
|
|
|
"campaignId",
|
|
|
|
|
}
|
2025-09-09 14:37:32 +05:30
|
|
|
sanitized_filters = []
|
|
|
|
|
|
|
|
|
|
if filters:
|
|
|
|
|
for filter_item in filters:
|
|
|
|
|
attribute = filter_item.get("attribute")
|
|
|
|
|
|
|
|
|
|
# Only process allowed filters
|
|
|
|
|
if attribute in allowed_filters:
|
|
|
|
|
sanitized_filters.append(filter_item)
|
|
|
|
|
|
|
|
|
|
# Apply filters using the common filter function
|
|
|
|
|
query = apply_workflow_run_filters(query, sanitized_filters)
|
|
|
|
|
|
|
|
|
|
# Get total count
|
|
|
|
|
count_result = await session.execute(
|
|
|
|
|
select(func.count()).select_from(query.subquery())
|
|
|
|
|
)
|
|
|
|
|
total_count = count_result.scalar()
|
|
|
|
|
|
|
|
|
|
results = await session.execute(
|
|
|
|
|
query.options(joinedload(WorkflowRunModel.workflow))
|
|
|
|
|
.limit(limit)
|
|
|
|
|
.offset(offset)
|
|
|
|
|
)
|
|
|
|
|
runs = results.scalars().all()
|
|
|
|
|
|
|
|
|
|
# Format runs
|
|
|
|
|
formatted_runs = []
|
|
|
|
|
total_tokens = 0
|
|
|
|
|
total_duration_seconds = 0
|
|
|
|
|
for run in runs:
|
2026-06-12 14:55:30 +05:30
|
|
|
dograh_tokens = 0
|
|
|
|
|
call_duration = (run.usage_info or {}).get("call_duration_seconds", 0)
|
2025-09-09 14:37:32 +05:30
|
|
|
total_tokens += dograh_tokens
|
|
|
|
|
total_duration_seconds += int(round(call_duration))
|
|
|
|
|
|
2026-05-08 14:48:53 +05:30
|
|
|
ic = run.initial_context or {}
|
|
|
|
|
caller_number = ic.get("caller_number")
|
|
|
|
|
called_number = ic.get("called_number") or ic.get("phone_number")
|
|
|
|
|
# DEPRECATED: phone_number — use caller_number/called_number.
|
2026-04-17 11:47:20 +05:30
|
|
|
# Inbound runs only have caller_number/called_number; the
|
|
|
|
|
# caller_number is the customer. Outbound runs use the
|
|
|
|
|
# phone_number key written by the dispatchers.
|
|
|
|
|
if run.call_type == "inbound":
|
2026-05-08 14:48:53 +05:30
|
|
|
phone_number = caller_number
|
2026-04-17 11:47:20 +05:30
|
|
|
else:
|
|
|
|
|
phone_number = ic.get("phone_number")
|
2025-09-09 14:37:32 +05:30
|
|
|
|
|
|
|
|
# Extract disposition from gathered_context
|
|
|
|
|
disposition = None
|
|
|
|
|
if run.gathered_context:
|
|
|
|
|
disposition = run.gathered_context.get("mapped_call_disposition")
|
|
|
|
|
|
|
|
|
|
run_data = {
|
|
|
|
|
"id": run.id,
|
|
|
|
|
"workflow_id": run.workflow_id,
|
|
|
|
|
"workflow_name": run.workflow.name if run.workflow else None,
|
|
|
|
|
"name": run.name,
|
|
|
|
|
"created_at": run.created_at.isoformat(),
|
|
|
|
|
"dograh_token_usage": dograh_tokens,
|
|
|
|
|
"call_duration_seconds": int(round(call_duration)),
|
|
|
|
|
"recording_url": run.recording_url,
|
|
|
|
|
"transcript_url": run.transcript_url,
|
2026-06-16 15:19:49 +05:30
|
|
|
"user_recording_url": get_recording_storage_key(run.extra, "user"),
|
|
|
|
|
"bot_recording_url": get_recording_storage_key(run.extra, "bot"),
|
|
|
|
|
"extra": run.extra,
|
2026-05-26 13:24:12 +05:30
|
|
|
"public_access_token": run.public_access_token,
|
2025-09-09 14:37:32 +05:30
|
|
|
"phone_number": phone_number,
|
2026-05-08 14:48:53 +05:30
|
|
|
"caller_number": caller_number,
|
|
|
|
|
"called_number": called_number,
|
2026-04-17 11:47:20 +05:30
|
|
|
"call_type": run.call_type,
|
2026-05-21 15:20:02 +05:30
|
|
|
"mode": run.mode,
|
2025-09-09 14:37:32 +05:30
|
|
|
"disposition": disposition,
|
|
|
|
|
"initial_context": run.initial_context,
|
|
|
|
|
"gathered_context": run.gathered_context,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# Add USD cost if available in cost_info
|
|
|
|
|
if run.cost_info and "charge_usd" in run.cost_info:
|
|
|
|
|
run_data["charge_usd"] = run.cost_info["charge_usd"]
|
|
|
|
|
|
|
|
|
|
formatted_runs.append(run_data)
|
|
|
|
|
|
|
|
|
|
return formatted_runs, total_count, total_tokens, total_duration_seconds
|
|
|
|
|
|
2026-05-08 14:48:53 +05:30
|
|
|
async def get_usage_runs_for_report(
|
|
|
|
|
self,
|
|
|
|
|
organization_id: int,
|
|
|
|
|
start_date: Optional[datetime] = None,
|
|
|
|
|
end_date: Optional[datetime] = None,
|
|
|
|
|
filters: Optional[list[dict]] = None,
|
|
|
|
|
) -> list:
|
|
|
|
|
"""Get filtered runs for an organization-scoped usage CSV report.
|
|
|
|
|
|
|
|
|
|
Mirrors the filter allowlist used by `get_usage_history`, but selects
|
|
|
|
|
only the columns needed by `build_run_report_csv` and returns every
|
|
|
|
|
matching run (no pagination).
|
|
|
|
|
"""
|
|
|
|
|
async with self.async_session() as session:
|
|
|
|
|
query = (
|
|
|
|
|
select(
|
|
|
|
|
WorkflowRunModel.id,
|
|
|
|
|
WorkflowRunModel.workflow_id,
|
|
|
|
|
WorkflowRunModel.definition_id,
|
|
|
|
|
WorkflowRunModel.campaign_id,
|
|
|
|
|
WorkflowRunModel.created_at,
|
|
|
|
|
WorkflowRunModel.initial_context,
|
|
|
|
|
WorkflowRunModel.gathered_context,
|
|
|
|
|
WorkflowRunModel.cost_info,
|
2026-06-12 14:55:30 +05:30
|
|
|
WorkflowRunModel.usage_info,
|
2026-05-08 14:48:53 +05:30
|
|
|
WorkflowRunModel.public_access_token,
|
|
|
|
|
)
|
|
|
|
|
.join(WorkflowModel, WorkflowRunModel.workflow_id == WorkflowModel.id)
|
|
|
|
|
.join(UserModel, WorkflowModel.user_id == UserModel.id)
|
|
|
|
|
.where(
|
|
|
|
|
UserModel.selected_organization_id == organization_id,
|
2026-06-12 14:55:30 +05:30
|
|
|
WorkflowRunModel.usage_info.isnot(None),
|
2026-05-08 14:48:53 +05:30
|
|
|
)
|
|
|
|
|
.order_by(WorkflowRunModel.created_at.desc())
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
if start_date:
|
|
|
|
|
query = query.where(WorkflowRunModel.created_at >= start_date)
|
|
|
|
|
if end_date:
|
|
|
|
|
query = query.where(WorkflowRunModel.created_at <= end_date)
|
|
|
|
|
|
|
|
|
|
allowed_filters = {
|
|
|
|
|
"duration",
|
|
|
|
|
"dispositionCode",
|
|
|
|
|
"callerNumber",
|
|
|
|
|
"calledNumber",
|
|
|
|
|
"runId",
|
|
|
|
|
"workflowId",
|
|
|
|
|
"campaignId",
|
|
|
|
|
}
|
|
|
|
|
sanitized_filters = []
|
|
|
|
|
if filters:
|
|
|
|
|
for filter_item in filters:
|
|
|
|
|
if filter_item.get("attribute") in allowed_filters:
|
|
|
|
|
sanitized_filters.append(filter_item)
|
|
|
|
|
|
|
|
|
|
query = apply_workflow_run_filters(query, sanitized_filters)
|
|
|
|
|
|
|
|
|
|
result = await session.execute(query)
|
|
|
|
|
return list(result.all())
|
|
|
|
|
|
2025-09-09 14:37:32 +05:30
|
|
|
async def get_daily_usage_breakdown(
|
|
|
|
|
self,
|
|
|
|
|
organization_id: int,
|
|
|
|
|
start_date: datetime,
|
|
|
|
|
end_date: datetime,
|
|
|
|
|
price_per_second_usd: float,
|
|
|
|
|
user_id: Optional[int] = None,
|
|
|
|
|
) -> dict:
|
|
|
|
|
"""Get daily usage breakdown for an organization with pricing."""
|
|
|
|
|
|
|
|
|
|
async with self.async_session() as session:
|
2026-06-09 16:10:26 +05:30
|
|
|
# Get org timezone preference first, then fall back to legacy user config.
|
2025-09-09 14:37:32 +05:30
|
|
|
user_timezone = "UTC" # Default timezone
|
2026-06-09 16:10:26 +05:30
|
|
|
pref_result = await session.execute(
|
|
|
|
|
select(OrganizationConfigurationModel).where(
|
|
|
|
|
OrganizationConfigurationModel.organization_id == organization_id,
|
|
|
|
|
OrganizationConfigurationModel.key.in_(
|
|
|
|
|
[
|
|
|
|
|
OrganizationConfigurationKey.ORGANIZATION_PREFERENCES.value,
|
|
|
|
|
OrganizationConfigurationKey.MODEL_CONFIGURATION_PREFERENCES.value,
|
|
|
|
|
]
|
|
|
|
|
),
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
pref_rows = pref_result.scalars().all()
|
|
|
|
|
pref_by_key = {pref.key: pref for pref in pref_rows}
|
|
|
|
|
pref_obj = pref_by_key.get(
|
|
|
|
|
OrganizationConfigurationKey.ORGANIZATION_PREFERENCES.value
|
|
|
|
|
) or pref_by_key.get(
|
|
|
|
|
OrganizationConfigurationKey.MODEL_CONFIGURATION_PREFERENCES.value
|
|
|
|
|
)
|
|
|
|
|
if pref_obj and pref_obj.value:
|
|
|
|
|
user_timezone = pref_obj.value.get("timezone") or user_timezone
|
|
|
|
|
|
2025-09-09 14:37:32 +05:30
|
|
|
if user_id:
|
|
|
|
|
config_result = await session.execute(
|
|
|
|
|
select(UserConfigurationModel).where(
|
feat: UI refresh and user onboarding (#430)
* docs: design spec for lead-gen surfaces (Credits & Billing, Hire-an-Expert, Top-up, Enterprise)
Add brainstorming spec for: sidebar OBSERVE→MANAGE rename + Credits & Billing
link + Hire-an-Expert footer button; new /billing page with extracted Dograh
Model Credits card + CTAs; Top-up / Hire-an-Expert / Enterprise intake modals
with inline math captcha; and a workflow-builder Hire-an-Expert nudge. Frontend
only; submissions fire PostHog events via a submitLead() seam for a future
MongoDB endpoint. Also gitignore .superpowers/ brainstorm mockups.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* docs: implementation plan for user-onboarding lead-gen surfaces
14 bite-sized tasks: PostHog events, shared helpers (field options,
work-email blocklist, submitLead seam, math captcha), three intake modals
(enterprise/hire/top-up), LeadFormsProvider context, AppLayout mount, sidebar
MANAGE rename + Credits & Billing link + footer Hire button, extracted
DograhCreditsCard, /billing page, credits removal from Agent Runs, builder
nudge, and a full verification/dogfood pass.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): register PostHog events for lead-gen surfaces
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): shared field options, work-email validation, and submit seam
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): inline math captcha field
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): enterprise intake modal
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): hire-an-expert modal with enterprise link
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): top-up modal with >20k volume-pricing gate
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): shared lead-forms context provider
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): mount LeadFormsProvider in app layout
Wrap the sidebar branch of AppLayout with LeadFormsProvider so the shared
lead modals are available to the sidebar, billing card, and builder nudge.
Includes eslint import-order auto-fixes in TopUpModal and LeadFormsContext.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): rename OBSERVE to MANAGE, add Credits & Billing link and Hire-an-Expert footer button
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): extract DograhCreditsCard with top-up + hire CTAs
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): add Credits & Billing page
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* refactor(lead-gen): move Dograh Model Credits card out of Agent Runs to /billing
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(lead-gen): delayed Hire-an-Expert nudge on the workflow builder
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* ci(ui): add lint:lead-flow guard script
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(ui): restructure lead forms, self-serve Buy Credits, dialog blur
Revised lead-capture surfaces and credits bar:
- Dialog overlay gains backdrop blur (bg-black/60 backdrop-blur-sm).
- Shared primitives: LeadModalShell (icon/eyebrow header, scrollable body,
sticky footer, trust-line slot), PhoneField (react-international-phone,
dark, E.164 out), FormTrustLine ("Average response: under 10 minutes...").
- HireExpertModal: Name, Company, Job title, agent goal, Phone (required),
monthly volume. EnterpriseModal: + work email (required logged-out),
conditional deployment (yes/no/maybe, source-gated), agent goal.
OnboardingModal: drop useCase. Phone mandatory except onboarding.
- Volume buckets match the backend qualifier (0-5k/5k-100k/100k+/not-sure).
- Delete TopUpModal; DograhCreditsCard now self-serve Buy Credits (amount
chips $5/$10/$25/$50/$100 + custom min $5 → startTopUp seam) + Hire an
Expert + dashed custom-pricing link opening Enterprise (billing_custom_pricing).
- PostHog events: drop topup_*, add buy_credits_clicked,
buy_credits_amount_selected, custom_pricing_clicked. LeadFormsContext
drops topup; LeadKind/LeadSource updated.
- Introduce a single --cta warm accent token (CTAs + focus rings only).
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* feat(ui): split-screen auth + enterprise CTA + dark theme default
- AuthShell: dark two-column auth layout (brand/value panel with CSS-only
waveform motif + proof points + Bland-style enterprise CTA block on the
left, zinc-900 form card on the right; single-column on mobile).
- AuthEnterpriseCTA: "Talk to our team" → dograh.com/contact?intent=enterprise.
- stack-theme: dark StackTheme token overrides synced to globals.css.
- page.tsx: wrap StackHandler (non-fullPage) in AuthShell + StackTheme;
local-auth fallback preserved inside the shell. BackButton slimmed for the card.
- Dark locked as default: <html className="dark">, next-themes ThemeProvider
(defaultTheme="dark", enableSystem=false); inline no-FOUC script defaults dark.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* ui rezig, onboarding, billing, hire us & on prem cues
* ui changes
* chore: update comment
* chore: untrack docs/superpowers and gitignore it
* feat: refactor user configuration table
* feat(ui): 'check your email' confirmation on lead forms
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* added email and country in form submissions
* chore: update leads api
* fix: wrap dograh model config in card
---------
Co-authored-by: Pritesh <pritesh@dograh.com>
Co-authored-by: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
2026-06-17 19:49:33 +05:30
|
|
|
UserConfigurationModel.user_id == user_id,
|
|
|
|
|
UserConfigurationModel.key
|
|
|
|
|
== UserConfigurationKey.MODEL_CONFIGURATION.value,
|
2025-09-09 14:37:32 +05:30
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
config_obj = config_result.scalar_one_or_none()
|
|
|
|
|
if config_obj and config_obj.configuration:
|
2026-06-12 14:55:30 +05:30
|
|
|
effective_config = EffectiveAIModelConfiguration.model_validate(
|
2025-09-09 14:37:32 +05:30
|
|
|
config_obj.configuration
|
|
|
|
|
)
|
2026-06-12 14:55:30 +05:30
|
|
|
if effective_config.timezone and user_timezone == "UTC":
|
|
|
|
|
user_timezone = effective_config.timezone
|
2025-09-09 14:37:32 +05:30
|
|
|
|
|
|
|
|
# Validate timezone string
|
|
|
|
|
try:
|
|
|
|
|
# Test if timezone is valid
|
|
|
|
|
ZoneInfo(user_timezone)
|
|
|
|
|
except Exception:
|
|
|
|
|
# Fallback to UTC if timezone is invalid
|
|
|
|
|
user_timezone = "UTC"
|
|
|
|
|
# Query to get daily aggregates
|
|
|
|
|
# Use AT TIME ZONE to convert to user's timezone before grouping by date
|
|
|
|
|
date_expr = cast(
|
|
|
|
|
func.timezone(user_timezone, WorkflowRunModel.created_at), Date
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
daily_usage = await session.execute(
|
|
|
|
|
select(
|
|
|
|
|
date_expr.label("date"),
|
|
|
|
|
func.sum(
|
2026-06-12 14:55:30 +05:30
|
|
|
WorkflowRunModel.usage_info["call_duration_seconds"].as_float()
|
2025-09-09 14:37:32 +05:30
|
|
|
).label("total_seconds"),
|
|
|
|
|
func.count(WorkflowRunModel.id).label("call_count"),
|
|
|
|
|
)
|
|
|
|
|
.join(WorkflowModel, WorkflowModel.id == WorkflowRunModel.workflow_id)
|
|
|
|
|
.join(UserModel, UserModel.id == WorkflowModel.user_id)
|
|
|
|
|
.where(
|
|
|
|
|
UserModel.selected_organization_id == organization_id,
|
|
|
|
|
WorkflowRunModel.created_at >= start_date,
|
|
|
|
|
WorkflowRunModel.created_at <= end_date,
|
|
|
|
|
WorkflowRunModel.is_completed == True,
|
|
|
|
|
)
|
|
|
|
|
.group_by(date_expr)
|
|
|
|
|
.order_by(date_expr.desc())
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
breakdown = []
|
|
|
|
|
total_minutes = 0
|
|
|
|
|
total_cost_usd = 0
|
|
|
|
|
total_dograh_tokens = 0
|
|
|
|
|
|
|
|
|
|
for row in daily_usage:
|
|
|
|
|
seconds = row.total_seconds or 0
|
|
|
|
|
minutes = seconds / 60
|
|
|
|
|
cost_usd = seconds * price_per_second_usd
|
|
|
|
|
dograh_tokens = cost_usd * 100 # 1 cent = 1 token
|
|
|
|
|
|
|
|
|
|
total_minutes += minutes
|
|
|
|
|
total_cost_usd += cost_usd
|
|
|
|
|
total_dograh_tokens += dograh_tokens
|
|
|
|
|
|
|
|
|
|
breakdown.append(
|
|
|
|
|
{
|
|
|
|
|
"date": row.date.isoformat(),
|
|
|
|
|
"minutes": round(minutes, 1),
|
|
|
|
|
"cost_usd": round(cost_usd, 2),
|
|
|
|
|
"dograh_tokens": round(dograh_tokens, 0),
|
|
|
|
|
"call_count": row.call_count,
|
|
|
|
|
}
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
"breakdown": breakdown,
|
|
|
|
|
"total_minutes": round(total_minutes, 1),
|
|
|
|
|
"total_cost_usd": round(total_cost_usd, 2),
|
|
|
|
|
"total_dograh_tokens": round(total_dograh_tokens, 0),
|
|
|
|
|
"currency": "USD",
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-12 14:55:30 +05:30
|
|
|
def _calculate_current_period(self) -> tuple[datetime, datetime]:
|
|
|
|
|
"""Calculate the current calendar-month reporting period."""
|
2025-09-09 14:37:32 +05:30
|
|
|
now = datetime.now(timezone.utc)
|
|
|
|
|
|
2026-06-12 14:55:30 +05:30
|
|
|
period_start = now.replace(day=1, hour=0, minute=0, second=0, microsecond=0)
|
|
|
|
|
period_end = period_start + relativedelta(months=1) - relativedelta(seconds=1)
|
2025-09-09 14:37:32 +05:30
|
|
|
|
|
|
|
|
return period_start, period_end
|