1
0
Fork 0
mirror of https://git.rwth-aachen.de/acs/public/villas/node/ synced 2025-03-09 00:00:00 +01:00
VILLASnode/lib/api/requests/status.cpp

163 lines
4.8 KiB
C++
Raw Permalink Normal View History

2020-08-17 17:03:54 +02:00
/** The "stats" API request.
*
* @author Steffen Vogel <stvogel@eonerc.rwth-aachen.de>
2020-01-20 17:17:00 +01:00
* @copyright 2014-2020, Institute for Automation of Complex Power Systems, EONERC
* @license GNU General Public License (version 3)
*
* VILLASnode
*
* 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
* 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 <http://www.gnu.org/licenses/>.
*********************************************************************************/
2020-10-15 14:47:43 +02:00
#include <time.h>
#include <uuid/uuid.h>
#include <sys/utsname.h>
#include <sys/sysinfo.h>
2020-10-15 14:47:43 +02:00
#include <villas/timing.h>
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
typedef char uuid_string_t[37];
2020-10-15 14:47:43 +02:00
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
2020-10-15 14:47:43 +02:00
protected:
#ifdef LWS_WITH_SERVER_STATUS
json_t * getLwsStatus()
{
int ret;
struct lws_context *ctx = lws_get_context(session->wsi);
char buf[4096];
ret = lws_json_dump_context(ctx, buf, sizeof(buf), 0);
2021-01-08 19:29:20 +01:00
if (ret <= 0)
throw Error(HTTP_STATUS_INTERNAL_SERVER_ERROR, "Failed to dump LWS context");
2020-10-15 14:47:43 +02:00
return json_loads(buf, 0, nullptr);
}
#endif /* LWS_WITH_SERVER_STATUS */
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;
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();
uuid_t uuid;
uuid_string_t uuid_str;
2020-10-15 14:47:43 +02:00
struct utsname uts;
struct sysinfo sinfo;
char hname[128];
sn->getUUID(uuid);
uuid_unparse_lower(uuid, uuid_str);
auto now = time_now();
auto started = sn->getStartTime();
ret = gethostname(hname, sizeof(hname));
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
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 } } }",
"state", state_print(sn->getState()),
"version", PROJECT_VERSION_STR,
"release", PROJECT_RELEASE,
"build_id", PROJECT_BUILD_ID,
"build_date", PROJECT_BUILD_DATE,
"hostname", hname,
"uuid", uuid_str,
"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(),
"procecces", (json_int_t) sinfo.procs,
"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);
#ifdef LWS_WITH_SERVER_STATUS
json_object_set(json_status, "lws", getLwsStatus());
#endif /* LWS_WITH_SERVER_STATUS */
2018-10-20 14:20:06 +02:00
return new JsonResponse(session, HTTP_STATUS_OK, json_status);
2018-10-20 14:20:06 +02:00
}
};
2020-08-17 17:03:54 +02:00
/* Register API request */
static char n[] = "status";
2020-08-17 17:03:54 +02:00
static char r[] = "/status";
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
} /* namespace api */
} /* namespace node */
} /* namespace villas */