mirror of https://github.com/interlegis/sapl.git
Browse Source
feat: Adiciona health check endpoints Co-authored-by: Edward <9326037+edwardoliveira@users.noreply.github.com>pull/3800/merge
committed by
GitHub
8 changed files with 167 additions and 21 deletions
@ -0,0 +1,106 @@ |
|||||
|
import logging |
||||
|
import traceback |
||||
|
|
||||
|
from django.http import HttpResponse, JsonResponse |
||||
|
from django.utils import timezone |
||||
|
from rest_framework.views import APIView |
||||
|
from rest_framework.response import Response |
||||
|
from rest_framework import status |
||||
|
from django.conf import settings |
||||
|
from sapl.health import check_app, check_db, check_cache |
||||
|
|
||||
|
COMMON_HEADERS = { |
||||
|
"Cache-Control": "no-store, no-cache, must-revalidate, max-age=0", |
||||
|
"Pragma": "no-cache", |
||||
|
} |
||||
|
|
||||
|
|
||||
|
def _format_plain(ok: bool) -> HttpResponse: |
||||
|
return HttpResponse("OK\n" if ok else "UNHEALTHY\n", |
||||
|
status=status.HTTP_200_OK if ok else status.HTTP_503_SERVICE_UNAVAILABLE, |
||||
|
content_type="text/plain") |
||||
|
|
||||
|
|
||||
|
class HealthzView(APIView): |
||||
|
authentication_classes = [] |
||||
|
permission_classes = [] |
||||
|
|
||||
|
logger = logging.getLogger(__name__) |
||||
|
|
||||
|
def get(self, request): |
||||
|
try: |
||||
|
ok, msg, ms = check_app() |
||||
|
payload = { |
||||
|
"status": "OK" if ok else "UNHEALTHY", |
||||
|
"checks": {"app": {"ok": ok, "latency_ms": round(ms, 1), "error": msg}}, |
||||
|
"version": settings.SAPL_VERSION, |
||||
|
"time": timezone.now().isoformat(), |
||||
|
} |
||||
|
if request.query_params.get("fmt") == "txt": |
||||
|
return _format_plain(ok) |
||||
|
return Response(payload, |
||||
|
status=status.HTTP_200_OK if ok else status.HTTP_503_SERVICE_UNAVAILABLE, |
||||
|
headers=COMMON_HEADERS) |
||||
|
except Exception as e: |
||||
|
self.logger.error(traceback.format_exc()) |
||||
|
return "An internal error has occurred!" |
||||
|
|
||||
|
|
||||
|
|
||||
|
class ReadyzView(APIView): |
||||
|
authentication_classes = [] |
||||
|
permission_classes = [] |
||||
|
|
||||
|
logger = logging.getLogger(__name__) |
||||
|
|
||||
|
def get(self, request): |
||||
|
try: |
||||
|
checks = { |
||||
|
"app": check_app(), |
||||
|
"db": check_db(), |
||||
|
"cache": check_cache(), |
||||
|
} |
||||
|
payload_checks = { |
||||
|
name: {"ok": r[0], "latency_ms": round(r[2], 1), "error": r[1]} |
||||
|
for name, r in checks.items() |
||||
|
} |
||||
|
ok = all(r[0] for r in checks.values()) |
||||
|
payload = { |
||||
|
"status": "ok" if ok else "unhealthy", |
||||
|
"checks": payload_checks, |
||||
|
"version": settings.SAPL_VERSION, |
||||
|
"time": timezone.now().isoformat(), |
||||
|
} |
||||
|
if request.query_params.get("fmt") == "txt": |
||||
|
return _format_plain(ok) |
||||
|
return Response(payload, |
||||
|
status=status.HTTP_200_OK if ok else status.HTTP_503_SERVICE_UNAVAILABLE, |
||||
|
headers=COMMON_HEADERS) |
||||
|
except Exception as e: |
||||
|
self.logger.error(traceback.format_exc()) |
||||
|
return "An internal error has occurred!" |
||||
|
|
||||
|
|
||||
|
class AppzVersionView(APIView): |
||||
|
authentication_classes = [] |
||||
|
permission_classes = [] |
||||
|
|
||||
|
logger = logging.getLogger(__name__) |
||||
|
|
||||
|
def get(self, request): |
||||
|
try: |
||||
|
payload = { |
||||
|
'name': 'SAPL', |
||||
|
'description': 'Sistema de Apoio ao Processo Legislativo', |
||||
|
'version': settings.SAPL_VERSION, |
||||
|
} |
||||
|
if request.query_params.get("fmt") == "txt": |
||||
|
return HttpResponse(f"{payload['version']} {payload['name']}", |
||||
|
status=status.HTTP_200_OK, |
||||
|
content_type="text/plain") |
||||
|
return Response(payload, |
||||
|
status=status.HTTP_200_OK, |
||||
|
headers=COMMON_HEADERS) |
||||
|
except Exception as e: |
||||
|
self.logger.error(traceback.format_exc()) |
||||
|
return "An internal error has occurred!" |
@ -0,0 +1,36 @@ |
|||||
|
# core/health.py |
||||
|
import logging |
||||
|
import time |
||||
|
from typing import Tuple, Optional |
||||
|
from django.db import connection |
||||
|
from django.core.cache import cache |
||||
|
|
||||
|
logger = logging.getLogger(__name__) |
||||
|
|
||||
|
|
||||
|
def check_app() -> Tuple[bool, Optional[str], float]: |
||||
|
t0 = time.monotonic() |
||||
|
return True, None, (time.monotonic() - t0) * 1000 |
||||
|
|
||||
|
|
||||
|
def check_db() -> Tuple[bool, Optional[str], float]: |
||||
|
t0 = time.monotonic() |
||||
|
try: |
||||
|
with connection.cursor() as cur: |
||||
|
cur.execute("SELECT 1") |
||||
|
cur.fetchone() |
||||
|
return True, None, (time.monotonic() - t0) * 1000 |
||||
|
except Exception as e: |
||||
|
logging.error(e) |
||||
|
return False, "An internal error has occurred!", (time.monotonic() - t0) * 1000 |
||||
|
|
||||
|
|
||||
|
def check_cache() -> Tuple[bool, Optional[str], float]: |
||||
|
t0 = time.monotonic() |
||||
|
try: |
||||
|
cache.set("_hc", "1", 5) |
||||
|
ok = cache.get("_hc") == "1" |
||||
|
return ok, None if ok else "Cache get/set failed", (time.monotonic() - t0) * 1000 |
||||
|
except Exception as e: |
||||
|
logging.error(e) |
||||
|
return False, "An internal error has occurred!", (time.monotonic() - t0) * 1000 |
@ -0,0 +1,12 @@ |
|||||
|
from prometheus_client import CollectorRegistry, Gauge, generate_latest, CONTENT_TYPE_LATEST |
||||
|
|
||||
|
|
||||
|
def health_registry(check_results: dict) -> bytes: |
||||
|
""" |
||||
|
check_results: {"app": True/False, "db": True/False, ...} |
||||
|
""" |
||||
|
reg = CollectorRegistry() |
||||
|
g = Gauge("app_health", "1 if healthy, 0 if unhealthy", ["component"], registry=reg) |
||||
|
for comp, ok in check_results.items(): |
||||
|
g.labels(component=comp).set(1 if ok else 0) |
||||
|
return generate_latest(reg) |
Loading…
Reference in new issue