refactoring

This commit is contained in:
Devaev Maxim 2020-08-30 23:30:45 +03:00
parent 164f2620cf
commit 146a44844a
4 changed files with 74 additions and 49 deletions

View File

@ -0,0 +1,72 @@
# ========================================================================== #
# #
# KVMD - The main Pi-KVM daemon. #
# #
# Copyright (C) 2018 Maxim Devaev <mdevaev@gmail.com> #
# #
# This program is free software: you can redistribute it and/or modify #
# it under the terms of the GNU General Public License as published by #
# the Free Software Foundation, either version 3 of the License, or #
# (at your option) any later version. #
# #
# This program is distributed in the hope that it will be useful, #
# but WITHOUT ANY WARRANTY; without even the implied warranty of #
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the #
# GNU General Public License for more details. #
# #
# You should have received a copy of the GNU General Public License #
# along with this program. If not, see <https://www.gnu.org/licenses/>. #
# #
# ========================================================================== #
import asyncio
import operator
from typing import Any
from typing import List
from aiohttp.web import Request
from aiohttp.web import Response
from ....plugins.atx import BaseAtx
from ..info import InfoManager
from ..http import exposed_http
# =====
class ExportApi:
def __init__(self, info_manager: InfoManager, atx: BaseAtx) -> None:
self.__info_manager = info_manager
self.__atx = atx
# =====
@exposed_http("GET", "/export/prometheus/metrics")
async def __prometheus_metrics_handler(self, _: Request) -> Response:
(atx_state, hw_state) = await asyncio.gather(*[
self.__atx.get_state(),
self.__info_manager.get_submanager("hw").get_state(),
])
rows: List[str] = []
self.__append_prometheus_rows(rows, atx_state["enabled"], "pikvm_atx_enabled")
self.__append_prometheus_rows(rows, atx_state["leds"]["power"], "pikvm_atx_power")
if hw_state is not None:
self.__append_prometheus_rows(rows, hw_state["health"], "pikvm_hw")
return Response(text="\n".join(rows))
def __append_prometheus_rows(self, rows: List[str], value: Any, path: str) -> None:
if isinstance(value, bool):
value = int(value)
if isinstance(value, (int, float)):
rows.extend([
f"# TYPE {path} gauge",
f"{path} {value}",
"",
])
elif isinstance(value, dict):
for (sub_key, sub_value) in sorted(value.items(), key=operator.itemgetter(0)):
sub_path = (f"{path}_{sub_key}" if sub_key != "parsed_flags" else path)
self.__append_prometheus_rows(rows, sub_value, sub_path)

View File

@ -22,8 +22,6 @@
import asyncio
from typing import Any
from typing import Dict
from typing import List
from aiohttp.web import Request
@ -36,22 +34,6 @@ 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)
# =====
@ -77,17 +59,3 @@ 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))

View File

@ -147,23 +147,6 @@ 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()

View File

@ -90,6 +90,7 @@ from .api.hid import HidApi
from .api.atx import AtxApi
from .api.msd import MsdApi
from .api.streamer import StreamerApi
from .api.export import ExportApi
# =====
@ -190,6 +191,7 @@ class KvmdServer(HttpServer): # pylint: disable=too-many-arguments,too-many-ins
AtxApi(atx),
MsdApi(msd, sync_chunk_size),
StreamerApi(streamer),
ExportApi(info_manager, atx),
]
self.__ws_handlers: Dict[str, Callable] = {}