# HG changeset patch # User Sebastien Decugis # Date 1362229384 -3600 # Node ID e5a09fab5ef3f3a45db27aba4f02421f0a158456 # Parent 877592751fee97896606a70ce5277bda84eee86f Add support for 'include' in freeDiameter configuration file diff -r 877592751fee -r e5a09fab5ef3 doc/freediameter.conf.sample --- a/doc/freediameter.conf.sample Thu Feb 14 17:52:57 2013 +0100 +++ b/doc/freediameter.conf.sample Sat Mar 02 14:03:04 2013 +0100 @@ -2,6 +2,10 @@ # Only the "TLS_Cred" directive is really mandatory in this file. +# It is possible to use "include" keyword to import additional files +# e.g.: include "/etc/freeDiameter.d/*.conf" + + ############################################################## ## Peer identity and realm diff -r 877592751fee -r e5a09fab5ef3 libfdcore/fdd.l --- a/libfdcore/fdd.l Thu Feb 14 17:52:57 2013 +0100 +++ b/libfdcore/fdd.l Sat Mar 02 14:03:04 2013 +0100 @@ -66,17 +66,39 @@ /* %option noinput ? */ #define YY_NO_INPUT + +/* Additional for files inclusion */ +#include +#include + +#define MAX_NESTED_CONF_FILES 5 + +struct nested_conffiles_t { + YY_BUFFER_STATE parent_level_state; + glob_t filelist; + int current_file; +} nested_conffiles[MAX_NESTED_CONF_FILES]; + +int current_nested_level = 0; + +int globerrfct(const char *epath, int eerrno) +{ + TRACE_DEBUG_ERROR("Failed to scan %s: %s\n", epath, strerror(eerrno)); + return 1; +} + %} %option bison-bridge bison-locations %option noyywrap %option nounput +%x in_include + /* Quoted string. Multilines do not match. */ qstring \"[^\"\n]*\" %% - <*>\n { /* Update the line count */ yylloc->first_line++; @@ -87,6 +109,114 @@ <*>([[:space:]]{-}[\n])+ ; /* Eat all spaces, not new lines */ <*>#.*$ ; /* Eat all comments */ + +include BEGIN(in_include); + /* Following an "include" keyword */ +{ +{qstring} { /* Name of the file to include. This is directly sent to glob. */ + int globerror=0; + char * buf = strdup(yytext+1); + if (buf[yyleng-2] != '"') + { + TRACE_DEBUG_ERROR("Unterminated string: %s\n", yytext); + return LEX_ERROR; + } + buf[yyleng-2] = '\0'; + + if (current_nested_level >= MAX_NESTED_CONF_FILES) + { + TRACE_DEBUG_ERROR("Too many recursion levels in configuration files includes\n"); + return LEX_ERROR; + } + + /* glob the include */ + globerror = glob(buf, GLOB_ERR, globerrfct, &nested_conffiles[current_nested_level].filelist); + + if (globerror == GLOB_NOSPACE) + { + TRACE_DEBUG_ERROR("Not enough memory to parse include directive.\n"); + return LEX_ERROR; + } + if (globerror == GLOB_ABORTED) + { + TRACE_DEBUG_ERROR("An error was encountered in include directive.\n"); + return LEX_ERROR; + } + if (globerror == GLOB_NOMATCH) + { + globfree(&nested_conffiles[current_nested_level].filelist); + goto nomatch; + } + if (globerror) + { + TRACE_DEBUG_ERROR("Unexpected error in glob (%d).\n", globerror); + return LEX_ERROR; + } + + /* We have a list of files to include. */ + + /* save the current buffer for returning when this include has been parsed */ + nested_conffiles[current_nested_level].parent_level_state = YY_CURRENT_BUFFER; + + /* Start with the first match */ + nested_conffiles[current_nested_level].current_file = 0; + + yyin = fopen( nested_conffiles[current_nested_level].filelist.gl_pathv[0], "r" ); + + if ( ! yyin ) + { + TRACE_DEBUG_ERROR("Error in %s: %s", nested_conffiles[current_nested_level].filelist.gl_pathv[0], strerror(errno)); + return LEX_ERROR; + } + + yy_switch_to_buffer(yy_create_buffer( yyin, YY_BUF_SIZE )); + + /* In case of recursive includes */ + current_nested_level++; + +nomatch: + BEGIN(INITIAL); + } +} + +<> { + if (current_nested_level == 0) + { + /* We are at the end of parsing */ + yyterminate(); + } + + /* Otherwise we are doing an include statement */ + --current_nested_level; + yy_delete_buffer(YY_CURRENT_BUFFER); + + /* Go to next file, if any */ + nested_conffiles[current_nested_level].current_file++; + if ( nested_conffiles[current_nested_level].filelist.gl_pathv[nested_conffiles[current_nested_level].current_file] == NULL ) + { + /* We have finished with this list of includes */ + globfree(&nested_conffiles[current_nested_level].filelist); + yy_switch_to_buffer(nested_conffiles[current_nested_level].parent_level_state); + } + else + { + /* Proceed to next included file */ + yyin = fopen( nested_conffiles[current_nested_level].filelist.gl_pathv[nested_conffiles[current_nested_level].current_file], "r" ); + + if ( ! yyin ) + { + TRACE_DEBUG_ERROR("Error in %s: %s", nested_conffiles[current_nested_level].filelist.gl_pathv[nested_conffiles[current_nested_level].current_file], strerror(errno)); + return LEX_ERROR; + } + + yy_switch_to_buffer(yy_create_buffer( yyin, YY_BUF_SIZE )); + + /* In case of recursive includes */ + current_nested_level++; + } + +} + {qstring} { /* First copy the string without the quotes for use in the yacc parser */ CHECK_MALLOC_DO( yylval->string = strdup(yytext+1), /* This allocates one useless tail char but... it's easier :D */