From 4e8af6617782210485a26e7a6ae35913c71b19d6 Mon Sep 17 00:00:00 2001 From: Bostjan Skufca Date: Fri, 24 Oct 2014 17:06:27 +0000 Subject: [PATCH 1/3] Added iniparser v3.1 from http://ndevilla.free.fr/iniparser/ or https://github.com/ndevilla/iniparser; Just compiled-in for now, not used at all yet --- .gitignore | 9 +- configure.ac | 1 + src/Makefile.am | 3 +- src/lib/Makefile.am | 16 + src/lib/iniparser/AUTHORS | 6 + src/lib/iniparser/INSTALL | 15 + src/lib/iniparser/LICENSE | 21 + src/lib/iniparser/README | 12 + src/lib/iniparser/SNOOPY_LIB_URI | 2 + src/lib/iniparser/SNOOPY_LIB_VERSION | 1 + src/lib/iniparser/src/dictionary.c | 398 ++++++++++++++ src/lib/iniparser/src/dictionary.h | 165 ++++++ src/lib/iniparser/src/iniparser.c | 748 +++++++++++++++++++++++++++ src/lib/iniparser/src/iniparser.h | 307 +++++++++++ 14 files changed, 1697 insertions(+), 7 deletions(-) create mode 100644 src/lib/Makefile.am create mode 100644 src/lib/iniparser/AUTHORS create mode 100644 src/lib/iniparser/INSTALL create mode 100644 src/lib/iniparser/LICENSE create mode 100644 src/lib/iniparser/README create mode 100644 src/lib/iniparser/SNOOPY_LIB_URI create mode 100644 src/lib/iniparser/SNOOPY_LIB_VERSION create mode 100644 src/lib/iniparser/src/dictionary.c create mode 100644 src/lib/iniparser/src/dictionary.h create mode 100644 src/lib/iniparser/src/iniparser.c create mode 100644 src/lib/iniparser/src/iniparser.h diff --git a/.gitignore b/.gitignore index f70e1fed..dcc3b6c5 100644 --- a/.gitignore +++ b/.gitignore @@ -19,16 +19,13 @@ Makefile ### Files produced by build process # +.dirstamp *.a *.o *.la *.lo -/.deps/* -/*/.deps/* -/*/*/.deps/* -/.libs/* -/*/.libs/* -/*/*/.libs/* +.deps +.libs /src/libsnoopy.so /bin/snoopy-test-output /bin/snoopy-detect diff --git a/configure.ac b/configure.ac index 96ca5dba..dfe80859 100644 --- a/configure.ac +++ b/configure.ac @@ -41,6 +41,7 @@ AC_CONFIG_FILES([Makefile bin/Makefile contrib/sles/snoopy.spec src/Makefile + src/lib/Makefile src/filter/Makefile src/input/Makefile]) diff --git a/src/Makefile.am b/src/Makefile.am index 3d122056..8f52dc05 100644 --- a/src/Makefile.am +++ b/src/Makefile.am @@ -6,7 +6,7 @@ include $(top_srcdir)/build/Makefile.am.common ### First process all required subdirs # -SUBDIRS = filter input +SUBDIRS = lib filter input @@ -27,6 +27,7 @@ libsnoopy_no_execve_la_SOURCES = \ snoopy.h libsnoopy_no_execve_la_LIBADD = \ + lib/libiniparser.la \ filter/libsnoopy_filters_all.la \ input/libsnoopy_inputs_all.la diff --git a/src/lib/Makefile.am b/src/lib/Makefile.am new file mode 100644 index 00000000..5009bfc8 --- /dev/null +++ b/src/lib/Makefile.am @@ -0,0 +1,16 @@ +### Include common Makefile configuration +# +include $(top_srcdir)/build/Makefile.am.common + + + +### INI file parser +# +noinst_LTLIBRARIES = \ + libiniparser.la + +libiniparser_la_SOURCES = \ + iniparser/src/dictionary.c \ + iniparser/src/dictionary.h \ + iniparser/src/iniparser.c \ + iniparser/src/iniparser.h diff --git a/src/lib/iniparser/AUTHORS b/src/lib/iniparser/AUTHORS new file mode 100644 index 00000000..d5a3f6b2 --- /dev/null +++ b/src/lib/iniparser/AUTHORS @@ -0,0 +1,6 @@ +Author: Nicolas Devillard + +This tiny library has received countless contributions and I have +not kept track of all the people who contributed. Let them be thanked +for their ideas, code, suggestions, corrections, enhancements! + diff --git a/src/lib/iniparser/INSTALL b/src/lib/iniparser/INSTALL new file mode 100644 index 00000000..a5b05d0e --- /dev/null +++ b/src/lib/iniparser/INSTALL @@ -0,0 +1,15 @@ + +iniParser installation instructions +----------------------------------- + +- Modify the Makefile to suit your environment. +- Type 'make' to make the library. +- Type 'make check' to make the test program. +- Type 'test/iniexample' to launch the test program. +- Type 'test/parse' to launch torture tests. + + + +Enjoy! +N. Devillard +Wed Mar 2 21:14:17 CET 2011 diff --git a/src/lib/iniparser/LICENSE b/src/lib/iniparser/LICENSE new file mode 100644 index 00000000..5a3a80ba --- /dev/null +++ b/src/lib/iniparser/LICENSE @@ -0,0 +1,21 @@ +Copyright (c) 2000-2011 by Nicolas Devillard. +MIT License + +Permission is hereby granted, free of charge, to any person obtaining a +copy of this software and associated documentation files (the "Software"), +to deal in the Software without restriction, including without limitation +the rights to use, copy, modify, merge, publish, distribute, sublicense, +and/or sell copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER +DEALINGS IN THE SOFTWARE. + diff --git a/src/lib/iniparser/README b/src/lib/iniparser/README new file mode 100644 index 00000000..bc697876 --- /dev/null +++ b/src/lib/iniparser/README @@ -0,0 +1,12 @@ + +Welcome to iniParser -- version 3.1 +released 08 Apr 2012 + +This modules offers parsing of ini files from the C level. +See a complete documentation in HTML format, from this directory +open the file html/index.html with any HTML-capable browser. + +Enjoy! + +N.Devillard +Sun Apr 8 16:38:09 CEST 2012 diff --git a/src/lib/iniparser/SNOOPY_LIB_URI b/src/lib/iniparser/SNOOPY_LIB_URI new file mode 100644 index 00000000..3b98432e --- /dev/null +++ b/src/lib/iniparser/SNOOPY_LIB_URI @@ -0,0 +1,2 @@ +http://ndevilla.free.fr/iniparser/ +https://github.com/ndevilla/iniparser diff --git a/src/lib/iniparser/SNOOPY_LIB_VERSION b/src/lib/iniparser/SNOOPY_LIB_VERSION new file mode 100644 index 00000000..8c50098d --- /dev/null +++ b/src/lib/iniparser/SNOOPY_LIB_VERSION @@ -0,0 +1 @@ +3.1 diff --git a/src/lib/iniparser/src/dictionary.c b/src/lib/iniparser/src/dictionary.c new file mode 100644 index 00000000..3f0f5cfa --- /dev/null +++ b/src/lib/iniparser/src/dictionary.c @@ -0,0 +1,398 @@ +/*-------------------------------------------------------------------------*/ +/** + @file dictionary.c + @author N. Devillard + @brief Implements a dictionary for string variables. + + This module implements a simple dictionary object, i.e. a list + of string/string associations. This object is useful to store e.g. + informations retrieved from a configuration file (ini files). +*/ +/*--------------------------------------------------------------------------*/ + +/*--------------------------------------------------------------------------- + Includes + ---------------------------------------------------------------------------*/ +#include "dictionary.h" + +#include +#include +#include +#include + +/** Maximum value size for integers and doubles. */ +#define MAXVALSZ 1024 + +/** Minimal allocated number of entries in a dictionary */ +#define DICTMINSZ 128 + +/** Invalid key token */ +#define DICT_INVALID_KEY ((char*)-1) + +/*--------------------------------------------------------------------------- + Private functions + ---------------------------------------------------------------------------*/ + +/* Doubles the allocated size associated to a pointer */ +/* 'size' is the current allocated size. */ +static void * mem_double(void * ptr, int size) +{ + void * newptr ; + + newptr = calloc(2*size, 1); + if (newptr==NULL) { + return NULL ; + } + memcpy(newptr, ptr, size); + free(ptr); + return newptr ; +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Duplicate a string + @param s String to duplicate + @return Pointer to a newly allocated string, to be freed with free() + + This is a replacement for strdup(). This implementation is provided + for systems that do not have it. + */ +/*--------------------------------------------------------------------------*/ +static char * xstrdup(const char * s) +{ + char * t ; + if (!s) + return NULL ; + t = (char*)malloc(strlen(s)+1) ; + if (t) { + strcpy(t,s); + } + return t ; +} + +/*--------------------------------------------------------------------------- + Function codes + ---------------------------------------------------------------------------*/ +/*-------------------------------------------------------------------------*/ +/** + @brief Compute the hash key for a string. + @param key Character string to use for key. + @return 1 unsigned int on at least 32 bits. + + This hash function has been taken from an Article in Dr Dobbs Journal. + This is normally a collision-free function, distributing keys evenly. + The key is stored anyway in the struct so that collision can be avoided + by comparing the key itself in last resort. + */ +/*--------------------------------------------------------------------------*/ +unsigned dictionary_hash(const char * key) +{ + int len ; + unsigned hash ; + int i ; + + len = strlen(key); + for (hash=0, i=0 ; i>6) ; + } + hash += (hash <<3); + hash ^= (hash >>11); + hash += (hash <<15); + return hash ; +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Create a new dictionary object. + @param size Optional initial size of the dictionary. + @return 1 newly allocated dictionary objet. + + This function allocates a new dictionary object of given size and returns + it. If you do not know in advance (roughly) the number of entries in the + dictionary, give size=0. + */ +/*--------------------------------------------------------------------------*/ +dictionary * dictionary_new(int size) +{ + dictionary * d ; + + /* If no size was specified, allocate space for DICTMINSZ */ + if (sizesize = size ; + d->val = (char **)calloc(size, sizeof(char*)); + d->key = (char **)calloc(size, sizeof(char*)); + d->hash = (unsigned int *)calloc(size, sizeof(unsigned)); + return d ; +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Delete a dictionary object + @param d dictionary object to deallocate. + @return void + + Deallocate a dictionary object and all memory associated to it. + */ +/*--------------------------------------------------------------------------*/ +void dictionary_del(dictionary * d) +{ + int i ; + + if (d==NULL) return ; + for (i=0 ; isize ; i++) { + if (d->key[i]!=NULL) + free(d->key[i]); + if (d->val[i]!=NULL) + free(d->val[i]); + } + free(d->val); + free(d->key); + free(d->hash); + free(d); + return ; +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Get a value from a dictionary. + @param d dictionary object to search. + @param key Key to look for in the dictionary. + @param def Default value to return if key not found. + @return 1 pointer to internally allocated character string. + + This function locates a key in a dictionary and returns a pointer to its + value, or the passed 'def' pointer if no such key can be found in + dictionary. The returned character pointer points to data internal to the + dictionary object, you should not try to free it or modify it. + */ +/*--------------------------------------------------------------------------*/ +char * dictionary_get(dictionary * d, const char * key, char * def) +{ + unsigned hash ; + int i ; + + hash = dictionary_hash(key); + for (i=0 ; isize ; i++) { + if (d->key[i]==NULL) + continue ; + /* Compare hash */ + if (hash==d->hash[i]) { + /* Compare string, to avoid hash collisions */ + if (!strcmp(key, d->key[i])) { + return d->val[i] ; + } + } + } + return def ; +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Set a value in a dictionary. + @param d dictionary object to modify. + @param key Key to modify or add. + @param val Value to add. + @return int 0 if Ok, anything else otherwise + + If the given key is found in the dictionary, the associated value is + replaced by the provided one. If the key cannot be found in the + dictionary, it is added to it. + + It is Ok to provide a NULL value for val, but NULL values for the dictionary + or the key are considered as errors: the function will return immediately + in such a case. + + Notice that if you dictionary_set a variable to NULL, a call to + dictionary_get will return a NULL value: the variable will be found, and + its value (NULL) is returned. In other words, setting the variable + content to NULL is equivalent to deleting the variable from the + dictionary. It is not possible (in this implementation) to have a key in + the dictionary without value. + + This function returns non-zero in case of failure. + */ +/*--------------------------------------------------------------------------*/ +int dictionary_set(dictionary * d, const char * key, const char * val) +{ + int i ; + unsigned hash ; + + if (d==NULL || key==NULL) return -1 ; + + /* Compute hash for this key */ + hash = dictionary_hash(key) ; + /* Find if value is already in dictionary */ + if (d->n>0) { + for (i=0 ; isize ; i++) { + if (d->key[i]==NULL) + continue ; + if (hash==d->hash[i]) { /* Same hash value */ + if (!strcmp(key, d->key[i])) { /* Same key */ + /* Found a value: modify and return */ + if (d->val[i]!=NULL) + free(d->val[i]); + d->val[i] = val ? xstrdup(val) : NULL ; + /* Value has been modified: return */ + return 0 ; + } + } + } + } + /* Add a new value */ + /* See if dictionary needs to grow */ + if (d->n==d->size) { + + /* Reached maximum size: reallocate dictionary */ + d->val = (char **)mem_double(d->val, d->size * sizeof(char*)) ; + d->key = (char **)mem_double(d->key, d->size * sizeof(char*)) ; + d->hash = (unsigned int *)mem_double(d->hash, d->size * sizeof(unsigned)) ; + if ((d->val==NULL) || (d->key==NULL) || (d->hash==NULL)) { + /* Cannot grow dictionary */ + return -1 ; + } + /* Double size */ + d->size *= 2 ; + } + + /* Insert key in the first empty slot. Start at d->n and wrap at + d->size. Because d->n < d->size this will necessarily + terminate. */ + for (i=d->n ; d->key[i] ; ) { + if(++i == d->size) i = 0; + } + /* Copy key */ + d->key[i] = xstrdup(key); + d->val[i] = val ? xstrdup(val) : NULL ; + d->hash[i] = hash; + d->n ++ ; + return 0 ; +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Delete a key in a dictionary + @param d dictionary object to modify. + @param key Key to remove. + @return void + + This function deletes a key in a dictionary. Nothing is done if the + key cannot be found. + */ +/*--------------------------------------------------------------------------*/ +void dictionary_unset(dictionary * d, const char * key) +{ + unsigned hash ; + int i ; + + if (key == NULL) { + return; + } + + hash = dictionary_hash(key); + for (i=0 ; isize ; i++) { + if (d->key[i]==NULL) + continue ; + /* Compare hash */ + if (hash==d->hash[i]) { + /* Compare string, to avoid hash collisions */ + if (!strcmp(key, d->key[i])) { + /* Found key */ + break ; + } + } + } + if (i>=d->size) + /* Key not found */ + return ; + + free(d->key[i]); + d->key[i] = NULL ; + if (d->val[i]!=NULL) { + free(d->val[i]); + d->val[i] = NULL ; + } + d->hash[i] = 0 ; + d->n -- ; + return ; +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Dump a dictionary to an opened file pointer. + @param d Dictionary to dump + @param f Opened file pointer. + @return void + + Dumps a dictionary onto an opened file pointer. Key pairs are printed out + as @c [Key]=[Value], one per line. It is Ok to provide stdout or stderr as + output file pointers. + */ +/*--------------------------------------------------------------------------*/ +void dictionary_dump(dictionary * d, FILE * out) +{ + int i ; + + if (d==NULL || out==NULL) return ; + if (d->n<1) { + fprintf(out, "empty dictionary\n"); + return ; + } + for (i=0 ; isize ; i++) { + if (d->key[i]) { + fprintf(out, "%20s\t[%s]\n", + d->key[i], + d->val[i] ? d->val[i] : "UNDEF"); + } + } + return ; +} + + +/* Test code */ +#ifdef TESTDIC +#define NVALS 20000 +int main(int argc, char *argv[]) +{ + dictionary * d ; + char * val ; + int i ; + char cval[90] ; + + /* Allocate dictionary */ + printf("allocating...\n"); + d = dictionary_new(0); + + /* Set values in dictionary */ + printf("setting %d values...\n", NVALS); + for (i=0 ; in != 0) { + printf("error deleting values\n"); + } + printf("deallocating...\n"); + dictionary_del(d); + return 0 ; +} +#endif +/* vim: set ts=4 et sw=4 tw=75 */ diff --git a/src/lib/iniparser/src/dictionary.h b/src/lib/iniparser/src/dictionary.h new file mode 100644 index 00000000..34c4b825 --- /dev/null +++ b/src/lib/iniparser/src/dictionary.h @@ -0,0 +1,165 @@ + +/*-------------------------------------------------------------------------*/ +/** + @file dictionary.h + @author N. Devillard + @brief Implements a dictionary for string variables. + + This module implements a simple dictionary object, i.e. a list + of string/string associations. This object is useful to store e.g. + informations retrieved from a configuration file (ini files). +*/ +/*--------------------------------------------------------------------------*/ + +#ifndef _DICTIONARY_H_ +#define _DICTIONARY_H_ + +/*--------------------------------------------------------------------------- + Includes + ---------------------------------------------------------------------------*/ + +#include +#include +#include +#include + +/*--------------------------------------------------------------------------- + New types + ---------------------------------------------------------------------------*/ + + +/*-------------------------------------------------------------------------*/ +/** + @brief Dictionary object + + This object contains a list of string/string associations. Each + association is identified by a unique string key. Looking up values + in the dictionary is speeded up by the use of a (hopefully collision-free) + hash function. + */ +/*-------------------------------------------------------------------------*/ +typedef struct _dictionary_ { + int n ; /** Number of entries in dictionary */ + int size ; /** Storage size */ + char ** val ; /** List of string values */ + char ** key ; /** List of string keys */ + unsigned * hash ; /** List of hash values for keys */ +} dictionary ; + + +/*--------------------------------------------------------------------------- + Function prototypes + ---------------------------------------------------------------------------*/ + +/*-------------------------------------------------------------------------*/ +/** + @brief Compute the hash key for a string. + @param key Character string to use for key. + @return 1 unsigned int on at least 32 bits. + + This hash function has been taken from an Article in Dr Dobbs Journal. + This is normally a collision-free function, distributing keys evenly. + The key is stored anyway in the struct so that collision can be avoided + by comparing the key itself in last resort. + */ +/*--------------------------------------------------------------------------*/ +unsigned dictionary_hash(const char * key); + +/*-------------------------------------------------------------------------*/ +/** + @brief Create a new dictionary object. + @param size Optional initial size of the dictionary. + @return 1 newly allocated dictionary objet. + + This function allocates a new dictionary object of given size and returns + it. If you do not know in advance (roughly) the number of entries in the + dictionary, give size=0. + */ +/*--------------------------------------------------------------------------*/ +dictionary * dictionary_new(int size); + +/*-------------------------------------------------------------------------*/ +/** + @brief Delete a dictionary object + @param d dictionary object to deallocate. + @return void + + Deallocate a dictionary object and all memory associated to it. + */ +/*--------------------------------------------------------------------------*/ +void dictionary_del(dictionary * vd); + +/*-------------------------------------------------------------------------*/ +/** + @brief Get a value from a dictionary. + @param d dictionary object to search. + @param key Key to look for in the dictionary. + @param def Default value to return if key not found. + @return 1 pointer to internally allocated character string. + + This function locates a key in a dictionary and returns a pointer to its + value, or the passed 'def' pointer if no such key can be found in + dictionary. The returned character pointer points to data internal to the + dictionary object, you should not try to free it or modify it. + */ +/*--------------------------------------------------------------------------*/ +char * dictionary_get(dictionary * d, const char * key, char * def); + + +/*-------------------------------------------------------------------------*/ +/** + @brief Set a value in a dictionary. + @param d dictionary object to modify. + @param key Key to modify or add. + @param val Value to add. + @return int 0 if Ok, anything else otherwise + + If the given key is found in the dictionary, the associated value is + replaced by the provided one. If the key cannot be found in the + dictionary, it is added to it. + + It is Ok to provide a NULL value for val, but NULL values for the dictionary + or the key are considered as errors: the function will return immediately + in such a case. + + Notice that if you dictionary_set a variable to NULL, a call to + dictionary_get will return a NULL value: the variable will be found, and + its value (NULL) is returned. In other words, setting the variable + content to NULL is equivalent to deleting the variable from the + dictionary. It is not possible (in this implementation) to have a key in + the dictionary without value. + + This function returns non-zero in case of failure. + */ +/*--------------------------------------------------------------------------*/ +int dictionary_set(dictionary * vd, const char * key, const char * val); + +/*-------------------------------------------------------------------------*/ +/** + @brief Delete a key in a dictionary + @param d dictionary object to modify. + @param key Key to remove. + @return void + + This function deletes a key in a dictionary. Nothing is done if the + key cannot be found. + */ +/*--------------------------------------------------------------------------*/ +void dictionary_unset(dictionary * d, const char * key); + + +/*-------------------------------------------------------------------------*/ +/** + @brief Dump a dictionary to an opened file pointer. + @param d Dictionary to dump + @param f Opened file pointer. + @return void + + Dumps a dictionary onto an opened file pointer. Key pairs are printed out + as @c [Key]=[Value], one per line. It is Ok to provide stdout or stderr as + output file pointers. + */ +/*--------------------------------------------------------------------------*/ +void dictionary_dump(dictionary * d, FILE * out); + +#endif diff --git a/src/lib/iniparser/src/iniparser.c b/src/lib/iniparser/src/iniparser.c new file mode 100644 index 00000000..d5579557 --- /dev/null +++ b/src/lib/iniparser/src/iniparser.c @@ -0,0 +1,748 @@ + +/*-------------------------------------------------------------------------*/ +/** + @file iniparser.c + @author N. Devillard + @brief Parser for ini files. +*/ +/*--------------------------------------------------------------------------*/ +/*---------------------------- Includes ------------------------------------*/ +#include +#include "iniparser.h" + +/*---------------------------- Defines -------------------------------------*/ +#define ASCIILINESZ (1024) +#define INI_INVALID_KEY ((char*)-1) + +/*--------------------------------------------------------------------------- + Private to this module + ---------------------------------------------------------------------------*/ +/** + * This enum stores the status for each parsed line (internal use only). + */ +typedef enum _line_status_ { + LINE_UNPROCESSED, + LINE_ERROR, + LINE_EMPTY, + LINE_COMMENT, + LINE_SECTION, + LINE_VALUE +} line_status ; + +/*-------------------------------------------------------------------------*/ +/** + @brief Convert a string to lowercase. + @param s String to convert. + @return ptr to statically allocated string. + + This function returns a pointer to a statically allocated string + containing a lowercased version of the input string. Do not free + or modify the returned string! Since the returned string is statically + allocated, it will be modified at each function call (not re-entrant). + */ +/*--------------------------------------------------------------------------*/ +static char * strlwc(const char * s) +{ + static char l[ASCIILINESZ+1]; + int i ; + + if (s==NULL) return NULL ; + memset(l, 0, ASCIILINESZ+1); + i=0 ; + while (s[i] && i l) { + if (!isspace((int)*(last-1))) + break ; + last -- ; + } + *last = (char)0; + return (char*)l ; +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Get number of sections in a dictionary + @param d Dictionary to examine + @return int Number of sections found in dictionary + + This function returns the number of sections found in a dictionary. + The test to recognize sections is done on the string stored in the + dictionary: a section name is given as "section" whereas a key is + stored as "section:key", thus the test looks for entries that do not + contain a colon. + + This clearly fails in the case a section name contains a colon, but + this should simply be avoided. + + This function returns -1 in case of error. + */ +/*--------------------------------------------------------------------------*/ +int iniparser_getnsec(dictionary * d) +{ + int i ; + int nsec ; + + if (d==NULL) return -1 ; + nsec=0 ; + for (i=0 ; isize ; i++) { + if (d->key[i]==NULL) + continue ; + if (strchr(d->key[i], ':')==NULL) { + nsec ++ ; + } + } + return nsec ; +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Get name for section n in a dictionary. + @param d Dictionary to examine + @param n Section number (from 0 to nsec-1). + @return Pointer to char string + + This function locates the n-th section in a dictionary and returns + its name as a pointer to a string statically allocated inside the + dictionary. Do not free or modify the returned string! + + This function returns NULL in case of error. + */ +/*--------------------------------------------------------------------------*/ +char * iniparser_getsecname(dictionary * d, int n) +{ + int i ; + int foundsec ; + + if (d==NULL || n<0) return NULL ; + foundsec=0 ; + for (i=0 ; isize ; i++) { + if (d->key[i]==NULL) + continue ; + if (strchr(d->key[i], ':')==NULL) { + foundsec++ ; + if (foundsec>n) + break ; + } + } + if (foundsec<=n) { + return NULL ; + } + return d->key[i] ; +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Dump a dictionary to an opened file pointer. + @param d Dictionary to dump. + @param f Opened file pointer to dump to. + @return void + + This function prints out the contents of a dictionary, one element by + line, onto the provided file pointer. It is OK to specify @c stderr + or @c stdout as output files. This function is meant for debugging + purposes mostly. + */ +/*--------------------------------------------------------------------------*/ +void iniparser_dump(dictionary * d, FILE * f) +{ + int i ; + + if (d==NULL || f==NULL) return ; + for (i=0 ; isize ; i++) { + if (d->key[i]==NULL) + continue ; + if (d->val[i]!=NULL) { + fprintf(f, "[%s]=[%s]\n", d->key[i], d->val[i]); + } else { + fprintf(f, "[%s]=UNDEF\n", d->key[i]); + } + } + return ; +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Save a dictionary to a loadable ini file + @param d Dictionary to dump + @param f Opened file pointer to dump to + @return void + + This function dumps a given dictionary into a loadable ini file. + It is Ok to specify @c stderr or @c stdout as output files. + */ +/*--------------------------------------------------------------------------*/ +void iniparser_dump_ini(dictionary * d, FILE * f) +{ + int i ; + int nsec ; + char * secname ; + + if (d==NULL || f==NULL) return ; + + nsec = iniparser_getnsec(d); + if (nsec<1) { + /* No section in file: dump all keys as they are */ + for (i=0 ; isize ; i++) { + if (d->key[i]==NULL) + continue ; + fprintf(f, "%s = %s\n", d->key[i], d->val[i]); + } + return ; + } + for (i=0 ; isize ; j++) { + if (d->key[j]==NULL) + continue ; + if (!strncmp(d->key[j], keym, seclen+1)) { + fprintf(f, + "%-30s = %s\n", + d->key[j]+seclen+1, + d->val[j] ? d->val[j] : ""); + } + } + fprintf(f, "\n"); + return ; +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Get the number of keys in a section of a dictionary. + @param d Dictionary to examine + @param s Section name of dictionary to examine + @return Number of keys in section + */ +/*--------------------------------------------------------------------------*/ +int iniparser_getsecnkeys(dictionary * d, char * s) +{ + int seclen, nkeys ; + char keym[ASCIILINESZ+1]; + int j ; + + nkeys = 0; + + if (d==NULL) return nkeys; + if (! iniparser_find_entry(d, s)) return nkeys; + + seclen = (int)strlen(s); + sprintf(keym, "%s:", s); + + for (j=0 ; jsize ; j++) { + if (d->key[j]==NULL) + continue ; + if (!strncmp(d->key[j], keym, seclen+1)) + nkeys++; + } + + return nkeys; + +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Get the number of keys in a section of a dictionary. + @param d Dictionary to examine + @param s Section name of dictionary to examine + @return pointer to statically allocated character strings + + This function queries a dictionary and finds all keys in a given section. + Each pointer in the returned char pointer-to-pointer is pointing to + a string allocated in the dictionary; do not free or modify them. + + This function returns NULL in case of error. + */ +/*--------------------------------------------------------------------------*/ +char ** iniparser_getseckeys(dictionary * d, char * s) +{ + + char **keys; + + int i, j ; + char keym[ASCIILINESZ+1]; + int seclen, nkeys ; + + keys = NULL; + + if (d==NULL) return keys; + if (! iniparser_find_entry(d, s)) return keys; + + nkeys = iniparser_getsecnkeys(d, s); + + keys = (char**) malloc(nkeys*sizeof(char*)); + + seclen = (int)strlen(s); + sprintf(keym, "%s:", s); + + i = 0; + + for (j=0 ; jsize ; j++) { + if (d->key[j]==NULL) + continue ; + if (!strncmp(d->key[j], keym, seclen+1)) { + keys[i] = d->key[j]; + i++; + } + } + + return keys; + +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Get the string associated to a key + @param d Dictionary to search + @param key Key string to look for + @param def Default value to return if key not found. + @return pointer to statically allocated character string + + This function queries a dictionary for a key. A key as read from an + ini file is given as "section:key". If the key cannot be found, + the pointer passed as 'def' is returned. + The returned char pointer is pointing to a string allocated in + the dictionary, do not free or modify it. + */ +/*--------------------------------------------------------------------------*/ +char * iniparser_getstring(dictionary * d, const char * key, char * def) +{ + char * lc_key ; + char * sval ; + + if (d==NULL || key==NULL) + return def ; + + lc_key = strlwc(key); + sval = dictionary_get(d, lc_key, def); + return sval ; +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Get the string associated to a key, convert to an int + @param d Dictionary to search + @param key Key string to look for + @param notfound Value to return in case of error + @return integer + + This function queries a dictionary for a key. A key as read from an + ini file is given as "section:key". If the key cannot be found, + the notfound value is returned. + + Supported values for integers include the usual C notation + so decimal, octal (starting with 0) and hexadecimal (starting with 0x) + are supported. Examples: + + "42" -> 42 + "042" -> 34 (octal -> decimal) + "0x42" -> 66 (hexa -> decimal) + + Warning: the conversion may overflow in various ways. Conversion is + totally outsourced to strtol(), see the associated man page for overflow + handling. + + Credits: Thanks to A. Becker for suggesting strtol() + */ +/*--------------------------------------------------------------------------*/ +int iniparser_getint(dictionary * d, const char * key, int notfound) +{ + char * str ; + + str = iniparser_getstring(d, key, INI_INVALID_KEY); + if (str==INI_INVALID_KEY) return notfound ; + return (int)strtol(str, NULL, 0); +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Get the string associated to a key, convert to a double + @param d Dictionary to search + @param key Key string to look for + @param notfound Value to return in case of error + @return double + + This function queries a dictionary for a key. A key as read from an + ini file is given as "section:key". If the key cannot be found, + the notfound value is returned. + */ +/*--------------------------------------------------------------------------*/ +double iniparser_getdouble(dictionary * d, const char * key, double notfound) +{ + char * str ; + + str = iniparser_getstring(d, key, INI_INVALID_KEY); + if (str==INI_INVALID_KEY) return notfound ; + return atof(str); +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Get the string associated to a key, convert to a boolean + @param d Dictionary to search + @param key Key string to look for + @param notfound Value to return in case of error + @return integer + + This function queries a dictionary for a key. A key as read from an + ini file is given as "section:key". If the key cannot be found, + the notfound value is returned. + + A true boolean is found if one of the following is matched: + + - A string starting with 'y' + - A string starting with 'Y' + - A string starting with 't' + - A string starting with 'T' + - A string starting with '1' + + A false boolean is found if one of the following is matched: + + - A string starting with 'n' + - A string starting with 'N' + - A string starting with 'f' + - A string starting with 'F' + - A string starting with '0' + + The notfound value returned if no boolean is identified, does not + necessarily have to be 0 or 1. + */ +/*--------------------------------------------------------------------------*/ +int iniparser_getboolean(dictionary * d, const char * key, int notfound) +{ + char * c ; + int ret ; + + c = iniparser_getstring(d, key, INI_INVALID_KEY); + if (c==INI_INVALID_KEY) return notfound ; + if (c[0]=='y' || c[0]=='Y' || c[0]=='1' || c[0]=='t' || c[0]=='T') { + ret = 1 ; + } else if (c[0]=='n' || c[0]=='N' || c[0]=='0' || c[0]=='f' || c[0]=='F') { + ret = 0 ; + } else { + ret = notfound ; + } + return ret; +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Finds out if a given entry exists in a dictionary + @param ini Dictionary to search + @param entry Name of the entry to look for + @return integer 1 if entry exists, 0 otherwise + + Finds out if a given entry exists in the dictionary. Since sections + are stored as keys with NULL associated values, this is the only way + of querying for the presence of sections in a dictionary. + */ +/*--------------------------------------------------------------------------*/ +int iniparser_find_entry( + dictionary * ini, + const char * entry +) +{ + int found=0 ; + if (iniparser_getstring(ini, entry, INI_INVALID_KEY)!=INI_INVALID_KEY) { + found = 1 ; + } + return found ; +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Set an entry in a dictionary. + @param ini Dictionary to modify. + @param entry Entry to modify (entry name) + @param val New value to associate to the entry. + @return int 0 if Ok, -1 otherwise. + + If the given entry can be found in the dictionary, it is modified to + contain the provided value. If it cannot be found, -1 is returned. + It is Ok to set val to NULL. + */ +/*--------------------------------------------------------------------------*/ +int iniparser_set(dictionary * ini, const char * entry, const char * val) +{ + return dictionary_set(ini, strlwc(entry), val) ; +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Delete an entry in a dictionary + @param ini Dictionary to modify + @param entry Entry to delete (entry name) + @return void + + If the given entry can be found, it is deleted from the dictionary. + */ +/*--------------------------------------------------------------------------*/ +void iniparser_unset(dictionary * ini, const char * entry) +{ + dictionary_unset(ini, strlwc(entry)); +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Load a single line from an INI file + @param input_line Input line, may be concatenated multi-line input + @param section Output space to store section + @param key Output space to store key + @param value Output space to store value + @return line_status value + */ +/*--------------------------------------------------------------------------*/ +static line_status iniparser_line( + const char * input_line, + char * section, + char * key, + char * value) +{ + line_status sta ; + char line[ASCIILINESZ+1]; + int len ; + + strcpy(line, strstrip(input_line)); + len = (int)strlen(line); + + sta = LINE_UNPROCESSED ; + if (len<1) { + /* Empty line */ + sta = LINE_EMPTY ; + } else if (line[0]=='#' || line[0]==';') { + /* Comment line */ + sta = LINE_COMMENT ; + } else if (line[0]=='[' && line[len-1]==']') { + /* Section name */ + sscanf(line, "[%[^]]", section); + strcpy(section, strstrip(section)); + strcpy(section, strlwc(section)); + sta = LINE_SECTION ; + } else if (sscanf (line, "%[^=] = \"%[^\"]\"", key, value) == 2 + || sscanf (line, "%[^=] = '%[^\']'", key, value) == 2 + || sscanf (line, "%[^=] = %[^;#]", key, value) == 2) { + /* Usual key=value, with or without comments */ + strcpy(key, strstrip(key)); + strcpy(key, strlwc(key)); + strcpy(value, strstrip(value)); + /* + * sscanf cannot handle '' or "" as empty values + * this is done here + */ + if (!strcmp(value, "\"\"") || (!strcmp(value, "''"))) { + value[0]=0 ; + } + sta = LINE_VALUE ; + } else if (sscanf(line, "%[^=] = %[;#]", key, value)==2 + || sscanf(line, "%[^=] %[=]", key, value) == 2) { + /* + * Special cases: + * key= + * key=; + * key=# + */ + strcpy(key, strstrip(key)); + strcpy(key, strlwc(key)); + value[0]=0 ; + sta = LINE_VALUE ; + } else { + /* Generate syntax error */ + sta = LINE_ERROR ; + } + return sta ; +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Parse an ini file and return an allocated dictionary object + @param ininame Name of the ini file to read. + @return Pointer to newly allocated dictionary + + This is the parser for ini files. This function is called, providing + the name of the file to be read. It returns a dictionary object that + should not be accessed directly, but through accessor functions + instead. + + The returned dictionary must be freed using iniparser_freedict(). + */ +/*--------------------------------------------------------------------------*/ +dictionary * iniparser_load(const char * ininame) +{ + FILE * in ; + + char line [ASCIILINESZ+1] ; + char section [ASCIILINESZ+1] ; + char key [ASCIILINESZ+1] ; + char tmp [ASCIILINESZ+1] ; + char val [ASCIILINESZ+1] ; + + int last=0 ; + int len ; + int lineno=0 ; + int errs=0; + + dictionary * dict ; + + if ((in=fopen(ininame, "r"))==NULL) { + fprintf(stderr, "iniparser: cannot open %s\n", ininame); + return NULL ; + } + + dict = dictionary_new(0) ; + if (!dict) { + fclose(in); + return NULL ; + } + + memset(line, 0, ASCIILINESZ); + memset(section, 0, ASCIILINESZ); + memset(key, 0, ASCIILINESZ); + memset(val, 0, ASCIILINESZ); + last=0 ; + + while (fgets(line+last, ASCIILINESZ-last, in)!=NULL) { + lineno++ ; + len = (int)strlen(line)-1; + if (len==0) + continue; + /* Safety check against buffer overflows */ + if (line[len]!='\n') { + fprintf(stderr, + "iniparser: input line too long in %s (%d)\n", + ininame, + lineno); + dictionary_del(dict); + fclose(in); + return NULL ; + } + /* Get rid of \n and spaces at end of line */ + while ((len>=0) && + ((line[len]=='\n') || (isspace(line[len])))) { + line[len]=0 ; + len-- ; + } + /* Detect multi-line */ + if (line[len]=='\\') { + /* Multi-line value */ + last=len ; + continue ; + } else { + last=0 ; + } + switch (iniparser_line(line, section, key, val)) { + case LINE_EMPTY: + case LINE_COMMENT: + break ; + + case LINE_SECTION: + errs = dictionary_set(dict, section, NULL); + break ; + + case LINE_VALUE: + sprintf(tmp, "%s:%s", section, key); + errs = dictionary_set(dict, tmp, val) ; + break ; + + case LINE_ERROR: + fprintf(stderr, "iniparser: syntax error in %s (%d):\n", + ininame, + lineno); + fprintf(stderr, "-> %s\n", line); + errs++ ; + break; + + default: + break ; + } + memset(line, 0, ASCIILINESZ); + last=0; + if (errs<0) { + fprintf(stderr, "iniparser: memory allocation failure\n"); + break ; + } + } + if (errs) { + dictionary_del(dict); + dict = NULL ; + } + fclose(in); + return dict ; +} + +/*-------------------------------------------------------------------------*/ +/** + @brief Free all memory associated to an ini dictionary + @param d Dictionary to free + @return void + + Free all memory associated to an ini dictionary. + It is mandatory to call this function before the dictionary object + gets out of the current context. + */ +/*--------------------------------------------------------------------------*/ +void iniparser_freedict(dictionary * d) +{ + dictionary_del(d); +} + +/* vim: set ts=4 et sw=4 tw=75 */ diff --git a/src/lib/iniparser/src/iniparser.h b/src/lib/iniparser/src/iniparser.h new file mode 100644 index 00000000..ecd73bf9 --- /dev/null +++ b/src/lib/iniparser/src/iniparser.h @@ -0,0 +1,307 @@ + +/*-------------------------------------------------------------------------*/ +/** + @file iniparser.h + @author N. Devillard + @brief Parser for ini files. +*/ +/*--------------------------------------------------------------------------*/ + +#ifndef _INIPARSER_H_ +#define _INIPARSER_H_ + +/*--------------------------------------------------------------------------- + Includes + ---------------------------------------------------------------------------*/ + +#include +#include +#include + +/* + * The following #include is necessary on many Unixes but not Linux. + * It is not needed for Windows platforms. + * Uncomment it if needed. + */ +/* #include */ + +#include "dictionary.h" + +/*-------------------------------------------------------------------------*/ +/** + @brief Get number of sections in a dictionary + @param d Dictionary to examine + @return int Number of sections found in dictionary + + This function returns the number of sections found in a dictionary. + The test to recognize sections is done on the string stored in the + dictionary: a section name is given as "section" whereas a key is + stored as "section:key", thus the test looks for entries that do not + contain a colon. + + This clearly fails in the case a section name contains a colon, but + this should simply be avoided. + + This function returns -1 in case of error. + */ +/*--------------------------------------------------------------------------*/ + +int iniparser_getnsec(dictionary * d); + + +/*-------------------------------------------------------------------------*/ +/** + @brief Get name for section n in a dictionary. + @param d Dictionary to examine + @param n Section number (from 0 to nsec-1). + @return Pointer to char string + + This function locates the n-th section in a dictionary and returns + its name as a pointer to a string statically allocated inside the + dictionary. Do not free or modify the returned string! + + This function returns NULL in case of error. + */ +/*--------------------------------------------------------------------------*/ + +char * iniparser_getsecname(dictionary * d, int n); + + +/*-------------------------------------------------------------------------*/ +/** + @brief Save a dictionary to a loadable ini file + @param d Dictionary to dump + @param f Opened file pointer to dump to + @return void + + This function dumps a given dictionary into a loadable ini file. + It is Ok to specify @c stderr or @c stdout as output files. + */ +/*--------------------------------------------------------------------------*/ + +void iniparser_dump_ini(dictionary * d, FILE * f); + +/*-------------------------------------------------------------------------*/ +/** + @brief Save a dictionary section to a loadable ini file + @param d Dictionary to dump + @param s Section name of dictionary to dump + @param f Opened file pointer to dump to + @return void + + This function dumps a given section of a given dictionary into a loadable ini + file. It is Ok to specify @c stderr or @c stdout as output files. + */ +/*--------------------------------------------------------------------------*/ + +void iniparser_dumpsection_ini(dictionary * d, char * s, FILE * f); + +/*-------------------------------------------------------------------------*/ +/** + @brief Dump a dictionary to an opened file pointer. + @param d Dictionary to dump. + @param f Opened file pointer to dump to. + @return void + + This function prints out the contents of a dictionary, one element by + line, onto the provided file pointer. It is OK to specify @c stderr + or @c stdout as output files. This function is meant for debugging + purposes mostly. + */ +/*--------------------------------------------------------------------------*/ +void iniparser_dump(dictionary * d, FILE * f); + +/*-------------------------------------------------------------------------*/ +/** + @brief Get the number of keys in a section of a dictionary. + @param d Dictionary to examine + @param s Section name of dictionary to examine + @return Number of keys in section + */ +/*--------------------------------------------------------------------------*/ +int iniparser_getsecnkeys(dictionary * d, char * s); + +/*-------------------------------------------------------------------------*/ +/** + @brief Get the number of keys in a section of a dictionary. + @param d Dictionary to examine + @param s Section name of dictionary to examine + @return pointer to statically allocated character strings + + This function queries a dictionary and finds all keys in a given section. + Each pointer in the returned char pointer-to-pointer is pointing to + a string allocated in the dictionary; do not free or modify them. + + This function returns NULL in case of error. + */ +/*--------------------------------------------------------------------------*/ +char ** iniparser_getseckeys(dictionary * d, char * s); + +/*-------------------------------------------------------------------------*/ +/** + @brief Get the string associated to a key + @param d Dictionary to search + @param key Key string to look for + @param def Default value to return if key not found. + @return pointer to statically allocated character string + + This function queries a dictionary for a key. A key as read from an + ini file is given as "section:key". If the key cannot be found, + the pointer passed as 'def' is returned. + The returned char pointer is pointing to a string allocated in + the dictionary, do not free or modify it. + */ +/*--------------------------------------------------------------------------*/ +char * iniparser_getstring(dictionary * d, const char * key, char * def); + +/*-------------------------------------------------------------------------*/ +/** + @brief Get the string associated to a key, convert to an int + @param d Dictionary to search + @param key Key string to look for + @param notfound Value to return in case of error + @return integer + + This function queries a dictionary for a key. A key as read from an + ini file is given as "section:key". If the key cannot be found, + the notfound value is returned. + + Supported values for integers include the usual C notation + so decimal, octal (starting with 0) and hexadecimal (starting with 0x) + are supported. Examples: + + - "42" -> 42 + - "042" -> 34 (octal -> decimal) + - "0x42" -> 66 (hexa -> decimal) + + Warning: the conversion may overflow in various ways. Conversion is + totally outsourced to strtol(), see the associated man page for overflow + handling. + + Credits: Thanks to A. Becker for suggesting strtol() + */ +/*--------------------------------------------------------------------------*/ +int iniparser_getint(dictionary * d, const char * key, int notfound); + +/*-------------------------------------------------------------------------*/ +/** + @brief Get the string associated to a key, convert to a double + @param d Dictionary to search + @param key Key string to look for + @param notfound Value to return in case of error + @return double + + This function queries a dictionary for a key. A key as read from an + ini file is given as "section:key". If the key cannot be found, + the notfound value is returned. + */ +/*--------------------------------------------------------------------------*/ +double iniparser_getdouble(dictionary * d, const char * key, double notfound); + +/*-------------------------------------------------------------------------*/ +/** + @brief Get the string associated to a key, convert to a boolean + @param d Dictionary to search + @param key Key string to look for + @param notfound Value to return in case of error + @return integer + + This function queries a dictionary for a key. A key as read from an + ini file is given as "section:key". If the key cannot be found, + the notfound value is returned. + + A true boolean is found if one of the following is matched: + + - A string starting with 'y' + - A string starting with 'Y' + - A string starting with 't' + - A string starting with 'T' + - A string starting with '1' + + A false boolean is found if one of the following is matched: + + - A string starting with 'n' + - A string starting with 'N' + - A string starting with 'f' + - A string starting with 'F' + - A string starting with '0' + + The notfound value returned if no boolean is identified, does not + necessarily have to be 0 or 1. + */ +/*--------------------------------------------------------------------------*/ +int iniparser_getboolean(dictionary * d, const char * key, int notfound); + + +/*-------------------------------------------------------------------------*/ +/** + @brief Set an entry in a dictionary. + @param ini Dictionary to modify. + @param entry Entry to modify (entry name) + @param val New value to associate to the entry. + @return int 0 if Ok, -1 otherwise. + + If the given entry can be found in the dictionary, it is modified to + contain the provided value. If it cannot be found, -1 is returned. + It is Ok to set val to NULL. + */ +/*--------------------------------------------------------------------------*/ +int iniparser_set(dictionary * ini, const char * entry, const char * val); + + +/*-------------------------------------------------------------------------*/ +/** + @brief Delete an entry in a dictionary + @param ini Dictionary to modify + @param entry Entry to delete (entry name) + @return void + + If the given entry can be found, it is deleted from the dictionary. + */ +/*--------------------------------------------------------------------------*/ +void iniparser_unset(dictionary * ini, const char * entry); + +/*-------------------------------------------------------------------------*/ +/** + @brief Finds out if a given entry exists in a dictionary + @param ini Dictionary to search + @param entry Name of the entry to look for + @return integer 1 if entry exists, 0 otherwise + + Finds out if a given entry exists in the dictionary. Since sections + are stored as keys with NULL associated values, this is the only way + of querying for the presence of sections in a dictionary. + */ +/*--------------------------------------------------------------------------*/ +int iniparser_find_entry(dictionary * ini, const char * entry) ; + +/*-------------------------------------------------------------------------*/ +/** + @brief Parse an ini file and return an allocated dictionary object + @param ininame Name of the ini file to read. + @return Pointer to newly allocated dictionary + + This is the parser for ini files. This function is called, providing + the name of the file to be read. It returns a dictionary object that + should not be accessed directly, but through accessor functions + instead. + + The returned dictionary must be freed using iniparser_freedict(). + */ +/*--------------------------------------------------------------------------*/ +dictionary * iniparser_load(const char * ininame); + +/*-------------------------------------------------------------------------*/ +/** + @brief Free all memory associated to an ini dictionary + @param d Dictionary to free + @return void + + Free all memory associated to an ini dictionary. + It is mandatory to call this function before the dictionary object + gets out of the current context. + */ +/*--------------------------------------------------------------------------*/ +void iniparser_freedict(dictionary * d); + +#endif From 6a7458b116f0920e9bd803f38521936332993157 Mon Sep 17 00:00:00 2001 From: Bostjan Skufca Date: Fri, 24 Oct 2014 18:44:41 +0000 Subject: [PATCH 2/3] Added configuration option for config file, debugged proper installation and non-overwriting of existing file --- Makefile.am | 2 +- config.h.in | 3 +++ configure.ac | 51 ++++++++++++++++++++++++++++++++++++++++++------- etc/Makefile.am | 14 ++++++++++++++ etc/snoopy.ini | 2 ++ 5 files changed, 64 insertions(+), 8 deletions(-) create mode 100644 etc/Makefile.am create mode 100644 etc/snoopy.ini diff --git a/Makefile.am b/Makefile.am index 70c0d718..860b5c26 100644 --- a/Makefile.am +++ b/Makefile.am @@ -3,7 +3,7 @@ include $(top_srcdir)/build/Makefile.am.common AUTOMAKE_OPTIONS = foreign subdir-objects ACLOCAL_AMFLAGS = -I m4 -SUBDIRS = src bin +SUBDIRS = src bin etc EXTRA_DIST +=\ autogen.sh \ diff --git a/config.h.in b/config.h.in index 36e3dca2..6b481d0a 100644 --- a/config.h.in +++ b/config.h.in @@ -77,6 +77,9 @@ /* Define to the version of this package. */ #undef PACKAGE_VERSION +/* INI configuration file path to use */ +#undef SNOOPY_CONF_CONFIG_FILE + /* Custom filter chain to use */ #undef SNOOPY_CONF_FILTER_CHAIN_custom diff --git a/configure.ac b/configure.ac index dfe80859..2537f217 100644 --- a/configure.ac +++ b/configure.ac @@ -7,12 +7,12 @@ AC_DEFINE(PACKAGE_URL, [], "https://github.com/a2o/snoopy/") AC_CONFIG_SRCDIR([config.h.in]) AC_CONFIG_HEADERS([config.h]) -# We are using automake and libtool +dnl We are using automake and libtool AC_CONFIG_MACRO_DIR([m4]) AM_INIT_AUTOMAKE([1.11 gnu silent-rules subdir-objects -Wall -Werror]) LT_INIT([disable-static]) -# Checks for programs. +dnl Checks for programs. AC_PROG_CXX AC_PROG_AWK AC_PROG_CC @@ -22,24 +22,26 @@ AC_PROG_LN_S AC_PROG_MAKE_SET AC_PROG_RANLIB -# Checks for libraries. -# FIXME: Replace `main' with a function in `-ldl': +dnl Checks for libraries. +dnl FIXME: Replace `main' with a function in `-ldl': AC_CHECK_LIB([dl], [main]) -# Checks for header files. +dnl Checks for header files. AC_CHECK_HEADERS([limits.h stdlib.h string.h syslog.h unistd.h]) -# Checks for typedefs, structures, and compiler characteristics. +dnl Checks for typedefs, structures, and compiler characteristics. AC_C_INLINE AC_TYPE_SIZE_T -# Checks for library functions. +dnl Checks for library functions. AC_FUNC_MALLOC AC_CHECK_FUNCS([getcwd getsid strstr]) +dnl Generate these (Make)files AC_CONFIG_FILES([Makefile bin/Makefile contrib/sles/snoopy.spec + etc/Makefile src/Makefile src/lib/Makefile src/filter/Makefile @@ -47,6 +49,23 @@ AC_CONFIG_FILES([Makefile +dnl ============================================================================ + +dnl Decide where things are installed +if test "x$prefix" = "xNONE" ; then + PREFIX="/usr/local" +else + PREFIX="$prefix" +fi + +if test "x$sysconfdir" = "x\${prefix}/etc" ; then + SYSCONFDIR="$PREFIX/etc" +else + SYSCONFDIR="$sysconfdir" +fi + + + dnl ============================================================================ AC_ARG_ENABLE(root-only, [AC_HELP_STRING( @@ -130,5 +149,23 @@ AC_DEFINE_UNQUOTED(SNOOPY_SYSLOG_LEVEL, [$with_syslog_level], [Syslog level to u +dnl ============================================================================ +AC_ARG_WITH(config-file, + [AC_HELP_STRING( + [--with-config-file=PATH], + [enable INI configuration file parsing [default=disabled, if enabled then default path is PREFIX/etc/snoopy.ini]] + )], + [ + if test "$with_config_file" == "yes"; then + with_config_file_path="$SYSCONFDIR/snoopy.ini" + else + with_config_file_path="$with_config_file" + fi + AC_DEFINE_UNQUOTED(SNOOPY_CONF_CONFIG_FILE, [$with_config_file_path], [INI configuration file path to use]) + ] +) + + + dnl ============================================================================ AC_OUTPUT diff --git a/etc/Makefile.am b/etc/Makefile.am new file mode 100644 index 00000000..1b80d0cf --- /dev/null +++ b/etc/Makefile.am @@ -0,0 +1,14 @@ +include $(top_srcdir)/build/Makefile.am.common + +AUTOMAKE_OPTIONS = foreign subdir-objects +ACLOCAL_AMFLAGS = -I m4 + +install-exec-local: + if [ -f $(sysconfdir)/snoopy.ini ]; then \ + echo ; \ + echo "CONFIG FILE EXISTS, skipping: $(sysconfdir)/snoopy.ini" ; \ + echo ; \ + else \ + echo "Installing: $(sysconfdir)/snoopy.ini" ; \ + install snoopy.ini $(sysconfdir)/snoopy.ini ; \ + fi diff --git a/etc/snoopy.ini b/etc/snoopy.ini new file mode 100644 index 00000000..586f698e --- /dev/null +++ b/etc/snoopy.ini @@ -0,0 +1,2 @@ +[snoopy] +log_format = "[uid:%{uid} sid:%{sid} tty:%{tty} cwd:%{cwd} filename:%{filename}]: %{cmdline}" From bdca2c03195c33ee8a95902713dd25508d4429cb Mon Sep 17 00:00:00 2001 From: Bostjan Skufca Date: Sat, 25 Oct 2014 01:06:03 +0000 Subject: [PATCH 3/3] Added configuration file options support, updated changelog, bumped revision number --- ChangeLog | 5 +- README.md | 51 +++++------ bin/snoopy-test-output.c | 38 +++++--- configure.ac | 6 +- etc/snoopy.ini | 52 ++++++++++- src/Makefile.am | 2 + src/configuration.c | 193 +++++++++++++++++++++++++++++++++++++++ src/configuration.h | 50 ++++++++++ src/log.c | 31 ++++--- src/snoopy.h | 13 ++- 10 files changed, 384 insertions(+), 57 deletions(-) create mode 100644 src/configuration.c create mode 100644 src/configuration.h diff --git a/ChangeLog b/ChangeLog index 82455a58..7a1db528 100644 --- a/ChangeLog +++ b/ChangeLog @@ -3,7 +3,7 @@ ------------------------------------------------------------------------------- -2014-FIXME - Version 2.0.0rc5 +2014-FIXME - Version 2.0.0rc6 --------------------------- o Feature: Added support for custom message format specification at configuration time @@ -20,6 +20,8 @@ o Feature: Implemented internal filtering, with the following filters available: - exclude_uid - only_root - only_uid +o Feature: Added optional support for INI configuration file + o Refactoring: Separated data gathering into separate functions, to be used at will - now they are called "input providers" o Refactoring: Removed external filtering, to be replaced with internal @@ -32,6 +34,7 @@ o Refactoring: removed --enable-cwd-logging (superseded by message formatting options and input provider) o Refactoring: snoopy library is now called libsnoopy.so instead of snoopy.so, to conform to autotools syntax + o Documentation: added into about writing new input providers o Bugfix: fixed compilation failure on debian in inputs/cwd.c (thanks Sébastien Gross) diff --git a/README.md b/README.md index 3a42be61..dc55f363 100644 --- a/README.md +++ b/README.md @@ -64,38 +64,33 @@ in question. ### Configuring filtering -Snoopy supports message filtering. Filtering must be configured at -build time, here is an example: +Snoopy supports message filtering. Filtering support must be +enabled at build time, here is an example: # REQUIRED TO ENABLE FILTERING FEATURE --enable-filter - # HOW TO DEFINE FILTERS - --with-filter-chain="exclude_uid:0" # Log all commands, except the ones executed by root - --with-filter-chain="exclude_uid:1,2,3" # Log all commands, except those executed by users with UIDs 1, 2 and 3 - --with-filter-chain="only_uid:0" # Log only root commands - --with-filter-chain="filter1:arg11;filter2:arg21,arg22;filter3:arg31,32,33" - -Here you have four filter definitions for your reference, they are -quite self-explanatory. As you probably noted in the last example, -multiple filters may be defined in a chain, separated by semicolon. - -Each filter chains can contain multiple filter definitions. They are -processed in order of appearance. If any of the filters decides the -message should be dropped, the filter chain processing is immediately -interrupted and message is not passed to syslog. - -If filter requires an argument, they may be passed to them by -specifying a colon after filter name, followed by an argument. -Argument is passed to the filter as-is. If passing of multiple -arguments to filter is required, they are passed as single string -and must be parsed/tokenized by the filter itself (see "only_uid" -filter for example). - -Filter chain specification may not contain any spaces. -(Acutally spaces are allowed in arguments, but not in filter names -and between semicolons and filter names, nor between filter names -and following colons.) + # HOW TO DEFINE FILTER CHAINS + --with-filter-chain="FILTER_CHAIN_SPEC" + +By default, if FILTER_CHAIN_SPEC is not configured, empty string is +used, which effectively disables filtering. + +See sample configuration file etc/snoopy.ini for list and description +of supported filter configurations. + + +### Optional configuration file support + +Snoopy supports optional configuration file, which may help with +development and/or configuration endeavours. Configuration file must +be enabled at build time: + + --with-config-file[=PATH] + +If PATH is not specified, default path SYSCONFDIR/snoopy.ini is used +instead. See sample configuration file etc/snoopy.ini for list and +description of supported configuration directives. diff --git a/bin/snoopy-test-output.c b/bin/snoopy-test-output.c index 38a8da54..89e36f1d 100644 --- a/bin/snoopy-test-output.c +++ b/bin/snoopy-test-output.c @@ -20,6 +20,7 @@ */ #include "snoopy.h" #include "log.h" +#include "configuration.h" #include "inputdatastorage.h" #include "filterregistry.h" #include @@ -38,30 +39,43 @@ int main (int argc, char **argv) snoopy_inputdatastorage_store_filename(argv[0]); snoopy_inputdatastorage_store_argv(argv); - snoopy_log_message_generate(logMessage, SNOOPY_LOG_MESSAGE_FORMAT); + snoopy_configuration_ctor(); + if (SNOOPY_TRUE == snoopy_configuration.config_file_enabled) { + printf("Configuration file is enabled: %s\n", snoopy_configuration.config_file_path); + if (SNOOPY_TRUE == snoopy_configuration.config_file_parsed) { + printf("Configuration file was parsed sucessfully.\n"); + } else { + printf("WARNING: Configuration file parsing FAILED!\n"); + } + } else { + printf("INFO: Configuration file is NOT enabled.\n"); + } + + snoopy_log_message_generate(logMessage, snoopy_configuration.message_format); printf("Message generated:\n"); printf("\n"); printf("%s\n", logMessage); printf("\n"); -#if defined(SNOOPY_FILTER_ENABLED) - if (SNOOPY_FILTER_PASS == snoopy_log_filter_check_chain(logMessage, SNOOPY_FILTER_CHAIN)) { - /* Send it to syslog */ + if (SNOOPY_TRUE == snoopy_configuration.filter_enabled) { + if (SNOOPY_FILTER_PASS == snoopy_log_filter_check_chain(logMessage, snoopy_configuration.filter_chain)) { + /* Send it to syslog */ + snoopy_log_send_to_syslog(logMessage); + printf("Message sent to syslog, check your syslog output.\n"); + printf("If snoopy is already enabled on your system, you should see two identical messages.\n"); + printf("If you are testing snoopy via LD_PRELOAD environmental variable, you will see another identical message.\n"); + } else { + printf("Message NOT sent to syslog. One of the filters dropped it.\n"); + } + } else { snoopy_log_send_to_syslog(logMessage); printf("Message sent to syslog, check your syslog output.\n"); printf("If snoopy is already enabled on your system, you should see two identical messages.\n"); printf("If you are testing snoopy via LD_PRELOAD environmental variable, you will see another identical message.\n"); - } else { - printf("Message NOT sent to syslog. One of the filters dropped it.\n"); } -#else - snoopy_log_send_to_syslog(logMessage); - printf("Message sent to syslog, check your syslog output.\n"); - printf("If snoopy is already enabled on your system, you should see two identical messages.\n"); - printf("If you are testing snoopy via LD_PRELOAD environmental variable, you will see another identical message.\n"); -#endif /* Housekeeping */ + snoopy_configuration_dtor(); free(logMessage); return 0; } diff --git a/configure.ac b/configure.ac index 2537f217..a7c5915d 100644 --- a/configure.ac +++ b/configure.ac @@ -2,7 +2,7 @@ # Process this file with autoconf to produce a configure script. AC_PREREQ([2.63]) -AC_INIT([Snoopy Logger], [2.0.0rc5], [https://github.com/a2o/snoopy/], [snoopy]) +AC_INIT([Snoopy Logger], [2.0.0rc6], [https://github.com/a2o/snoopy/], [snoopy]) AC_DEFINE(PACKAGE_URL, [], "https://github.com/a2o/snoopy/") AC_CONFIG_SRCDIR([config.h.in]) AC_CONFIG_HEADERS([config.h]) @@ -153,7 +153,7 @@ dnl ============================================================================ AC_ARG_WITH(config-file, [AC_HELP_STRING( [--with-config-file=PATH], - [enable INI configuration file parsing [default=disabled, if enabled then default path is PREFIX/etc/snoopy.ini]] + [enable INI configuration file parsing [default=disabled, if enabled then default path is SYSCONFDIR/snoopy.ini]] )], [ if test "$with_config_file" == "yes"; then @@ -161,7 +161,7 @@ AC_ARG_WITH(config-file, else with_config_file_path="$with_config_file" fi - AC_DEFINE_UNQUOTED(SNOOPY_CONF_CONFIG_FILE, [$with_config_file_path], [INI configuration file path to use]) + AC_DEFINE_UNQUOTED(SNOOPY_CONF_CONFIG_FILE, "$with_config_file_path", [INI configuration file path to use]) ] ) diff --git a/etc/snoopy.ini b/etc/snoopy.ini index 586f698e..2a1e9405 100644 --- a/etc/snoopy.ini +++ b/etc/snoopy.ini @@ -1,2 +1,52 @@ +;;; REQUIRED Section +; [snoopy] -log_format = "[uid:%{uid} sid:%{sid} tty:%{tty} cwd:%{cwd} filename:%{filename}]: %{cmdline}" + + +;;; Log Message Format specification +; +; May consist of: +; - arbitrary text, is copied to log message as-is, +; - calls to input providers without arguments: %{input} +; - calls to input providers with argument : %{input:arg1} +; - calls to input providers with arguments: %{input:arg1,arg2} <--- if input provider supports it +; +; Default value: +; "[uid:%{uid} sid:%{sid} tty:%{tty} cwd:%{cwd} filename:%{filename}]: %{cmdline}" +; +message_format = "[uid:%{uid} sid:%{sid} tty:%{tty} cwd:%{cwd} filename:%{filename}]: %{cmdline}" + + +;;; Filter Chain specification +; +; Example definitions: +; - filter_chain = "exclude_uid:0" # Log all commands, except the ones executed by root +; - filter_chain = "exclude_uid:1,2,3" # Log all commands, except those executed by users with UIDs 1, 2 and 3 +; - filter_chain = "only_uid:0" # Log only root commands +; - filter_chain = "filter1:arg11;filter2:arg21,arg22;filter3:arg31,32,33" +; +; Here you have four filter definitions for your reference, they are +; quite self-explanatory. As you probably noted in the last example, +; multiple filters may be defined in a chain, separated by semicolon. +; +; Each filter chains can contain multiple filter definitions. They are +; processed in order of appearance. If any of the filters decides the +; message should be dropped, the filter chain processing is immediately +; interrupted and message is not passed to syslog. +; +; If filter requires an argument, they may be passed to them by +; specifying a colon after filter name, followed by an argument. +; Argument is passed to the filter as-is. If passing of multiple +; arguments to filter is required, they are passed as single string +; and must be parsed/tokenized by the filter itself (see "only_uid" +; filter for example). +; +; Filter chain specification may not contain any spaces. +; (Acutally spaces are allowed in arguments, but not in filter names +; and between semicolons and filter names, nor between filter names +; and following colons.) +; +; Default value: +; "" (empty string) +; +filter_chain = "" diff --git a/src/Makefile.am b/src/Makefile.am index 8f52dc05..22c4ab38 100644 --- a/src/Makefile.am +++ b/src/Makefile.am @@ -16,6 +16,8 @@ noinst_LTLIBRARIES = \ libsnoopy_no_execve.la libsnoopy_no_execve_la_SOURCES = \ + configuration.c \ + configuration.h \ filterregistry.c \ filterregistry.h \ inputdatastorage.c \ diff --git a/src/configuration.c b/src/configuration.c new file mode 100644 index 00000000..b6aa29af --- /dev/null +++ b/src/configuration.c @@ -0,0 +1,193 @@ +/* + * SNOOPY LOGGER + * + * File: configuration.c + * + * Copyright (c) 2014 bostjan@a2o.si + * + * 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 2, 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 General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + */ + + + +/* + * Include all required C resources + */ +#include +#include +#include +#include +#include +#include + + + +/* + * Include all snoopy-related resources + */ +#include "snoopy.h" +#include "configuration.h" + + + +/* + * Include iniparser-related resources + */ +#include "lib/iniparser/src/iniparser.h" + + + +/* + * Storage of snoopy configuration + */ +struct snoopy_configuration_type snoopy_configuration; + + + +/* + * snoopy_configuration_ctor + * + * Description: + * Populates snoopy_configuration config variable storage with + * correct values, either from configuration file (if enabled) + * or from ./configure arguments, or defaults are used as last + * case scenario. + * + * Params: + * (none) + * + * Return: + * void + */ +void snoopy_configuration_ctor () +{ + /* Initialize default variables first */ + snoopy_configuration_load_defaults(); + + /* Parse INI file if enabled */ +#ifdef SNOOPY_CONFIG_FILE + snoopy_configuration_load_file(SNOOPY_CONFIG_FILE); +#endif +} + + + +/* + * snoopy_configuration_dtor + * + * Description: + * Frees all configuration-related malloced resources. + * + * Params: + * (none) + * + * Return: + * void + */ +void snoopy_configuration_dtor () +{ + if (SNOOPY_TRUE == snoopy_configuration.message_format_malloced) { + free(snoopy_configuration.message_format); + } + if (SNOOPY_TRUE == snoopy_configuration.filter_chain_malloced) { + free(snoopy_configuration.filter_chain); + } +} + + + +/* + * snoopy_configuration_load_defaults + * + * Description: + * Initializes configuration struct with default configuration, + * either hard-coded or ./configured. + * + * Params: + * (none) + * + * Return: + * void + */ +void snoopy_configuration_load_defaults () +{ + snoopy_configuration.config_file_enabled = SNOOPY_FALSE; + snoopy_configuration.config_file_path = ""; + snoopy_configuration.config_file_parsed = SNOOPY_FALSE; + + snoopy_configuration.message_format = SNOOPY_LOG_MESSAGE_FORMAT; + snoopy_configuration.message_format_malloced = SNOOPY_FALSE; + +#ifdef SNOOPY_FILTER_ENABLED + snoopy_configuration.filter_enabled = SNOOPY_TRUE; +#else + snoopy_configuration.filter_enabled = SNOOPY_FALSE; +#endif + snoopy_configuration.filter_chain = SNOOPY_FILTER_CHAIN; + snoopy_configuration.filter_chain_malloced = SNOOPY_FALSE; +} + + + +/* + * snoopy_configuration_load_file + * + * Description: + * Parses INI configuration file and overrides snoopy + * configuration with changed values. + * + * Params: + * file Path log INI configuration file + * + * Return: + * int 0 on success, -1 on error openinf file, other int for other errors + */ +int snoopy_configuration_load_file ( + char *iniFilePath +) { + dictionary *ini ; + char *confValString; // Temporary query result space + + /* Tell snoopy we are using configuration file */ + snoopy_configuration.config_file_enabled = SNOOPY_TRUE; + snoopy_configuration.config_file_path = iniFilePath; + + /* Parse the INI configuration file first */ + ini = iniparser_load(iniFilePath); + if (NULL == ini) { + // TODO snoopy error handling + return -1; + } + + + /* Pick out snoopy configuration variables */ + confValString = iniparser_getstring(ini, "snoopy:message_format", NULL); + if (NULL != confValString) { + snoopy_configuration.message_format = strdup(confValString); + snoopy_configuration.message_format_malloced = SNOOPY_TRUE; + } + + confValString = iniparser_getstring(ini, "snoopy:filter_chain", NULL); + if (NULL != confValString) { + snoopy_configuration.filter_chain = strdup(confValString); + snoopy_configuration.filter_chain_malloced = SNOOPY_TRUE; + } + + + /* Housekeeping */ + snoopy_configuration.config_file_parsed = SNOOPY_TRUE; // We have sucessfully parsed configuration file + iniparser_freedict(ini); + return 0; +} diff --git a/src/configuration.h b/src/configuration.h new file mode 100644 index 00000000..300af5c6 --- /dev/null +++ b/src/configuration.h @@ -0,0 +1,50 @@ +/* + * SNOOPY LOGGER + * + * File: configuration.h + * + * Copyright (c) 2014 bostjan@a2o.si + * + * 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 2, 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 General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + */ + + + +/* + * Two arrays holding data about filter functions + */ +struct snoopy_configuration_type { + int config_file_enabled; + char *config_file_path; + int config_file_parsed; + + char *message_format; + int message_format_malloced; + + int filter_enabled; + char *filter_chain; + int filter_chain_malloced; +}; +extern struct snoopy_configuration_type snoopy_configuration; + + + +/* + * Functions to manage and utilise configuration + */ +void snoopy_configuration_ctor(); +void snoopy_configuration_dtor(); +void snoopy_configuration_load_defaults(); +int snoopy_configuration_load_file(char *iniFilePath); diff --git a/src/log.c b/src/log.c index 9d60559e..bea6aa0a 100644 --- a/src/log.c +++ b/src/log.c @@ -19,8 +19,12 @@ * along with this program; if not, write to the Free Software Foundation, * Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ -#include "snoopy.h" -#include "config.h" + + + +/* + * Include all required C resources + */ #include #include #include @@ -31,9 +35,11 @@ /* - * Include all input-related resources + * Include all snoopy-related resources */ +#include "snoopy.h" #include "log.h" +#include "configuration.h" #include "inputdatastorage.h" #include "inputregistry.h" #include "filterregistry.h" @@ -429,19 +435,22 @@ void snoopy_log_syscall ( logMessage = malloc(SNOOPY_LOG_MESSAGE_MAX_SIZE); logMessage[0] = '\0'; + /* Initialize snoopy configuration */ + snoopy_configuration_ctor(); + /* Generate log message in specified format */ - snoopy_log_message_generate(logMessage, SNOOPY_LOG_MESSAGE_FORMAT); + snoopy_log_message_generate(logMessage, snoopy_configuration.message_format); -#if defined(SNOOPY_FILTER_ENABLED) /* Should message be passed to syslog or not? */ - if (SNOOPY_FILTER_PASS == snoopy_log_filter_check_chain(logMessage, SNOOPY_FILTER_CHAIN)) { -#endif - /* Send it to syslog */ - snoopy_log_send_to_syslog(logMessage); -#if defined(SNOOPY_FILTER_ENABLED) + if (SNOOPY_TRUE == snoopy_configuration.filter_enabled) { + if (SNOOPY_FILTER_PASS == snoopy_log_filter_check_chain(logMessage, snoopy_configuration.filter_chain)) { + snoopy_log_send_to_syslog(logMessage); + } + } else { + snoopy_log_send_to_syslog(logMessage); } -#endif /* Housekeeping */ + snoopy_configuration_dtor(); free(logMessage); } diff --git a/src/snoopy.h b/src/snoopy.h index 4b09c0db..58396938 100644 --- a/src/snoopy.h +++ b/src/snoopy.h @@ -22,7 +22,7 @@ /** - * Include configured constants + * Include ./configured constants */ #include "config.h" @@ -181,3 +181,14 @@ +/** + * SNOOPY_CONFIG_FILE + * + * Path to INI configuration file, if enabled + * + * If configuration file is enabled, this constant is defined + * and holds absolute path to it + */ +#ifdef SNOOPY_CONF_CONFIG_FILE +#define SNOOPY_CONFIG_FILE SNOOPY_CONF_CONFIG_FILE +#endif