mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-11-03 20:38:59 +00:00 
			
		
		
		
	ASTERISK_REGISTER_FILE no longer has any purpose so this commit removes all traces of it. Previously exported symbols removed: * __ast_register_file * __ast_unregister_file * ast_complete_source_filename This also removes the mtx_prof static variable that was declared when MTX_PROFILE was enabled. This variable was only used in lock.c so it is now initialized in that file only. ASTERISK-26480 #close Change-Id: I1074af07d71f9e159c48ef36631aa432c86f9966
		
			
				
	
	
		
			268 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			268 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * Asterisk -- An open source telephony toolkit.
 | 
						|
 *
 | 
						|
 * Copyright (C) 1999 - 2005, Digium, Inc.
 | 
						|
 *
 | 
						|
 * Martin Pycko <martinp@digium.com>
 | 
						|
 *
 | 
						|
 * 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 Applications connected with CDR engine
 | 
						|
 *
 | 
						|
 * \author Martin Pycko <martinp@digium.com>
 | 
						|
 *
 | 
						|
 * \ingroup applications
 | 
						|
 */
 | 
						|
 | 
						|
/*** MODULEINFO
 | 
						|
	<support_level>core</support_level>
 | 
						|
 ***/
 | 
						|
 | 
						|
#include "asterisk.h"
 | 
						|
 | 
						|
#include "asterisk/channel.h"
 | 
						|
#include "asterisk/module.h"
 | 
						|
#include "asterisk/app.h"
 | 
						|
#include "asterisk/stasis.h"
 | 
						|
#include "asterisk/stasis_message_router.h"
 | 
						|
 | 
						|
/*** DOCUMENTATION
 | 
						|
	<application name="NoCDR" language="en_US">
 | 
						|
		<synopsis>
 | 
						|
			Tell Asterisk to not maintain a CDR for this channel.
 | 
						|
		</synopsis>
 | 
						|
		<syntax />
 | 
						|
		<description>
 | 
						|
			<para>This application will tell Asterisk not to maintain a CDR for
 | 
						|
			the current channel. This does <emphasis>NOT</emphasis> mean that
 | 
						|
			information is not tracked; rather, if the channel is hung up no
 | 
						|
			CDRs will be created for that channel.</para>
 | 
						|
			<para>If a subsequent call to ResetCDR occurs, all non-finalized
 | 
						|
			CDRs created for the channel will be enabled.</para>
 | 
						|
			<note><para>This application is deprecated. Please use the CDR_PROP
 | 
						|
			function to disable CDRs on a channel.</para></note>
 | 
						|
		</description>
 | 
						|
		<see-also>
 | 
						|
			<ref type="application">ResetCDR</ref>
 | 
						|
			<ref type="function">CDR_PROP</ref>
 | 
						|
		</see-also>
 | 
						|
	</application>
 | 
						|
	<application name="ResetCDR" language="en_US">
 | 
						|
		<synopsis>
 | 
						|
			Resets the Call Data Record.
 | 
						|
		</synopsis>
 | 
						|
		<syntax>
 | 
						|
			<parameter name="options">
 | 
						|
				<optionlist>
 | 
						|
					<option name="v">
 | 
						|
						<para>Save the CDR variables during the reset.</para>
 | 
						|
					</option>
 | 
						|
					<option name="e">
 | 
						|
						<para>Enable the CDRs for this channel only (negate
 | 
						|
						effects of NoCDR).</para>
 | 
						|
					</option>
 | 
						|
				</optionlist>
 | 
						|
			</parameter>
 | 
						|
		</syntax>
 | 
						|
		<description>
 | 
						|
			<para>This application causes the Call Data Record to be reset.
 | 
						|
			Depending on the flags passed in, this can have several effects.
 | 
						|
			With no options, a reset does the following:</para>
 | 
						|
			<para>1. The <literal>start</literal> time is set to the current time.</para>
 | 
						|
			<para>2. If the channel is answered, the <literal>answer</literal> time is set to the
 | 
						|
			current time.</para>
 | 
						|
			<para>3. All variables are wiped from the CDR. Note that this step
 | 
						|
			can be prevented with the <literal>v</literal> option.</para>
 | 
						|
			<para>On the other hand, if the <literal>e</literal> option is
 | 
						|
			specified, the effects of the NoCDR application will be lifted. CDRs
 | 
						|
			will be re-enabled for this channel.</para>
 | 
						|
			<note><para>The <literal>e</literal> option is deprecated. Please
 | 
						|
			use the CDR_PROP function instead.</para></note>
 | 
						|
		</description>
 | 
						|
		<see-also>
 | 
						|
			<ref type="application">ForkCDR</ref>
 | 
						|
			<ref type="application">NoCDR</ref>
 | 
						|
			<ref type="function">CDR_PROP</ref>
 | 
						|
		</see-also>
 | 
						|
	</application>
 | 
						|
 ***/
 | 
						|
 | 
						|
static const char nocdr_app[] = "NoCDR";
 | 
						|
static const char resetcdr_app[] = "ResetCDR";
 | 
						|
 | 
						|
enum reset_cdr_options {
 | 
						|
	OPT_DISABLE_DISPATCH = (1 << 0),
 | 
						|
	OPT_KEEP_VARS = (1 << 1),
 | 
						|
	OPT_ENABLE = (1 << 2),
 | 
						|
};
 | 
						|
 | 
						|
AST_APP_OPTIONS(resetcdr_opts, {
 | 
						|
	AST_APP_OPTION('v', AST_CDR_FLAG_KEEP_VARS),
 | 
						|
	AST_APP_OPTION('e', AST_CDR_FLAG_DISABLE_ALL),
 | 
						|
});
 | 
						|
 | 
						|
STASIS_MESSAGE_TYPE_DEFN_LOCAL(appcdr_message_type);
 | 
						|
 | 
						|
/*! \internal \brief Payload for the Stasis message sent to manipulate a CDR */
 | 
						|
struct app_cdr_message_payload {
 | 
						|
	/*! The name of the channel to be manipulated */
 | 
						|
	const char *channel_name;
 | 
						|
	/*! Disable the CDR for this channel */
 | 
						|
	int disable:1;
 | 
						|
	/*! Re-enable the CDR for this channel */
 | 
						|
	int reenable:1;
 | 
						|
	/*! Reset the CDR */
 | 
						|
	int reset:1;
 | 
						|
	/*! If reseting the CDR, keep the variables */
 | 
						|
	int keep_variables:1;
 | 
						|
};
 | 
						|
 | 
						|
static void appcdr_callback(void *data, struct stasis_subscription *sub, struct stasis_message *message)
 | 
						|
{
 | 
						|
	struct app_cdr_message_payload *payload;
 | 
						|
 | 
						|
	if (stasis_message_type(message) != appcdr_message_type()) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	payload = stasis_message_data(message);
 | 
						|
	if (!payload) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	if (payload->disable) {
 | 
						|
		if (ast_cdr_set_property(payload->channel_name, AST_CDR_FLAG_DISABLE_ALL)) {
 | 
						|
			ast_log(AST_LOG_WARNING, "Failed to disable CDRs on channel %s\n",
 | 
						|
				payload->channel_name);
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if (payload->reenable) {
 | 
						|
		if (ast_cdr_clear_property(payload->channel_name, AST_CDR_FLAG_DISABLE_ALL)) {
 | 
						|
			ast_log(AST_LOG_WARNING, "Failed to enable CDRs on channel %s\n",
 | 
						|
				payload->channel_name);
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if (payload->reset) {
 | 
						|
		if (ast_cdr_reset(payload->channel_name, payload->keep_variables)) {
 | 
						|
			ast_log(AST_LOG_WARNING, "Failed to reset CDRs on channel %s\n",
 | 
						|
				payload->channel_name);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static int publish_app_cdr_message(struct ast_channel *chan, struct app_cdr_message_payload *payload)
 | 
						|
{
 | 
						|
	RAII_VAR(struct stasis_message *, message, NULL, ao2_cleanup);
 | 
						|
	RAII_VAR(struct stasis_message_router *, router, ast_cdr_message_router(), ao2_cleanup);
 | 
						|
 | 
						|
	if (!router) {
 | 
						|
		ast_log(AST_LOG_WARNING, "Failed to manipulate CDR for channel %s: no message router\n",
 | 
						|
			ast_channel_name(chan));
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
 | 
						|
	message = stasis_message_create(appcdr_message_type(), payload);
 | 
						|
	if (!message) {
 | 
						|
		ast_log(AST_LOG_WARNING, "Failed to manipulate CDR for channel %s: unable to create message\n",
 | 
						|
			payload->channel_name);
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
	stasis_message_router_publish_sync(router, message);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int resetcdr_exec(struct ast_channel *chan, const char *data)
 | 
						|
{
 | 
						|
	RAII_VAR(struct app_cdr_message_payload *, payload,
 | 
						|
		ao2_alloc(sizeof(*payload), NULL), ao2_cleanup);
 | 
						|
	char *args;
 | 
						|
	struct ast_flags flags = { 0 };
 | 
						|
 | 
						|
	if (!payload) {
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
 | 
						|
	if (!ast_strlen_zero(data)) {
 | 
						|
		args = ast_strdupa(data);
 | 
						|
		ast_app_parse_options(resetcdr_opts, &flags, NULL, args);
 | 
						|
	}
 | 
						|
 | 
						|
	payload->channel_name = ast_channel_name(chan);
 | 
						|
	payload->reset = 1;
 | 
						|
 | 
						|
	if (ast_test_flag(&flags, AST_CDR_FLAG_DISABLE_ALL)) {
 | 
						|
		payload->reenable = 1;
 | 
						|
	}
 | 
						|
 | 
						|
	if (ast_test_flag(&flags, AST_CDR_FLAG_KEEP_VARS)) {
 | 
						|
		payload->keep_variables = 1;
 | 
						|
	}
 | 
						|
 | 
						|
	return publish_app_cdr_message(chan, payload);
 | 
						|
}
 | 
						|
 | 
						|
static int nocdr_exec(struct ast_channel *chan, const char *data)
 | 
						|
{
 | 
						|
	RAII_VAR(struct app_cdr_message_payload *, payload,
 | 
						|
		ao2_alloc(sizeof(*payload), NULL), ao2_cleanup);
 | 
						|
 | 
						|
	if (!payload) {
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
 | 
						|
	payload->channel_name = ast_channel_name(chan);
 | 
						|
	payload->disable = 1;
 | 
						|
 | 
						|
	return publish_app_cdr_message(chan, payload);
 | 
						|
}
 | 
						|
 | 
						|
static int unload_module(void)
 | 
						|
{
 | 
						|
	RAII_VAR(struct stasis_message_router *, router, ast_cdr_message_router(), ao2_cleanup);
 | 
						|
 | 
						|
	if (router) {
 | 
						|
		stasis_message_router_remove(router, appcdr_message_type());
 | 
						|
	}
 | 
						|
	STASIS_MESSAGE_TYPE_CLEANUP(appcdr_message_type);
 | 
						|
	ast_unregister_application(nocdr_app);
 | 
						|
	ast_unregister_application(resetcdr_app);
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int load_module(void)
 | 
						|
{
 | 
						|
	RAII_VAR(struct stasis_message_router *, router, ast_cdr_message_router(), ao2_cleanup);
 | 
						|
	int res = 0;
 | 
						|
 | 
						|
	if (!router) {
 | 
						|
		return AST_MODULE_LOAD_FAILURE;
 | 
						|
	}
 | 
						|
 | 
						|
	res |= STASIS_MESSAGE_TYPE_INIT(appcdr_message_type);
 | 
						|
	res |= ast_register_application_xml(nocdr_app, nocdr_exec);
 | 
						|
	res |= ast_register_application_xml(resetcdr_app, resetcdr_exec);
 | 
						|
	res |= stasis_message_router_add(router, appcdr_message_type(),
 | 
						|
	                                 appcdr_callback, NULL);
 | 
						|
 | 
						|
	if (res) {
 | 
						|
		return AST_MODULE_LOAD_FAILURE;
 | 
						|
	}
 | 
						|
	return AST_MODULE_LOAD_SUCCESS;
 | 
						|
}
 | 
						|
 | 
						|
AST_MODULE_INFO_STANDARD(ASTERISK_GPL_KEY, "Tell Asterisk to not maintain a CDR for the current call");
 |