Skip to content
Snippets Groups Projects
func_cdr.c 9.06 KiB
Newer Older
  • Learn to ignore specific revisions
  •  * Asterisk -- An open source telephony toolkit.
     *
    
     * Copyright (C) 1999-2006, Digium, Inc.
    
    Kevin P. Fleming's avatar
    Kevin P. Fleming committed
     * Portions Copyright (C) 2005, Anthony Minessale II
    
     * 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.
     */
    
    
     * \brief  Call Detail Record related dialplan functions
    
     *
     * \author Anthony Minessale II 
    
    Olle Johansson's avatar
    Olle Johansson committed
     *
     * \ingroup functions
    
    /*** MODULEINFO
    	<support_level>core</support_level>
     ***/
    
    
    Kevin P. Fleming's avatar
    Kevin P. Fleming committed
    #include "asterisk.h"
    
    
    ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
    
    #include "asterisk/channel.h"
    #include "asterisk/pbx.h"
    #include "asterisk/utils.h"
    #include "asterisk/app.h"
    #include "asterisk/cdr.h"
    
    
    /*** DOCUMENTATION
    	<function name="CDR" language="en_US">
    		<synopsis>
    			Gets or sets a CDR variable.
    		</synopsis>	
    		<syntax>
    			<parameter name="name" required="true">
    				<para>CDR field name:</para>
    				<enumlist>
    					<enum name="clid">
    						<para>Caller ID.</para>
    					</enum>
    					<enum name="lastdata">
    						<para>Last application arguments.</para>
    					</enum>
    					<enum name="disposition">
    
    						<para>ANSWERED, NO ANSWER, BUSY, FAILED.</para>
    
    					</enum>
    					<enum name="src">
    						<para>Source.</para>
    					</enum>
    					<enum name="start">
    						<para>Time the call started.</para>
    					</enum>
    					<enum name="amaflags">
    						<para>DOCUMENTATION, BILL, IGNORE, etc.</para>
    					</enum>
    					<enum name="dst">
    						<para>Destination.</para>
    					</enum>
    					<enum name="answer">
    						<para>Time the call was answered.</para>
    					</enum>
    					<enum name="accountcode">
    						<para>The channel's account code.</para>
    					</enum>
    					<enum name="dcontext">
    						<para>Destination context.</para>
    					</enum>
    					<enum name="end">
    						<para>Time the call ended.</para>
    					</enum>
    					<enum name="uniqueid">
    						<para>The channel's unique id.</para>
    					</enum>
    					<enum name="dstchannel">
    						<para>Destination channel.</para>
    					</enum>
    					<enum name="duration">
    						<para>Duration of the call.</para>
    					</enum>
    					<enum name="userfield">
    						<para>The channel's user specified field.</para>
    					</enum>
    					<enum name="lastapp">
    						<para>Last application.</para>
    					</enum>
    					<enum name="billsec">
    						<para>Duration of the call once it was answered.</para>
    					</enum>
    					<enum name="channel">
    						<para>Channel name.</para>
    					</enum>
    
    Mark Michelson's avatar
    Mark Michelson committed
    						<para>CDR sequence number.</para>
    
    				</enumlist>
    			</parameter>
    			<parameter name="options" required="false">
    				<optionlist>
    
    					<option name="f">
    						<para>Returns billsec or duration fields as floating point values.</para>
    					</option>
    
    					<option name="l">
    						<para>Uses the most recent CDR on a channel with multiple records</para>
    					</option>
    					<option name="r">
    						<para>Searches the entire stack of CDRs on the channel.</para>
    					</option>
    					<option name="s">
    						<para>Skips any CDR's that are marked 'LOCKED' due to forkCDR() calls.
    						(on setting/writing CDR vars only)</para>
    					</option>
    					<option name="u">
    						<para>Retrieves the raw, unprocessed value.</para>
    						<para>For example, 'start', 'answer', and 'end' will be retrieved as epoch
    						values, when the <literal>u</literal> option is passed, but formatted as YYYY-MM-DD HH:MM:SS
    						otherwise.  Similarly, disposition and amaflags will return their raw
    						integral values.</para>
    					</option>
    				</optionlist>
    			</parameter>
    		</syntax>
    		<description>
    			<para>All of the CDR field names are read-only, except for <literal>accountcode</literal>,
    			<literal>userfield</literal>, and <literal>amaflags</literal>. You may, however, supply
    			a name not on the above list, and create your own variable, whose value can be changed
    			with this function, and this variable will be stored on the cdr.</para>
    			<note><para>For setting CDR values, the <literal>l</literal> flag does not apply to
    			setting the <literal>accountcode</literal>, <literal>userfield</literal>, or
    
    			<literal>amaflags</literal>.</para><para>CDRs can only be modified before the bridge
    			between two channels is torn down. For example, CDRs may not be modified after the
    			<literal>Dial</literal> application has returned.</para></note>
    
    			<para>Raw values for <literal>disposition</literal>:</para>
    			<enumlist>
    
    					<para>NO ANSWER</para>
    				</enum>
    
    				<enum name="1">
    					<para>NO ANSWER (NULL record)</para>
    
    					<para>FAILED</para>
    				</enum>
    				<enum name="4">
    
    					<para>BUSY</para>
    				</enum>
    				<enum name="8">
    
    					<para>ANSWERED</para>
    				</enum>
    			</enumlist>
    			<para>Raw values for <literal>amaflags</literal>:</para>
    			<enumlist>
    				<enum name="1">
    					<para>OMIT</para>
    				</enum>
    				<enum name="2">
    					<para>BILLING</para>
    				</enum>
    				<enum name="3">
    					<para>DOCUMENTATION</para>
    				</enum>
    			</enumlist>
    			<para>Example: exten => 1,1,Set(CDR(userfield)=test)</para>
    		</description>
    	</function>
     ***/
    
    
    enum cdr_option_flags {
    
    	OPT_RECURSIVE = (1 << 0),
    
    	OPT_UNPARSED = (1 << 1),
    
    	OPT_SKIPLOCKED = (1 << 3),
    
    	OPT_FLOAT = (1 << 4),
    
    
    AST_APP_OPTIONS(cdr_func_options, {
    
    	AST_APP_OPTION('f', OPT_FLOAT),
    
    	AST_APP_OPTION('r', OPT_RECURSIVE),
    
    	AST_APP_OPTION('s', OPT_SKIPLOCKED),
    
    	AST_APP_OPTION('u', OPT_UNPARSED),
    
    static int cdr_read(struct ast_channel *chan, const char *cmd, char *parse,
    
    		    char *buf, size_t len)
    
    	struct ast_flags flags = { 0 };
    
    	struct ast_cdr *cdr;
    
    	AST_DECLARE_APP_ARGS(args,
    
    			     AST_APP_ARG(variable);
    			     AST_APP_ARG(options);
    
    	if (ast_strlen_zero(parse) || !chan)
    
    		return -1;
    
    	ast_channel_lock(chan);
    
    	cdr = ast_channel_cdr(chan);
    
    	if (!cdr) {
    		ast_channel_unlock(chan);
    
    		return -1;
    
    	AST_STANDARD_APP_ARGS(args, parse);
    
    
    	if (!ast_strlen_zero(args.options))
    
    		ast_app_parse_options(cdr_func_options, &flags, NULL, args.options);
    
    	if (ast_test_flag(&flags, OPT_LAST))
    		while (cdr->next)
    			cdr = cdr->next;
    
    
    	if (ast_test_flag(&flags, OPT_SKIPLOCKED))
    		while (ast_test_flag(cdr, AST_CDR_FLAG_LOCKED) && cdr->next)
    			cdr = cdr->next;
    
    
    	if (!strcasecmp("billsec", args.variable) && ast_test_flag(&flags, OPT_FLOAT)) {
    		if (!ast_tvzero(cdr->answer)) {
    			double hrtime;
    
    			if(!ast_tvzero(cdr->end))
    				hrtime = (double)(ast_tvdiff_us(cdr->end, cdr->answer) / 1000000.0);
    			else
    				hrtime = (double)(ast_tvdiff_us(ast_tvnow(), cdr->answer) / 1000000.0);
    
    			snprintf(buf, len, "%lf", hrtime);
    		} else {
    			snprintf(buf, len, "%lf", 0.0);
    		}
    		ret = buf;
    	} else if (!strcasecmp("duration", args.variable) && ast_test_flag(&flags, OPT_FLOAT)) {
    			double hrtime;
    
    			if(!ast_tvzero(cdr->end))
    				hrtime = (double)(ast_tvdiff_us(cdr->end, cdr->start) / 1000000.0);
    			else
    				hrtime = (double)(ast_tvdiff_us(ast_tvnow(), cdr->start) / 1000000.0);
    
    			snprintf(buf, len, "%lf", hrtime);
    
    		if (!ast_strlen_zero(buf)) {
    			ret = buf;
    		}
    	} else {
    		ast_cdr_getvar(cdr, args.variable, &ret, buf, len,
    			       ast_test_flag(&flags, OPT_RECURSIVE),
    				   ast_test_flag(&flags, OPT_UNPARSED));
    	}
    
    	ast_channel_unlock(chan);
    
    	return ret ? 0 : -1;
    
    static int cdr_write(struct ast_channel *chan, const char *cmd, char *parse,
    
    		     const char *value)
    
    	struct ast_cdr *cdr;
    
    	struct ast_flags flags = { 0 };
    	AST_DECLARE_APP_ARGS(args,
    			     AST_APP_ARG(variable);
    			     AST_APP_ARG(options);
    	);
    
    	if (ast_strlen_zero(parse) || !value || !chan)
    
    		return -1;
    
    	ast_channel_lock(chan);
    
    	cdr = ast_channel_cdr(chan);
    
    	if (!cdr) {
    		ast_channel_unlock(chan);
    
    	AST_STANDARD_APP_ARGS(args, parse);
    
    	if (!ast_strlen_zero(args.options))
    
    		ast_app_parse_options(cdr_func_options, &flags, NULL, args.options);
    
    	if (ast_test_flag(&flags, OPT_LAST))
    		while (cdr->next)
    			cdr = cdr->next;
    
    	if (!strcasecmp(args.variable, "accountcode"))  /* the 'l' flag doesn't apply to setting the accountcode, userfield, or amaflags */
    
    	else if (!strcasecmp(args.variable, "peeraccount"))
    		ast_cdr_setpeeraccount(chan, value);
    
    	else if (!strcasecmp(args.variable, "userfield"))
    
    		ast_cdr_setuserfield(chan, value);
    
    	else if (!strcasecmp(args.variable, "amaflags"))
    
    	else
    		ast_cdr_setvar(cdr, args.variable, value, ast_test_flag(&flags, OPT_RECURSIVE));
    
    		/* No need to worry about the u flag, as all fields for which setting
    		 * 'u' would do anything are marked as readonly. */
    
    	ast_channel_unlock(chan);
    
    	return 0;
    
    static struct ast_custom_function cdr_function = {
    
    	.read = cdr_read,
    	.write = cdr_write,
    
    static int unload_module(void)
    
    	return ast_custom_function_unregister(&cdr_function);
    
    static int load_module(void)
    
    	return ast_custom_function_register(&cdr_function);
    
    AST_MODULE_INFO_STANDARD(ASTERISK_GPL_KEY, "Call Detail Record (CDR) dialplan function");