prometheus-pve-sd/prometheuspvesd/cli.py

189 lines
6.0 KiB
Python
Raw Permalink Normal View History

2021-06-09 18:44:10 +00:00
#!/usr/bin/env python3
"""Entrypoint and CLI handler."""
import argparse
import json
import shutil
import signal
import tempfile
from os import chmod
2021-06-09 18:44:10 +00:00
from time import sleep
from prometheus_client import start_http_server
2021-06-09 18:44:10 +00:00
import prometheuspvesd.exception
from prometheuspvesd import __version__
from prometheuspvesd.config import SingleConfig
from prometheuspvesd.discovery import Discovery
2021-06-09 20:57:48 +00:00
from prometheuspvesd.exception import APIError
from prometheuspvesd.logger import SingleLog
2021-06-09 18:44:10 +00:00
from prometheuspvesd.model import HostList
class PrometheusSD:
"""Main Prometheus SD object."""
def __init__(self):
self.log = SingleLog()
self.logger = self.log.logger
self.args = self._cli_args()
self.config = self._get_config()
2021-06-09 20:57:48 +00:00
signal.signal(signal.SIGINT, self._terminate)
signal.signal(signal.SIGTERM, self._terminate)
2021-06-09 21:18:14 +00:00
2021-06-09 20:57:48 +00:00
while True:
try:
self.discovery = Discovery()
except APIError as e:
2021-06-09 21:18:14 +00:00
if not self.config.config["service"]:
self.log.sysexit_with_message(f"Proxmoxer API error: {str(e).strip()}")
2021-06-09 21:18:14 +00:00
self.logger.error(f"Proxmoxer API error: {str(e).strip()}")
2021-06-09 21:18:14 +00:00
sleep(60)
2021-06-09 20:57:48 +00:00
continue
2021-06-09 21:18:14 +00:00
else:
break
2021-06-09 20:57:48 +00:00
2021-06-09 18:44:10 +00:00
self._fetch()
def _cli_args(self):
"""
Use argparse for parsing CLI arguments.
:return: args objec
"""
parser = argparse.ArgumentParser(description="Prometheus Service Discovery for Proxmox VE")
parser.add_argument(
"-c",
"--config",
dest="config_file",
action="store",
help="location of configuration file",
2021-06-09 18:44:10 +00:00
)
parser.add_argument(
"-o", "--output", dest="output_file", action="store", help="output file"
)
parser.add_argument(
"-m", "--mode", dest="output_file_mode", action="store", help="output file mode"
)
parser.add_argument(
"-d",
"--loop-delay",
dest="loop_delay",
action="store",
2021-06-09 21:18:14 +00:00
type=int,
help="delay between discovery runs",
)
parser.add_argument(
"--no-service",
dest="service",
action="store_false",
default=None,
help="run discovery only once",
)
parser.add_argument(
"-f",
"--log-format",
dest="logging.format",
metavar="LOG_FORMAT",
action="store",
help="used log format",
)
2021-06-09 18:44:10 +00:00
parser.add_argument(
"-v", dest="logging.level", action="append_const", const=-1, help="increase log level"
)
parser.add_argument(
"-q", dest="logging.level", action="append_const", const=1, help="decrease log level"
)
parser.add_argument("--version", action="version", version=f"%(prog)s {__version__}")
2021-06-09 18:44:10 +00:00
return parser.parse_args().__dict__
def _get_config(self):
try:
config = SingleConfig(args=self.args)
except prometheuspvesd.exception.ConfigError as e:
self.log.sysexit_with_message(e)
try:
self.log.update_logger(
config.config["logging"]["level"], config.config["logging"]["format"]
)
2021-06-09 18:44:10 +00:00
except ValueError as e:
self.log.sysexit_with_message(f"Can not set log level.\n{e!s}")
2021-06-09 18:44:10 +00:00
required = [
("pve.server", config.config["pve"]["server"]),
("pve.user", config.config["pve"]["user"]),
]
2021-06-09 18:44:10 +00:00
for name, value in required:
if not value:
self.log.sysexit_with_message(f"Option '{name}' is required but not set")
2021-06-09 18:44:10 +00:00
if not config.config["pve"]["password"] and not (
config.config["pve"]["token_name"] and config.config["pve"]["token_value"]
):
self.log.sysexit_with_message(
"Either 'pve.password' or 'pve.token_name' and 'pve.token_value' "
"are required but not set"
)
self.logger.info(f"Using config file {config.config_file}")
2021-06-09 18:44:10 +00:00
return config
def _fetch(self):
self.logger.info("Writes targets to {}".format(self.config.config["output_file"]))
self.logger.debug("Propagate from PVE")
2021-06-09 18:44:10 +00:00
if self.config.config["service"] and self.config.config["metrics"]["enabled"]:
self.logger.info(
"Starting metrics http endpoint on port {}".format(
self.config.config["metrics"]["port"]
)
)
start_http_server(
self.config.config["metrics"]["port"],
addr=self.config.config["metrics"]["address"],
)
2021-06-09 18:44:10 +00:00
while True:
2021-06-09 20:57:48 +00:00
try:
inventory = self.discovery.propagate()
except APIError as e:
self.logger.error(f"Proxmoxer API error: {str(e).strip()}")
2021-06-09 20:57:48 +00:00
except Exception as e: # noqa
self.logger.error(f"Unknown error: {str(e).strip()}")
2021-06-09 20:57:48 +00:00
else:
self._write(inventory)
2021-06-09 18:44:10 +00:00
if not self.config.config["service"]:
break
self.logger.info(
"Waiting {} seconds for next discovery loop".format(
self.config.config["loop_delay"]
)
)
2021-06-09 18:44:10 +00:00
sleep(self.config.config["loop_delay"])
def _write(self, host_list: HostList):
output = []
for host in host_list.hosts:
output.append(host.to_sd_json())
# Write to tmp file and move after write
2024-10-21 18:25:28 +00:00
with tempfile.NamedTemporaryFile(mode="w", prefix="prometheus-pve-sd", delete=False) as tf:
2021-06-09 18:44:10 +00:00
json.dump(output, tf, indent=4)
2024-10-21 18:25:28 +00:00
shutil.move(tf.name, self.config.config["output_file"])
chmod(self.config.config["output_file"], int(self.config.config["output_file_mode"], 8))
2021-06-09 18:44:10 +00:00
def _terminate(self, signal, frame): # noqa
2021-06-09 18:44:10 +00:00
self.log.sysexit_with_message("Terminating", code=0)
def main():
PrometheusSD()