| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Asterisk -- A telephony toolkit for Linux. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Configuration File Parser | 
					
						
							|  |  |  |  *  | 
					
						
							| 
									
										
										
										
											1999-12-11 20:09:45 +00:00
										 |  |  |  * Copyright (C) 1999, Mark Spencer | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Mark Spencer <markster@linux-support.net> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software, distributed under the terms of | 
					
						
							|  |  |  |  * the GNU General Public License | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <unistd.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <errno.h>
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | #include <time.h>
 | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | #include <asterisk/config.h>
 | 
					
						
							|  |  |  | #include <asterisk/options.h>
 | 
					
						
							|  |  |  | #include <asterisk/logger.h>
 | 
					
						
							|  |  |  | #include "asterisk.h"
 | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | #include "astconf.h"
 | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | #define MAX_INCLUDE_LEVEL 10
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | struct ast_category { | 
					
						
							|  |  |  | 	char name[80]; | 
					
						
							|  |  |  | 	struct ast_variable *root; | 
					
						
							|  |  |  | 	struct ast_category *next; | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 	struct ast_comment *precomments; | 
					
						
							|  |  |  | 	struct ast_comment *sameline; | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif	
 | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ast_config { | 
					
						
							|  |  |  | 	/* Maybe this structure isn't necessary but we'll keep it
 | 
					
						
							|  |  |  | 	   for now */ | 
					
						
							|  |  |  | 	struct ast_category *root; | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 	struct ast_category *prev; | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 	struct ast_comment *trailingcomments; | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif	
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | struct ast_comment_struct | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_comment *root; | 
					
						
							|  |  |  | 	struct ast_comment *prev; | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static char *strip(char *buf) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *start; | 
					
						
							|  |  |  | 	/* Strip off trailing whitespace, returns, etc */ | 
					
						
							|  |  |  | 	while(strlen(buf) && (buf[strlen(buf)-1]<33)) | 
					
						
							|  |  |  | 		buf[strlen(buf)-1] = '\0'; | 
					
						
							|  |  |  | 	start = buf; | 
					
						
							|  |  |  | 	/* Strip off leading whitespace, returns, etc */ | 
					
						
							|  |  |  | 	while(*start && (*start < 33)) | 
					
						
							|  |  |  | 		*start++ = '\0'; | 
					
						
							|  |  |  | 	return start; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | static void free_comments(struct ast_comment *com) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_comment *l; | 
					
						
							|  |  |  | 	while (com) { | 
					
						
							|  |  |  | 		l = com; | 
					
						
							|  |  |  | 		com = com->next; | 
					
						
							|  |  |  | 		free(l); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | void ast_destroy(struct ast_config *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_category *cat, *catn; | 
					
						
							|  |  |  | 	struct ast_variable *v, *vn; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-03-06 17:32:03 +00:00
										 |  |  | 	if (!ast) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | 	cat = ast->root; | 
					
						
							|  |  |  | 	while(cat) { | 
					
						
							|  |  |  | 		v = cat->root; | 
					
						
							|  |  |  | 		while(v) { | 
					
						
							|  |  |  | 			vn = v; | 
					
						
							|  |  |  | 			free(v->name); | 
					
						
							|  |  |  | 			free(v->value); | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 			free_comments(v->precomments); | 
					
						
							|  |  |  | 			free_comments(v->sameline); | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif			
 | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | 			v = v->next; | 
					
						
							|  |  |  | 			free(vn); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		catn = cat; | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 		free_comments(cat->precomments); | 
					
						
							|  |  |  | 		free_comments(cat->sameline); | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif		
 | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | 		cat = cat->next; | 
					
						
							|  |  |  | 		free(catn); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 	free_comments(ast->trailingcomments); | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif	
 | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | 	free(ast); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_true(char *s) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2000-01-06 10:19:32 +00:00
										 |  |  | 	if (!s) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | 	/* Determine if this is a true value */ | 
					
						
							|  |  |  | 	if (!strcasecmp(s, "yes") || | 
					
						
							|  |  |  | 	    !strcasecmp(s, "true") || | 
					
						
							|  |  |  | 		!strcasecmp(s, "y") || | 
					
						
							|  |  |  | 		!strcasecmp(s, "t") || | 
					
						
							|  |  |  | 		!strcasecmp(s, "1")) | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ast_variable *ast_variable_browse(struct ast_config *config, char *category) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_category *cat; | 
					
						
							|  |  |  | 	cat = config->root; | 
					
						
							| 
									
										
										
										
											2001-10-09 14:54:42 +00:00
										 |  |  | 	while(cat) { | 
					
						
							|  |  |  | 		if (cat->name == category) | 
					
						
							|  |  |  | 			return cat->root; | 
					
						
							|  |  |  | 		cat = cat->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	cat = config->root; | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | 	while(cat) { | 
					
						
							|  |  |  | 		if (!strcasecmp(cat->name, category)) | 
					
						
							|  |  |  | 			return cat->root; | 
					
						
							|  |  |  | 		cat = cat->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | char *ast_variable_retrieve(struct ast_config *config, char *category, char *value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_variable *v; | 
					
						
							| 
									
										
										
										
											1999-12-11 20:09:45 +00:00
										 |  |  | 	if (category) { | 
					
						
							| 
									
										
										
										
											2001-10-09 14:54:42 +00:00
										 |  |  | 		v = ast_variable_browse(config, category); | 
					
						
							|  |  |  | 		while (v) { | 
					
						
							|  |  |  | 			if (value == v->name) | 
					
						
							|  |  |  | 				return v->value; | 
					
						
							|  |  |  | 			v=v->next; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											1999-12-11 20:09:45 +00:00
										 |  |  | 		v = ast_variable_browse(config, category); | 
					
						
							|  |  |  | 		while (v) { | 
					
						
							|  |  |  | 			if (!strcasecmp(value, v->name)) | 
					
						
							|  |  |  | 				return v->value; | 
					
						
							|  |  |  | 			v=v->next; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		struct ast_category *cat; | 
					
						
							|  |  |  | 		cat = config->root; | 
					
						
							|  |  |  | 		while(cat) { | 
					
						
							|  |  |  | 			v = cat->root; | 
					
						
							|  |  |  | 			while (v) { | 
					
						
							|  |  |  | 				if (!strcasecmp(value, v->name)) | 
					
						
							|  |  |  | 					return v->value; | 
					
						
							|  |  |  | 				v=v->next; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			cat = cat->next; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | int ast_variable_delete(struct ast_config *cfg, char *category, char *variable, char *value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_variable *v, *pv, *bv, *bpv; | 
					
						
							|  |  |  | 	struct ast_category *cat; | 
					
						
							|  |  |  | 	cat = cfg->root; | 
					
						
							|  |  |  | 	while(cat) { | 
					
						
							|  |  |  | 		if (cat->name == category) { | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		cat = cat->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (!cat) { | 
					
						
							|  |  |  | 		cat = cfg->root; | 
					
						
							|  |  |  | 		while(cat) { | 
					
						
							|  |  |  | 			if (!strcasecmp(cat->name, category)) { | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			cat = cat->next; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (!cat) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	v = cat->root; | 
					
						
							|  |  |  | 	pv = NULL; | 
					
						
							|  |  |  | 	while (v) { | 
					
						
							|  |  |  | 		if ((variable == v->name) && (!value || !strcmp(v->value, value))) | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		pv = v; | 
					
						
							|  |  |  | 		v=v->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (!v) { | 
					
						
							|  |  |  | 		/* Get the last one that looks like it */ | 
					
						
							|  |  |  | 		bv = NULL; | 
					
						
							|  |  |  | 		bpv = NULL; | 
					
						
							|  |  |  | 		v = cat->root; | 
					
						
							|  |  |  | 		pv = NULL; | 
					
						
							|  |  |  | 		while (v) { | 
					
						
							|  |  |  | 			if (!strcasecmp(variable, v->name) && (!value || !strcmp(v->value, value))) { | 
					
						
							|  |  |  | 				bv = v; | 
					
						
							|  |  |  | 				bpv = pv; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			pv = v; | 
					
						
							|  |  |  | 			v=v->next; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		v = bv; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (v) { | 
					
						
							|  |  |  | 		/* Unlink from original position */ | 
					
						
							|  |  |  | 		if (pv)  | 
					
						
							|  |  |  | 			pv->next = v->next; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			cat->root = v->next; | 
					
						
							|  |  |  | 		v->next = NULL; | 
					
						
							|  |  |  | 		free(v->name); | 
					
						
							|  |  |  | 		if (v->value) | 
					
						
							|  |  |  | 			free(v->value); | 
					
						
							|  |  |  | 		free_comments(v->sameline); | 
					
						
							|  |  |  | 		free_comments(v->precomments); | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_category_delete(struct ast_config *cfg, char *category) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_variable *v, *pv; | 
					
						
							|  |  |  | 	struct ast_category *cat, *cprev; | 
					
						
							|  |  |  | 	cat = cfg->root; | 
					
						
							|  |  |  | 	cprev = NULL; | 
					
						
							|  |  |  | 	while(cat) { | 
					
						
							|  |  |  | 		if (cat->name == category) { | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		cprev = cat; | 
					
						
							|  |  |  | 		cat = cat->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (!cat) { | 
					
						
							|  |  |  | 		cat = cfg->root; | 
					
						
							|  |  |  | 		cprev = NULL; | 
					
						
							|  |  |  | 		while(cat) { | 
					
						
							|  |  |  | 			if (!strcasecmp(cat->name, category)) { | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			cprev = cat; | 
					
						
							|  |  |  | 			cat = cat->next; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (!cat) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	/* Unlink it */ | 
					
						
							|  |  |  | 	if (cprev) | 
					
						
							|  |  |  | 		cprev->next = cat->next; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		cfg->root = cat->next; | 
					
						
							|  |  |  | 	v = cat->root; | 
					
						
							|  |  |  | 	while (v) { | 
					
						
							|  |  |  | 		pv = v; | 
					
						
							|  |  |  | 		v=v->next; | 
					
						
							|  |  |  | 		if (pv->value) | 
					
						
							|  |  |  | 			free(pv->value); | 
					
						
							|  |  |  | 		if (pv->name) | 
					
						
							|  |  |  | 			free(pv->name); | 
					
						
							|  |  |  | 		free_comments(pv->sameline); | 
					
						
							|  |  |  | 		free_comments(pv->precomments); | 
					
						
							|  |  |  | 		free(pv); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	free_comments(cat->sameline); | 
					
						
							|  |  |  | 	free_comments(cat->precomments); | 
					
						
							|  |  |  | 	free(cat); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ast_variable *ast_variable_append_modify(struct ast_config *config, char *category, char *variable, char *value, int newcat, int newvar, int move) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2003-04-25 20:33:27 +00:00
										 |  |  | 	struct ast_variable *v, *pv=NULL, *bv, *bpv; | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 	struct ast_category *cat, *pcat; | 
					
						
							|  |  |  | 	cat = config->root; | 
					
						
							|  |  |  | 	if (!newcat) { | 
					
						
							|  |  |  | 		while(cat) { | 
					
						
							|  |  |  | 			if (cat->name == category) { | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			cat = cat->next; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (!cat) { | 
					
						
							|  |  |  | 			cat = config->root; | 
					
						
							|  |  |  | 			while(cat) { | 
					
						
							|  |  |  | 				if (!strcasecmp(cat->name, category)) { | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				cat = cat->next; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (!cat) { | 
					
						
							|  |  |  | 		cat = malloc(sizeof(struct ast_category)); | 
					
						
							|  |  |  | 		if (!cat) | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 		memset(cat, 0, sizeof(struct ast_category)); | 
					
						
							|  |  |  | 		strncpy(cat->name, category, sizeof(cat->name)); | 
					
						
							|  |  |  | 		if (config->root) { | 
					
						
							|  |  |  | 			/* Put us at the end */ | 
					
						
							|  |  |  | 			pcat = config->root; | 
					
						
							|  |  |  | 			while(pcat->next) | 
					
						
							|  |  |  | 				pcat = pcat->next; | 
					
						
							|  |  |  | 			pcat->next = cat; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			/* We're the first one */ | 
					
						
							|  |  |  | 			config->root = cat; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (!newvar) { | 
					
						
							|  |  |  | 		v = cat->root; | 
					
						
							|  |  |  | 		pv = NULL; | 
					
						
							|  |  |  | 		while (v) { | 
					
						
							|  |  |  | 			if (variable == v->name) | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			pv = v; | 
					
						
							|  |  |  | 			v=v->next; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (!v) { | 
					
						
							|  |  |  | 			/* Get the last one that looks like it */ | 
					
						
							|  |  |  | 			bv = NULL; | 
					
						
							|  |  |  | 			bpv = NULL; | 
					
						
							|  |  |  | 			v = cat->root; | 
					
						
							|  |  |  | 			pv = NULL; | 
					
						
							|  |  |  | 			while (v) { | 
					
						
							|  |  |  | 				if (!strcasecmp(variable, v->name)) { | 
					
						
							|  |  |  | 					bv = v; | 
					
						
							|  |  |  | 					bpv = pv; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				pv = v; | 
					
						
							|  |  |  | 				v=v->next; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			v = bv; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else v = NULL; | 
					
						
							|  |  |  | 	if (v && move) { | 
					
						
							|  |  |  | 		/* Unlink from original position */ | 
					
						
							|  |  |  | 		if (pv)  | 
					
						
							|  |  |  | 			pv->next = v->next; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			cat->root = v->next; | 
					
						
							|  |  |  | 		v->next = NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (!v) { | 
					
						
							|  |  |  | 		v = malloc(sizeof(struct ast_variable)); | 
					
						
							|  |  |  | 		if (!v) | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 		memset(v, 0, sizeof(struct ast_variable)); | 
					
						
							|  |  |  | 		v->name = strdup(variable); | 
					
						
							|  |  |  | 		move = 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (v->value) | 
					
						
							|  |  |  | 		free(v->value); | 
					
						
							|  |  |  | 	if (value) | 
					
						
							|  |  |  | 		v->value = strdup(value); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		v->value = strdup(""); | 
					
						
							|  |  |  | 	if (move) { | 
					
						
							|  |  |  | 		if (cat->root) { | 
					
						
							|  |  |  | 			pv = cat->root; | 
					
						
							|  |  |  | 			while (pv->next)  | 
					
						
							|  |  |  | 				pv = pv->next; | 
					
						
							|  |  |  | 			pv->next = v; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			cat->root = v; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return v; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif		
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-23 16:50:12 +00:00
										 |  |  | int ast_category_exist(struct ast_config *config, char *category_name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_category *category = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	category = config->root; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while(category) { | 
					
						
							|  |  |  | 		if (!strcasecmp(category->name,category_name))  | 
					
						
							|  |  |  | 			return 1; | 
					
						
							|  |  |  | 		category = category->next; | 
					
						
							|  |  |  | 	}  | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | static struct ast_comment *build_comment(char *cmt) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_comment *c; | 
					
						
							| 
									
										
										
										
											2003-04-25 20:33:27 +00:00
										 |  |  | 	int len = strlen(cmt) + 1; | 
					
						
							|  |  |  | 	c = malloc(sizeof(struct ast_comment) + len); | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 	if (c) { | 
					
						
							| 
									
										
										
										
											2003-04-25 20:33:27 +00:00
										 |  |  | 		/* Memset the header */ | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 		memset(c, 0, sizeof(struct ast_comment)); | 
					
						
							| 
									
										
										
										
											2003-04-25 20:33:27 +00:00
										 |  |  | 		/* Copy the rest */ | 
					
						
							|  |  |  | 		strcpy(c->cmt, cmt); | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return c; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct ast_config *__ast_load(char *configfile, struct ast_config *tmp, struct ast_category **_tmpc, struct ast_variable **_last, int includelevel | 
					
						
							|  |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							|  |  |  | , struct ast_comment_struct *acs | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | ); | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | static int cfg_process(struct ast_config *tmp, struct ast_category **_tmpc, struct ast_variable **_last, char *buf, int lineno, char *configfile, int includelevel  | 
					
						
							|  |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							|  |  |  | ,struct ast_comment_struct *acs | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | ) | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	char *c; | 
					
						
							|  |  |  | 	char *cur; | 
					
						
							|  |  |  | 	struct ast_variable *v; | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 	struct ast_comment *com = NULL; | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif	
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 	int object; | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | 	/* Strip off lines using ; as comment */ | 
					
						
							|  |  |  | 	c = strchr(buf, ';'); | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 	if (c) { | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | 		*c = '\0'; | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 		c++; | 
					
						
							|  |  |  | 		if (*c != '!') | 
					
						
							|  |  |  | 			com = build_comment(c); | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif			
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | 	cur = strip(buf); | 
					
						
							|  |  |  | 	if (strlen(cur)) { | 
					
						
							|  |  |  | 		/* Actually parse the entry */ | 
					
						
							|  |  |  | 		if (cur[0] == '[') { | 
					
						
							|  |  |  | 			/* A category header */ | 
					
						
							|  |  |  | 			c = strchr(cur, ']'); | 
					
						
							|  |  |  | 			if (c) { | 
					
						
							|  |  |  | 				*c = 0; | 
					
						
							|  |  |  | 				*_tmpc = malloc(sizeof(struct ast_category)); | 
					
						
							|  |  |  | 				if (!*_tmpc) { | 
					
						
							|  |  |  | 					ast_destroy(tmp); | 
					
						
							|  |  |  | 					ast_log(LOG_WARNING, | 
					
						
							|  |  |  | 						"Out of memory, line %d\n", lineno); | 
					
						
							|  |  |  | 					return -1; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2002-04-22 15:14:18 +00:00
										 |  |  | 				memset(*_tmpc, 0, sizeof(struct ast_category)); | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | 				strncpy((*_tmpc)->name, cur+1, sizeof((*_tmpc)->name) - 1); | 
					
						
							|  |  |  | 				(*_tmpc)->root =  NULL; | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 				(*_tmpc)->precomments = acs->root; | 
					
						
							|  |  |  | 				(*_tmpc)->sameline = com; | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif				
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 				if (!tmp->prev) | 
					
						
							|  |  |  | 					tmp->root = *_tmpc; | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					tmp->prev->next = *_tmpc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				tmp->prev = *_tmpc; | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 				acs->root = NULL; | 
					
						
							|  |  |  | 				acs->prev = NULL; | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif				
 | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | 				*_last =  NULL; | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				ast_log(LOG_WARNING,  | 
					
						
							|  |  |  | 					"parse error: no closing ']', line %d of %s\n", lineno, configfile); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} else if (cur[0] == '#') { | 
					
						
							|  |  |  | 			/* A directive */ | 
					
						
							|  |  |  | 			cur++; | 
					
						
							|  |  |  | 			c = cur; | 
					
						
							|  |  |  | 			while(*c && (*c > 32)) c++; | 
					
						
							|  |  |  | 			if (*c) { | 
					
						
							|  |  |  | 				*c = '\0'; | 
					
						
							|  |  |  | 				c++; | 
					
						
							|  |  |  | 				/* Find real argument */ | 
					
						
							|  |  |  | 				while(*c  && (*c < 33)) c++; | 
					
						
							|  |  |  | 				if (!*c) | 
					
						
							|  |  |  | 					c = NULL; | 
					
						
							|  |  |  | 			} else  | 
					
						
							|  |  |  | 				c = NULL; | 
					
						
							|  |  |  | 			if (!strcasecmp(cur, "include")) { | 
					
						
							|  |  |  | 				/* A #include */ | 
					
						
							|  |  |  | 				if (c) { | 
					
						
							|  |  |  | 					while((*c == '<') || (*c == '>') || (*c == '\"')) c++; | 
					
						
							|  |  |  | 					/* Get rid of leading mess */ | 
					
						
							|  |  |  | 					cur = c; | 
					
						
							|  |  |  | 					while(strlen(cur)) { | 
					
						
							|  |  |  | 						c = cur + strlen(cur) - 1; | 
					
						
							|  |  |  | 						if ((*c == '>') || (*c == '<') || (*c == '\"')) | 
					
						
							|  |  |  | 							*c = '\0'; | 
					
						
							|  |  |  | 						else | 
					
						
							|  |  |  | 							break; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					if (includelevel < MAX_INCLUDE_LEVEL) { | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | 						__ast_load(cur, tmp, _tmpc, _last, includelevel + 1 | 
					
						
							|  |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							|  |  |  | 						,acs | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 						); | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | 					} else  | 
					
						
							|  |  |  | 						ast_log(LOG_WARNING, "Maximum Include level (%d) exceeded\n", includelevel); | 
					
						
							|  |  |  | 				} else | 
					
						
							|  |  |  | 					ast_log(LOG_WARNING, "Directive '#include' needs an argument (filename) at line %d of %s\n", lineno, configfile); | 
					
						
							|  |  |  | 				/* Strip off leading and trailing "'s and <>'s */ | 
					
						
							|  |  |  | 			} else  | 
					
						
							|  |  |  | 				ast_log(LOG_WARNING, "Unknown directive '%s' at line %d of %s\n", cur, lineno, configfile); | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			/* Just a line (variable = value) */ | 
					
						
							|  |  |  | 			if (!*_tmpc) { | 
					
						
							|  |  |  | 				ast_log(LOG_WARNING, | 
					
						
							|  |  |  | 					"parse error: No category context for line %d of %s\n", lineno, configfile); | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 				ast_destroy(tmp); | 
					
						
							|  |  |  | 				return -1; | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			c = strchr(cur, '='); | 
					
						
							|  |  |  | 			if (c) { | 
					
						
							|  |  |  | 				*c = 0; | 
					
						
							|  |  |  | 				c++; | 
					
						
							|  |  |  | 				/* Ignore > in => */ | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 				if (*c== '>') { | 
					
						
							|  |  |  | 					object = 1; | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | 					c++; | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 				} else | 
					
						
							|  |  |  | 					object = 0; | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | 				v = malloc(sizeof(struct ast_variable)); | 
					
						
							|  |  |  | 				if (v) { | 
					
						
							| 
									
										
										
										
											2002-04-22 15:14:18 +00:00
										 |  |  | 					memset(v, 0, sizeof(struct ast_variable)); | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | 					v->next = NULL; | 
					
						
							|  |  |  | 					v->name = strdup(strip(cur)); | 
					
						
							|  |  |  | 					v->value = strdup(strip(c)); | 
					
						
							|  |  |  | 					v->lineno = lineno; | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 					v->object = object; | 
					
						
							|  |  |  | 					/* Put and reset comments */ | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 					v->precomments = acs->root; | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | 					v->sameline = com; | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 					acs->prev = NULL; | 
					
						
							|  |  |  | 					acs->root = NULL; | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif					
 | 
					
						
							|  |  |  | 					v->blanklines = 0; | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 					if (*_last) | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | 						(*_last)->next = v; | 
					
						
							|  |  |  | 					else | 
					
						
							|  |  |  | 						(*_tmpc)->root = v; | 
					
						
							|  |  |  | 					*_last = v; | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					ast_destroy(tmp); | 
					
						
							|  |  |  | 					ast_log(LOG_WARNING, "Out of memory, line %d\n", lineno); | 
					
						
							|  |  |  | 					return -1; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				ast_log(LOG_WARNING, "No '=' (equal sign) in line %d of %s\n", lineno, configfile); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 														 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		/* store any comments if there are any */ | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 		if (com) { | 
					
						
							|  |  |  | 			if (acs->prev) | 
					
						
							|  |  |  | 				acs->prev->next = com; | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				acs->root = com; | 
					
						
							|  |  |  | 			acs->prev = com; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 		if (*_last)  | 
					
						
							|  |  |  | 			(*_last)->blanklines++; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | static void dump_comments(FILE *f, struct ast_comment *comment) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	while (comment) { | 
					
						
							| 
									
										
										
										
											2003-04-25 20:33:27 +00:00
										 |  |  | 		fprintf(f, ";%s", comment->cmt); | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 		comment = comment->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | int ast_save(char *configfile, struct ast_config *cfg, char *generator) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	FILE *f; | 
					
						
							|  |  |  | 	char fn[256]; | 
					
						
							|  |  |  | 	char date[256]; | 
					
						
							|  |  |  | 	time_t t; | 
					
						
							|  |  |  | 	struct ast_variable *var; | 
					
						
							|  |  |  | 	struct ast_category *cat; | 
					
						
							|  |  |  | 	int blanklines = 0; | 
					
						
							|  |  |  | 	if (configfile[0] == '/') { | 
					
						
							|  |  |  | 		strncpy(fn, configfile, sizeof(fn)-1); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		snprintf(fn, sizeof(fn), "%s/%s", AST_CONFIG_DIR, configfile); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	time(&t); | 
					
						
							|  |  |  | 	strncpy(date, ctime(&t), sizeof(date)); | 
					
						
							|  |  |  | 	if ((f = fopen(fn, "w"))) { | 
					
						
							|  |  |  | 		if ((option_verbose > 1) && !option_debug) | 
					
						
							|  |  |  | 			ast_verbose(  VERBOSE_PREFIX_2 "Saving '%s': ", fn); | 
					
						
							|  |  |  | 		fprintf(f, ";!\n"); | 
					
						
							|  |  |  | 		fprintf(f, ";! Automatically generated configuration file\n"); | 
					
						
							|  |  |  | 		fprintf(f, ";! Filename: %s (%s)\n", configfile, fn); | 
					
						
							|  |  |  | 		fprintf(f, ";! Generator: %s\n", generator); | 
					
						
							|  |  |  | 		fprintf(f, ";! Creation Date: %s", date); | 
					
						
							|  |  |  | 		fprintf(f, ";!\n"); | 
					
						
							|  |  |  | 		cat = cfg->root; | 
					
						
							|  |  |  | 		while(cat) { | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 			/* Dump any precomments */ | 
					
						
							|  |  |  | 			dump_comments(f, cat->precomments); | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 			/* Dump section with any appropriate comment */ | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 			if (cat->sameline)  | 
					
						
							| 
									
										
										
										
											2003-04-25 20:33:27 +00:00
										 |  |  | 				fprintf(f, "[%s]  ; %s\n", cat->name, cat->sameline->cmt); | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 			else | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 				fprintf(f, "[%s]\n", cat->name); | 
					
						
							|  |  |  | 			var = cat->root; | 
					
						
							|  |  |  | 			while(var) { | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 				dump_comments(f, var->precomments); | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif				
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 				if (var->sameline)  | 
					
						
							| 
									
										
										
										
											2003-04-25 20:33:27 +00:00
										 |  |  | 					fprintf(f, "%s %s %s  ; %s\n", var->name, (var->object ? "=>" : "="), var->value, var->sameline->cmt); | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 				else	 | 
					
						
							|  |  |  | 					fprintf(f, "%s %s %s\n", var->name, (var->object ? "=>" : "="), var->value); | 
					
						
							|  |  |  | 				if (var->blanklines) { | 
					
						
							|  |  |  | 					blanklines = var->blanklines; | 
					
						
							|  |  |  | 					while (blanklines) { | 
					
						
							|  |  |  | 						fprintf(f, "\n"); | 
					
						
							|  |  |  | 						blanklines--; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 					 | 
					
						
							|  |  |  | 				var = var->next; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 			/* Put an empty line */ | 
					
						
							|  |  |  | 			fprintf(f, "\n"); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 			cat = cat->next; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 		dump_comments(f, cfg->trailingcomments); | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif		
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		if (option_debug) | 
					
						
							|  |  |  | 			printf("Unable to open for writing: %s\n", fn); | 
					
						
							|  |  |  | 		else if (option_verbose > 1) | 
					
						
							|  |  |  | 			printf( "Unable to write (%s)", strerror(errno)); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	fclose(f); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | static struct ast_config *__ast_load(char *configfile, struct ast_config *tmp, struct ast_category **_tmpc, struct ast_variable **_last, int includelevel | 
					
						
							|  |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							|  |  |  | , struct ast_comment_struct *acs | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | ) | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	char fn[256]; | 
					
						
							| 
									
										
										
										
											2003-06-04 18:27:36 +00:00
										 |  |  | 	char buf[512]; | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | 	FILE *f; | 
					
						
							|  |  |  | 	int lineno=0; | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 	int master=0; | 
					
						
							| 
									
										
										
										
											2001-04-23 16:50:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | 	if (configfile[0] == '/') { | 
					
						
							| 
									
										
										
										
											2001-11-02 01:22:37 +00:00
										 |  |  | 		strncpy(fn, configfile, sizeof(fn)-1); | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 		snprintf(fn, sizeof(fn), "%s/%s", (char *)ast_config_AST_CONFIG_DIR, configfile); | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if ((option_verbose > 1) && !option_debug) { | 
					
						
							|  |  |  | 		ast_verbose(  VERBOSE_PREFIX_2 "Parsing '%s': ", fn); | 
					
						
							|  |  |  | 		fflush(stdout); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if ((f = fopen(fn, "r"))) { | 
					
						
							|  |  |  | 		if (option_debug) | 
					
						
							|  |  |  | 			ast_log(LOG_DEBUG, "Parsing %s\n", fn); | 
					
						
							|  |  |  | 		else if (option_verbose > 1) | 
					
						
							|  |  |  | 			ast_verbose( "Found\n"); | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | 		if (!tmp) { | 
					
						
							|  |  |  | 			tmp = malloc(sizeof(struct ast_config)); | 
					
						
							| 
									
										
										
										
											2002-04-22 15:14:18 +00:00
										 |  |  | 			if (tmp) | 
					
						
							|  |  |  | 				memset(tmp, 0, sizeof(struct ast_config)); | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			master = 1; | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | 		if (!tmp) { | 
					
						
							|  |  |  | 			ast_log(LOG_WARNING, "Out of memory\n"); | 
					
						
							|  |  |  | 			fclose(f); | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		while(!feof(f)) { | 
					
						
							|  |  |  | 			fgets(buf, sizeof(buf), f); | 
					
						
							|  |  |  | 			lineno++; | 
					
						
							|  |  |  | 			if (!feof(f)) { | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | 				if (cfg_process(tmp, _tmpc, _last, buf, lineno, configfile, includelevel | 
					
						
							|  |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							|  |  |  | 				, acs | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 				)) { | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | 					fclose(f); | 
					
						
							|  |  |  | 					return NULL; | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		fclose(f);		 | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		if (option_debug) | 
					
						
							|  |  |  | 			ast_log(LOG_DEBUG, "No file to parse: %s\n", fn); | 
					
						
							|  |  |  | 		else if (option_verbose > 1) | 
					
						
							| 
									
										
										
										
											2003-07-01 04:08:25 +00:00
										 |  |  | 			ast_verbose( "Not found (%s)\n", strerror(errno)); | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 	if (master) { | 
					
						
							|  |  |  | 		/* Keep trailing comments */ | 
					
						
							|  |  |  | 		tmp->trailingcomments = acs->root; | 
					
						
							|  |  |  | 		acs->root = NULL; | 
					
						
							|  |  |  | 		acs->prev = NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | 	return tmp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | struct ast_config *ast_load(char *configfile) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_category *tmpc=NULL; | 
					
						
							|  |  |  | 	struct ast_variable *last = NULL; | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							| 
									
										
										
										
											2003-03-09 06:00:18 +00:00
										 |  |  | 	struct ast_comment_struct acs = { NULL, NULL }; | 
					
						
							| 
									
										
										
										
											2003-04-28 19:58:43 +00:00
										 |  |  | #endif	
 | 
					
						
							|  |  |  | 	return __ast_load(configfile, NULL, &tmpc, &last, 0  | 
					
						
							|  |  |  | #ifdef PRESERVE_COMMENTS
 | 
					
						
							|  |  |  | 	,&acs | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	); | 
					
						
							| 
									
										
										
										
											2002-03-05 23:57:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | char *ast_category_browse(struct ast_config *config, char *prev) | 
					
						
							|  |  |  | {	 | 
					
						
							|  |  |  | 	struct ast_category *cat; | 
					
						
							|  |  |  | 	if (!prev) { | 
					
						
							|  |  |  | 		if (config->root) | 
					
						
							|  |  |  | 			return config->root->name; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	cat = config->root; | 
					
						
							| 
									
										
										
										
											2001-10-09 14:54:42 +00:00
										 |  |  | 	while(cat) { | 
					
						
							|  |  |  | 		if (cat->name == prev) { | 
					
						
							|  |  |  | 			if (cat->next) | 
					
						
							|  |  |  | 				return cat->next->name; | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				return NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		cat = cat->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	cat = config->root; | 
					
						
							| 
									
										
										
										
											1999-10-24 18:00:20 +00:00
										 |  |  | 	while(cat) { | 
					
						
							|  |  |  | 		if (!strcasecmp(cat->name, prev)) { | 
					
						
							|  |  |  | 			if (cat->next) | 
					
						
							|  |  |  | 				return cat->next->name; | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				return NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		cat = cat->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } |