prometheus.py 1.8 KB

123456789101112131415161718192021222324252627282930313233343536373839
  1. import inspect
  2. import logging
  3. import uuid
  4. from fastapi import FastAPI, Request
  5. from starlette.middleware.base import BaseHTTPMiddleware
  6. from starlette.middleware.sessions import SessionMiddleware
  7. EXCLUDED_USER_AGENTS = ('bot', 'spider', 'crawler', 'monitor', 'curl',
  8. 'wget', 'python-requests', 'kuma', 'health check')
  9. def start_monitor(app: FastAPI) -> None:
  10. try:
  11. import prometheus_client
  12. visits = prometheus_client.Counter('nicegui_page_visits', 'Number of real page visits',
  13. ['path', 'session', 'origin'])
  14. class PrometheusMiddleware(BaseHTTPMiddleware):
  15. async def dispatch(self, request: Request, call_next):
  16. if 'id' not in request.session:
  17. request.session['id'] = str(uuid.uuid4())
  18. response = await call_next(request)
  19. if response.headers.get('x-nicegui-content') == 'page':
  20. agent = request.headers.get('user-agent', 'unknown').lower()
  21. # ignore monitoring, web crawlers and the like
  22. if not any(s in agent for s in EXCLUDED_USER_AGENTS):
  23. origin_url = request.headers.get('referer', 'unknown')
  24. print(request.get('path'), agent, request.session['id'], origin_url, flush=True)
  25. visits.labels(request.get('path'), request.session['id'], origin_url).inc()
  26. return response
  27. if inspect.stack()[-2].filename.endswith('spawn.py'):
  28. prometheus_client.start_http_server(9062)
  29. app.add_middleware(PrometheusMiddleware)
  30. app.add_middleware(SessionMiddleware, secret_key='NiceGUI is awesome!')
  31. except ModuleNotFoundError:
  32. logging.info('Prometheus not installed, skipping monitoring')