diff --git a/daras_ai_v2/icons.py b/daras_ai_v2/icons.py index 30e90f77e..821f999fe 100644 --- a/daras_ai_v2/icons.py +++ b/daras_ai_v2/icons.py @@ -35,6 +35,7 @@ member = '' chevron_right = '' check = '' +analytics = '' # brands github = '' diff --git a/routers/account.py b/routers/account.py index 22b4376ad..b725e4353 100644 --- a/routers/account.py +++ b/routers/account.py @@ -109,6 +109,12 @@ def account_route(request: Request): ) +@gui.route(app, "/account/analytics") +def analytics_route(request: Request): + with account_page_wrapper(request, AccountTabs.analytics) as current_workspace: + analytics_tab(request) + + @gui.route(app, "/account/profile/") def profile_route(request: Request): with account_page_wrapper(request, AccountTabs.profile) as current_workspace: @@ -227,6 +233,7 @@ class AccountTabs(TabData, Enum): saved = TabData(title=f"{icons.save} Saved", route=saved_route) api_keys = TabData(title=f"{icons.api} API Keys", route=api_keys_route) billing = TabData(title=f"{icons.billing} Billing", route=account_route) + analytics = TabData(title=f"{icons.analytics} Analytics", route=analytics_route) @property def url_path(self) -> str: @@ -241,10 +248,12 @@ def get_tabs_for_user( if workspace.is_personal: ret.remove(cls.members) + ret.remove(cls.analytics) else: ret.remove(cls.profile) if not workspace.memberships.get(user=user).can_edit_workspace(): ret.remove(cls.billing) + ret.remove(cls.analytics) return ret @@ -345,6 +354,61 @@ def _render_run(pr: PublishedRun): paginate_button(url=request.url, cursor=cursor) +def analytics_tab(request: Request): + workspace = get_current_workspace(request.user, request.session) + gui.write("# Usage & Limits") + gui.caption( + f"Member, API & Integration usage for **{workspace.display_name(request.user)}**." + ) + + with gui.div(className="table-responsive"), gui.tag("table", className="table"): + with gui.tag("thead"), gui.tag("tr"): + with gui.tag("th", scope="col"): + gui.html("Name") + with gui.tag("th", scope="col"): + gui.html("Type") + with gui.tag("th", scope="col"): + gui.html("Runs") + with gui.tag("th", scope="col"): + gui.html("Credits Used") + with gui.tag("th", scope="col"): + gui.html("") + + with gui.tag("tbody"): + for m in workspace.memberships.all(): + with gui.tag("tr", className="no-margin"): + with gui.tag("td"): + gui.write(m.user.full_name()) + with gui.tag("td"): + gui.html("User") + with gui.tag("td"): + gui.html(f"{m.get_run_count()}") + with gui.tag("td"): + gui.html(f"{m.get_credit_usage()} Cr") + with gui.tag("td"): + gui.html("") + + with gui.tag("tr", className="no-margin"): + with gui.tag("td"): + gui.write(f"[API Keys]({get_route_path(api_keys_route)})") + with gui.tag("td"): + gui.html("API Key") + with gui.tag("td"): + gui.html(f"{workspace.get_api_key_run_count()}") + with gui.tag("td"): + gui.html(f"{workspace.get_api_key_credit_usage()} Cr") + + with gui.tag("tr", className="no-margin"): + with gui.tag("td"): + gui.write("Bot Integrations") + with gui.tag("td"): + gui.html("Bot") + with gui.tag("td"): + gui.html(f"{workspace.get_bot_run_count()}") + with gui.tag("td"): + gui.html(f"{workspace.get_bot_credit_usage()} Cr") + + def api_keys_tab(request: Request): gui.write("# 🔐 API Keys") workspace = get_current_workspace(request.user, request.session) diff --git a/workspaces/models.py b/workspaces/models.py index e4fb17331..3e621f485 100644 --- a/workspaces/models.py +++ b/workspaces/models.py @@ -12,6 +12,7 @@ from django.db import models, transaction, IntegrityError from django.db.backends.base.schema import logger from django.db.models.aggregates import Sum +from django.db.models.functions import Abs from django.db.models.query_utils import Q from django.utils import timezone from django.utils.text import slugify @@ -256,6 +257,35 @@ def add_balance( pass raise + def get_api_key_run_count(self) -> int: + return ( + self.saved_runs.filter(is_api_call=True, price__gt=0) + .exclude(messages__isnull=False) + .count() + ) + + def get_api_key_credit_usage(self) -> int: + return ( + self.saved_runs.filter(is_api_call=True) + .exclude(messages__isnull=False) # exclude bot-integration runs + .aggregate(total=Sum("price")) + .get("total") + or 0 + ) + + def get_bot_run_count(self) -> int: + return self.saved_runs.filter( + is_api_call=True, price__gt=0, messages__isnull=False + ).count() + + def get_bot_credit_usage(self) -> int: + return ( + self.saved_runs.filter(is_api_call=True, messages__isnull=False) + .aggregate(total=Sum("price")) + .get("total") + or 0 + ) + def get_or_create_stripe_customer(self) -> stripe.Customer: customer = None @@ -439,6 +469,21 @@ def can_invite(self): and not self.workspace.is_personal ) + def get_run_count(self) -> int: + return self.workspace.saved_runs.filter( + uid=self.user.uid, + price__gt=0, # proxy for successful runs + is_api_call=False, + ).count() + + def get_credit_usage(self) -> int: + return ( + self.workspace.saved_runs.filter( + uid=self.user.uid, price__gt=0, is_api_call=False + ).aggregate(total=Sum("price"))["total"] + or 0 + ) + class WorkspaceInviteQuerySet(models.QuerySet): def create_and_send_invite(