freebsd-dev/usr.sbin/prometheus_sysctl_exporter/prometheus_sysctl_exporter.c
Ed Schouten 4f6a15dd39 Be a bit more liberal about sysctl naming.
On the systems on which I tested this exporter, I never ran into metrics
that were named in such a way that they couldn't be exported to
Prometheus metrics directly. Now it turns out that on systems with NUMA,
the sysctl tree contains metrics named dev.${driver}.${index}.%domain.
For these metrics, the % in the name is problematic, as Prometheus
doesn't allow this symbol to be used.

Remove the assertions that were originally put in place to prevent the
exporter from generating malformed output and add code to deal with it
accordingly. For metric names, convert any unsupported character to an
underscore. For label values, perform string escaping.

PR:		https://bugs.freebsd.org/bugzilla/show_bug.cgi?id=221035
Reported by:	lifanov@
2017-07-29 08:35:07 +00:00

658 lines
15 KiB
C

/*-
* Copyright (c) 2016-2017 Nuxi, https://nuxi.nl/
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*/
#include <sys/cdefs.h>
__FBSDID("$FreeBSD$");
#include <sys/param.h>
#include <sys/resource.h>
#include <sys/socket.h>
#include <sys/sysctl.h>
#include <assert.h>
#include <ctype.h>
#include <err.h>
#include <errno.h>
#include <math.h>
#include <stdbool.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <zlib.h>
/*
* Cursor for iterating over all of the system's sysctl OIDs.
*/
struct oid {
int id[CTL_MAXNAME];
size_t len;
};
/* Initializes the cursor to point to start of the tree. */
static void
oid_get_root(struct oid *o)
{
o->id[0] = 1;
o->len = 1;
}
/* Obtains the OID for a sysctl by name. */
static void
oid_get_by_name(struct oid *o, const char *name)
{
o->len = nitems(o->id);
if (sysctlnametomib(name, o->id, &o->len) != 0)
err(1, "sysctl(%s)", name);
}
/* Returns whether an OID is placed below another OID. */
static bool
oid_is_beneath(struct oid *oa, struct oid *ob)
{
return (oa->len >= ob->len &&
memcmp(oa->id, ob->id, ob->len * sizeof(oa->id[0])) == 0);
}
/* Advances the cursor to the next OID. */
static bool
oid_get_next(const struct oid *cur, struct oid *next)
{
int lookup[CTL_MAXNAME + 2];
size_t nextsize;
lookup[0] = 0;
lookup[1] = 2;
memcpy(lookup + 2, cur->id, cur->len * sizeof(lookup[0]));
nextsize = sizeof(next->id);
if (sysctl(lookup, 2 + cur->len, &next->id, &nextsize, 0, 0) != 0) {
if (errno == ENOENT)
return (false);
err(1, "sysctl(next)");
}
next->len = nextsize / sizeof(next->id[0]);
return (true);
}
/*
* OID formatting metadata.
*/
struct oidformat {
unsigned int kind;
char format[BUFSIZ];
};
/* Returns whether the OID represents a temperature value. */
static bool
oidformat_is_temperature(const struct oidformat *of)
{
return (of->format[0] == 'I' && of->format[1] == 'K');
}
/* Returns whether the OID represents a timeval structure. */
static bool
oidformat_is_timeval(const struct oidformat *of)
{
return (strcmp(of->format, "S,timeval") == 0);
}
/* Fetches the formatting metadata for an OID. */
static bool
oid_get_format(const struct oid *o, struct oidformat *of)
{
int lookup[CTL_MAXNAME + 2];
size_t oflen;
lookup[0] = 0;
lookup[1] = 4;
memcpy(lookup + 2, o->id, o->len * sizeof(lookup[0]));
oflen = sizeof(*of);
if (sysctl(lookup, 2 + o->len, of, &oflen, 0, 0) != 0) {
if (errno == ENOENT)
return (false);
err(1, "sysctl(oidfmt)");
}
return (true);
}
/*
* Container for holding the value of an OID.
*/
struct oidvalue {
enum { SIGNED, UNSIGNED, FLOAT } type;
union {
intmax_t s;
uintmax_t u;
double f;
} value;
};
/* Extracts the value of an OID, converting it to a floating-point number. */
static double
oidvalue_get_float(const struct oidvalue *ov)
{
switch (ov->type) {
case SIGNED:
return (ov->value.s);
case UNSIGNED:
return (ov->value.u);
case FLOAT:
return (ov->value.f);
default:
assert(0 && "Unknown value type");
}
}
/* Sets the value of an OID as a signed integer. */
static void
oidvalue_set_signed(struct oidvalue *ov, intmax_t s)
{
ov->type = SIGNED;
ov->value.s = s;
}
/* Sets the value of an OID as an unsigned integer. */
static void
oidvalue_set_unsigned(struct oidvalue *ov, uintmax_t u)
{
ov->type = UNSIGNED;
ov->value.u = u;
}
/* Sets the value of an OID as a floating-point number. */
static void
oidvalue_set_float(struct oidvalue *ov, double f)
{
ov->type = FLOAT;
ov->value.f = f;
}
/* Prints the value of an OID to a file stream. */
static void
oidvalue_print(const struct oidvalue *ov, FILE *fp)
{
switch (ov->type) {
case SIGNED:
fprintf(fp, "%jd", ov->value.s);
break;
case UNSIGNED:
fprintf(fp, "%ju", ov->value.u);
break;
case FLOAT:
switch (fpclassify(ov->value.f)) {
case FP_INFINITE:
if (signbit(ov->value.f))
fprintf(fp, "-Inf");
else
fprintf(fp, "+Inf");
break;
case FP_NAN:
fprintf(fp, "Nan");
break;
default:
fprintf(fp, "%.6f", ov->value.f);
break;
}
break;
}
}
/* Fetches the value of an OID. */
static bool
oid_get_value(const struct oid *o, const struct oidformat *of,
struct oidvalue *ov)
{
switch (of->kind & CTLTYPE) {
#define GET_VALUE(ctltype, type) \
case (ctltype): { \
type value; \
size_t valuesize; \
\
valuesize = sizeof(value); \
if (sysctl(o->id, o->len, &value, &valuesize, 0, 0) != 0) \
return (false); \
if ((type)-1 > 0) \
oidvalue_set_unsigned(ov, value); \
else \
oidvalue_set_signed(ov, value); \
break; \
}
GET_VALUE(CTLTYPE_INT, int);
GET_VALUE(CTLTYPE_UINT, unsigned int);
GET_VALUE(CTLTYPE_LONG, long);
GET_VALUE(CTLTYPE_ULONG, unsigned long);
GET_VALUE(CTLTYPE_S8, int8_t);
GET_VALUE(CTLTYPE_U8, uint8_t);
GET_VALUE(CTLTYPE_S16, int16_t);
GET_VALUE(CTLTYPE_U16, uint16_t);
GET_VALUE(CTLTYPE_S32, int32_t);
GET_VALUE(CTLTYPE_U32, uint32_t);
GET_VALUE(CTLTYPE_S64, int64_t);
GET_VALUE(CTLTYPE_U64, uint64_t);
#undef GET_VALUE
case CTLTYPE_OPAQUE:
if (oidformat_is_timeval(of)) {
struct timeval tv;
size_t tvsize;
tvsize = sizeof(tv);
if (sysctl(o->id, o->len, &tv, &tvsize, 0, 0) != 0)
return (false);
oidvalue_set_float(ov,
(double)tv.tv_sec + (double)tv.tv_usec / 1000000);
return (true);
} else if (strcmp(of->format, "S,loadavg") == 0) {
struct loadavg la;
size_t lasize;
/*
* Only return the one minute load average, as
* the others can be inferred using avg_over_time().
*/
lasize = sizeof(la);
if (sysctl(o->id, o->len, &la, &lasize, 0, 0) != 0)
return (false);
oidvalue_set_float(ov,
(double)la.ldavg[0] / (double)la.fscale);
return (true);
}
return (false);
default:
return (false);
}
/* Convert temperatures from decikelvin to degrees Celcius. */
if (oidformat_is_temperature(of)) {
double v;
int e;
v = oidvalue_get_float(ov);
if (v < 0) {
oidvalue_set_float(ov, NAN);
} else {
e = of->format[2] >= '0' && of->format[2] <= '9' ?
of->format[2] - '0' : 1;
oidvalue_set_float(ov, v / pow(10, e) - 273.15);
}
}
return (true);
}
/*
* The full name of an OID, stored as a series of components.
*/
struct oidname {
struct oid oid;
char names[BUFSIZ];
char labels[BUFSIZ];
};
/*
* Initializes the OID name object with an empty value.
*/
static void
oidname_init(struct oidname *on)
{
on->oid.len = 0;
}
/* Fetches the name and labels of an OID, reusing the previous results. */
static void
oid_get_name(const struct oid *o, struct oidname *on)
{
int lookup[CTL_MAXNAME + 2];
char *c, *label;
size_t i, len;
/* Fetch the name and split it up in separate components. */
lookup[0] = 0;
lookup[1] = 1;
memcpy(lookup + 2, o->id, o->len * sizeof(lookup[0]));
len = sizeof(on->names);
if (sysctl(lookup, 2 + o->len, on->names, &len, 0, 0) != 0)
err(1, "sysctl(name)");
for (c = strchr(on->names, '.'); c != NULL; c = strchr(c + 1, '.'))
*c = '\0';
/* No need to fetch labels for components that we already have. */
label = on->labels;
for (i = 0; i < o->len && i < on->oid.len && o->id[i] == on->oid.id[i];
++i)
label += strlen(label) + 1;
/* Fetch the remaining labels. */
lookup[1] = 6;
for (; i < o->len; ++i) {
len = on->labels + sizeof(on->labels) - label;
if (sysctl(lookup, 2 + i + 1, label, &len, 0, 0) == 0) {
label += len;
} else if (errno == ENOENT) {
*label++ = '\0';
} else {
err(1, "sysctl(oidlabel)");
}
}
on->oid = *o;
}
/* Prints the name and labels of an OID to a file stream. */
static void
oidname_print(const struct oidname *on, const struct oidformat *of,
FILE *fp)
{
const char *name, *label;
size_t i;
char separator;
/* Print the name of the metric. */
fprintf(fp, "sysctl");
name = on->names;
label = on->labels;
for (i = 0; i < on->oid.len; ++i) {
if (*label == '\0') {
fputc('_', fp);
while (*name != '\0') {
/* Map unsupported characters to underscores. */
fputc(isalnum(*name) ? *name : '_', fp);
++name;
}
}
name += strlen(name) + 1;
label += strlen(label) + 1;
}
if (oidformat_is_temperature(of))
fprintf(fp, "_celcius");
else if (oidformat_is_timeval(of))
fprintf(fp, "_seconds");
/* Print the labels of the metric. */
name = on->names;
label = on->labels;
separator = '{';
for (i = 0; i < on->oid.len; ++i) {
if (*label != '\0') {
assert(label[strspn(label,
"abcdefghijklmnopqrstuvwxyz"
"ABCDEFGHIJKLMNOPQRSTUVWXYZ"
"0123456789_")] == '\0');
fprintf(fp, "%c%s=\"", separator, label);
while (*name != '\0') {
/* Escape backslashes and double quotes. */
if (*name == '\\' || *name == '"')
fputc('\\', fp);
fputc(*name++, fp);
}
fputc('"', fp);
separator = ',';
}
name += strlen(name) + 1;
label += strlen(label) + 1;
}
if (separator != '{')
fputc('}', fp);
}
/* Returns whether the OID name has any labels associated to it. */
static bool
oidname_has_labels(const struct oidname *on)
{
size_t i;
for (i = 0; i < on->oid.len; ++i)
if (on->labels[i] != 0)
return (true);
return (false);
}
/*
* The description of an OID.
*/
struct oiddescription {
char description[BUFSIZ];
};
/*
* Fetches the description of an OID.
*/
static bool
oid_get_description(const struct oid *o, struct oiddescription *od)
{
int lookup[CTL_MAXNAME + 2];
char *newline;
size_t odlen;
lookup[0] = 0;
lookup[1] = 5;
memcpy(lookup + 2, o->id, o->len * sizeof(lookup[0]));
odlen = sizeof(od->description);
if (sysctl(lookup, 2 + o->len, &od->description, &odlen, 0, 0) != 0) {
if (errno == ENOENT)
return (false);
err(1, "sysctl(oiddescr)");
}
newline = strchr(od->description, '\n');
if (newline != NULL)
*newline = '\0';
return (*od->description != '\0');
}
/* Prints the description of an OID to a file stream. */
static void
oiddescription_print(const struct oiddescription *od, FILE *fp)
{
fprintf(fp, "%s", od->description);
}
static void
oid_print(const struct oid *o, struct oidname *on, bool print_description,
FILE *fp)
{
struct oidformat of;
struct oidvalue ov;
struct oiddescription od;
if (!oid_get_format(o, &of) || !oid_get_value(o, &of, &ov))
return;
oid_get_name(o, on);
/*
* Print the line with the description. Prometheus expects a
* single unique description for every metric, which cannot be
* guaranteed by sysctl if labels are present. Omit the
* description if labels are present.
*/
if (print_description && !oidname_has_labels(on) &&
oid_get_description(o, &od)) {
fprintf(fp, "# HELP ");
oidname_print(on, &of, fp);
fputc(' ', fp);
oiddescription_print(&od, fp);
fputc('\n', fp);
}
/* Print the line with the value. */
oidname_print(on, &of, fp);
fputc(' ', fp);
oidvalue_print(&ov, fp);
fputc('\n', fp);
}
/* Gzip compresses a buffer of memory. */
static bool
buf_gzip(const char *in, size_t inlen, char *out, size_t *outlen)
{
z_stream stream = {
.next_in = __DECONST(unsigned char *, in),
.avail_in = inlen,
.next_out = (unsigned char *)out,
.avail_out = *outlen,
};
if (deflateInit2(&stream, Z_DEFAULT_COMPRESSION, Z_DEFLATED,
MAX_WBITS + 16, 8, Z_DEFAULT_STRATEGY) != Z_OK ||
deflate(&stream, Z_FINISH) != Z_STREAM_END) {
return (false);
}
*outlen = stream.total_out;
return (deflateEnd(&stream) == Z_OK);
}
static void
usage(void)
{
fprintf(stderr,
"usage: prometheus_sysctl_exporter [-dgh] [prefix ...]\n");
exit(1);
}
int
main(int argc, char *argv[])
{
struct oidname on;
char *http_buf;
FILE *fp;
size_t http_buflen;
int ch;
bool gzip_mode, http_mode, print_descriptions;
/* Parse command line flags. */
gzip_mode = http_mode = print_descriptions = false;
while ((ch = getopt(argc, argv, "dgh")) != -1) {
switch (ch) {
case 'd':
print_descriptions = true;
break;
case 'g':
gzip_mode = true;
break;
case 'h':
http_mode = true;
break;
default:
usage();
}
}
argc -= optind;
argv += optind;
/* HTTP output: cache metrics in buffer. */
if (http_mode) {
fp = open_memstream(&http_buf, &http_buflen);
if (fp == NULL)
err(1, "open_memstream");
} else {
fp = stdout;
}
oidname_init(&on);
if (argc == 0) {
struct oid o;
/* Print all OIDs. */
oid_get_root(&o);
do {
oid_print(&o, &on, print_descriptions, fp);
} while (oid_get_next(&o, &o));
} else {
int i;
/* Print only trees provided as arguments. */
for (i = 0; i < argc; ++i) {
struct oid o, root;
oid_get_by_name(&root, argv[i]);
o = root;
do {
oid_print(&o, &on, print_descriptions, fp);
} while (oid_get_next(&o, &o) &&
oid_is_beneath(&o, &root));
}
}
if (http_mode) {
const char *content_encoding = "";
if (ferror(fp) || fclose(fp) != 0)
err(1, "Cannot generate output");
/* Gzip compress the output. */
if (gzip_mode) {
char *buf;
size_t buflen;
buflen = http_buflen;
buf = malloc(buflen);
if (buf == NULL)
err(1, "Cannot allocate compression buffer");
if (buf_gzip(http_buf, http_buflen, buf, &buflen)) {
content_encoding = "Content-Encoding: gzip\r\n";
free(http_buf);
http_buf = buf;
http_buflen = buflen;
} else {
free(buf);
}
}
/* Print HTTP header and metrics. */
dprintf(STDOUT_FILENO,
"HTTP/1.1 200 OK\r\n"
"Connection: close\r\n"
"%s"
"Content-Length: %zu\r\n"
"Content-Type: text/plain; version=0.0.4\r\n"
"\r\n",
content_encoding, http_buflen);
write(STDOUT_FILENO, http_buf, http_buflen);
free(http_buf);
/* Drain output. */
if (shutdown(STDIN_FILENO, SHUT_WR) == 0) {
char buf[1024];
while (read(STDIN_FILENO, buf, sizeof(buf)) > 0) {
}
}
}
return (0);
}