X-Git-Url: https://git.octo.it/?p=collection4.git;a=blobdiff_plain;f=src%2Fgraph.c;h=544a606d80cf06b1d2d728650e5cde56e6cf74ed;hp=02b3ebf7bc36b692218ada66700d6ddeab043cd5;hb=33e56e5052a805658254653ea7721bbd95a7fd51;hpb=2ab3d5bed954cccd4486ed3c8255f27b0a6ab7eb diff --git a/src/graph.c b/src/graph.c index 02b3ebf..544a606 100644 --- a/src/graph.c +++ b/src/graph.c @@ -1,3 +1,26 @@ +/** + * collection4 - graph.c + * Copyright (C) 2010 Florian octo Forster + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 2.1 of the License, or (at your option) 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 + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, + * Boston, MA 02110-1301 USA + * + * Authors: + * Florian octo Forster + **/ + #include #include #include @@ -8,8 +31,9 @@ #include #include "graph.h" -#include "graph_list.h" #include "graph_ident.h" +#include "graph_instance.h" +#include "graph_list.h" #include "graph_def.h" #include "graph_config.h" #include "common.h" @@ -164,6 +188,25 @@ int graph_config_add (const oconfig_item_t *ci) /* {{{ */ return (0); } /* }}} graph_config_add */ +int graph_add_inst (graph_config_t *graph, graph_instance_t *inst) /* {{{ */ +{ + graph_instance_t **tmp; + + if ((graph == NULL) || (inst == NULL)) + return (EINVAL); + + tmp = realloc (graph->instances, + sizeof (*graph->instances) * (graph->instances_num + 1)); + if (tmp == NULL) + return (ENOMEM); + graph->instances = tmp; + + graph->instances[graph->instances_num] = inst; + graph->instances_num++; + + return (0); +} /* }}} int graph_add_inst */ + int graph_add_file (graph_config_t *cfg, const graph_ident_t *file) /* {{{ */ { graph_instance_t *inst; @@ -171,20 +214,11 @@ int graph_add_file (graph_config_t *cfg, const graph_ident_t *file) /* {{{ */ inst = graph_inst_find_matching (cfg, file); if (inst == NULL) { - graph_instance_t **tmp; - - tmp = realloc (cfg->instances, - sizeof (*cfg->instances) * (cfg->instances_num + 1)); - if (tmp == NULL) - return (ENOMEM); - cfg->instances = tmp; - inst = inst_create (cfg, file); if (inst == NULL) return (ENOMEM); - cfg->instances[cfg->instances_num] = inst; - cfg->instances_num++; + graph_add_inst (cfg, inst); } return (inst_add_file (inst, file)); @@ -216,7 +250,7 @@ int graph_get_params (graph_config_t *cfg, /* {{{ */ #define COPY_FIELD(field) do { \ const char *str = ident_get_##field (cfg->select); \ char uri_str[1024]; \ - uri_escape (uri_str, str, sizeof (uri_str)); \ + uri_escape_copy (uri_str, str, sizeof (uri_str)); \ strlcat (buffer, #field, buffer_size); \ strlcat (buffer, "=", buffer_size); \ strlcat (buffer, uri_str, buffer_size); \ @@ -255,6 +289,8 @@ graph_def_t *graph_get_defs (graph_config_t *cfg) /* {{{ */ int graph_add_def (graph_config_t *cfg, graph_def_t *def) /* {{{ */ { + graph_def_t *tmp; + if ((cfg == NULL) || (def == NULL)) return (EINVAL); @@ -264,17 +300,45 @@ int graph_add_def (graph_config_t *cfg, graph_def_t *def) /* {{{ */ return (0); } - return (def_append (cfg->defs, def)); + /* Insert in reverse order. This makes the order in the config file and the + * order of the DEFs in the graph more natural. Really. */ + tmp = cfg->defs; + cfg->defs = def; + return (def_append (cfg->defs, tmp)); } /* }}} int graph_add_def */ -_Bool graph_matches_ident (graph_config_t *cfg, const graph_ident_t *ident) /* {{{ */ +_Bool graph_ident_matches (graph_config_t *cfg, const graph_ident_t *ident) /* {{{ */ { +#if C4_DEBUG if ((cfg == NULL) || (ident == NULL)) return (0); +#endif return (ident_matches (cfg->select, ident)); +} /* }}} _Bool graph_ident_matches */ + +_Bool graph_matches_ident (graph_config_t *cfg, /* {{{ */ + const graph_ident_t *selector) +{ +#if C4_DEBUG + if ((cfg == NULL) || (selector == NULL)) + return (0); +#endif + + return (ident_matches (selector, cfg->select)); } /* }}} _Bool graph_matches_ident */ +_Bool graph_ident_intersect (graph_config_t *cfg, /* {{{ */ + const graph_ident_t *selector) +{ +#if C4_DEBUG + if ((cfg == NULL) || (selector == NULL)) + return (0); +#endif + + return (ident_intersect (cfg->select, selector)); +} /* }}} _Bool graph_ident_intersect */ + _Bool graph_matches_field (graph_config_t *cfg, /* {{{ */ graph_ident_field_t field, const char *field_value) { @@ -335,13 +399,137 @@ graph_instance_t *graph_inst_find_matching (graph_config_t *cfg, /* {{{ */ return (NULL); for (i = 0; i < cfg->instances_num; i++) - if (inst_matches_ident (cfg->instances[i], ident)) + if (inst_ident_matches (cfg->instances[i], ident)) return (cfg->instances[i]); return (NULL); } /* }}} graph_instance_t *graph_inst_find_matching */ -int graph_inst_search (graph_config_t *cfg, const char *term, /* {{{ */ +int graph_inst_find_all_matching (graph_config_t *cfg, /* {{{ */ + const graph_ident_t *ident, + graph_inst_callback_t callback, void *user_data) +{ + size_t i; + + if ((cfg == NULL) || (ident == NULL) || (callback == NULL)) + return (EINVAL); + + for (i = 0; i < cfg->instances_num; i++) + { + int status; + + if (!inst_matches_ident (cfg->instances[i], ident)) + continue; + + status = (*callback) (cfg, cfg->instances[i], user_data); + if (status != 0) + return (status); + } + + return (0); +} /* }}} int graph_inst_find_all_matching */ + +/* Lightweight variant of the "graph_search_inst" which is used if the + * search_info_t doesn't select any field values explicitely. */ +static int graph_search_inst_noselector (graph_config_t *cfg, /* {{{ */ + search_info_t *si, graph_inst_callback_t cb, void *user_data) +{ + char title[1024]; + int status; + size_t i; + + /* parameters have already been checked in "graph_search_inst" */ + + status = graph_get_title (cfg, title, sizeof (title)); + if (status != 0) + { + fprintf (stderr, "graph_search_inst_noselector: " + "graph_get_title failed\n"); + return (status); + } + strtolower (title); + + for (i = 0; i < cfg->instances_num; i++) + { + if (search_graph_inst_matches (si, cfg, cfg->instances[i], title)) + { + status = (*cb) (cfg, cfg->instances[i], user_data); + if (status != 0) + return (status); + } + } /* for (cfg->instances_num) */ + + return (0); +} /* }}} int graph_search_inst_noselector */ + +/* When this function is called from graph_list, it will already have checked + * that the selector of the graph does not contradict the field selections contained in + * the search_info_t. We still have to check if the instance contradicts the + * search parameters, though, since the "ANY" wildcard is filled in now - + * possibly with contradicting values. */ +int graph_search_inst (graph_config_t *cfg, search_info_t *si, /* {{{ */ + graph_inst_callback_t cb, + void *user_data) +{ + char title[1024]; + int status; + size_t i; + graph_ident_t *search_selector; + + if ((cfg == NULL) || (si == NULL) || (cb == NULL)) + return (EINVAL); + + if (!search_has_selector (si)) + return (graph_search_inst_noselector (cfg, si, cb, user_data)); + + search_selector = search_to_ident (si); + if (search_selector == NULL) + return (ENOMEM); + + status = graph_get_title (cfg, title, sizeof (title)); + if (status != 0) + { + ident_destroy (search_selector); + fprintf (stderr, "graph_search_inst: graph_get_title failed\n"); + return (status); + } + strtolower (title); + + for (i = 0; i < cfg->instances_num; i++) + { + graph_ident_t *inst_selector; + + inst_selector = inst_get_selector (cfg->instances[i]); + if (inst_selector == NULL) + continue; + + /* If the two selectors contradict one another, there is no point in + * calling the (more costly) "search_graph_inst_matches" function. */ + if (!ident_intersect (search_selector, inst_selector)) + { + ident_destroy (inst_selector); + continue; + } + + if (search_graph_inst_matches (si, cfg, cfg->instances[i], title)) + { + status = (*cb) (cfg, cfg->instances[i], user_data); + if (status != 0) + { + ident_destroy (search_selector); + ident_destroy (inst_selector); + return (status); + } + } + + ident_destroy (inst_selector); + } /* for (cfg->instances_num) */ + + ident_destroy (search_selector); + return (0); +} /* }}} int graph_search_inst */ + +int graph_search_inst_string (graph_config_t *cfg, const char *term, /* {{{ */ graph_inst_callback_t cb, void *user_data) { @@ -352,7 +540,7 @@ int graph_inst_search (graph_config_t *cfg, const char *term, /* {{{ */ status = graph_get_title (cfg, buffer, sizeof (buffer)); if (status != 0) { - fprintf (stderr, "graph_inst_search: graph_get_title failed\n"); + fprintf (stderr, "graph_search_inst_string: graph_get_title failed\n"); return (status); } @@ -381,13 +569,15 @@ int graph_inst_search (graph_config_t *cfg, const char *term, /* {{{ */ } return (0); -} /* }}} int graph_inst_search */ +} /* }}} int graph_search_inst_string */ int graph_inst_search_field (graph_config_t *cfg, /* {{{ */ graph_ident_field_t field, const char *field_value, graph_inst_callback_t callback, void *user_data) { size_t i; + const char *selector_field; + _Bool need_check_instances = 0; if ((cfg == NULL) || (field_value == NULL) || (callback == NULL)) return (EINVAL); @@ -395,16 +585,24 @@ int graph_inst_search_field (graph_config_t *cfg, /* {{{ */ if (!graph_matches_field (cfg, field, field_value)) return (0); + selector_field = ident_get_field (cfg->select, field); + if (selector_field == NULL) + return (-1); + + if (IS_ALL (selector_field) || IS_ANY (selector_field)) + need_check_instances = 1; + for (i = 0; i < cfg->instances_num; i++) { - if (inst_matches_field (cfg->instances[i], field, field_value)) - { - int status; + int status; - status = (*callback) (cfg, cfg->instances[i], user_data); - if (status != 0) - return (status); - } + if (need_check_instances + && !inst_matches_field (cfg->instances[i], field, field_value)) + continue; + + status = (*callback) (cfg, cfg->instances[i], user_data); + if (status != 0) + return (status); } return (0); @@ -418,6 +616,116 @@ int graph_compare (graph_config_t *cfg, const graph_ident_t *ident) /* {{{ */ return (ident_compare (cfg->select, ident)); } /* }}} int graph_compare */ +size_t graph_num_instances (graph_config_t *cfg) /* {{{ */ +{ + if (cfg == NULL) + return ((size_t) -1); + + return (cfg->instances_num); +} /* }}} size_t graph_num_instances */ + +int graph_to_json (const graph_config_t *cfg, /* {{{ */ + yajl_gen handler) +{ + size_t i; + + if ((cfg == NULL) || (handler == NULL)) + return (EINVAL); + + yajl_gen_map_open (handler); + + yajl_gen_string (handler, + (unsigned char *) "title", + (unsigned int) strlen ("title")); + yajl_gen_string (handler, + (unsigned char *) cfg->title, + (unsigned int) strlen (cfg->title)); + + yajl_gen_string (handler, + (unsigned char *) "select", + (unsigned int) strlen ("select")); + ident_to_json (cfg->select, handler); + + yajl_gen_string (handler, + (unsigned char *) "instances", + (unsigned int) strlen ("instances")); + yajl_gen_array_open (handler); + for (i = 0; i < cfg->instances_num; i++) + inst_to_json (cfg->instances[i], handler); + yajl_gen_array_close (handler); + + yajl_gen_map_close (handler); + + return (0); +} /* }}} int graph_to_json */ + +int graph_def_to_json (graph_config_t *cfg, /* {{{ */ + graph_instance_t *inst, + yajl_gen handler) +{ +#define yajl_gen_string_cast(h,p,l) \ + yajl_gen_string (h, (unsigned char *) p, (unsigned int) l) + + if ((cfg == NULL) || (handler == NULL)) + return (EINVAL); + + yajl_gen_map_open (handler); + + yajl_gen_string_cast (handler, "select", strlen ("select")); + ident_to_json (cfg->select, handler); + if (cfg->title != NULL) + { + yajl_gen_string_cast (handler, "title", strlen ("title")); + yajl_gen_string_cast (handler, cfg->title, strlen (cfg->title)); + } + if (cfg->vertical_label != NULL) + { + yajl_gen_string_cast (handler, "vertical_label", strlen ("vertical_label")); + yajl_gen_string_cast (handler, cfg->vertical_label, strlen (cfg->vertical_label)); + } + yajl_gen_string_cast (handler, "show_zero", strlen ("show_zero")); + yajl_gen_bool (handler, cfg->show_zero); + + yajl_gen_string_cast (handler, "defs", strlen ("defs")); + if (cfg->defs == NULL) + { + graph_def_t *defs; + + defs = inst_get_default_defs (cfg, inst); + def_to_json (defs, handler); + def_destroy (defs); + } + else + { + def_to_json (cfg->defs, handler); + } + + yajl_gen_map_close (handler); + + return (0); +#undef yajl_gen_string_cast +} /* }}} int graph_def_to_json */ + +static int graph_sort_instances_cb (const void *v0, const void *v1) /* {{{ */ +{ + return (inst_compare (*(graph_instance_t * const *) v0, + *(graph_instance_t * const *) v1)); +} /* }}} int graph_sort_instances_cb */ + +int graph_sort_instances (graph_config_t *cfg) /* {{{ */ +{ + if (cfg == NULL) + return (EINVAL); + + if (cfg->instances_num < 2) + return (0); + + qsort (cfg->instances, cfg->instances_num, sizeof (*cfg->instances), + graph_sort_instances_cb); + + return (0); +} /* }}} int graph_sort_instances */ + int graph_clear_instances (graph_config_t *cfg) /* {{{ */ { size_t i; @@ -435,27 +743,27 @@ int graph_clear_instances (graph_config_t *cfg) /* {{{ */ } /* }}} int graph_clear_instances */ int graph_get_rrdargs (graph_config_t *cfg, graph_instance_t *inst, /* {{{ */ - str_array_t *args) + rrd_args_t *args) { if ((cfg == NULL) || (inst == NULL) || (args == NULL)) return (EINVAL); if (cfg->title != NULL) { - array_append (args, "-t"); - array_append (args, cfg->title); + array_append (args->options, "-t"); + array_append (args->options, cfg->title); } if (cfg->vertical_label != NULL) { - array_append (args, "-v"); - array_append (args, cfg->vertical_label); + array_append (args->options, "-v"); + array_append (args->options, cfg->vertical_label); } if (cfg->show_zero) { - array_append (args, "-l"); - array_append (args, "0"); + array_append (args->options, "-l"); + array_append (args->options, "0"); } return (0);