2021-10-04 21:34:49 +02:00
|
|
|
/** The "status" API request.
|
2017-09-06 20:31:36 +02:00
|
|
|
*
|
2022-12-14 17:41:58 +01:00
|
|
|
* @author Steffen Vogel <post@steffenvogel.de>
|
2022-03-15 09:28:57 -04:00
|
|
|
* @copyright 2014-2022, Institute for Automation of Complex Power Systems, EONERC
|
2022-07-04 18:20:03 +02:00
|
|
|
* @license Apache 2.0
|
2017-09-06 20:31:36 +02:00
|
|
|
*********************************************************************************/
|
|
|
|
|
2020-10-15 14:47:43 +02:00
|
|
|
#include <time.h>
|
|
|
|
#include <sys/utsname.h>
|
|
|
|
#include <sys/sysinfo.h>
|
2017-09-06 20:31:36 +02:00
|
|
|
|
2023-06-30 13:00:01 +02:00
|
|
|
#include <villas/uuid.hpp>
|
2021-08-10 10:12:48 -04:00
|
|
|
#include <villas/timing.hpp>
|
2020-08-17 17:03:54 +02:00
|
|
|
#include <villas/api/request.hpp>
|
|
|
|
#include <villas/api/response.hpp>
|
2018-10-20 14:20:06 +02:00
|
|
|
|
|
|
|
namespace villas {
|
|
|
|
namespace node {
|
|
|
|
namespace api {
|
|
|
|
|
2020-08-17 17:03:54 +02:00
|
|
|
class StatusRequest : public Request {
|
2018-10-20 14:20:06 +02:00
|
|
|
|
|
|
|
public:
|
2020-08-17 17:03:54 +02:00
|
|
|
using Request::Request;
|
2019-01-21 10:34:22 +01:00
|
|
|
|
2020-08-17 17:03:54 +02:00
|
|
|
virtual Response * execute()
|
2018-10-20 14:20:06 +02:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
2020-09-30 17:56:12 +02:00
|
|
|
if (method != Session::Method::GET)
|
2020-08-17 17:03:54 +02:00
|
|
|
throw InvalidMethod(this);
|
|
|
|
|
|
|
|
if (body != nullptr)
|
|
|
|
throw BadRequest("Status endpoint does not accept any body data");
|
|
|
|
|
2020-10-15 14:47:43 +02:00
|
|
|
auto *sn = session->getSuperNode();
|
|
|
|
|
|
|
|
struct utsname uts;
|
|
|
|
struct sysinfo sinfo;
|
|
|
|
char hname[128];
|
|
|
|
|
|
|
|
auto now = time_now();
|
|
|
|
auto started = sn->getStartTime();
|
|
|
|
|
|
|
|
ret = gethostname(hname, sizeof(hname));
|
2019-03-26 14:10:26 +01:00
|
|
|
if (ret)
|
2020-10-15 14:47:43 +02:00
|
|
|
throw Error(HTTP_STATUS_INTERNAL_SERVER_ERROR, "Failed to get system hostname");
|
|
|
|
|
|
|
|
ret = uname(&uts);
|
|
|
|
if (ret)
|
|
|
|
throw Error(HTTP_STATUS_INTERNAL_SERVER_ERROR, "Failed to get kernel information");
|
|
|
|
|
|
|
|
ret = sysinfo(&sinfo);
|
|
|
|
if (ret)
|
|
|
|
throw Error(HTTP_STATUS_INTERNAL_SERVER_ERROR, "Failed to get system information");
|
|
|
|
|
|
|
|
float f_load = 1.f / (1 << SI_LOAD_SHIFT);
|
2018-10-20 14:20:06 +02:00
|
|
|
|
2021-10-15 09:32:12 +02:00
|
|
|
tzset();
|
|
|
|
|
2020-10-15 14:47:43 +02:00
|
|
|
json_error_t err;
|
|
|
|
json_t *json_status = json_pack_ex(&err, 0, "{ s: s, s: s, s: s, s: s, s: s, s: s, s: s, s: f, s: f, s: { s: s, s: I, s: b }, s: { s: s, s: s, s: s, s: s, s: s, s: s}, s: { s: i, s: i, s: I, s: I, s: [ f, f, f ], s: { s: I, s, I, s: I, s: I }, s: { s: I, s: I }, s: { s: I, s: I } } }",
|
2021-10-04 21:30:53 +02:00
|
|
|
"state", stateToString(sn->getState()).c_str(),
|
2020-10-15 14:47:43 +02:00
|
|
|
"version", PROJECT_VERSION_STR,
|
|
|
|
"release", PROJECT_RELEASE,
|
|
|
|
"build_id", PROJECT_BUILD_ID,
|
|
|
|
"build_date", PROJECT_BUILD_DATE,
|
|
|
|
"hostname", hname,
|
2023-06-30 13:00:01 +02:00
|
|
|
"uuid", uuid::toString(sn->getUuid()).c_str(),
|
2020-10-15 14:47:43 +02:00
|
|
|
"time_now", time_to_double(&now),
|
|
|
|
"time_started", time_to_double(&started),
|
|
|
|
"timezone",
|
|
|
|
"name", tzname[daylight],
|
|
|
|
"offset", (json_int_t) timezone,
|
|
|
|
"dst", daylight,
|
|
|
|
"kernel",
|
|
|
|
"sysname", uts.sysname,
|
|
|
|
"nodename", uts.nodename,
|
|
|
|
"release", uts.release,
|
|
|
|
"version", uts.version,
|
|
|
|
"machine", uts.machine,
|
|
|
|
"domainname", uts.domainname,
|
|
|
|
"system",
|
|
|
|
"cores_configured", get_nprocs_conf(),
|
|
|
|
"cores", get_nprocs(),
|
2021-10-04 21:31:02 +02:00
|
|
|
"processes", (json_int_t) sinfo.procs,
|
2020-10-15 14:47:43 +02:00
|
|
|
"uptime", (json_int_t) sinfo.uptime,
|
|
|
|
"load",
|
|
|
|
f_load * sinfo.loads[0],
|
|
|
|
f_load * sinfo.loads[1],
|
|
|
|
f_load * sinfo.loads[2],
|
|
|
|
"ram",
|
|
|
|
"total", (json_int_t) (sinfo.totalram * sinfo.mem_unit),
|
|
|
|
"free", (json_int_t) (sinfo.freeram * sinfo.mem_unit),
|
|
|
|
"shared", (json_int_t) (sinfo.sharedram * sinfo.mem_unit),
|
|
|
|
"buffer", (json_int_t) (sinfo.bufferram * sinfo.mem_unit),
|
|
|
|
"swap",
|
|
|
|
"total", (json_int_t) (sinfo.totalswap * sinfo.mem_unit),
|
|
|
|
"free", (json_int_t) (sinfo.freeswap * sinfo.mem_unit),
|
|
|
|
"highmem",
|
|
|
|
"total", (json_int_t) (sinfo.totalhigh * sinfo.mem_unit),
|
|
|
|
"free", (json_int_t) (sinfo.freehigh * sinfo.mem_unit)
|
|
|
|
);
|
|
|
|
if (!json_status)
|
|
|
|
throw Error(HTTP_STATUS_INTERNAL_SERVER_ERROR, "Failed to prepare response: {}", err.text);
|
|
|
|
|
2020-10-20 22:17:55 +02:00
|
|
|
return new JsonResponse(session, HTTP_STATUS_OK, json_status);
|
2018-10-20 14:20:06 +02:00
|
|
|
}
|
2017-09-06 20:31:36 +02:00
|
|
|
};
|
|
|
|
|
2020-08-17 17:03:54 +02:00
|
|
|
/* Register API request */
|
2020-06-14 15:00:02 +02:00
|
|
|
static char n[] = "status";
|
2020-08-17 17:03:54 +02:00
|
|
|
static char r[] = "/status";
|
2020-06-14 15:00:02 +02:00
|
|
|
static char d[] = "get status and statistics of web server";
|
2020-08-17 17:03:54 +02:00
|
|
|
static RequestPlugin<StatusRequest, n, r, d> p;
|
2018-10-20 14:20:06 +02:00
|
|
|
|
2023-08-28 09:34:02 +02:00
|
|
|
} // namespace api
|
|
|
|
} // namespace node
|
|
|
|
} // namespace villas
|
2017-09-16 16:33:21 +02:00
|
|
|
|