X-Git-Url: https://git.verplant.org/?a=blobdiff_plain;f=common.c;h=2b6b5eb9180efa2272695de031d0d3d4c6d49ac0;hb=3dfc071fbc2e15897d1888e7d75c444a7a801189;hp=a2bfd090f612a255104874b770e7b2dec60a2215;hpb=9e5770370bb085ccfb19f3dd14707542bf1475f1;p=collection4.git diff --git a/common.c b/common.c index a2bfd09..2b6b5eb 100644 --- a/common.c +++ b/common.c @@ -1,145 +1,193 @@ #include #include +#include +#include #include #include #include #include #include #include +#include +#include + +#include #include "common.h" +#include "graph_list.h" + +#include +#include -static int foreach_rrd_file (const char *dir, /* {{{ */ - int (*callback) (const char *, void *), - void *user_data) +size_t c_strlcat (char *dst, const char *src, size_t size) /* {{{ */ { - DIR *dh; - struct dirent *entry; - int status; + size_t retval; + size_t dst_len; + size_t src_len; - if (callback == NULL) - return (EINVAL); + dst_len = strlen (dst); + src_len = strlen (src); + retval = dst_len + src_len; - dh = opendir (dir); - if (dh == NULL) - return (errno); + if ((dst_len + 1) >= size) + return (retval); - while ((entry = readdir (dh)) != NULL) - { - struct stat statbuf; - char abspath[PATH_MAX + 1]; - size_t d_name_len; + dst += dst_len; + size -= dst_len; + assert (size >= 2); - if (entry->d_name[0] == '.') - continue; + /* Result will be truncated. */ + if (src_len >= size) + src_len = size - 1; - d_name_len = strlen (entry->d_name); - if (d_name_len <= 4) - continue; + memcpy (dst, src, src_len); + dst[src_len] = 0; - if (strcasecmp (".rrd", entry->d_name + (d_name_len - 4)) != 0) - continue; + return (retval); +} /* }}} size_t c_strlcat */ - snprintf (abspath, sizeof (abspath), "%s/%s", dir, entry->d_name); - abspath[sizeof (abspath) - 1] = 0; +int ds_list_from_rrd_file (char *file, /* {{{ */ + size_t *ret_dses_num, char ***ret_dses) +{ + char *rrd_argv[] = { "info", file, NULL }; + int rrd_argc = (sizeof (rrd_argv) / sizeof (rrd_argv[0])) - 1; - memset (&statbuf, 0, sizeof (statbuf)); + rrd_info_t *info; + rrd_info_t *ptr; - status = stat (abspath, &statbuf); - if (status != 0) - continue; + char **dses = NULL; + size_t dses_num = 0; - if (!S_ISREG (statbuf.st_mode)) - continue; + info = rrd_info (rrd_argc, rrd_argv); + if (info == NULL) + { + printf ("%s: rrd_info (%s) failed.\n", __func__, file); + return (-1); + } - entry->d_name[d_name_len - 4] = 0; + for (ptr = info; ptr != NULL; ptr = ptr->next) + { + size_t keylen; + size_t dslen; + char *ds; + char **tmp; - status = (*callback) (entry->d_name, user_data); - if (status != 0) - break; - } /* while (readdir) */ + if (strncmp ("ds[", ptr->key, strlen ("ds[")) != 0) + continue; - closedir (dh); - return (status); -} /* }}} int foreach_rrd_file */ + keylen = strlen (ptr->key); + if (keylen < strlen ("ds[?].index")) + continue; -static int foreach_dir (const char *dir, /* {{{ */ - int (*callback) (const char *, void *), - void *user_data) -{ - DIR *dh; - struct dirent *entry; - int status; + dslen = keylen - strlen ("ds[].index"); + assert (dslen >= 1); - if (callback == NULL) - return (EINVAL); + if (strcmp ("].index", ptr->key + (strlen ("ds[") + dslen)) != 0) + continue; - dh = opendir (dir); - if (dh == NULL) - return (errno); + ds = malloc (dslen + 1); + if (ds == NULL) + continue; - while ((entry = readdir (dh)) != NULL) - { - struct stat statbuf; - char abspath[PATH_MAX + 1]; + memcpy (ds, ptr->key + strlen ("ds["), dslen); + ds[dslen] = 0; - if (entry->d_name[0] == '.') + tmp = realloc (dses, sizeof (*dses) * (dses_num + 1)); + if (tmp == NULL) + { + free (ds); continue; + } + dses = tmp; - snprintf (abspath, sizeof (abspath), "%s/%s", dir, entry->d_name); - abspath[sizeof (abspath) - 1] = 0; + dses[dses_num] = ds; + dses_num++; + } - memset (&statbuf, 0, sizeof (statbuf)); + rrd_info_free (info); - status = stat (abspath, &statbuf); - if (status != 0) - continue; + if (dses_num < 1) + { + assert (dses == NULL); + return (ENOENT); + } - if (!S_ISDIR (statbuf.st_mode)) - continue; + *ret_dses_num = dses_num; + *ret_dses = dses; - status = (*callback) (entry->d_name, user_data); - if (status != 0) - break; - } /* while (readdir) */ + return (0); +} /* }}} int ds_list_from_rrd_file */ - closedir (dh); - return (status); -} /* }}} int foreach_dir */ +static int hsv_to_rgb (double *hsv, double *rgb) /* {{{ */ +{ + double c = hsv[2] * hsv[1]; + double h = hsv[0] / 60.0; + double x = c * (1.0 - fabs (fmod (h, 2.0) - 1)); + double m = hsv[2] - c; + + rgb[0] = 0.0; + rgb[1] = 0.0; + rgb[2] = 0.0; + + if ((0.0 <= h) && (h < 1.0)) { rgb[0] = 1.0; rgb[1] = x; rgb[2] = 0.0; } + else if ((1.0 <= h) && (h < 2.0)) { rgb[0] = x; rgb[1] = 1.0; rgb[2] = 0.0; } + else if ((2.0 <= h) && (h < 3.0)) { rgb[0] = 0.0; rgb[1] = 1.0; rgb[2] = x; } + else if ((3.0 <= h) && (h < 4.0)) { rgb[0] = 0.0; rgb[1] = x; rgb[2] = 1.0; } + else if ((4.0 <= h) && (h < 5.0)) { rgb[0] = x; rgb[1] = 0.0; rgb[2] = 1.0; } + else if ((5.0 <= h) && (h < 6.0)) { rgb[0] = 1.0; rgb[1] = 0.0; rgb[2] = x; } + + rgb[0] += m; + rgb[1] += m; + rgb[2] += m; + + return (0); +} /* }}} int hsv_to_rgb */ + +static uint32_t rgb_to_uint32 (double *rgb) /* {{{ */ +{ + uint8_t r; + uint8_t g; + uint8_t b; + + r = (uint8_t) (255.0 * rgb[0]); + g = (uint8_t) (255.0 * rgb[1]); + b = (uint8_t) (255.0 * rgb[2]); + + return ((((uint32_t) r) << 16) + | (((uint32_t) g) << 8) + | ((uint32_t) b)); +} /* }}} uint32_t rgb_to_uint32 */ -int foreach_type (const char *host, const char *plugin, /* {{{ */ - callback_type_t callback, void *user_data) +uint32_t get_random_color (void) /* {{{ */ { - char abspath[PATH_MAX + 1]; + double hsv[3] = { 0.0, 1.0, 1.0 }; + double rgb[3] = { 0.0, 0.0, 0.0 }; - if ((host == NULL) || (plugin == NULL)) - return (EINVAL); + hsv[0] = 360.0 * ((double) rand ()) / (((double) RAND_MAX) + 1.0); - snprintf (abspath, sizeof (abspath), "%s/%s/%s", DATA_DIR, host, plugin); - abspath[sizeof (abspath) - 1] = 0; + hsv_to_rgb (hsv, rgb); - return (foreach_rrd_file (abspath, callback, user_data)); -} /* }}} int foreach_type */ + return (rgb_to_uint32 (rgb)); +} /* }}} uint32_t get_random_color */ -int foreach_plugin (const char *host, /* {{{ */ - callback_plugin_t callback, - void *user_data) +int print_debug (const char *format, ...) /* {{{ */ { - char abspath[PATH_MAX + 1]; + static _Bool have_header = 0; - if (host == NULL) - return (EINVAL); + va_list ap; + int status; - snprintf (abspath, sizeof (abspath), "%s/%s", DATA_DIR, host); - abspath[sizeof (abspath) - 1] = 0; + if (!have_header) + { + printf ("Content-Type: text/plain\n\n"); + have_header = 1; + } - return (foreach_dir (abspath, callback, user_data)); -} /* }}} int foreach_plugin */ + va_start (ap, format); + status = vprintf (format, ap); + va_end (ap); -int foreach_host (callback_host_t callback, /* {{{ */ - void *user_data) -{ - return (foreach_dir (DATA_DIR, callback, user_data)); -} /* }}} int foreach_host */ + return (status); +} /* }}} int print_debug */ /* vim: set sw=2 sts=2 et fdm=marker : */