Spaces:
Running
Running
Commit ·
7da14b7
1
Parent(s): 9fdf42c
Implement mandatory authentication and usage tracking for OpenAI API protection
Browse files- Enable strict authentication requirement for all users
- Add comprehensive usage tracking middleware
- Monitor OpenAI API calls with detailed logging
- Create beautiful login page explaining security requirements
- Add usage summary endpoint for monitoring
- Protect against abuse and track costs
- backend/app.py +4 -0
- backend/middleware/__init__.py +2 -1
- backend/middleware/auth.py +17 -12
- backend/middleware/usage_tracker.py +134 -0
- backend/routers/auth.py +46 -8
- backend/routers/observability.py +33 -0
backend/app.py
CHANGED
|
@@ -14,6 +14,7 @@ from fastapi.middleware.cors import CORSMiddleware
|
|
| 14 |
from starlette.middleware.sessions import SessionMiddleware
|
| 15 |
from fastapi.responses import RedirectResponse, HTMLResponse
|
| 16 |
from backend.middleware.auth import ConditionalAuthMiddleware
|
|
|
|
| 17 |
from utils.environment import should_enable_auth, debug_environment
|
| 18 |
|
| 19 |
|
|
@@ -51,6 +52,9 @@ app.add_middleware(
|
|
| 51 |
max_age=86400, # 24 hours
|
| 52 |
)
|
| 53 |
|
|
|
|
|
|
|
|
|
|
| 54 |
# Add conditional authentication middleware
|
| 55 |
app.add_middleware(ConditionalAuthMiddleware)
|
| 56 |
|
|
|
|
| 14 |
from starlette.middleware.sessions import SessionMiddleware
|
| 15 |
from fastapi.responses import RedirectResponse, HTMLResponse
|
| 16 |
from backend.middleware.auth import ConditionalAuthMiddleware
|
| 17 |
+
from backend.middleware.usage_tracker import UsageTrackingMiddleware
|
| 18 |
from utils.environment import should_enable_auth, debug_environment
|
| 19 |
|
| 20 |
|
|
|
|
| 52 |
max_age=86400, # 24 hours
|
| 53 |
)
|
| 54 |
|
| 55 |
+
# Add usage tracking middleware (before auth, to track all requests)
|
| 56 |
+
app.add_middleware(UsageTrackingMiddleware)
|
| 57 |
+
|
| 58 |
# Add conditional authentication middleware
|
| 59 |
app.add_middleware(ConditionalAuthMiddleware)
|
| 60 |
|
backend/middleware/__init__.py
CHANGED
|
@@ -3,5 +3,6 @@ Middleware package for AgentGraph backend.
|
|
| 3 |
"""
|
| 4 |
|
| 5 |
from .auth import ConditionalAuthMiddleware
|
|
|
|
| 6 |
|
| 7 |
-
__all__ = ["ConditionalAuthMiddleware"]
|
|
|
|
| 3 |
"""
|
| 4 |
|
| 5 |
from .auth import ConditionalAuthMiddleware
|
| 6 |
+
from .usage_tracker import UsageTrackingMiddleware
|
| 7 |
|
| 8 |
+
__all__ = ["ConditionalAuthMiddleware", "UsageTrackingMiddleware"]
|
backend/middleware/auth.py
CHANGED
|
@@ -75,20 +75,25 @@ class ConditionalAuthMiddleware(BaseHTTPMiddleware):
|
|
| 75 |
# Check user authentication
|
| 76 |
user = await self._get_current_user(request)
|
| 77 |
if not user:
|
| 78 |
-
#
|
| 79 |
-
#
|
| 80 |
-
# This makes the auth "optional" rather than "required"
|
| 81 |
|
| 82 |
-
|
| 83 |
-
logger.info(f"Unauthenticated access to {request.url.path} in HF Spaces")
|
| 84 |
|
| 85 |
-
#
|
| 86 |
-
|
| 87 |
-
|
| 88 |
-
|
| 89 |
-
|
| 90 |
-
|
| 91 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 92 |
|
| 93 |
# Add user info to request state
|
| 94 |
request.state.user = user
|
|
|
|
| 75 |
# Check user authentication
|
| 76 |
user = await self._get_current_user(request)
|
| 77 |
if not user:
|
| 78 |
+
# 🔐 MANDATORY AUTHENTICATION: Protect OpenAI API usage
|
| 79 |
+
# All users must be authenticated to prevent abuse of OpenAI resources
|
|
|
|
| 80 |
|
| 81 |
+
logger.warning(f"🚫 Unauthorized access attempt to {request.url.path} from {request.client.host if request.client else 'unknown'}")
|
|
|
|
| 82 |
|
| 83 |
+
# For API calls, return JSON error with login instructions
|
| 84 |
+
if request.url.path.startswith("/api/"):
|
| 85 |
+
return JSONResponse(
|
| 86 |
+
status_code=401,
|
| 87 |
+
content={
|
| 88 |
+
"error": "Authentication required to access OpenAI-powered features",
|
| 89 |
+
"message": "Please log in with your Hugging Face account to use this service",
|
| 90 |
+
"login_url": "/auth/login",
|
| 91 |
+
"reason": "API access requires user authentication for security and usage tracking"
|
| 92 |
+
}
|
| 93 |
+
)
|
| 94 |
+
|
| 95 |
+
# For web requests, redirect to login page
|
| 96 |
+
return RedirectResponse(url="/auth/login-page", status_code=302)
|
| 97 |
|
| 98 |
# Add user info to request state
|
| 99 |
request.state.user = user
|
backend/middleware/usage_tracker.py
ADDED
|
@@ -0,0 +1,134 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
"""
|
| 2 |
+
Usage Tracking Middleware
|
| 3 |
+
|
| 4 |
+
Tracks user API usage for security and monitoring purposes.
|
| 5 |
+
Especially important for OpenAI API calls which cost money.
|
| 6 |
+
"""
|
| 7 |
+
|
| 8 |
+
import logging
|
| 9 |
+
import time
|
| 10 |
+
from typing import Dict, Any, Optional
|
| 11 |
+
from fastapi import Request, Response
|
| 12 |
+
from starlette.middleware.base import BaseHTTPMiddleware
|
| 13 |
+
from utils.environment import is_huggingface_space
|
| 14 |
+
import json
|
| 15 |
+
from datetime import datetime
|
| 16 |
+
|
| 17 |
+
logger = logging.getLogger(__name__)
|
| 18 |
+
|
| 19 |
+
|
| 20 |
+
class UsageTrackingMiddleware(BaseHTTPMiddleware):
|
| 21 |
+
"""
|
| 22 |
+
Middleware to track user API usage, especially for OpenAI-powered endpoints.
|
| 23 |
+
"""
|
| 24 |
+
|
| 25 |
+
def __init__(self, app):
|
| 26 |
+
super().__init__(app)
|
| 27 |
+
|
| 28 |
+
# Endpoints that use OpenAI API (and thus cost money)
|
| 29 |
+
self.openai_endpoints = [
|
| 30 |
+
"/api/knowledge-graphs/extract",
|
| 31 |
+
"/api/knowledge-graphs/analyze",
|
| 32 |
+
"/api/methods/",
|
| 33 |
+
"/api/traces/analyze",
|
| 34 |
+
"/api/causal/",
|
| 35 |
+
]
|
| 36 |
+
|
| 37 |
+
# Endpoints that should be monitored for usage patterns
|
| 38 |
+
self.monitored_endpoints = self.openai_endpoints + [
|
| 39 |
+
"/api/traces/",
|
| 40 |
+
"/api/tasks/",
|
| 41 |
+
"/api/perturbation/",
|
| 42 |
+
]
|
| 43 |
+
|
| 44 |
+
async def dispatch(self, request: Request, call_next):
|
| 45 |
+
"""Track API usage and log user activity."""
|
| 46 |
+
start_time = time.time()
|
| 47 |
+
|
| 48 |
+
# Get user info from request state (set by auth middleware)
|
| 49 |
+
user = getattr(request.state, "user", None)
|
| 50 |
+
user_id = user.get("username", "anonymous") if user else "anonymous"
|
| 51 |
+
user_auth_method = user.get("auth_method", "none") if user else "none"
|
| 52 |
+
|
| 53 |
+
# Track the request
|
| 54 |
+
should_track = any(
|
| 55 |
+
request.url.path.startswith(endpoint)
|
| 56 |
+
for endpoint in self.monitored_endpoints
|
| 57 |
+
)
|
| 58 |
+
|
| 59 |
+
is_openai_call = any(
|
| 60 |
+
request.url.path.startswith(endpoint)
|
| 61 |
+
for endpoint in self.openai_endpoints
|
| 62 |
+
)
|
| 63 |
+
|
| 64 |
+
# Log the request if it's being tracked
|
| 65 |
+
if should_track:
|
| 66 |
+
client_ip = request.client.host if request.client else "unknown"
|
| 67 |
+
logger.info(
|
| 68 |
+
f"📊 API Usage: {user_id} ({user_auth_method}) -> "
|
| 69 |
+
f"{request.method} {request.url.path} from {client_ip} "
|
| 70 |
+
f"{'💰 [OpenAI]' if is_openai_call else ''}"
|
| 71 |
+
)
|
| 72 |
+
|
| 73 |
+
# Process the request
|
| 74 |
+
response = await call_next(request)
|
| 75 |
+
|
| 76 |
+
# Calculate duration
|
| 77 |
+
duration = time.time() - start_time
|
| 78 |
+
|
| 79 |
+
# Log completion for important endpoints
|
| 80 |
+
if should_track:
|
| 81 |
+
status_emoji = "✅" if response.status_code < 400 else "❌"
|
| 82 |
+
cost_warning = " 💸 COST INCURRED" if is_openai_call and response.status_code < 400 else ""
|
| 83 |
+
|
| 84 |
+
logger.info(
|
| 85 |
+
f"{status_emoji} API Complete: {user_id} -> "
|
| 86 |
+
f"{request.method} {request.url.path} "
|
| 87 |
+
f"[{response.status_code}] in {duration:.2f}s{cost_warning}"
|
| 88 |
+
)
|
| 89 |
+
|
| 90 |
+
# Log detailed usage for OpenAI calls
|
| 91 |
+
if is_openai_call:
|
| 92 |
+
self._log_openai_usage(user_id, user_auth_method, request, response, duration)
|
| 93 |
+
|
| 94 |
+
return response
|
| 95 |
+
|
| 96 |
+
def _log_openai_usage(
|
| 97 |
+
self,
|
| 98 |
+
user_id: str,
|
| 99 |
+
auth_method: str,
|
| 100 |
+
request: Request,
|
| 101 |
+
response: Response,
|
| 102 |
+
duration: float
|
| 103 |
+
):
|
| 104 |
+
"""Log detailed information about OpenAI API usage."""
|
| 105 |
+
|
| 106 |
+
usage_record = {
|
| 107 |
+
"timestamp": datetime.now().isoformat(),
|
| 108 |
+
"user_id": user_id,
|
| 109 |
+
"auth_method": auth_method,
|
| 110 |
+
"endpoint": request.url.path,
|
| 111 |
+
"method": request.method,
|
| 112 |
+
"status_code": response.status_code,
|
| 113 |
+
"duration_seconds": round(duration, 2),
|
| 114 |
+
"client_ip": request.client.host if request.client else "unknown",
|
| 115 |
+
"user_agent": request.headers.get("User-Agent", "unknown"),
|
| 116 |
+
"environment": "hf_spaces" if is_huggingface_space() else "local",
|
| 117 |
+
}
|
| 118 |
+
|
| 119 |
+
# Log as structured data for easy parsing/analysis
|
| 120 |
+
logger.warning(
|
| 121 |
+
f"💰 OPENAI_USAGE: {json.dumps(usage_record, separators=(',', ':'))}"
|
| 122 |
+
)
|
| 123 |
+
|
| 124 |
+
# Also log a human-readable summary
|
| 125 |
+
if response.status_code >= 400:
|
| 126 |
+
logger.error(
|
| 127 |
+
f"🚨 OpenAI API Error: User {user_id} got {response.status_code} "
|
| 128 |
+
f"on {request.url.path} - potential abuse or misconfiguration"
|
| 129 |
+
)
|
| 130 |
+
else:
|
| 131 |
+
logger.info(
|
| 132 |
+
f"💰 OpenAI API Success: User {user_id} used {request.url.path} "
|
| 133 |
+
f"({duration:.2f}s) - track costs and usage patterns"
|
| 134 |
+
)
|
backend/routers/auth.py
CHANGED
|
@@ -28,7 +28,7 @@ async def auth_status(request: Request):
|
|
| 28 |
"auth_enabled": should_enable_auth(),
|
| 29 |
"environment": "huggingface_spaces" if is_huggingface_space() else "local_development",
|
| 30 |
"oauth_available": bool(config),
|
| 31 |
-
"login_required":
|
| 32 |
"user_authenticated": bool(user),
|
| 33 |
"user_info": {
|
| 34 |
"auth_method": user.get("auth_method") if user else None,
|
|
@@ -189,7 +189,7 @@ async def get_current_user(request: Request):
|
|
| 189 |
@router.get("/login-page")
|
| 190 |
async def login_page(request: Request):
|
| 191 |
"""
|
| 192 |
-
Serve a
|
| 193 |
"""
|
| 194 |
if not should_enable_auth():
|
| 195 |
return RedirectResponse(url="/", status_code=302)
|
|
@@ -198,18 +198,56 @@ async def login_page(request: Request):
|
|
| 198 |
<!DOCTYPE html>
|
| 199 |
<html>
|
| 200 |
<head>
|
| 201 |
-
<title>AgentGraph -
|
| 202 |
<style>
|
| 203 |
-
body {
|
| 204 |
-
|
| 205 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 206 |
</style>
|
| 207 |
</head>
|
| 208 |
<body>
|
| 209 |
<div class="login-container">
|
|
|
|
| 210 |
<h1>🕸️ AgentGraph</h1>
|
| 211 |
-
<
|
| 212 |
-
<
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 213 |
</div>
|
| 214 |
</body>
|
| 215 |
</html>
|
|
|
|
| 28 |
"auth_enabled": should_enable_auth(),
|
| 29 |
"environment": "huggingface_spaces" if is_huggingface_space() else "local_development",
|
| 30 |
"oauth_available": bool(config),
|
| 31 |
+
"login_required": True, # Mandatory for OpenAI API protection
|
| 32 |
"user_authenticated": bool(user),
|
| 33 |
"user_info": {
|
| 34 |
"auth_method": user.get("auth_method") if user else None,
|
|
|
|
| 189 |
@router.get("/login-page")
|
| 190 |
async def login_page(request: Request):
|
| 191 |
"""
|
| 192 |
+
Serve a login page explaining why authentication is required.
|
| 193 |
"""
|
| 194 |
if not should_enable_auth():
|
| 195 |
return RedirectResponse(url="/", status_code=302)
|
|
|
|
| 198 |
<!DOCTYPE html>
|
| 199 |
<html>
|
| 200 |
<head>
|
| 201 |
+
<title>AgentGraph - Authentication Required</title>
|
| 202 |
<style>
|
| 203 |
+
body {
|
| 204 |
+
font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif;
|
| 205 |
+
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
|
| 206 |
+
margin: 0; padding: 0; min-height: 100vh; display: flex; align-items: center; justify-content: center;
|
| 207 |
+
}
|
| 208 |
+
.login-container {
|
| 209 |
+
background: white; max-width: 500px; margin: 0 auto; padding: 40px;
|
| 210 |
+
border-radius: 12px; box-shadow: 0 10px 30px rgba(0,0,0,0.3); text-align: center;
|
| 211 |
+
}
|
| 212 |
+
.login-btn {
|
| 213 |
+
background: #ff6b35; color: white; padding: 14px 28px; text-decoration: none;
|
| 214 |
+
border-radius: 8px; font-weight: bold; display: inline-block; margin-top: 20px;
|
| 215 |
+
transition: background 0.3s ease;
|
| 216 |
+
}
|
| 217 |
+
.login-btn:hover { background: #e55a2b; }
|
| 218 |
+
.icon { font-size: 48px; margin-bottom: 20px; }
|
| 219 |
+
.subtitle { color: #666; margin: 20px 0; line-height: 1.6; }
|
| 220 |
+
.security-note {
|
| 221 |
+
background: #f8f9fa; padding: 15px; border-radius: 8px; margin: 20px 0;
|
| 222 |
+
border-left: 4px solid #ff6b35; text-align: left; font-size: 14px;
|
| 223 |
+
}
|
| 224 |
</style>
|
| 225 |
</head>
|
| 226 |
<body>
|
| 227 |
<div class="login-container">
|
| 228 |
+
<div class="icon">🔐</div>
|
| 229 |
<h1>🕸️ AgentGraph</h1>
|
| 230 |
+
<h2>Authentication Required</h2>
|
| 231 |
+
<p class="subtitle">
|
| 232 |
+
AgentGraph uses advanced AI models (OpenAI GPT) to provide knowledge graph extraction
|
| 233 |
+
and analysis capabilities. To ensure responsible usage and prevent abuse,
|
| 234 |
+
we require user authentication.
|
| 235 |
+
</p>
|
| 236 |
+
|
| 237 |
+
<div class="security-note">
|
| 238 |
+
<strong>🛡️ Why authentication is required:</strong><br>
|
| 239 |
+
• Prevents unauthorized access to AI resources<br>
|
| 240 |
+
• Enables usage tracking and abuse prevention<br>
|
| 241 |
+
• Ensures fair access for all legitimate users<br>
|
| 242 |
+
• Maintains service quality and availability
|
| 243 |
+
</div>
|
| 244 |
+
|
| 245 |
+
<p>Please log in with your Hugging Face account to continue.</p>
|
| 246 |
+
<a href="/auth/login" class="login-btn">🚀 Login with Hugging Face</a>
|
| 247 |
+
|
| 248 |
+
<p style="margin-top: 30px; font-size: 12px; color: #888;">
|
| 249 |
+
By logging in, you agree to use this service responsibly and in accordance with our usage policies.
|
| 250 |
+
</p>
|
| 251 |
</div>
|
| 252 |
</body>
|
| 253 |
</html>
|
backend/routers/observability.py
CHANGED
|
@@ -883,6 +883,39 @@ async def get_environment():
|
|
| 883 |
}
|
| 884 |
|
| 885 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 886 |
@router.get("/health-check")
|
| 887 |
async def health_check():
|
| 888 |
"""Comprehensive health check for the system."""
|
|
|
|
| 883 |
}
|
| 884 |
|
| 885 |
|
| 886 |
+
@router.get("/usage-summary")
|
| 887 |
+
async def get_usage_summary(request: Request):
|
| 888 |
+
"""
|
| 889 |
+
Get a summary of recent API usage for monitoring purposes.
|
| 890 |
+
This helps track OpenAI API costs and detect potential abuse.
|
| 891 |
+
"""
|
| 892 |
+
# Only authenticated users can see usage data
|
| 893 |
+
user = getattr(request.state, "user", None)
|
| 894 |
+
if not user:
|
| 895 |
+
raise HTTPException(status_code=401, detail="Authentication required")
|
| 896 |
+
|
| 897 |
+
# In a production system, you'd query a database or log aggregation service
|
| 898 |
+
# For now, we'll return a summary based on recent log entries
|
| 899 |
+
|
| 900 |
+
return {
|
| 901 |
+
"message": "Usage tracking is active",
|
| 902 |
+
"tracking_enabled": True,
|
| 903 |
+
"openai_endpoints_monitored": [
|
| 904 |
+
"/api/knowledge-graphs/extract",
|
| 905 |
+
"/api/knowledge-graphs/analyze",
|
| 906 |
+
"/api/methods/",
|
| 907 |
+
"/api/traces/analyze",
|
| 908 |
+
"/api/causal/",
|
| 909 |
+
],
|
| 910 |
+
"current_user": {
|
| 911 |
+
"username": user.get("username", "unknown"),
|
| 912 |
+
"auth_method": user.get("auth_method", "unknown"),
|
| 913 |
+
},
|
| 914 |
+
"note": "Detailed usage logs are available in the application logs for administrator review",
|
| 915 |
+
"timestamp": datetime.now().isoformat()
|
| 916 |
+
}
|
| 917 |
+
|
| 918 |
+
|
| 919 |
@router.get("/health-check")
|
| 920 |
async def health_check():
|
| 921 |
"""Comprehensive health check for the system."""
|