| 
									
										
										
										
											2005-05-05 05:39:33 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2005-09-14 20:46:50 +00:00
										 |  |  |  * Asterisk -- An open source telephony toolkit. | 
					
						
							| 
									
										
										
										
											2005-05-05 05:39:33 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Copyright (C) 2005, Digium, Inc. | 
					
						
							| 
									
										
										
										
											2005-05-05 12:48:52 +00:00
										 |  |  |  * Portions Copyright (C) 2005, Tilghman Lesher.  All rights reserved. | 
					
						
							|  |  |  |  * Portions Copyright (C) 2005, Anthony Minessale II | 
					
						
							| 
									
										
										
										
											2005-05-05 05:39:33 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-09-14 20:46:50 +00:00
										 |  |  |  * See http://www.asterisk.org for more information about
 | 
					
						
							|  |  |  |  * the Asterisk project. Please do not directly contact | 
					
						
							|  |  |  |  * any of the maintainers of this project for assistance; | 
					
						
							|  |  |  |  * the project provides a web site, mailing lists and IRC | 
					
						
							|  |  |  |  * channels for your use. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-05-05 05:39:33 +00:00
										 |  |  |  * This program is free software, distributed under the terms of | 
					
						
							| 
									
										
										
										
											2005-09-14 20:46:50 +00:00
										 |  |  |  * the GNU General Public License Version 2. See the LICENSE file | 
					
						
							|  |  |  |  * at the top of the source tree. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * String manipulation dialplan functions | 
					
						
							|  |  |  |  *  | 
					
						
							| 
									
										
										
										
											2005-05-05 05:39:33 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <sys/types.h>
 | 
					
						
							|  |  |  | #include <regex.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-06-06 22:12:19 +00:00
										 |  |  | #include "asterisk.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ASTERISK_FILE_VERSION(__FILE__, "$Revision$") */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-05-05 05:39:33 +00:00
										 |  |  | #include "asterisk/channel.h"
 | 
					
						
							|  |  |  | #include "asterisk/pbx.h"
 | 
					
						
							|  |  |  | #include "asterisk/logger.h"
 | 
					
						
							|  |  |  | #include "asterisk/utils.h"
 | 
					
						
							|  |  |  | #include "asterisk/app.h"
 | 
					
						
							| 
									
										
										
										
											2005-05-05 05:50:11 +00:00
										 |  |  | #include "asterisk/localtime.h"
 | 
					
						
							| 
									
										
										
										
											2005-05-05 05:39:33 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static char *function_fieldqty(struct ast_channel *chan, char *cmd, char *data, char *buf, size_t len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *varname, *varval, workspace[256]; | 
					
						
							|  |  |  | 	char *delim = ast_strdupa(data); | 
					
						
							|  |  |  | 	int fieldcount = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (delim) { | 
					
						
							|  |  |  | 		varname = strsep(&delim, "|"); | 
					
						
							|  |  |  | 		pbx_retrieve_variable(chan, varname, &varval, workspace, sizeof(workspace), NULL); | 
					
						
							|  |  |  | 		while (strsep(&varval, delim)) | 
					
						
							|  |  |  | 			fieldcount++; | 
					
						
							|  |  |  | 		snprintf(buf, len, "%d", fieldcount); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "Out of memory\n"); | 
					
						
							|  |  |  | 		strncpy(buf, "1", len); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return buf; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef BUILTIN_FUNC
 | 
					
						
							|  |  |  | static | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | struct ast_custom_function fieldqty_function = { | 
					
						
							|  |  |  | 	.name = "FIELDQTY", | 
					
						
							|  |  |  | 	.synopsis = "Count the fields, with an arbitrary delimiter", | 
					
						
							|  |  |  | 	.syntax = "FIELDQTY(<varname>,<delim>)", | 
					
						
							|  |  |  | 	.read = function_fieldqty, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char *builtin_function_regex(struct ast_channel *chan, char *cmd, char *data, char *buf, size_t len)  | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *ret_true = "1", *ret_false = "0", *ret; | 
					
						
							|  |  |  | 	char *arg, *earg, *tmp, errstr[256] = ""; | 
					
						
							|  |  |  | 	int errcode; | 
					
						
							|  |  |  | 	regex_t regexbuf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ret = ret_false; /* convince me otherwise */ | 
					
						
							|  |  |  | 	tmp = ast_strdupa(data); | 
					
						
							|  |  |  | 	if (tmp) { | 
					
						
							|  |  |  | 		/* Regex in quotes */ | 
					
						
							|  |  |  | 		arg = strchr(tmp, '"'); | 
					
						
							|  |  |  | 		if (arg) { | 
					
						
							|  |  |  | 			arg++; | 
					
						
							|  |  |  | 			earg = strrchr(arg, '"'); | 
					
						
							|  |  |  | 			if (earg) { | 
					
						
							|  |  |  | 				*earg = '\0'; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			arg = tmp; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if ((errcode = regcomp(®exbuf, arg, REG_EXTENDED | REG_NOSUB))) { | 
					
						
							|  |  |  | 			regerror(errcode, ®exbuf, errstr, sizeof(errstr)); | 
					
						
							|  |  |  | 			ast_log(LOG_WARNING, "Malformed input %s(%s): %s\n", cmd, data, errstr); | 
					
						
							|  |  |  | 			ret = NULL; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			ret = regexec(®exbuf, data, 0, NULL, 0) ? ret_false : ret_true; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		regfree(®exbuf); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "Out of memory in %s(%s)\n", cmd, data); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef BUILTIN_FUNC
 | 
					
						
							|  |  |  | static | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | struct ast_custom_function regex_function = { | 
					
						
							|  |  |  | 	.name = "REGEX", | 
					
						
							|  |  |  | 	.synopsis = "Regular Expression: Returns 1 if data matches regular expression.", | 
					
						
							|  |  |  | 	.syntax = "REGEX(\"<regular expression>\" <data>)", | 
					
						
							|  |  |  | 	.read = builtin_function_regex, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char *builtin_function_len(struct ast_channel *chan, char *cmd, char *data, char *buf, size_t len)  | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int length = 0; | 
					
						
							|  |  |  | 	if (data) { | 
					
						
							|  |  |  | 		length = strlen(data); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	snprintf(buf, len, "%d", length); | 
					
						
							|  |  |  | 	return buf; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef BUILTIN_FUNC
 | 
					
						
							|  |  |  | static | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | struct ast_custom_function len_function = { | 
					
						
							|  |  |  | 	.name = "LEN", | 
					
						
							|  |  |  | 	.synopsis = "Returns the length of the argument given", | 
					
						
							|  |  |  | 	.syntax = "LEN(<string>)", | 
					
						
							|  |  |  | 	.read = builtin_function_len, | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2005-05-05 05:50:11 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static char *acf_strftime(struct ast_channel *chan, char *cmd, char *data, char *buf, size_t len)  | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *format, *epoch, *timezone; | 
					
						
							|  |  |  | 	long epochi; | 
					
						
							|  |  |  | 	struct tm time; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (data) { | 
					
						
							|  |  |  | 		format = ast_strdupa(data); | 
					
						
							|  |  |  | 		if (format) { | 
					
						
							|  |  |  | 			epoch = strsep(&format, "|"); | 
					
						
							|  |  |  | 			timezone = strsep(&format, "|"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (epoch && !ast_strlen_zero(epoch) && sscanf(epoch, "%ld", &epochi) == 1) { | 
					
						
							|  |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2005-07-15 23:00:47 +00:00
										 |  |  | 				struct timeval tv = ast_tvnow(); | 
					
						
							|  |  |  | 				epochi = tv.tv_sec; | 
					
						
							| 
									
										
										
										
											2005-05-05 05:50:11 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			ast_localtime(&epochi, &time, timezone); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (!format) { | 
					
						
							|  |  |  | 				format = "%c"; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			buf[0] = '\0'; | 
					
						
							|  |  |  | 			if (! strftime(buf, len, format, &time)) { | 
					
						
							|  |  |  | 				ast_log(LOG_WARNING, "C function strftime() output nothing?!!\n"); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			buf[len - 1] = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			return buf; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			ast_log(LOG_ERROR, "Out of memory\n"); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "Asterisk function STRFTIME() requires an argument.\n"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return ""; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef BUILTIN_FUNC
 | 
					
						
							|  |  |  | static | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | struct ast_custom_function strftime_function = { | 
					
						
							|  |  |  | 	.name = "STRFTIME", | 
					
						
							|  |  |  | 	.synopsis = "Returns the current date/time in a specified format.", | 
					
						
							|  |  |  | 	.syntax = "STRFTIME([<epoch>][,[timezone][,format]])", | 
					
						
							|  |  |  | 	.read = acf_strftime, | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2005-05-15 23:53:11 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static char *function_eval(struct ast_channel *chan, char *cmd, char *data, char *buf, size_t len)  | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2005-09-25 20:47:00 +00:00
										 |  |  | 	memset(buf, 0, len); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-05-15 23:53:11 +00:00
										 |  |  | 	if (!data || ast_strlen_zero(data)) { | 
					
						
							| 
									
										
										
										
											2005-09-25 20:47:00 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "EVAL requires an argument: EVAL(<string>)\n"); | 
					
						
							| 
									
										
										
										
											2005-05-15 23:53:11 +00:00
										 |  |  | 		return buf; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-09-25 20:47:00 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-05-15 23:53:11 +00:00
										 |  |  | 	pbx_substitute_variables_helper(chan, data, buf, len - 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return buf; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef BUILTIN_FUNC
 | 
					
						
							|  |  |  | static | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | struct ast_custom_function eval_function = { | 
					
						
							|  |  |  | 	.name = "EVAL", | 
					
						
							|  |  |  | 	.synopsis = "Evaluate stored variables.", | 
					
						
							|  |  |  | 	.syntax = "EVAL(<variable>)", | 
					
						
							|  |  |  | 	.desc = "Using EVAL basically causes a string to be evaluated twice.\n" | 
					
						
							|  |  |  | 		"When a variable or expression is in the dialplan, it will be\n" | 
					
						
							|  |  |  | 		"evaluated at runtime. However, if the result of the evaluation\n" | 
					
						
							|  |  |  | 		"is in fact a variable or expression, using EVAL will have it\n" | 
					
						
							|  |  |  | 		"evaluated a second time. For example, if the variable ${MYVAR}\n" | 
					
						
							|  |  |  | 		"contains \"${OTHERVAR}\", then the result of putting ${EVAL(${MYVAR})}\n" | 
					
						
							|  |  |  | 		"in the dialplan will be the contents of the variable, OTHERVAR.\n" | 
					
						
							|  |  |  | 		"Normally, by just putting ${MYVAR} in the dialplan, you would be\n" | 
					
						
							|  |  |  | 		"left with \"${OTHERVAR}\".\n",  | 
					
						
							|  |  |  | 	.read = function_eval, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 |