blob: ee72a427b52107f49bf7c9e8261d05af584deb1e [file] [log] [blame]
# Author: Alex Savatieiev (osavatieiev@mirantis.com; a.savex@gmail.com)
# Copyright 2019-2022 Mirantis, Inc.
from gevent import monkey, pywsgi
monkey.patch_all()
import falcon # noqa E402
import os # noqa E402
import json # noqa E402
from copy import deepcopy # noqa E402
from platform import system, release, node # noqa E402
from cfg_checker.common.log import logger # noqa E402
from cfg_checker.common.settings import pkg_dir # noqa E402
from cfg_checker.common.exception import CheckerException # noqa E402
from cfg_checker.helpers.falcon_jinja2 import FalconTemplate # noqa E402
from .fio_runner import FioProcessShellRun, get_time # noqa E402
template = FalconTemplate(
path=os.path.join(pkg_dir, "templates")
)
_module_status = {
"status": "unknown",
"healthcheck": {},
"actions": [],
"options": {},
"uri": "<agent_uri>/api/<modile_name>",
}
_action = {
"module": "<name>",
"action": "<action>",
"options": "<options_dict>"
}
_modules = {
"fio": deepcopy(_module_status)
}
_status = {
"agent": {
"started": get_time()
},
"modules": list(_modules.keys()),
"help": {
".../api": {
"GET": "<this_status>",
"POST": json.dumps(_action)
},
".../api/<module_name>": {
"GET": "returns healthcheck and module help"
}
}
}
# Populate modules
_fio = FioProcessShellRun()
def _init_status(mod):
_modules[mod]["uri"] = "<agent_uri>/api/fio"
_modules[mod]["actions"] = list(_fio.actions.keys())
def _update_status(mod):
_modules[mod]["healthcheck"] = _fio.hchk
_modules[mod]["options"] = _fio.get_options()
_modules[mod].update(_fio.status())
class FioStatus:
_name = "fio"
def on_get(self, req, resp):
# Hacky way to handle empty request
_m = req.get_media(default_when_empty={})
if "fast" in _m and _m["fast"]:
resp.status = falcon.HTTP_200
resp.content_type = "application/json"
resp.text = json.dumps(_fio.status())
else:
_update_status(self._name)
resp.status = falcon.HTTP_200
resp.content_type = "application/json"
resp.text = json.dumps(_modules[self._name])
class Api:
def on_get(self, req, resp):
# return api status
resp.status = falcon.HTTP_200
resp.content_type = "application/json"
resp.text = json.dumps(_status)
def on_post(self, req, resp):
def _resp(resp, code, msg, opt={}):
resp.status = code
resp.content_type = "application/json"
resp.text = json.dumps({"error": msg, "options": opt})
# Handle actions
logger.info("Getting media")
try:
_m = req.stream.read()
_m = json.loads(_m)
except json.JSONDecodeError:
_msg = "Incorrect input data"
logger.error(_msg)
_resp(resp, falcon.HTTP_400, _msg)
return
if _m:
logger.debug("got media object:\n{}".format(json.dumps(_m)))
# Validate action structure
_module = _m.get('module', "")
_action = _m.get('action', "")
_options = _m.get('options', {})
if not _module or _module not in list(_modules.keys()):
logger.error("invalid module '{}'".format(_module))
_resp(
resp,
falcon.HTTP_400,
"Invalid module '{}'".format(_module)
)
return
elif not _action or _action not in _modules[_module]['actions']:
logger.error("invalid action '{}'".format(_action))
_resp(
resp,
falcon.HTTP_400,
"Invalid action '{}'".format(_action)
)
return
else:
# Handle command
_a = _fio.actions[_action]
try:
if _action == 'get_options':
logger.info("returning options")
resp.status = falcon.HTTP_200
resp.content_type = "application/json"
resp.text = json.dumps({"options": _a()})
elif _action == 'get_resultlist':
logger.info("getting results")
resp.status = falcon.HTTP_200
resp.content_type = "application/json"
resp.text = json.dumps({"resultlist": _a()})
elif _action == 'get_result':
if 'time' not in _options:
_msg = "No 'time' found for '{}'".format(_action)
logger.error(_msg)
_resp(
resp,
falcon.HTTP_400,
_msg
)
return
_time = _options['time']
logger.info("getting results for '{}'".format(_time))
resp.status = falcon.HTTP_200
resp.content_type = "application/json"
# TODO: get timeline too?
resp.text = json.dumps({_time: _a(_time)})
elif _action == 'do_singlerun':
logger.info("executing single run")
_a(_options)
resp.status = falcon.HTTP_200
resp.content_type = "application/json"
resp.text = json.dumps({"ok": True})
elif _action == 'do_scheduledrun':
logger.info("executing scheduled run")
# prepare scheduled run
# Run it
_a(_options)
resp.status = falcon.HTTP_200
resp.content_type = "application/json"
resp.text = json.dumps({"ok": True})
else:
_msg = "Unknown error happened for '{}/{}/{}'".format(
_module,
_action,
_options
)
logger.error(_msg)
_resp(resp, falcon.HTTP_500, _msg)
except CheckerException as e:
_msg = "Error for '{}/{}':\n{}".format(
_module,
_action,
e
)
logger.error(_msg)
_resp(resp, falcon.HTTP_500, _msg, opt=_options)
return
else:
_msg = "Empty request body"
logger.error(_msg)
_resp(resp, falcon.HTTP_400, _msg)
class Index:
@template.render("agent_index_html.j2")
def on_get(self, request, response):
# prepare template context
_context = {
"gen_date": get_time(),
"system": system(),
"release": release(),
"hostname": node()
}
_context.update(_status)
# creating response
response.status = falcon.HTTP_200
response.content_type = "text/html"
response.context = _context
def agent_server(host="0.0.0.0", port=8765):
# init api
api = falcon.API()
# populate pages
api.add_route("/", Index())
api.add_route("/api/", Api())
# Populate modules list
_active_modules = [FioStatus]
# init modules
for mod in _active_modules:
_init_status(mod._name)
_update_status(mod._name)
api.add_route("/api/"+mod._name, mod())
# init and start server
server = pywsgi.WSGIServer((host, port), api)
server.serve_forever()