--- /dev/null
+/**
+ * collectd - src/config_list.c
+ * Copyright (C) 2006 Lubos Stanek <lubek at users.sourceforge.net>
+ *
+ * This program is free software; you can redistribute it and/
+ * or modify it under the terms of the GNU General Public Li-
+ * cence as published by the Free Software Foundation; either
+ * version 2 of the Licence, or any later version.
+ *
+ * This program is distributed in the hope that it will be use-
+ * ful, but WITHOUT ANY WARRANTY; without even the implied war-
+ * ranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public Licence for more details.
+ *
+ * You should have received a copy of the GNU General Public
+ * Licence along with this program; if not, write to the Free
+ * Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139,
+ * USA.
+ *
+ * Authors:
+ * Lubos Stanek <lubek at users.sourceforge.net>
+ **/
+/**
+ * configlist handles plugin's list of configured collectable
+ * entries with global ignore action
+ **/
+/**
+ * Usage:
+ *
+ * Define plugin's global variable of type configlist_t:
+ * configlist_t *myconfig_ignore;
+ * If you know the state of global ignore (IgnoreSelected),
+ * allocate the variable with:
+ * myconfig_ignore = configlist_create (YourKnownIgnore);
+ * If you do not know the state of the global ignore (IgnoreSelected),
+ * initialize the global variable and set the ignore flag later:
+ * myconfig_ignore = configlist_init ();
+ * Append single entries in your cf_register'ed callback function:
+ * configlist_add (myconfig_ignore, newentry);
+ * When you hit the IgnoreSelected config option,
+ * offer it to the list:
+ * configlist_ignore (myconfig_ignore, instantly_got_value_of_ignore);
+ * That ia all for the configlist initialization.
+ * Later during read and write (plugin's registered functions) get
+ * the information whether this entry would be collected or not:
+ * if (configlist_ignored (myconfig_ignore, thisentry))
+ * return;
+ **/
+
+#include "common.h"
+#include "utils_debug.h"
+#include "config_list.h"
+
+#define BUFSIZE 512
+
+/* private prototypes */
+
+struct configentry_s;
+typedef struct configentry_s configentry_t;
+
+struct configlist_s {
+ int ignore; /* ignore entries */
+ int num; /* number of entries */
+ configentry_t *next; /* pointer to the first entry */
+};
+
+struct configentry_s {
+#if HAVE_REGEX_H
+ regex_t *rmatch; /* regular expression entry identification */
+#endif
+ char *smatch; /* string entry identification */
+ configentry_t *next;
+};
+
+
+/* *** *** *** ********************************************* *** *** *** */
+/* *** *** *** *** *** *** private functions *** *** *** *** *** *** */
+/* *** *** *** ********************************************* *** *** *** */
+
+#if HAVE_REGEX_H
+static int configlist_regappend(configlist_t *conflist, const char *entry)
+{
+ int rcompile;
+ regex_t *regtemp;
+ char regerr[BUFSIZE];
+ configentry_t *new;
+
+ /* create buffer */
+ if ((regtemp = malloc(sizeof(regex_t))) == NULL)
+ {
+ syslog (LOG_ERR, "cannot allocate new config entry");
+ regfree (regtemp);
+ return (0);
+ }
+ memset (regtemp, '\0', sizeof(regex_t));
+
+ /* compile regex */
+ if ((rcompile = regcomp (regtemp, entry, REG_EXTENDED)) != 0)
+ {
+ if (regerror(rcompile, regtemp, regerr, sizeof(regerr)))
+ syslog (LOG_ERR, "cannot compile regex %s: %i/%s",
+ entry, rcompile, regerr);
+ else
+ syslog (LOG_ERR, "cannot compile regex %s: %i",
+ entry, rcompile);
+ regfree (regtemp);
+ return (0);
+ }
+ DBG("regex compiled: %s - %i", entry, rcompile);
+
+ /* create new entry */
+ if ((new = malloc(sizeof(configentry_t))) == NULL)
+ {
+ syslog (LOG_ERR, "cannot allocate new config entry");
+ regfree (regtemp);
+ return (0);
+ }
+ memset (new, '\0', sizeof(configentry_t));
+ new->rmatch = regtemp;
+#if COLLECTD_DEBUG
+ new->smatch = sstrdup(entry);
+#endif
+ /* append new entry */
+ if (conflist->next == NULL)
+ {
+ conflist->next=new;
+ }
+ else
+ {
+ new->next=conflist->next;
+ conflist->next=new;
+ }
+ conflist->num++;
+ return (1);
+} /* int configlist_regappend(configlist_t *conflist, const char *entry) */
+#endif
+
+static int configlist_strappend(configlist_t *conflist, const char *entry)
+{
+ configentry_t *new;
+
+ /* create new entry */
+ if ((new = malloc(sizeof(configentry_t))) == NULL )
+ {
+ syslog (LOG_ERR, "cannot allocate new entry");
+ return (0);
+ }
+ memset (new, '\0', sizeof(configentry_t));
+ new->smatch = sstrdup(entry);
+
+ /* append new entry */
+ if (conflist->next == NULL)
+ {
+ conflist->next=new;
+ }
+ else
+ {
+ new->next=conflist->next;
+ conflist->next=new;
+ }
+ conflist->num++;
+ return (1);
+} /* int configlist_strappend(configlist_t *conflist, const char *entry) */
+
+#if HAVE_REGEX_H
+/*
+ * check list for entry regex match
+ * return 1 if found
+ */
+static int configentry_rmatch (configentry_t *confentry, const char *entry)
+{
+ if (confentry == NULL)
+ return (0);
+
+ if (strlen (entry) == 0)
+ return (0);
+
+ if (confentry->rmatch == NULL)
+ return (0);
+
+ /* match regex */
+ if (regexec (confentry->rmatch, entry, 0, NULL, 0) == 0)
+ return (1);
+
+ return (0);
+} /* int configentry_rmatch (configentry_t *confentry, const char *entry) */
+#endif
+
+/*
+ * check list for entry string match
+ * return 1 if found
+ */
+static int configentry_smatch (configentry_t *confentry, const char *entry)
+{
+ if (confentry == NULL)
+ return (0);
+
+ if (strlen (entry) == 0)
+ return (0);
+
+ if ((confentry->smatch != NULL && strcmp (entry, confentry->smatch) == 0))
+ return (1);
+
+ return (0);
+} /* int configentry_smatch (configentry_t *confentry, const char *entry) */
+
+
+/* *** *** *** ******************************************** *** *** *** */
+/* *** *** *** *** *** *** public functions *** *** *** *** *** *** */
+/* *** *** *** ******************************************** *** *** *** */
+
+/*
+ * create the configlist_t with known ignore state
+ * return pointer to configlist_t
+ */
+configlist_t *configlist_create (int ignore)
+{
+ configlist_t *conflist;
+
+ if ((conflist = smalloc (sizeof (configlist_t))) == NULL)
+ {
+ syslog(LOG_ERR, "not enough memory to allocate configlist");
+ return (NULL);
+ }
+ DBG("configlist created 0x%p, ignore %i", (void *) conflist, ignore);
+ memset (conflist, '\0', sizeof (configlist_t));
+
+ if (ignore)
+ conflist->ignore = ignore;
+
+ return (conflist);
+} /* configlist_t *configlist_create (int ignore) */
+
+/*
+ * create configlist_t and initialize the ignore state to 0
+ * return pointer to configlist_t
+ */
+configlist_t *configlist_init (void)
+{
+ return (configlist_create (0));
+} /* configlist_t *configlist_init (void) */
+
+
+/*
+ * free memory used by configlist_t
+ */
+void configlist_free (configlist_t *conflist)
+{
+ configentry_t *this;
+ configentry_t *next;
+
+ DBG ("(conflist = 0x%p)", (void *) conflist);
+
+ if (conflist == NULL)
+ return;
+
+ for (this = conflist->next; this != NULL; this = next)
+ {
+ DBG ("free - confentry = 0x%p, numlist %i", (void *) this, conflist->num);
+ next = this->next;
+ conflist->num--;
+#if HAVE_REGEX_H
+ if (this->rmatch != NULL)
+ {
+ regfree (this->rmatch);
+ this->rmatch = NULL;
+ }
+#endif
+ if (this->smatch != NULL)
+ {
+ sfree (this->smatch);
+ this->smatch = NULL;
+ }
+ sfree (this);
+ }
+#if COLLECTD_DEBUG
+ if (conflist->num != 0)
+ DBG ("after free numlist: %i", conflist->num);
+#endif
+ conflist->num = 0;
+ sfree (conflist);
+ conflist = NULL;
+} /* void configlist_destroy (configlist_t *conflist) */
+
+/*
+ * set ignore state of the configlist_t
+ */
+void configlist_ignore (configlist_t *conflist, int ignore)
+{
+ if (conflist == NULL)
+ {
+ DBG("ignore call with configlist_t == NULL");
+ return;
+ }
+
+ conflist->ignore = ignore;
+} /* void configlist_ignore (configlist_t *conflist, int ignore) */
+
+/*
+ * get number of entries in the configlist_t
+ * return int number
+ */
+int configlist_num (configlist_t *conflist)
+{
+ if (conflist == NULL)
+ {
+ DBG("get num called with configlist_t == NULL");
+ return (0);
+ }
+
+ return (conflist->num);
+} /* int configlist_num (configlist_t *conflist) */
+
+/*
+ * append entry into configlist_t
+ * return 1 for success
+ */
+int configlist_add (configlist_t *conflist, const char *entry)
+{
+#if HAVE_REGEX_H
+ char *entrytemp;
+#endif
+ int restemp;
+
+ if (conflist == NULL)
+ {
+ DBG("add called with configlist_t == NULL");
+ return (0);
+ }
+
+ /* append nothing, report success */
+ if (strlen(entry) == 0)
+ {
+ DBG("not appending: empty entry");
+ return (1);
+ }
+
+#if HAVE_REGEX_H
+ /* regex string is enclosed in "|...|" */
+ if (entry[0] == '|' && strlen(entry) > 2 && entry[strlen(entry) - 1] == '|')
+ {
+ entrytemp = smalloc(strlen(entry) - 2);
+ sstrncpy(entrytemp, &entry[1], strlen(entry) - 1);
+ DBG("to add regex entry: %s", entrytemp);
+ restemp = configlist_regappend(conflist, entrytemp);
+ sfree (entrytemp);
+ }
+ else
+#endif
+ {
+ DBG("to add entry: %s", entry);
+ restemp = configlist_strappend(conflist, entry);
+ }
+ return (restemp);
+} /* int configlist_add (configlist_t *conflist, const char *entry) */
+
+/*
+ * check list for entry
+ * return 1 for ignored entry
+ */
+int configlist_ignored (configlist_t *conflist, const char *entry)
+{
+ configentry_t *traverse;
+
+ /* if no entries, collect all */
+ if (configlist_num(conflist) == 0)
+ return (0);
+
+ /* traverse list and check entries */
+ traverse = conflist->next;
+ while (traverse != NULL)
+ {
+#if HAVE_REGEX_H
+ if (traverse->rmatch != NULL)
+ {
+ if (configentry_rmatch (traverse, entry))
+ return (conflist->ignore);
+ }
+ else
+#endif
+ {
+ if (configentry_smatch (traverse, entry))
+ return (conflist->ignore);
+ }
+ traverse = traverse->next;
+ }
+
+ return (1 - conflist->ignore);
+} /* int configlist_ignored (configlist_t *conflist, const char *entry) */
+
--- /dev/null
+/**
+ * collectd - src/config_list.h
+ * Copyright (C) 2006 Lubos Stanek <lubek at users.sourceforge.net>
+ *
+ * This program is free software; you can redistribute it and/
+ * or modify it under the terms of the GNU General Public Li-
+ * cence as published by the Free Software Foundation; either
+ * version 2 of the Licence, or any later version.
+ *
+ * This program is distributed in the hope that it will be use-
+ * ful, but WITHOUT ANY WARRANTY; without even the implied war-
+ * ranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public Licence for more details.
+ *
+ * You should have received a copy of the GNU General Public
+ * Licence along with this program; if not, write to the Free
+ * Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139,
+ * USA.
+ *
+ * Authors:
+ * Lubos Stanek <lubek at users.sourceforge.net>
+ **/
+/**
+ * configlist handles plugin's list of configured collectable
+ * entries with global ignore action
+ **/
+
+#if !CONFIG_LIST_H
+#define CONFIG_LIST_H 1
+
+#include "common.h"
+
+#if HAVE_REGEX_H
+# include <regex.h>
+#endif
+
+/* public prototypes */
+
+struct configlist_s;
+typedef struct configlist_s configlist_t;
+
+/*
+ * create the configlist_t with known ignore state
+ * return pointer to configlist_t
+ */
+configlist_t *configlist_create (int ignore);
+
+/*
+ * create configlist_t and initialize the ignore state to 0
+ * return pointer to configlist_t
+ */
+configlist_t *configlist_init (void);
+
+/*
+ * free memory used by configlist_t
+ */
+void configlist_free (configlist_t *conflist);
+
+/*
+ * set ignore state of the configlist_t
+ */
+void configlist_ignore (configlist_t *conflist, int ignore);
+/*
+ * get number of entries in the configlist_t
+ * return int number
+ */
+int configlist_num (configlist_t *conflist);
+
+/*
+ * append entry to configlist_t
+ * return 1 for success
+ */
+int configlist_add (configlist_t *conflist, const char *entry);
+
+/*
+ * check list for entry
+ * return 1 for ignored entry
+ */
+int configlist_ignored (configlist_t *conflist, const char *entry);
+
+#endif /* !CONFIG_LIST_H */
+
#include "common.h"
#include "plugin.h"
#include "configfile.h"
+#include "config_list.h"
#include "utils_debug.h"
#define MODULE_NAME "sensors"
};
static int config_keys_num = 3;
-static char **sensor_list = NULL;
-static int sensor_list_num = 0;
-/*
- * sensor_list_action:
- * 0 => default is to collect selected sensors
- * 1 => ignore selected sensors
- */
-static int sensor_list_action = 0;
+static configlist_t *sensor_list;
+
/*
* sensor_extended_naming:
* 0 => default is to create chip-feature
static int sensors_config (char *key, char *value)
{
- char **temp;
+ if (sensor_list == NULL)
+ sensor_list = configlist_init();
if (strcasecmp (key, "Sensor") == 0)
{
- temp = (char **) realloc (sensor_list, (sensor_list_num + 1) * sizeof (char *));
- if (temp == NULL)
- {
- syslog (LOG_EMERG, "Cannot allocate more memory.");
- return (1);
- }
- sensor_list = temp;
-
- if ((sensor_list[sensor_list_num] = strdup (value)) == NULL)
+ if (!configlist_add (sensor_list, value))
{
- syslog (LOG_EMERG, "Cannot allocate memory.");
+ syslog (LOG_EMERG, "Cannot add value.");
return (1);
}
- sensor_list_num++;
}
else if (strcasecmp (key, "IgnoreSelected") == 0)
{
if ((strcasecmp (value, "True") == 0)
|| (strcasecmp (value, "Yes") == 0)
|| (strcasecmp (value, "On") == 0))
- sensor_list_action = 1;
- else
- sensor_list_action = 0;
+ configlist_ignore (sensor_list, 1);
}
else if (strcasecmp (key, "ExtendedSensorNaming") == 0)
{
return (0);
}
-/*
- * Check if this feature should be ignored. This is called from
- * both, `submit' and `write' to give client and server
- * the ability to ignore certain stuff...
- */
-static int config_get_ignored (const char *inst)
-{
- int i;
-
- /* If no ignored are given collect all features. */
- if (sensor_list_num < 1)
- return (0);
-
- for (i = 0; i < sensor_list_num; i++)
- if (strcasecmp (inst, sensor_list[i]) == 0)
- return (sensor_list_action);
- return (1 - sensor_list_action);
-}
-
static void collectd_sensors_init (void)
{
#ifdef HAVE_LIBSENSORS
int status;
/* skip ignored in our config */
- if (config_get_ignored (inst))
+ if (configlist_ignored (sensor_list, inst))
return;
/* extended sensor naming */
int status;
/* skip ignored in our config */
- if (config_get_ignored (inst))
+ if (configlist_ignored (sensor_list, inst))
return;
/* extended sensor naming */
return;
/* skip ignored in our config */
- if (config_get_ignored (inst))
+ if (configlist_ignored (sensor_list, inst))
return;
if (snprintf (buf, BUFSIZE, "%u:%.3f", (unsigned int) curtime,