mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-10-26 14:27:14 +00:00 
			
		
		
		
	An m option to Park and ParkAndAnnounce now allows specifying a music on hold class override. ASTERISK-30087 Change-Id: I03de8d97b100e451b2611b5a621d48750f5d6a9e
		
			
				
	
	
		
			1030 lines
		
	
	
		
			31 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1030 lines
		
	
	
		
			31 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Asterisk -- An open source telephony toolkit.
 | |
|  *
 | |
|  * Copyright (C) 2013, Digium, Inc.
 | |
|  *
 | |
|  * Jonathan Rose <jrose@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 Call Parking Applications
 | |
|  *
 | |
|  * \author Jonathan Rose <jrose@digium.com>
 | |
|  */
 | |
| 
 | |
| #include "asterisk.h"
 | |
| 
 | |
| #include "res_parking.h"
 | |
| #include "asterisk/config.h"
 | |
| #include "asterisk/config_options.h"
 | |
| #include "asterisk/utils.h"
 | |
| #include "asterisk/astobj2.h"
 | |
| #include "asterisk/features.h"
 | |
| #include "asterisk/module.h"
 | |
| #include "asterisk/app.h"
 | |
| #include "asterisk/say.h"
 | |
| #include "asterisk/bridge_basic.h"
 | |
| #include "asterisk/format_cache.h"
 | |
| 
 | |
| /*** DOCUMENTATION
 | |
| 	<application name="Park" language="en_US">
 | |
| 		<synopsis>
 | |
| 			Park yourself.
 | |
| 		</synopsis>
 | |
| 		<syntax>
 | |
| 			<parameter name="parking_lot_name">
 | |
| 				<para>Specify in which parking lot to park a call.</para>
 | |
| 				<para>The parking lot used is selected in the following order:</para>
 | |
| 				<para>1) parking_lot_name option to this application</para>
 | |
| 				<para>2) <variable>PARKINGLOT</variable> variable</para>
 | |
| 				<para>3) <literal>CHANNEL(parkinglot)</literal> function
 | |
| 				(Possibly preset by the channel driver.)</para>
 | |
| 				<para>4) Default parking lot.</para>
 | |
| 			</parameter>
 | |
| 			<parameter name="options">
 | |
| 				<para>A list of options for this parked call.</para>
 | |
| 				<optionlist>
 | |
| 					<option name="r">
 | |
| 						<para>Send ringing instead of MOH to the parked call.</para>
 | |
| 					</option>
 | |
| 					<option name="R">
 | |
| 						<para>Randomize the selection of a parking space.</para>
 | |
| 					</option>
 | |
| 					<option name="s">
 | |
| 						<para>Silence announcement of the parking space number.</para>
 | |
| 					</option>
 | |
| 					<option name="c" argsep=",">
 | |
| 						<argument name="context" required="false" />
 | |
| 						<argument name="extension" required="false" />
 | |
| 						<argument name="priority" required="true" />
 | |
| 						<para>If the parking times out, go to this place in the dialplan
 | |
| 							instead of where the parking lot defines the call should go.
 | |
| 						</para>
 | |
| 					</option>
 | |
| 					<option name="t">
 | |
| 						<argument name="duration" required="true" />
 | |
| 						<para>Use a timeout of <literal>duration</literal> seconds instead
 | |
| 							of the timeout specified by the parking lot.</para>
 | |
| 					</option>
 | |
| 				</optionlist>
 | |
| 			</parameter>
 | |
| 		</syntax>
 | |
| 		<description>
 | |
| 			<para>Used to park yourself (typically in combination with an attended
 | |
| 			transfer to know the parking space).</para>
 | |
| 			<para>If you set the <variable>PARKINGEXTEN</variable> variable to a
 | |
| 				parking space extension in the parking lot, Park() will attempt to park the
 | |
| 				call on that extension. If the extension is already in use then execution
 | |
| 				will continue at the next priority.
 | |
| 			</para>
 | |
| 			<para>If the <literal>parkeddynamic</literal> option is enabled in
 | |
| 				<filename>res_parking.conf</filename> the following variables can be
 | |
| 				used to dynamically create new parking lots. When using dynamic parking
 | |
| 				lots, be aware of the conditions as explained in the notes section
 | |
| 				below.
 | |
| 			</para>
 | |
| 			<para>The <variable>PARKINGDYNAMIC</variable> variable specifies the
 | |
| 				parking lot to use as a template to create a dynamic parking lot. It
 | |
| 				is an error to specify a non-existent parking lot for the template.
 | |
| 				If not set then the default parking lot is used as the template.
 | |
| 			</para>
 | |
| 			<para>The <variable>PARKINGDYNCONTEXT</variable> variable specifies the
 | |
| 				dialplan context to use for the newly created dynamic parking lot. If
 | |
| 				not set then the context from the parking lot template is used. The
 | |
| 				context is created if it does not already exist and the new parking lot
 | |
| 				needs to create extensions.
 | |
| 			</para>
 | |
| 			<para>The <variable>PARKINGDYNEXTEN</variable> variable specifies the
 | |
| 				<literal>parkext</literal> to use for the newly created dynamic
 | |
| 				parking lot. If not set then the <literal>parkext</literal> is used from
 | |
| 				the parking lot template. If the template does not specify a
 | |
| 				<literal>parkext</literal> then no extensions are created for the newly
 | |
| 				created parking lot. The dynamic parking lot cannot be created if it
 | |
| 				needs to create extensions that overlap existing parking lot extensions.
 | |
| 				The only exception to this is for the <literal>parkext</literal>
 | |
| 				extension and only if neither of the overlaping parking lot's
 | |
| 				<literal>parkext</literal> is exclusive.
 | |
| 			</para>
 | |
| 			<para>The <variable>PARKINGDYNPOS</variable> variable specifies the
 | |
| 				parking positions to use for the newly created dynamic parking lot. If
 | |
| 				not set then the <literal>parkpos</literal> from the parking lot template
 | |
| 				is used.
 | |
| 			</para>
 | |
| 			<note>
 | |
| 				<para>This application must be used as the first extension priority
 | |
| 					to be recognized as a parking access extension for blind transfers.
 | |
| 					Blind transfers and the DTMF one-touch parking feature need this
 | |
| 					distinction to operate properly. The parking access extension in
 | |
| 					this case is treated like a dialplan hint.
 | |
| 				</para>
 | |
| 			</note>
 | |
| 		</description>
 | |
| 		<see-also>
 | |
| 			<ref type="application">ParkedCall</ref>
 | |
| 		</see-also>
 | |
| 	</application>
 | |
| 
 | |
| 	<application name="ParkedCall" language="en_US">
 | |
| 		<synopsis>
 | |
| 			Retrieve a parked call.
 | |
| 		</synopsis>
 | |
| 		<syntax>
 | |
| 			<parameter name="parking_lot_name">
 | |
| 				<para>Specify from which parking lot to retrieve a parked call.</para>
 | |
| 				<para>The parking lot used is selected in the following order:</para>
 | |
| 				<para>1) parking_lot_name option</para>
 | |
| 				<para>2) <variable>PARKINGLOT</variable> variable</para>
 | |
| 				<para>3) <literal>CHANNEL(parkinglot)</literal> function
 | |
| 				(Possibly preset by the channel driver.)</para>
 | |
| 				<para>4) Default parking lot.</para>
 | |
| 			</parameter>
 | |
| 			<parameter name="parking_space">
 | |
| 				<para>Parking space to retrieve a parked call from.
 | |
| 				If not provided then the first available parked call in the
 | |
| 				parking lot will be retrieved.</para>
 | |
| 			</parameter>
 | |
| 		</syntax>
 | |
| 		<description>
 | |
| 			<para>Used to retrieve a parked call from a parking lot.</para>
 | |
| 			<note>
 | |
| 				<para>If a parking lot's parkext option is set, then Parking lots
 | |
| 				will automatically create and manage dialplan extensions in
 | |
| 				the parking lot context. If that is the case then you will not
 | |
| 				need to manage parking extensions yourself, just include the
 | |
| 				parking context of the parking lot.</para>
 | |
| 			</note>
 | |
| 		</description>
 | |
| 		<see-also>
 | |
| 			<ref type="application">Park</ref>
 | |
| 		</see-also>
 | |
| 	</application>
 | |
| 
 | |
| 	<application name="ParkAndAnnounce" language="en_US">
 | |
| 		<synopsis>
 | |
| 			Park and Announce.
 | |
| 		</synopsis>
 | |
| 		<syntax>
 | |
| 			<parameter name="parking_lot_name">
 | |
| 				<para>Specify in which parking lot to park a call.</para>
 | |
| 				<para>The parking lot used is selected in the following order:</para>
 | |
| 				<para>1) parking_lot_name option to this application</para>
 | |
| 				<para>2) <variable>PARKINGLOT</variable> variable</para>
 | |
| 				<para>3) <literal>CHANNEL(parkinglot)</literal> function
 | |
| 				(Possibly preset by the channel driver.)</para>
 | |
| 				<para>4) Default parking lot.</para>
 | |
| 			</parameter>
 | |
| 			<parameter name="options">
 | |
| 				<para>A list of options for this parked call.</para>
 | |
| 				<optionlist>
 | |
| 					<option name="r">
 | |
| 						<para>Send ringing instead of MOH to the parked call.</para>
 | |
| 					</option>
 | |
| 					<option name="R">
 | |
| 						<para>Randomize the selection of a parking space.</para>
 | |
| 					</option>
 | |
| 					<option name="c" argsep=",">
 | |
| 						<argument name="context" required="false" />
 | |
| 						<argument name="extension" required="false" />
 | |
| 						<argument name="priority" required="true" />
 | |
| 						<para>If the parking times out, go to this place in the dialplan
 | |
| 							instead of where the parking lot defines the call should go.
 | |
| 						</para>
 | |
| 					</option>
 | |
| 					<option name="t">
 | |
| 						<argument name="duration" required="true" />
 | |
| 						<para>Use a timeout of <literal>duration</literal> seconds instead
 | |
| 							of the timeout specified by the parking lot.</para>
 | |
| 					</option>
 | |
| 				</optionlist>
 | |
| 			</parameter>
 | |
| 			<parameter name="announce_template" required="true" argsep=":">
 | |
| 				<argument name="announce" required="true">
 | |
| 					<para>Colon-separated list of files to announce. The word
 | |
| 					<literal>PARKED</literal> will be replaced by a say_digits of the extension in which
 | |
| 					the call is parked.</para>
 | |
| 				</argument>
 | |
| 				<argument name="announce1" multiple="true" />
 | |
| 			</parameter>
 | |
| 			<parameter name="dial" required="true">
 | |
| 				<para>The app_dial style resource to call to make the
 | |
| 				announcement. Console/dsp calls the console.</para>
 | |
| 			</parameter>
 | |
| 		</syntax>
 | |
| 		<description>
 | |
| 			<para>Park a call into the parkinglot and announce the call to another channel.</para>
 | |
| 			<para>The variable <variable>PARKEDAT</variable> will contain the parking extension
 | |
| 			into which the call was placed.  Use with the Local channel to allow the dialplan to make
 | |
| 			use of this information.</para>
 | |
| 		</description>
 | |
| 		<see-also>
 | |
| 			<ref type="application">Park</ref>
 | |
| 			<ref type="application">ParkedCall</ref>
 | |
| 		</see-also>
 | |
| 	</application>
 | |
|  ***/
 | |
| 
 | |
| #define PARK_AND_ANNOUNCE_APPLICATION "ParkAndAnnounce"
 | |
| 
 | |
| /* Park a call */
 | |
| 
 | |
| enum park_args {
 | |
| 	OPT_ARG_COMEBACK,
 | |
| 	OPT_ARG_TIMEOUT,
 | |
| 	OPT_ARG_MUSICONHOLD,
 | |
| 	OPT_ARG_ARRAY_SIZE /* Always the last element of the enum */
 | |
| };
 | |
| 
 | |
| enum park_flags {
 | |
| 	MUXFLAG_RINGING = (1 << 0),
 | |
| 	MUXFLAG_RANDOMIZE = (1 << 1),
 | |
| 	MUXFLAG_NOANNOUNCE = (1 << 2),
 | |
| 	MUXFLAG_COMEBACK_OVERRIDE = (1 << 3),
 | |
| 	MUXFLAG_TIMEOUT_OVERRIDE = (1 << 4),
 | |
| 	MUXFLAG_MUSICONHOLD = (1 << 5),
 | |
| };
 | |
| 
 | |
| AST_APP_OPTIONS(park_opts, {
 | |
| 	AST_APP_OPTION('r', MUXFLAG_RINGING),
 | |
| 	AST_APP_OPTION('R', MUXFLAG_RANDOMIZE),
 | |
| 	AST_APP_OPTION('s', MUXFLAG_NOANNOUNCE),
 | |
| 	AST_APP_OPTION_ARG('c', MUXFLAG_COMEBACK_OVERRIDE, OPT_ARG_COMEBACK),
 | |
| 	AST_APP_OPTION_ARG('t', MUXFLAG_TIMEOUT_OVERRIDE, OPT_ARG_TIMEOUT),
 | |
| 	AST_APP_OPTION_ARG('m', MUXFLAG_MUSICONHOLD, OPT_ARG_MUSICONHOLD),
 | |
| });
 | |
| 
 | |
| static int apply_option_timeout (int *var, char *timeout_arg)
 | |
| {
 | |
| 	if (ast_strlen_zero(timeout_arg)) {
 | |
| 		ast_log(LOG_ERROR, "No duration value provided for the timeout ('t') option.\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	if (sscanf(timeout_arg, "%d", var) != 1 || *var < 0) {
 | |
| 		ast_log(LOG_ERROR, "Duration value provided for timeout ('t') option must be 0 or greater.\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int park_app_parse_data(const char *data, int *disable_announce, int *use_ringing, int *randomize, int *time_limit,
 | |
| 	char **comeback_override, char **lot_name, char **musicclass)
 | |
| {
 | |
| 	char *parse;
 | |
| 	struct ast_flags flags = { 0 };
 | |
| 
 | |
| 	AST_DECLARE_APP_ARGS(args,
 | |
| 		AST_APP_ARG(lot_name);
 | |
| 		AST_APP_ARG(options);
 | |
| 		AST_APP_ARG(other);	/* Any remaining unused arguments */
 | |
| 	);
 | |
| 
 | |
| 	parse = ast_strdupa(data);
 | |
| 	AST_STANDARD_APP_ARGS(args, parse);
 | |
| 
 | |
| 	if (args.options) {
 | |
| 		char *opts[OPT_ARG_ARRAY_SIZE] = { NULL, };
 | |
| 		ast_app_parse_options(park_opts, &flags, opts, args.options);
 | |
| 		if (ast_test_flag(&flags, MUXFLAG_TIMEOUT_OVERRIDE)) {
 | |
| 			if (apply_option_timeout(time_limit, opts[OPT_ARG_TIMEOUT])) {
 | |
| 				return -1;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if (ast_test_flag(&flags, MUXFLAG_COMEBACK_OVERRIDE)) {
 | |
| 			*comeback_override = ast_strdup(opts[OPT_ARG_COMEBACK]);
 | |
| 		}
 | |
| 
 | |
| 		if (ast_test_flag(&flags, MUXFLAG_NOANNOUNCE)) {
 | |
| 			if (disable_announce) {
 | |
| 				*disable_announce = 1;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if (ast_test_flag(&flags, MUXFLAG_MUSICONHOLD)) {
 | |
| 			*musicclass = ast_strdup(opts[OPT_ARG_MUSICONHOLD]);
 | |
| 		}
 | |
| 
 | |
| 		if (ast_test_flag(&flags, MUXFLAG_RINGING)) {
 | |
| 			*use_ringing = 1;
 | |
| 		}
 | |
| 
 | |
| 		if (ast_test_flag(&flags, MUXFLAG_RANDOMIZE)) {
 | |
| 			*randomize = 1;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (!ast_strlen_zero(args.lot_name)) {
 | |
| 		*lot_name = ast_strdup(args.lot_name);
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| void park_common_datastore_free(struct park_common_datastore *datastore)
 | |
| {
 | |
| 	if (!datastore) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	ast_free(datastore->parker_uuid);
 | |
| 	ast_free(datastore->parker_dial_string);
 | |
| 	ast_free(datastore->comeback_override);
 | |
| 	ast_free(datastore);
 | |
| }
 | |
| 
 | |
| static void park_common_datastore_destroy(void *data)
 | |
| {
 | |
| 	struct park_common_datastore *datastore = data;
 | |
| 	park_common_datastore_free(datastore);
 | |
| }
 | |
| 
 | |
| static const struct ast_datastore_info park_common_info = {
 | |
| 	.type = "park entry data",
 | |
| 	.destroy = park_common_datastore_destroy,
 | |
| };
 | |
| 
 | |
| static void wipe_park_common_datastore(struct ast_channel *chan)
 | |
| {
 | |
| 	struct ast_datastore *datastore;
 | |
| 
 | |
| 	ast_channel_lock(chan);
 | |
| 	datastore = ast_channel_datastore_find(chan, &park_common_info, NULL);
 | |
| 	if (datastore) {
 | |
| 		ast_channel_datastore_remove(chan, datastore);
 | |
| 		ast_datastore_free(datastore);
 | |
| 	}
 | |
| 	ast_channel_unlock(chan);
 | |
| }
 | |
| 
 | |
| static int setup_park_common_datastore(struct ast_channel *parkee, const char *parker_uuid, const char *comeback_override, int randomize, int time_limit, int silence_announce)
 | |
| {
 | |
| 	struct ast_datastore *datastore = NULL;
 | |
| 	struct park_common_datastore *park_datastore;
 | |
| 	const char *attended_transfer;
 | |
| 	const char *blind_transfer;
 | |
| 	char *parker_dial_string = NULL;
 | |
| 
 | |
| 	wipe_park_common_datastore(parkee);
 | |
| 
 | |
| 	if (!(datastore = ast_datastore_alloc(&park_common_info, NULL))) {
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	if (!(park_datastore = ast_calloc(1, sizeof(*park_datastore)))) {
 | |
| 		ast_datastore_free(datastore);
 | |
| 		return -1;
 | |
| 	}
 | |
| 	datastore->data = park_datastore;
 | |
| 
 | |
| 	park_datastore->parker_uuid = ast_strdup(parker_uuid);
 | |
| 	if (!park_datastore->parker_uuid) {
 | |
| 		ast_datastore_free(datastore);
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	ast_channel_lock(parkee);
 | |
| 	attended_transfer = pbx_builtin_getvar_helper(parkee, "ATTENDEDTRANSFER");
 | |
| 	blind_transfer = pbx_builtin_getvar_helper(parkee, "BLINDTRANSFER");
 | |
| 	if (!ast_strlen_zero(attended_transfer)) {
 | |
| 		parker_dial_string = ast_strdupa(attended_transfer);
 | |
| 	} else if (!ast_strlen_zero(blind_transfer)) {
 | |
| 		parker_dial_string = ast_strdupa(blind_transfer);
 | |
| 		/* Ensure that attended_transfer is NULL and not an empty string. */
 | |
| 		attended_transfer = NULL;
 | |
| 	}
 | |
| 	ast_channel_unlock(parkee);
 | |
| 
 | |
| 	if (!ast_strlen_zero(parker_dial_string)) {
 | |
| 		ast_channel_name_to_dial_string(parker_dial_string);
 | |
| 		ast_verb(4, "Setting Parker dial string to %s from %s value\n",
 | |
| 			parker_dial_string,
 | |
| 			attended_transfer ? "ATTENDEDTRANSFER" : "BLINDTRANSFER");
 | |
| 		park_datastore->parker_dial_string = ast_strdup(parker_dial_string);
 | |
| 		if (!park_datastore->parker_dial_string) {
 | |
| 			ast_datastore_free(datastore);
 | |
| 			return -1;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	park_datastore->randomize = randomize;
 | |
| 	park_datastore->time_limit = time_limit;
 | |
| 	park_datastore->silence_announce = silence_announce;
 | |
| 
 | |
| 	if (comeback_override) {
 | |
| 		park_datastore->comeback_override = ast_strdup(comeback_override);
 | |
| 		if (!park_datastore->comeback_override) {
 | |
| 			ast_datastore_free(datastore);
 | |
| 			return -1;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	ast_channel_lock(parkee);
 | |
| 	ast_channel_datastore_add(parkee, datastore);
 | |
| 	ast_channel_unlock(parkee);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| struct park_common_datastore *get_park_common_datastore_copy(struct ast_channel *parkee)
 | |
| {
 | |
| 	struct ast_datastore *datastore;
 | |
| 	struct park_common_datastore *data;
 | |
| 	struct park_common_datastore *data_copy;
 | |
| 
 | |
| 	SCOPED_CHANNELLOCK(lock, parkee);
 | |
| 
 | |
| 	if (!(datastore = ast_channel_datastore_find(parkee, &park_common_info, NULL))) {
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	data = datastore->data;
 | |
| 
 | |
| 	/* This data should always be populated if this datastore was appended to the channel */
 | |
| 	ast_assert(data != NULL);
 | |
| 
 | |
| 	data_copy = ast_calloc(1, sizeof(*data_copy));
 | |
| 	if (!data_copy) {
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	data_copy->parker_uuid = ast_strdup(data->parker_uuid);
 | |
| 	if (!data_copy->parker_uuid) {
 | |
| 		park_common_datastore_free(data_copy);
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	data_copy->randomize = data->randomize;
 | |
| 	data_copy->time_limit = data->time_limit;
 | |
| 	data_copy->silence_announce = data->silence_announce;
 | |
| 
 | |
| 	if (data->comeback_override) {
 | |
| 		data_copy->comeback_override = ast_strdup(data->comeback_override);
 | |
| 		if (!data_copy->comeback_override) {
 | |
| 			park_common_datastore_free(data_copy);
 | |
| 			return NULL;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (data->parker_dial_string) {
 | |
| 		data_copy->parker_dial_string = ast_strdup(data->parker_dial_string);
 | |
| 		if (!data_copy->parker_dial_string) {
 | |
| 			park_common_datastore_free(data_copy);
 | |
| 			return NULL;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return data_copy;
 | |
| }
 | |
| 
 | |
| static struct ast_bridge *park_common_setup2(struct ast_channel *parkee, struct ast_channel *parker,
 | |
| 		const char *lot_name, const char *comeback_override, const char *musicclass,
 | |
| 		int use_ringing, int randomize, int time_limit, int silence_announcements)
 | |
| {
 | |
| 	struct ast_bridge *parking_bridge;
 | |
| 	RAII_VAR(struct parking_lot *, lot, NULL, ao2_cleanup);
 | |
| 
 | |
| 	if (!parker) {
 | |
| 		parker = parkee;
 | |
| 	}
 | |
| 
 | |
| 	/* If the name of the parking lot isn't specified in the arguments, find it based on the channel. */
 | |
| 	if (ast_strlen_zero(lot_name)) {
 | |
| 		ast_channel_lock(parker);
 | |
| 		lot_name = ast_strdupa(find_channel_parking_lot_name(parker));
 | |
| 		ast_channel_unlock(parker);
 | |
| 	}
 | |
| 
 | |
| 	lot = parking_lot_find_by_name(lot_name);
 | |
| 	if (!lot) {
 | |
| 		lot = parking_create_dynamic_lot(lot_name, parker);
 | |
| 	}
 | |
| 	if (!lot) {
 | |
| 		ast_log(LOG_ERROR, "Could not find parking lot: '%s'\n", lot_name);
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	ao2_lock(lot);
 | |
| 	parking_bridge = parking_lot_get_bridge(lot);
 | |
| 	ao2_unlock(lot);
 | |
| 
 | |
| 	if (!parking_bridge) {
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	/* Apply relevant bridge roles and such to the parking channel */
 | |
| 	parking_channel_set_roles(parkee, lot, use_ringing);
 | |
| 	/* If requested, override the MOH class */
 | |
| 	if (!ast_strlen_zero(musicclass)) {
 | |
| 		ast_channel_set_bridge_role_option(parkee, "holding_participant", "moh_class", musicclass);
 | |
| 	}
 | |
| 	setup_park_common_datastore(parkee, ast_channel_uniqueid(parker), comeback_override, randomize, time_limit,
 | |
| 		silence_announcements);
 | |
| 	return parking_bridge;
 | |
| }
 | |
| 
 | |
| struct ast_bridge *park_common_setup(struct ast_channel *parkee, struct ast_channel *parker,
 | |
| 		const char *lot_name, const char *comeback_override,
 | |
| 		int use_ringing, int randomize, int time_limit, int silence_announcements)
 | |
| {
 | |
| 	return park_common_setup2(parkee, parker, lot_name, comeback_override, NULL, use_ringing, randomize, time_limit, silence_announcements);
 | |
| }
 | |
| 
 | |
| struct ast_bridge *park_application_setup(struct ast_channel *parkee, struct ast_channel *parker, const char *app_data,
 | |
| 		int *silence_announcements)
 | |
| {
 | |
| 	int use_ringing = 0;
 | |
| 	int randomize = 0;
 | |
| 	int time_limit = -1;
 | |
| 
 | |
| 	RAII_VAR(char *, comeback_override, NULL, ast_free);
 | |
| 	RAII_VAR(char *, lot_name_app_arg, NULL, ast_free);
 | |
| 	RAII_VAR(char *, musicclass, NULL, ast_free);
 | |
| 
 | |
| 	if (app_data) {
 | |
| 		park_app_parse_data(app_data, silence_announcements, &use_ringing, &randomize, &time_limit, &comeback_override, &lot_name_app_arg, &musicclass);
 | |
| 	}
 | |
| 
 | |
| 	return park_common_setup2(parkee, parker, lot_name_app_arg, comeback_override, musicclass, use_ringing,
 | |
| 		randomize, time_limit, silence_announcements ? *silence_announcements : 0);
 | |
| 
 | |
| }
 | |
| 
 | |
| static int park_app_exec(struct ast_channel *chan, const char *data)
 | |
| {
 | |
| 	RAII_VAR(struct ast_bridge *, parking_bridge, NULL, ao2_cleanup);
 | |
| 
 | |
| 	struct ast_bridge_features chan_features;
 | |
| 	int res;
 | |
| 	int silence_announcements = 0;
 | |
| 	int blind_transfer;
 | |
| 
 | |
| 	/* Answer the channel if needed */
 | |
| 	if (ast_channel_state(chan) != AST_STATE_UP) {
 | |
| 		ast_answer(chan);
 | |
| 	}
 | |
| 
 | |
| 	ast_channel_lock(chan);
 | |
| 	blind_transfer = !ast_strlen_zero(pbx_builtin_getvar_helper(chan, "BLINDTRANSFER"));
 | |
| 	ast_channel_unlock(chan);
 | |
| 
 | |
| 	/* Handle the common parking setup stuff */
 | |
| 	if (!(parking_bridge = park_application_setup(chan, NULL, data, &silence_announcements))) {
 | |
| 		if (!silence_announcements && !blind_transfer) {
 | |
| 			ast_stream_and_wait(chan, "pbx-parkingfailed", "");
 | |
| 		}
 | |
| 		publish_parked_call_failure(chan);
 | |
| 		return 0;
 | |
| 	}
 | |
| 
 | |
| 	/* Initialize bridge features for the channel. */
 | |
| 	res = ast_bridge_features_init(&chan_features);
 | |
| 	if (res) {
 | |
| 		ast_bridge_features_cleanup(&chan_features);
 | |
| 		publish_parked_call_failure(chan);
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	/* Now for the fun part... park it! */
 | |
| 	ast_bridge_join(parking_bridge, chan, NULL, &chan_features, NULL, 0);
 | |
| 
 | |
| 	/*
 | |
| 	 * If the bridge was broken for a hangup that isn't real, then
 | |
| 	 * don't run the h extension, because the channel isn't really
 | |
| 	 * hung up.  This should only happen with AST_SOFTHANGUP_ASYNCGOTO.
 | |
| 	 */
 | |
| 	res = -1;
 | |
| 
 | |
| 	ast_channel_lock(chan);
 | |
| 	if (ast_channel_softhangup_internal_flag(chan) & AST_SOFTHANGUP_ASYNCGOTO) {
 | |
| 		res = 0;
 | |
| 	}
 | |
| 	ast_channel_unlock(chan);
 | |
| 
 | |
| 	ast_bridge_features_cleanup(&chan_features);
 | |
| 
 | |
| 	return res;
 | |
| }
 | |
| 
 | |
| /* Retrieve a parked call */
 | |
| 
 | |
| static int parked_call_app_exec(struct ast_channel *chan, const char *data)
 | |
| {
 | |
| 	RAII_VAR(struct parking_lot *, lot, NULL, ao2_cleanup);
 | |
| 	RAII_VAR(struct parked_user *, pu, NULL, ao2_cleanup); /* Parked user being retrieved */
 | |
| 	struct ast_bridge *retrieval_bridge;
 | |
| 	int res;
 | |
| 	int target_space = -1;
 | |
| 	struct ast_bridge_features chan_features;
 | |
| 	char *parse;
 | |
| 	const char *lot_name;
 | |
| 
 | |
| 	AST_DECLARE_APP_ARGS(args,
 | |
| 		AST_APP_ARG(lot_name);
 | |
| 		AST_APP_ARG(parking_space);
 | |
| 		AST_APP_ARG(other);	/* Any remaining unused arguments */
 | |
| 	);
 | |
| 
 | |
| 	parse = ast_strdupa(data);
 | |
| 	AST_STANDARD_APP_ARGS(args, parse);
 | |
| 
 | |
| 	/* Answer the channel if needed */
 | |
| 	if (ast_channel_state(chan) != AST_STATE_UP) {
 | |
| 		ast_answer(chan);
 | |
| 	}
 | |
| 
 | |
| 	lot_name = args.lot_name;
 | |
| 
 | |
| 	/* If the name of the parking lot isn't in the arguments, find it based on the channel. */
 | |
| 	if (ast_strlen_zero(lot_name)) {
 | |
| 		ast_channel_lock(chan);
 | |
| 		lot_name = ast_strdupa(find_channel_parking_lot_name(chan));
 | |
| 		ast_channel_unlock(chan);
 | |
| 	}
 | |
| 
 | |
| 	lot = parking_lot_find_by_name(lot_name);
 | |
| 	if (!lot) {
 | |
| 		ast_log(LOG_ERROR, "Could not find the requested parking lot\n");
 | |
| 		ast_stream_and_wait(chan, "pbx-invalidpark", "");
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	if (!ast_strlen_zero(args.parking_space)) {
 | |
| 		if (sscanf(args.parking_space, "%d", &target_space) != 1 || target_space < 0) {
 | |
| 			ast_stream_and_wait(chan, "pbx-invalidpark", "");
 | |
| 			ast_log(LOG_ERROR, "value '%s' for parking_space argument is invalid. Must be an integer greater than 0.\n", args.parking_space);
 | |
| 			return -1;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/* Attempt to get the parked user from the parking lot */
 | |
| 	pu = parking_lot_retrieve_parked_user(lot, target_space);
 | |
| 	if (!pu) {
 | |
| 		ast_stream_and_wait(chan, "pbx-invalidpark", "");
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	/* The parked call needs to know who is retrieving it before we move it out of the parking bridge */
 | |
| 	ast_assert(pu->retriever == NULL);
 | |
| 	pu->retriever = ast_channel_snapshot_create(chan);
 | |
| 
 | |
| 	/* Create bridge */
 | |
| 	retrieval_bridge = ast_bridge_basic_new();
 | |
| 	if (!retrieval_bridge) {
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	/* Move the parkee into the new bridge */
 | |
| 	if (ast_bridge_move(retrieval_bridge, lot->parking_bridge, pu->chan, NULL, 0)) {
 | |
| 		ast_bridge_destroy(retrieval_bridge, 0);
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	/* Initialize our bridge features */
 | |
| 	res = ast_bridge_features_init(&chan_features);
 | |
| 	if (res) {
 | |
| 		ast_bridge_destroy(retrieval_bridge, 0);
 | |
| 		ast_bridge_features_cleanup(&chan_features);
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	/* Set the features */
 | |
| 	parked_call_retrieve_enable_features(chan, lot, AST_FEATURE_FLAG_BYCALLER);
 | |
| 
 | |
| 	/* If the parkedplay option is set for the caller to hear, play that tone now. */
 | |
| 	if (lot->cfg->parkedplay & AST_FEATURE_FLAG_BYCALLER) {
 | |
| 		ast_stream_and_wait(chan, lot->cfg->courtesytone, NULL);
 | |
| 	}
 | |
| 
 | |
| 	/* Now we should try to join the new bridge ourselves... */
 | |
| 	ast_bridge_join(retrieval_bridge, chan, NULL, &chan_features, NULL,
 | |
| 		AST_BRIDGE_JOIN_PASS_REFERENCE);
 | |
| 
 | |
| 	ast_bridge_features_cleanup(&chan_features);
 | |
| 
 | |
| 	/* Return -1 so that call does not continue in the dialplan. This is to make
 | |
| 	 * behavior consistent with Asterisk versions prior to 12.
 | |
| 	 */
 | |
| 	return -1;
 | |
| }
 | |
| 
 | |
| struct park_announce_subscription_data {
 | |
| 	char *parkee_uuid;
 | |
| 	char *dial_string;
 | |
| 	char *announce_string;
 | |
| };
 | |
| 
 | |
| static void park_announce_subscription_data_destroy(void *data)
 | |
| {
 | |
| 	struct park_announce_subscription_data *pa_data = data;
 | |
| 	ast_free(pa_data->parkee_uuid);
 | |
| 	ast_free(pa_data->dial_string);
 | |
| 	ast_free(pa_data->announce_string);
 | |
| 	ast_free(pa_data);
 | |
| }
 | |
| 
 | |
| static struct park_announce_subscription_data *park_announce_subscription_data_create(const char *parkee_uuid,
 | |
| 		const char *dial_string,
 | |
| 		const char *announce_string)
 | |
| {
 | |
| 	struct park_announce_subscription_data *pa_data;
 | |
| 
 | |
| 	if (!(pa_data = ast_calloc(1, sizeof(*pa_data)))) {
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	if (!(pa_data->parkee_uuid = ast_strdup(parkee_uuid))
 | |
| 		|| !(pa_data->dial_string = ast_strdup(dial_string))
 | |
| 		|| !(pa_data->announce_string = ast_strdup(announce_string))) {
 | |
| 		park_announce_subscription_data_destroy(pa_data);
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	return pa_data;
 | |
| }
 | |
| 
 | |
| /*! \internal
 | |
|  * \brief Gathers inheritable channel variables from a channel by name.
 | |
|  *
 | |
|  * \param oh outgoing helper struct we are bestowing inheritable variables to
 | |
|  * \param channel_id name or uniqueID of the channel to inherit variables from
 | |
|  */
 | |
| static void inherit_channel_vars_from_id(struct outgoing_helper *oh, const char *channel_id)
 | |
| {
 | |
| 	struct ast_channel *chan = ast_channel_get_by_name(channel_id);
 | |
| 	struct ast_var_t *current;
 | |
| 	struct ast_variable *newvar;
 | |
| 	const char *varname;
 | |
| 	int vartype;
 | |
| 
 | |
| 
 | |
| 	if (!chan) {
 | |
| 		/* Already gone */
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	ast_channel_lock(chan);
 | |
| 
 | |
| 	AST_LIST_TRAVERSE(ast_channel_varshead((struct ast_channel *) chan), current, entries) {
 | |
| 		varname = ast_var_full_name(current);
 | |
| 		if (!varname) {
 | |
| 			continue;
 | |
| 		}
 | |
| 
 | |
| 		vartype = 0;
 | |
| 		if (varname[0] == '_') {
 | |
| 			vartype = 1;
 | |
| 			if (varname[1] == '_') {
 | |
| 				vartype = 2;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		switch (vartype) {
 | |
| 		case 1:
 | |
| 			newvar = ast_variable_new(&varname[1], ast_var_value(current), "");
 | |
| 			break;
 | |
| 		case 2:
 | |
| 			newvar = ast_variable_new(varname, ast_var_value(current), "");
 | |
| 			break;
 | |
| 		default:
 | |
| 			continue;
 | |
| 		}
 | |
| 		if (newvar) {
 | |
| 			ast_debug(1, "Inheriting variable %s from %s.\n",
 | |
| 				newvar->name, ast_channel_name(chan));
 | |
| 			if (oh->vars) {
 | |
| 				newvar->next = oh->vars;
 | |
| 				oh->vars = newvar;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	ast_channel_unlock(chan);
 | |
| 	ast_channel_cleanup(chan);
 | |
| }
 | |
| 
 | |
| static void announce_to_dial(char *dial_string, char *announce_string, int parkingspace, struct ast_channel_snapshot *parkee_snapshot)
 | |
| {
 | |
| 	struct ast_channel *dchan;
 | |
| 	struct outgoing_helper oh = { 0, };
 | |
| 	int outstate;
 | |
| 	struct ast_format_cap *cap_slin = ast_format_cap_alloc(AST_FORMAT_CAP_FLAG_DEFAULT);
 | |
| 	char buf[13];
 | |
| 	char *dial_tech;
 | |
| 	char *cur_announce;
 | |
| 
 | |
| 	dial_tech = strsep(&dial_string, "/");
 | |
| 	ast_verb(3, "Dial Tech,String: (%s,%s)\n", dial_tech, dial_string);
 | |
| 
 | |
| 	if (!cap_slin) {
 | |
| 		ast_log(LOG_WARNING, "PARK: Failed to announce park.\n");
 | |
| 		goto announce_cleanup;
 | |
| 	}
 | |
| 	ast_format_cap_append(cap_slin, ast_format_slin, 0);
 | |
| 
 | |
| 	snprintf(buf, sizeof(buf), "%d", parkingspace);
 | |
| 	oh.vars = ast_variable_new("_PARKEDAT", buf, "");
 | |
| 
 | |
| 	inherit_channel_vars_from_id(&oh, parkee_snapshot->base->uniqueid);
 | |
| 
 | |
| 	dchan = __ast_request_and_dial(dial_tech, cap_slin, NULL, NULL, dial_string, 30000,
 | |
| 		&outstate,
 | |
| 		parkee_snapshot->caller->number,
 | |
| 		parkee_snapshot->caller->name,
 | |
| 		&oh);
 | |
| 
 | |
| 	ast_variables_destroy(oh.vars);
 | |
| 	if (!dchan) {
 | |
| 		ast_log(LOG_WARNING, "PARK: Unable to allocate announce channel.\n");
 | |
| 		goto announce_cleanup;
 | |
| 	}
 | |
| 
 | |
| 	ast_verb(4, "Announce Template: %s\n", announce_string);
 | |
| 
 | |
| 	for (cur_announce = strsep(&announce_string, ":"); cur_announce; cur_announce = strsep(&announce_string, ":")) {
 | |
| 		ast_verb(4, "Announce:%s\n", cur_announce);
 | |
| 		if (!strcmp(cur_announce, "PARKED")) {
 | |
| 			ast_say_digits(dchan, parkingspace, "", ast_channel_language(dchan));
 | |
| 		} else {
 | |
| 			int dres = ast_streamfile(dchan, cur_announce, ast_channel_language(dchan));
 | |
| 			if (!dres) {
 | |
| 				dres = ast_waitstream(dchan, "");
 | |
| 			} else {
 | |
| 				ast_log(LOG_WARNING, "ast_streamfile of %s failed on %s\n", cur_announce, ast_channel_name(dchan));
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	ast_stopstream(dchan);
 | |
| 	ast_hangup(dchan);
 | |
| 
 | |
| announce_cleanup:
 | |
| 	ao2_cleanup(cap_slin);
 | |
| }
 | |
| 
 | |
| static void park_announce_update_cb(void *data, struct stasis_subscription *sub, struct stasis_message *message)
 | |
| {
 | |
| 	struct park_announce_subscription_data *pa_data = data;
 | |
| 	char *dial_string = pa_data->dial_string;
 | |
| 
 | |
| 	struct ast_parked_call_payload *payload = stasis_message_data(message);
 | |
| 
 | |
| 	if (stasis_subscription_final_message(sub, message)) {
 | |
| 		park_announce_subscription_data_destroy(data);
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	if (ast_parked_call_type() != stasis_message_type(message)) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	if (payload->event_type != PARKED_CALL) {
 | |
| 		/* We are only concerned with calls parked */
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	if (strcmp(payload->parkee->base->uniqueid, pa_data->parkee_uuid)) {
 | |
| 		/* We are only concerned with the parkee we are subscribed for. */
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	if (!ast_strlen_zero(dial_string)) {
 | |
| 		announce_to_dial(dial_string, pa_data->announce_string, payload->parkingspace, payload->parkee);
 | |
| 	}
 | |
| 
 | |
| 	*dial_string = '\0'; /* If we observe this dial string on a second pass, we don't want to do anything with it. */
 | |
| }
 | |
| 
 | |
| static int park_and_announce_app_exec(struct ast_channel *chan, const char *data)
 | |
| {
 | |
| 	struct ast_bridge_features chan_features;
 | |
| 	char *parse;
 | |
| 	int res;
 | |
| 	int silence_announcements = 1;
 | |
| 
 | |
| 	struct stasis_subscription *parking_subscription;
 | |
| 	struct park_announce_subscription_data *pa_data;
 | |
| 
 | |
| 	RAII_VAR(struct ast_bridge *, parking_bridge, NULL, ao2_cleanup);
 | |
| 
 | |
| 	AST_DECLARE_APP_ARGS(args,
 | |
| 		AST_APP_ARG(lot_name);
 | |
| 		AST_APP_ARG(options);
 | |
| 		AST_APP_ARG(announce_template);
 | |
| 		AST_APP_ARG(dial);
 | |
| 		AST_APP_ARG(others);/* Any remaining unused arguments */
 | |
| 	);
 | |
| 
 | |
| 	if (ast_strlen_zero(data)) {
 | |
| 		ast_log(LOG_ERROR, "ParkAndAnnounce has required arguments. No arguments were provided.\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	parse = ast_strdupa(data);
 | |
| 	AST_STANDARD_APP_ARGS(args, parse);
 | |
| 
 | |
| 	if (ast_strlen_zero(args.announce_template)) {
 | |
| 		/* improperly configured arguments for the application */
 | |
| 		ast_log(LOG_ERROR, "ParkAndAnnounce requires the announce_template argument.\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	if (ast_strlen_zero(args.dial)) {
 | |
| 		/* improperly configured arguments */
 | |
| 		ast_log(LOG_ERROR, "ParkAndAnnounce requires the dial argument.\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	if (!strchr(args.dial, '/')) {
 | |
| 		ast_log(LOG_ERROR, "ParkAndAnnounce dial string '%s' is improperly formed.\n", args.dial);
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	/* Handle the common parking setup stuff */
 | |
| 	if (!(parking_bridge = park_application_setup(chan, NULL, data, &silence_announcements))) {
 | |
| 		return 0;
 | |
| 	}
 | |
| 
 | |
| 	/* Initialize bridge features for the channel. */
 | |
| 	res = ast_bridge_features_init(&chan_features);
 | |
| 	if (res) {
 | |
| 		ast_bridge_features_cleanup(&chan_features);
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	/* subscribe to the parking message so that we can announce once it is parked */
 | |
| 	pa_data = park_announce_subscription_data_create(ast_channel_uniqueid(chan), args.dial, args.announce_template);
 | |
| 	if (!pa_data) {
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	if (!(parking_subscription = stasis_subscribe_pool(ast_parking_topic(), park_announce_update_cb, pa_data))) {
 | |
| 		/* Failed to create subscription */
 | |
| 		park_announce_subscription_data_destroy(pa_data);
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	stasis_subscription_accept_message_type(parking_subscription, ast_parked_call_type());
 | |
| 	stasis_subscription_accept_message_type(parking_subscription, stasis_subscription_change_type());
 | |
| 	stasis_subscription_set_filter(parking_subscription, STASIS_SUBSCRIPTION_FILTER_SELECTIVE);
 | |
| 
 | |
| 	/* Now for the fun part... park it! */
 | |
| 	ast_bridge_join(parking_bridge, chan, NULL, &chan_features, NULL, 0);
 | |
| 
 | |
| 	/* Toss the subscription since we aren't bridged at this point. */
 | |
| 	stasis_unsubscribe(parking_subscription);
 | |
| 
 | |
| 	/*
 | |
| 	 * If the bridge was broken for a hangup that isn't real, then
 | |
| 	 * don't run the h extension, because the channel isn't really
 | |
| 	 * hung up.  This should only happen with AST_SOFTHANGUP_ASYNCGOTO.
 | |
| 	 */
 | |
| 	res = -1;
 | |
| 
 | |
| 	ast_channel_lock(chan);
 | |
| 	if (ast_channel_softhangup_internal_flag(chan) & AST_SOFTHANGUP_ASYNCGOTO) {
 | |
| 		res = 0;
 | |
| 	}
 | |
| 	ast_channel_unlock(chan);
 | |
| 
 | |
| 	ast_bridge_features_cleanup(&chan_features);
 | |
| 
 | |
| 	return res;
 | |
| }
 | |
| 
 | |
| int load_parking_applications(void)
 | |
| {
 | |
| 	if (ast_register_application_xml(PARK_APPLICATION, park_app_exec)) {
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	if (ast_register_application_xml(PARKED_CALL_APPLICATION, parked_call_app_exec)) {
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	if (ast_register_application_xml(PARK_AND_ANNOUNCE_APPLICATION, park_and_announce_app_exec)) {
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| void unload_parking_applications(void)
 | |
| {
 | |
| 	ast_unregister_application(PARK_APPLICATION);
 | |
| 	ast_unregister_application(PARKED_CALL_APPLICATION);
 | |
| 	ast_unregister_application(PARK_AND_ANNOUNCE_APPLICATION);
 | |
| }
 |