"""Tempo time-tracking integration service. Authentication model -------------------- Each user authenticates to Tempo with their own personal Tempo API token, stored in ``user.tempo_api_token``. This is different from the Jira API token. Obtain a Tempo token at: Jira → Apps → Tempo → Settings → API Integration. The global ``settings.TEMPO_ENABLED`` flag acts as a kill-switch. When False, all Tempo calls are silently skipped regardless of whether users have tokens. What goes to Tempo ------------------ Both **red team execution** and **blue team evaluation** time are logged to Tempo. Red team time runs from when the red_tech clicks "Start" to "Done". Blue team time runs from when the blue_tech clicks "Start Evaluation" (sets blue_work_started_at) to when they submit, so it reflects actual working time rather than queue time. """ import logging from typing import Optional from sqlalchemy.orm import Session from app.config import settings from app.domain.exceptions import InvalidOperationError from app.models.jira_link import JiraLink, JiraLinkEntityType logger = logging.getLogger(__name__) # Only red team execution time goes to Tempo. # Blue team evaluation time is tracked internally (worklogs table) for SLA # purposes but is NOT forwarded to Tempo — blue team has no Jira access. _TEMPO_ACTIVITY_TYPES = {"red_team_execution"} def has_tempo_configured(user) -> bool: """Return True if *user* has a personal Tempo API token stored.""" return bool(getattr(user, "tempo_api_token", None)) _TEMPO_DEFAULT_BASE_URL = "https://api.tempo.io/4" _TEMPO_EU_BASE_URL = "https://api.eu.tempo.io/4" # system_configs key for admin-configurable Tempo base URL _TEMPO_BASE_URL_CONFIG_KEY = "tempo.base_url" def _get_tempo_base_url(db=None) -> str: """Return the Tempo API base URL. Reads ``tempo.base_url`` from ``system_configs`` first (allows the admin to set the EU endpoint ``https://api.eu.tempo.io/4`` without redeploying). Falls back to ``settings.TEMPO_BASE_URL`` env-var, then the global default. """ if db is not None: try: from app.models.system_config import SystemConfig row = db.query(SystemConfig).filter( SystemConfig.key == _TEMPO_BASE_URL_CONFIG_KEY ).first() if row and row.value: return row.value.rstrip("/") except Exception: pass # DB unavailable — fall through to defaults env_url = getattr(settings, "TEMPO_BASE_URL", None) return (env_url or _TEMPO_DEFAULT_BASE_URL).rstrip("/") def get_user_tempo_client(user, db=None): """Return a Tempo API v4 client authenticated as *user*. Pass ``db`` to allow reading the admin-configured base URL from ``system_configs`` (needed for EU Tempo workspaces). Raises ``InvalidOperationError`` when the user has no token or the client library is not installed. """ token = getattr(user, "tempo_api_token", None) if not token: raise InvalidOperationError( "No Tempo API token configured. " "Add it in Settings → Profile → Tempo Integration." ) try: from tempoapiclient import client_v4 as tempo_client base_url = _get_tempo_base_url(db) logger.debug("Using Tempo base URL: %s", base_url) return tempo_client.Tempo(auth_token=token, base_url=base_url) except ImportError: raise InvalidOperationError( "tempo-api-python-client is not installed. " "Run: pip install tempo-api-python-client" ) def log_worklog( user, jira_issue_id: int, author_account_id: str, date: str, time_spent_seconds: int, description: str, db=None, ) -> dict: """Create a worklog entry in Tempo using *user*'s personal token. Note: tempoapiclient raises SystemExit (not Exception) on API errors, so we intercept BaseException and re-raise as RuntimeError to keep it non-fatal. """ tempo = get_user_tempo_client(user, db=db) try: return tempo.create_worklog( accountId=author_account_id, issueId=jira_issue_id, dateFrom=date, timeSpentSeconds=time_spent_seconds, description=description, ) except Exception: raise except BaseException as exc: # tempoapiclient raises SystemExit on HTTP errors (e.g. 400 Bad Request). # SystemExit is a BaseException, not Exception, so convert it so callers # can catch it with the usual `except Exception` pattern. raise RuntimeError(f"Tempo API error: {exc}") from exc def auto_log_test_worklog( db: Session, test, user, activity_type: str, duration_seconds: int, ) -> Optional[dict]: """Log *duration_seconds* to Tempo for the given test if conditions are met. ``duration_seconds`` must be the value already computed by the workflow layer (gross elapsed time minus any paused time). It is used as-is so the Tempo entry always matches the Aegis worklog — no re-calculation. Only ``red_team_execution`` activities are forwarded to Tempo. ``blue_team_evaluation`` is tracked internally but not sent. Returns the Tempo worklog response dict, or ``None`` if skipped. Completely non-fatal — errors are logged and swallowed. """ # Only whitelisted activity types go to Tempo if activity_type not in _TEMPO_ACTIVITY_TYPES: logger.debug( "Skipping Tempo sync for activity_type=%s (not in whitelist)", activity_type ) return None # Global kill-switch if not settings.TEMPO_ENABLED: return None if duration_seconds <= 0: logger.debug( "Skipping Tempo sync for test %s: duration=%ds", test.id, duration_seconds ) return None # Per-user token required if not has_tempo_configured(user): logger.debug( "User %s has no Tempo token; skipping worklog for test %s", getattr(user, "username", user), test.id, ) return None # Need a Jira link with a numeric issue ID link = ( db.query(JiraLink) .filter( JiraLink.entity_id == test.id, JiraLink.entity_type == JiraLinkEntityType.test, ) .first() ) if not link or not link.jira_issue_id: logger.debug("No Jira link for test %s, skipping Tempo worklog", test.id) return None jira_account_id = (getattr(user, "jira_account_id", "") or "").strip() if not jira_account_id: logger.debug( "User %s has no jira_account_id; skipping Tempo worklog", getattr(user, "username", user), ) return None try: # Use the phase start timestamp as the worklog date so it matches when # work actually happened (not the submission timestamp). if activity_type == "blue_team_evaluation": work_date = ( (test.blue_work_started_at or test.blue_started_at or test.created_at) .strftime("%Y-%m-%d") ) description = f"[Aegis] Blue Team evaluation: {test.name}" else: work_date = ( (test.red_started_at or getattr(test, "updated_at", None) or test.created_at) .strftime("%Y-%m-%d") ) description = f"[Aegis] Red Team execution: {test.name}" result = log_worklog( user=user, jira_issue_id=int(link.jira_issue_id), author_account_id=jira_account_id, date=work_date, time_spent_seconds=duration_seconds, description=description, db=db, ) logger.info( "Tempo worklog created for test %s by user %s: %ds on %s", test.id, getattr(user, "username", user), duration_seconds, work_date, ) return result except Exception as e: logger.warning( "Tempo worklog failed for test %s (user %s): %s", test.id, getattr(user, "username", user), e, exc_info=True, ) return None