diff options
author | Sergey Lukjanov <[email protected]> | 2020-08-29 21:49:48 -0700 |
---|---|---|
committer | Sergey Lukjanov <[email protected]> | 2020-08-30 11:35:25 -0700 |
commit | 8f56e1ab8b153dd6e7861fb3aa5d0f0295ddbaec (patch) | |
tree | c3687124286e5c915952a271aefcb56efeb831c0 /kvmd | |
parent | 7ce7a6d03526fb7c5d5acb8801e6340a6003ce6e (diff) |
Add minimal prometheus exporter support
Prometheus https://prometheus.io/ is one of the popular monitoring
systems. It pulls service's endpoint to get metrics in a simple
text format https://prometheus.io/docs/instrumenting/exposition_formats/
Diffstat (limited to 'kvmd')
-rw-r--r-- | kvmd/apps/kvmd/api/info.py | 32 | ||||
-rw-r--r-- | kvmd/apps/kvmd/http.py | 17 |
2 files changed, 49 insertions, 0 deletions
diff --git a/kvmd/apps/kvmd/api/info.py b/kvmd/apps/kvmd/api/info.py index b33a4b6c..478d26d8 100644 --- a/kvmd/apps/kvmd/api/info.py +++ b/kvmd/apps/kvmd/api/info.py @@ -22,6 +22,8 @@ import asyncio +from typing import Any +from typing import Dict from typing import List from aiohttp.web import Request @@ -34,6 +36,22 @@ from ..info import InfoManager from ..http import exposed_http from ..http import make_json_response +from ..http import make_text_response + + +# ==== +def _build_metrics(metrics: List[str], name: str, value: Any) -> None: + if isinstance(value, bool): + value = 1 if value else 0 + if isinstance(value, (int, float)): + metrics.append(f"# TYPE {name} gauge") + metrics.append(f"{name} {value}") + elif isinstance(value, dict): + for key, val in value.items(): + if key == "parsed_flags": + _build_metrics(metrics, name, val) + else: + _build_metrics(metrics, f"{name}_{key}", val) # ===== @@ -59,3 +77,17 @@ class InfoApi: subval=(lambda field: check_string_in_list(field, "info field", subs)), name="info fields list", ))) or subs) + + @exposed_http("GET", "/export/prometheus/metrics", False) + async def __metrics_handler(self, _: Request) -> Response: + data = await asyncio.gather(self.__info_manager.get_submanager("hw").get_state()) + if data is None: + return make_text_response("error", 500) + else: + data_exists: Dict[Any, Any] = data + health = data_exists[0]["health"] + + metrics: List[str] = [] + _build_metrics(metrics, "pikvm", health) + + return make_text_response("\n".join(metrics)) diff --git a/kvmd/apps/kvmd/http.py b/kvmd/apps/kvmd/http.py index 729c8498..0e870225 100644 --- a/kvmd/apps/kvmd/http.py +++ b/kvmd/apps/kvmd/http.py @@ -147,6 +147,23 @@ def make_json_exception(err: Exception, status: Optional[int]=None) -> aiohttp.w }, status=status) +def make_text_response( + result: str, + status: int=200, + set_cookies: Optional[Dict[str, str]]=None, +) -> aiohttp.web.Response: + + response = aiohttp.web.Response( + text=result, + status=status, + content_type="text/plain", + ) + if set_cookies: + for (key, value) in set_cookies.items(): + response.set_cookie(key, value) + return response + + # ===== async def get_multipart_field(reader: aiohttp.MultipartReader, name: str) -> aiohttp.BodyPartReader: field = await reader.next() |