mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-10-27 06:31:54 +00:00 
			
		
		
		
	(closes issue #14801) Reported by: festr git-svn-id: https://origsvn.digium.com/svn/asterisk/branches/1.6.1@192094 65c4cc65-6c06-0410-ace0-fbb531ad65f3
		
			
				
	
	
		
			254 lines
		
	
	
		
			9.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			254 lines
		
	
	
		
			9.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Asterisk -- An open source telephony toolkit.
 | |
|  *
 | |
|  * Copyright (C) 1999 - 2005, Anthony Minessale anthmct@yahoo.com
 | |
|  * Development of this app Sponsered/Funded  by TAAN Softworks Corp
 | |
|  *
 | |
|  * 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.
 | |
|  *
 | |
|  * This program is free software, distributed under the terms of
 | |
|  * the GNU General Public License Version 2. See the LICENSE file
 | |
|  * at the top of the source tree.
 | |
|  */
 | |
| 
 | |
| /*! \file
 | |
|  *
 | |
|  * \brief Fork CDR application
 | |
|  *
 | |
|  * \author Anthony Minessale anthmct@yahoo.com
 | |
|  *
 | |
|  * \note Development of this app Sponsored/Funded by TAAN Softworks Corp
 | |
|  * 
 | |
|  * \ingroup applications
 | |
|  */
 | |
| 
 | |
| #include "asterisk.h"
 | |
| 
 | |
| ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
 | |
| 
 | |
| #include "asterisk/file.h"
 | |
| #include "asterisk/channel.h"
 | |
| #include "asterisk/pbx.h"
 | |
| #include "asterisk/cdr.h"
 | |
| #include "asterisk/app.h"
 | |
| #include "asterisk/module.h"
 | |
| 
 | |
| static char *app = "ForkCDR";
 | |
| static char *synopsis = 
 | |
| "Forks the Call Data Record";
 | |
| static char *descrip = 
 | |
| "  ForkCDR([options]):  Causes the Call Data Record to fork an additional\n"
 | |
| "cdr record starting from the time of the fork call. This new cdr record will\n"
 | |
| "be linked to end of the list of cdr records attached to the channel. The original CDR\n"
 | |
| "has a LOCKED flag set, which forces most cdr operations to skip it, except\n"
 | |
| "for the functions that set the answer and end times, which ignore the LOCKED\n"
 | |
| "flag. This allows all the cdr records in the channel to be 'ended' together\n"
 | |
| "when the channel is closed.\n"
 | |
| "The CDR() func (when setting CDR values) normally ignores the LOCKED flag also,\n"
 | |
| "but has options to vary its behavior. The 'T' option (described below), can\n"
 | |
| "override this behavior, but beware the risks.\n"
 | |
| "\n"
 | |
| "Detailed Behavior Description:\n"
 | |
| "First, this app finds the last cdr record in the list, and makes\n"
 | |
| "a copy of it. This new copy will be the newly forked cdr record.\n"
 | |
| "Next, this new record is linked to the end of the cdr record list.\n"
 | |
| "Next, The new cdr record is RESET (unless you use an option to prevent this)\n"
 | |
| "This means that:\n"
 | |
| "   1. All flags are unset on the cdr record\n"
 | |
| "   2. the start, end, and answer times are all set to zero.\n"
 | |
| "   3. the billsec and duration fields are set to zero.\n"
 | |
| "   4. the start time is set to the current time.\n"
 | |
| "   5. the disposition is set to NULL.\n"
 | |
| "Next, unless you specified the 'v' option, all variables will be\n"
 | |
| "removed from the original cdr record. Thus, the 'v' option allows\n"
 | |
| "any CDR variables to be replicated to all new forked cdr records.\n"
 | |
| "Without the 'v' option, the variables on the original are effectively\n"
 | |
| "moved to the new forked cdr record.\n"
 | |
| "Next, if the 's' option is set, the provided variable and value\n"
 | |
| "are set on the original cdr record.\n"
 | |
| "Next, if the 'a' option is given, and the original cdr record has an\n"
 | |
| "answer time set, then the new forked cdr record will have its answer\n"
 | |
| "time set to its start time. If the old answer time were carried forward,\n"
 | |
| "the answer time would be earlier than the start time, giving strange\n"
 | |
| "duration and billsec times.\n"
 | |
| "Next, if the 'd' option was specified, the disposition is copied from\n"
 | |
| "the original cdr record to the new forked cdr.\n"
 | |
| "Next, if the 'D' option was specified, the destination channel field\n"
 | |
| "in the new forked CDR is erased.\n"
 | |
| "Next, if the 'e' option was specified, the 'end' time for the original\n"
 | |
| "cdr record is set to the current time. Future hang-up or ending events\n"
 | |
| "will not override this time stamp.\n"
 | |
| "Next, If the 'A' option is specified, the original cdr record will have\n"
 | |
| "it ANS_LOCKED flag set, which prevent future answer events\n"
 | |
| "from updating the original cdr record's disposition. Normally, an\n"
 | |
| "'ANSWERED' event would mark all cdr records in the chain as 'ANSWERED'.\n"
 | |
| "Next, if the 'T' option is specified, the original cdr record will have\n"
 | |
| "its 'DONT_TOUCH' flag set, which will force the cdr_answer, cdr_end, and\n"
 | |
| "cdr_setvar functions to leave that cdr record alone.\n"
 | |
| "And, last but not least, the original cdr record has its LOCKED flag\n"
 | |
| "set. Almost all internal CDR functions (except for the funcs that set\n"
 | |
| "the end, and answer times, and set a variable) will honor this flag\n"
 | |
| "and leave a LOCKED cdr record alone.\n"
 | |
| "This means that the newly created forked cdr record will be affected\n"
 | |
| "by events transpiring within Asterisk, with the previously noted\n"
 | |
| "exceptions.\n"
 | |
| "  Options:\n"
 | |
| "    a - update the answer time on the NEW CDR just after it's been inited.\n"
 | |
| "         The new CDR may have been answered already. The reset that forkcdr\n"
 | |
| "         does will erase the answer time. This will bring it back, but\n"
 | |
| "         the answer time will be a copy of the fork/start time. It will\n"
 | |
| "         only do this if the initial cdr was indeed already answered.\n"
 | |
| "    A - Lock the original CDR against the answer time being updated.\n"
 | |
| "         This will allow the disposition on the original CDR to remain the same.\n"
 | |
| "    d - Copy the disposition forward from the old cdr, after the \n"
 | |
| "         init.\n"
 | |
| "    D - Clear the dstchannel on the new CDR after reset.\n"
 | |
| "    e - end the original CDR. Do this after all the neccessary data\n"
 | |
| "         is copied from the original CDR to the new forked CDR.\n"
 | |
| "    R -  do NOT reset the new cdr.\n"
 | |
| "    s(name=val) - Set the CDR var 'name' in the original CDR, with value\n"
 | |
| "                  'val'.\n"
 | |
| "    T -  Mark the original CDR with a DONT_TOUCH flag. setvar, answer, and end\n"
 | |
| "          cdr funcs will obey this flag; normally they don't honor the LOCKED\n"
 | |
| "          flag set on the original CDR record.\n"
 | |
| "          Beware-- using this flag may cause CDR's not to have their end times\n"
 | |
| "          updated! It is suggested that if you specify this flag, you might\n"
 | |
| "          wish to use the 'e' flag as well!\n"
 | |
| "    v  - When the new CDR is forked, it gets a copy of the vars attached\n"
 | |
| "         to the current CDR. The vars attached to the original CDR are removed\n"
 | |
| "         unless this option is specified.\n";
 | |
| 
 | |
| 
 | |
| enum {
 | |
| 	OPT_SETANS =            (1 << 0),
 | |
| 	OPT_SETDISP =           (1 << 1),
 | |
| 	OPT_RESETDEST =         (1 << 2),
 | |
| 	OPT_ENDCDR =            (1 << 3),
 | |
| 	OPT_NORESET =           (1 << 4),
 | |
| 	OPT_KEEPVARS =          (1 << 5),
 | |
| 	OPT_VARSET =            (1 << 6),
 | |
| 	OPT_ANSLOCK =           (1 << 7),
 | |
| 	OPT_DONTOUCH =          (1 << 8),
 | |
| };
 | |
| 
 | |
| enum {
 | |
| 	OPT_ARG_VARSET = 0,
 | |
| 	/* note: this entry _MUST_ be the last one in the enum */
 | |
| 	OPT_ARG_ARRAY_SIZE,
 | |
| };
 | |
| 
 | |
| AST_APP_OPTIONS(forkcdr_exec_options, {
 | |
| 	AST_APP_OPTION('a', OPT_SETANS),
 | |
| 	AST_APP_OPTION('A', OPT_ANSLOCK),
 | |
| 	AST_APP_OPTION('d', OPT_SETDISP),
 | |
| 	AST_APP_OPTION('D', OPT_RESETDEST),
 | |
| 	AST_APP_OPTION('e', OPT_ENDCDR),
 | |
| 	AST_APP_OPTION('R', OPT_NORESET),
 | |
| 	AST_APP_OPTION_ARG('s', OPT_VARSET, OPT_ARG_VARSET),
 | |
| 	AST_APP_OPTION('T', OPT_DONTOUCH),
 | |
| 	AST_APP_OPTION('v', OPT_KEEPVARS),
 | |
| });
 | |
| 
 | |
| static void ast_cdr_fork(struct ast_channel *chan, struct ast_flags optflags, char *set) 
 | |
| {
 | |
| 	struct ast_cdr *cdr;
 | |
| 	struct ast_cdr *newcdr;
 | |
| 	struct ast_flags flags = { AST_CDR_FLAG_KEEP_VARS };
 | |
| 
 | |
| 	cdr = chan->cdr;
 | |
| 
 | |
| 	while (cdr->next)
 | |
| 		cdr = cdr->next;
 | |
| 	
 | |
| 	if (!(newcdr = ast_cdr_dup(cdr)))
 | |
| 		return;
 | |
| 	
 | |
| 	ast_cdr_append(cdr, newcdr);
 | |
| 
 | |
| 	if (!ast_test_flag(&optflags, OPT_NORESET))
 | |
| 		ast_cdr_reset(newcdr, &flags);
 | |
| 		
 | |
| 	if (!ast_test_flag(cdr, AST_CDR_FLAG_KEEP_VARS))
 | |
| 		ast_cdr_free_vars(cdr, 0);
 | |
| 	
 | |
| 	if (!ast_strlen_zero(set)) {
 | |
| 		char *varname = ast_strdupa(set), *varval;
 | |
| 		varval = strchr(varname,'=');
 | |
| 		if (varval) {
 | |
| 			*varval = 0;
 | |
| 			varval++;
 | |
| 			ast_cdr_setvar(cdr, varname, varval, 0);
 | |
| 		}
 | |
| 	}
 | |
| 	
 | |
| 	if (ast_test_flag(&optflags, OPT_SETANS) && !ast_tvzero(cdr->answer))
 | |
| 		newcdr->answer = newcdr->start;
 | |
| 
 | |
| 	if (ast_test_flag(&optflags, OPT_SETDISP))
 | |
| 		newcdr->disposition = cdr->disposition;
 | |
| 	
 | |
| 	if (ast_test_flag(&optflags, OPT_RESETDEST))
 | |
| 		newcdr->dstchannel[0] = 0;
 | |
| 	
 | |
| 	if (ast_test_flag(&optflags, OPT_ENDCDR))
 | |
| 		ast_cdr_end(cdr);
 | |
| 
 | |
| 	if (ast_test_flag(&optflags, OPT_ANSLOCK))
 | |
| 		ast_set_flag(cdr, AST_CDR_FLAG_ANSLOCKED);
 | |
| 	
 | |
| 	if (ast_test_flag(&optflags, OPT_DONTOUCH))
 | |
| 		ast_set_flag(cdr, AST_CDR_FLAG_DONT_TOUCH);
 | |
| 		
 | |
| 	ast_set_flag(cdr, AST_CDR_FLAG_CHILD | AST_CDR_FLAG_LOCKED);
 | |
| }
 | |
| 
 | |
| static int forkcdr_exec(struct ast_channel *chan, void *data)
 | |
| {
 | |
| 	int res = 0;
 | |
| 	char *argcopy = NULL;
 | |
| 	struct ast_flags flags = {0};
 | |
| 	char *opts[OPT_ARG_ARRAY_SIZE];
 | |
| 	AST_DECLARE_APP_ARGS(arglist,
 | |
| 		AST_APP_ARG(options);
 | |
| 	);
 | |
| 
 | |
| 	if (!chan->cdr) {
 | |
| 		ast_log(LOG_WARNING, "Channel does not have a CDR\n");
 | |
| 		return 0;
 | |
| 	}
 | |
| 
 | |
| 	argcopy = ast_strdupa(data);
 | |
| 
 | |
| 	AST_STANDARD_APP_ARGS(arglist, argcopy);
 | |
| 
 | |
| 	opts[OPT_ARG_VARSET] = 0;
 | |
| 
 | |
| 	if (!ast_strlen_zero(arglist.options))
 | |
| 		ast_app_parse_options(forkcdr_exec_options, &flags, opts, arglist.options);
 | |
| 
 | |
| 	if (!ast_strlen_zero(data)) {
 | |
| 		int keepvars = ast_test_flag(&flags, OPT_KEEPVARS) ? 1 : 0;
 | |
| 		ast_set2_flag(chan->cdr, keepvars, AST_CDR_FLAG_KEEP_VARS);
 | |
| 	}
 | |
| 	
 | |
| 	ast_cdr_fork(chan, flags, opts[OPT_ARG_VARSET]);
 | |
| 
 | |
| 	return res;
 | |
| }
 | |
| 
 | |
| static int unload_module(void)
 | |
| {
 | |
| 	return ast_unregister_application(app);
 | |
| }
 | |
| 
 | |
| static int load_module(void)
 | |
| {
 | |
| 	return ast_register_application(app, forkcdr_exec, synopsis, descrip);
 | |
| }
 | |
| 
 | |
| AST_MODULE_INFO_STANDARD(ASTERISK_GPL_KEY, "Fork The CDR into 2 separate entities");
 |