2 * Asterisk -- An open source telephony toolkit.
4 * Copyright (C) 2013, Digium, Inc.
6 * Jonathan Rose <jrose@digium.com>
8 * See http://www.asterisk.org for more information about
9 * the Asterisk project. Please do not directly contact
10 * any of the maintainers of this project for assistance;
11 * the project provides a web site, mailing lists and IRC
12 * channels for your use.
14 * This program is free software, distributed under the terms of
15 * the GNU General Public License Version 2. See the LICENSE file
16 * at the top of the source tree.
21 * \brief Parking Bridge Class
23 * \author Jonathan Rose <jrose@digium.com>
27 #include "asterisk/logger.h"
28 #include "res_parking.h"
29 #include "asterisk/astobj2.h"
30 #include "asterisk/features.h"
31 #include "asterisk/say.h"
32 #include "asterisk/term.h"
34 struct ast_bridge_parking
36 struct ast_bridge base;
38 /* private stuff for parking */
39 struct parking_lot *lot;
44 * \brief ast_bridge parking class destructor
47 * \param self Bridge to operate upon.
49 * \note XXX Stub... and it might go unused.
53 static void bridge_parking_destroy(struct ast_bridge_parking *self)
55 ast_bridge_base_v_table.destroy(&self->base);
58 static void bridge_parking_dissolving(struct ast_bridge_parking *self)
61 ast_bridge_base_v_table.dissolving(&self->base);
64 static void destroy_parked_user(void *obj)
66 struct parked_user *pu = obj;
69 ao2_cleanup(pu->retriever);
70 ast_free(pu->parker_dial_string);
73 /* Only call this on a parked user that hasn't had its parker_dial_string set already */
74 static int parked_user_set_parker_dial_string(struct parked_user *pu, struct ast_channel *parker)
76 char *dial_string = ast_strdupa(ast_channel_name(parker));
78 ast_channel_name_to_dial_string(dial_string);
79 pu->parker_dial_string = ast_strdup(dial_string);
81 if (!pu->parker_dial_string) {
91 * \brief Construct a parked_user struct assigned to the specified parking lot
93 * \param lot The parking lot we are assigning the user to
94 * \param parkee The channel being parked
95 * \param parker The channel performing the park operation (may be the same channel)
96 * \param parker_dial_string Takes priority over parker for setting the parker dial string if included
97 * \param use_random_space if true, prioritize using a random parking space instead
98 * of ${PARKINGEXTEN} and/or automatic assignment from the parking lot
99 * \param time_limit If using a custom timeout, this should be supplied so that the
100 * parked_user struct can provide this information for manager events. If <0,
101 * use the parking lot limit instead.
103 * \retval NULL on failure
104 * \retval reference to the parked user
106 * \note ao2_cleanup this reference when you are done using it or you'll cause leaks.
108 static struct parked_user *generate_parked_user(struct parking_lot *lot, struct ast_channel *chan, struct ast_channel *parker, const char *parker_dial_string, int use_random_space, int time_limit)
110 struct parked_user *new_parked_user;
111 int preferred_space = -1; /* Initialize to use parking lot defaults */
113 const char *parkingexten;
115 if (lot->mode == PARKINGLOT_DISABLED) {
116 ast_log(LOG_NOTICE, "Tried to park in a parking lot that is no longer able to be parked to.\n");
120 new_parked_user = ao2_alloc(sizeof(*new_parked_user), destroy_parked_user);
121 if (!new_parked_user) {
125 if (use_random_space) {
126 preferred_space = ast_random() % (lot->cfg->parking_stop - lot->cfg->parking_start + 1);
127 preferred_space += lot->cfg->parking_start;
129 ast_channel_lock(chan);
130 if ((parkingexten = pbx_builtin_getvar_helper(chan, "PARKINGEXTEN"))) {
131 parkingexten = ast_strdupa(parkingexten);
133 ast_channel_unlock(chan);
135 if (!ast_strlen_zero(parkingexten)) {
136 if (sscanf(parkingexten, "%30d", &preferred_space) != 1 || preferred_space <= 0) {
137 ast_log(LOG_WARNING, "PARKINGEXTEN='%s' is not a valid parking space.\n", parkingexten);
138 ao2_ref(new_parked_user, -1);
144 /* We need to keep the lot locked between parking_lot_get_space and actually placing it in the lot. Or until we decide not to. */
147 parking_space = parking_lot_get_space(lot, preferred_space);
148 if (parking_space == -1) {
149 ast_log(LOG_NOTICE, "Failed to get parking space in lot '%s'. All full.\n", lot->name);
150 ao2_ref(new_parked_user, -1);
155 lot->next_space = ((parking_space + 1) - lot->cfg->parking_start) % (lot->cfg->parking_stop - lot->cfg->parking_start + 1) + lot->cfg->parking_start;
156 new_parked_user->chan = chan;
157 new_parked_user->parking_space = parking_space;
159 /* Have the parked user take a reference to the parking lot. This reference should be immutable and released at destruction */
160 new_parked_user->lot = lot;
163 new_parked_user->start = ast_tvnow();
164 new_parked_user->time_limit = (time_limit >= 0) ? time_limit : lot->cfg->parkingtime;
166 if (parker_dial_string) {
167 new_parked_user->parker_dial_string = ast_strdup(parker_dial_string);
169 if (parked_user_set_parker_dial_string(new_parked_user, parker)) {
170 ao2_ref(new_parked_user, -1);
176 if (!new_parked_user->parker_dial_string) {
177 ao2_ref(new_parked_user, -1);
182 /* Insert into the parking lot's parked user list. We can unlock the lot now. */
183 ao2_link(lot->parked_users, new_parked_user);
186 return new_parked_user;
189 /* TODO CEL events for parking */
193 * \brief ast_bridge parking push method.
196 * \param self Bridge to operate upon
197 * \param bridge_channel Bridge channel to push
198 * \param swap Bridge channel to swap places with if not NULL
200 * \note On entry, self is already locked
202 * \retval 0 on success
203 * \retval -1 on failure
205 static int bridge_parking_push(struct ast_bridge_parking *self, struct ast_bridge_channel *bridge_channel, struct ast_bridge_channel *swap)
207 struct parked_user *pu;
208 const char *blind_transfer;
209 RAII_VAR(struct ast_channel *, parker, NULL, ao2_cleanup); /* XXX replace with ast_channel_cleanup when available */
210 RAII_VAR(struct park_common_datastore *, park_datastore, NULL, park_common_datastore_free);
212 ast_bridge_base_v_table.push(&self->base, bridge_channel, swap);
214 ast_assert(self->lot != NULL);
216 /* Answer the channel if needed */
217 if (ast_channel_state(bridge_channel->chan) != AST_STATE_UP) {
218 ast_answer(bridge_channel->chan);
224 pu = swap->bridge_pvt;
226 /* This should be impossible since the only way a channel can enter in the first place
227 * is if it has a parked user associated with it */
228 publish_parked_call_failure(bridge_channel->chan);
233 /* Give the swap channel's parked user reference to the incoming channel */
234 pu->chan = bridge_channel->chan;
235 bridge_channel->bridge_pvt = pu;
236 swap->bridge_pvt = NULL;
238 /* TODO Add a parked call swap message type to relay information about parked channel swaps */
240 if (ast_bridge_channel_has_role(swap, "holding_participant")) {
241 const char *idle_mode = ast_bridge_channel_get_role_option(swap, "holding_participant", "idle_mode");
242 if (!ast_strlen_zero(idle_mode) && !strcmp(idle_mode, "ringing")) {
249 parking_set_duration(bridge_channel->features, pu);
251 if (parking_channel_set_roles(bridge_channel->chan, self->lot, use_ringing)) {
252 ast_log(LOG_WARNING, "Failed to apply holding bridge roles to %s while joining the parking lot.\n",
253 ast_channel_name(bridge_channel->chan));
259 if (!(park_datastore = get_park_common_datastore_copy(bridge_channel->chan))) {
260 /* There was either a failure to apply the datastore when performing park common setup or else we had alloc failures while cloning. Abort. */
263 parker = ast_channel_get_by_name(park_datastore->parker_uuid);
265 /* If the parker and the parkee are the same channel pointer, then the channel entered using
266 * the park application. It's possible that the channel that transferred it is still alive (particularly
267 * when a multichannel bridge is parked), so try to get the real parker if possible. */
268 ast_channel_lock(bridge_channel->chan);
269 blind_transfer = S_OR(pbx_builtin_getvar_helper(bridge_channel->chan, "BLINDTRANSFER"),
270 ast_channel_name(bridge_channel->chan));
271 if (blind_transfer) {
272 blind_transfer = ast_strdupa(blind_transfer);
274 ast_channel_unlock(bridge_channel->chan);
276 if (parker == bridge_channel->chan) {
277 struct ast_channel *real_parker = ast_channel_get_by_name(blind_transfer);
280 parker = real_parker;
284 pu = generate_parked_user(self->lot, bridge_channel->chan, parker,
285 park_datastore->parker_dial_string, park_datastore->randomize, park_datastore->time_limit);
288 publish_parked_call_failure(bridge_channel->chan);
292 /* If a comeback_override was provided, set it for the parked user's comeback string. */
293 if (park_datastore->comeback_override) {
294 strncpy(pu->comeback, park_datastore->comeback_override, sizeof(pu->comeback));
295 pu->comeback[sizeof(pu->comeback) - 1] = '\0';
298 /* Generate ParkedCall Stasis Message */
299 publish_parked_call(pu, PARKED_CALL);
301 /* If the parkee and the parker are the same and silence_announce isn't set, play the announcement to the parkee */
302 if (!strcmp(blind_transfer, ast_channel_name(bridge_channel->chan)) && !park_datastore->silence_announce) {
304 snprintf(saynum_buf, sizeof(saynum_buf), "%u %u", 0, pu->parking_space);
305 ast_bridge_channel_queue_playfile(bridge_channel, say_parking_space, saynum_buf, NULL);
308 /* Apply parking duration limits */
309 parking_set_duration(bridge_channel->features, pu);
311 /* Set this to the bridge pvt so that we don't have to refind the parked user associated with this bridge channel again. */
312 bridge_channel->bridge_pvt = pu;
314 ast_verb(3, "Parking '" COLORIZE_FMT "' in '" COLORIZE_FMT "' at space %d\n",
315 COLORIZE(COLOR_BRMAGENTA, 0, ast_channel_name(bridge_channel->chan)),
316 COLORIZE(COLOR_BRMAGENTA, 0, self->lot->name),
319 parking_notify_metermaids(pu->parking_space, self->lot->cfg->parking_con, AST_DEVICE_INUSE);
326 * \brief ast_bridge parking pull method.
329 * \param self Bridge to operate upon.
330 * \param bridge_channel Bridge channel to pull.
332 * \note On entry, self is already locked.
336 static void bridge_parking_pull(struct ast_bridge_parking *self, struct ast_bridge_channel *bridge_channel)
338 RAII_VAR(struct parked_user *, pu, NULL, ao2_cleanup);
339 ast_bridge_base_v_table.pull(&self->base, bridge_channel);
341 /* Take over the bridge channel's pu reference. It will be released when we are done. */
342 pu = bridge_channel->bridge_pvt;
343 bridge_channel->bridge_pvt = NULL;
345 /* This should only happen if the exiting channel was swapped out */
350 /* If we got here without the resolution being set, that's because the call was hung up for some reason without
351 * timing out or being picked up. There may be some forcible park removals later, but the resolution should be
352 * handled in those cases */
354 if (pu->resolution == PARK_UNSET) {
355 pu->resolution = PARK_ABANDON;
359 parking_notify_metermaids(pu->parking_space, self->lot->cfg->parking_con, AST_DEVICE_NOT_INUSE);
361 switch (pu->resolution) {
363 /* This should be impossible now since the resolution is forcibly set to abandon if it was unset at this point. Resolution
364 isn't allowed to be changed when it isn't currently PARK_UNSET. */
367 /* Since the call was abandoned without additional handling, we need to issue the give up event and unpark the user. */
368 publish_parked_call(pu, PARKED_CALL_GIVEUP);
369 unpark_parked_user(pu);
372 /* PARK_FORCED is currently unused, but it is expected that it would be handled similar to PARK_ANSWERED.
373 * There is currently no event related to forced parked calls either */
376 /* If answered or forced, the channel should be pulled from the bridge as part of that process and unlinked from
377 * the parking lot afterwards. We do need to apply bridge features though and play the courtesy tone if set. */
378 publish_parked_call(pu, PARKED_CALL_UNPARKED);
379 parked_call_retrieve_enable_features(bridge_channel->chan, pu->lot, AST_FEATURE_FLAG_BYCALLEE);
381 if (pu->lot->cfg->parkedplay & AST_FEATURE_FLAG_BYCALLEE) {
382 ast_bridge_channel_queue_playfile(bridge_channel, NULL, pu->lot->cfg->courtesytone, NULL);
387 /* Timeout is similar to abandon because it simply sets the bridge state to end and doesn't
388 * actually pull the channel. Because of that, unpark should happen in here. */
389 publish_parked_call(pu, PARKED_CALL_TIMEOUT);
390 unpark_parked_user(pu);
397 * \brief ast_bridge parking notify_masquerade method.
400 * \param self Bridge to operate upon.
401 * \param bridge_channel Bridge channel that was masqueraded.
403 * \note On entry, self is already locked.
404 * \note XXX Stub... and it will probably go unused.
408 static void bridge_parking_notify_masquerade(struct ast_bridge_parking *self, struct ast_bridge_channel *bridge_channel)
410 ast_bridge_base_v_table.notify_masquerade(&self->base, bridge_channel);
413 static void bridge_parking_get_merge_priority(struct ast_bridge_parking *self)
415 ast_bridge_base_v_table.get_merge_priority(&self->base);
418 struct ast_bridge_methods ast_bridge_parking_v_table = {
420 .destroy = (ast_bridge_destructor_fn) bridge_parking_destroy,
421 .dissolving = (ast_bridge_dissolving_fn) bridge_parking_dissolving,
422 .push = (ast_bridge_push_channel_fn) bridge_parking_push,
423 .pull = (ast_bridge_pull_channel_fn) bridge_parking_pull,
424 .notify_masquerade = (ast_bridge_notify_masquerade_fn) bridge_parking_notify_masquerade,
425 .get_merge_priority = (ast_bridge_merge_priority_fn) bridge_parking_get_merge_priority,
428 static struct ast_bridge *ast_bridge_parking_init(struct ast_bridge_parking *self, struct parking_lot *bridge_lot)
434 /* If no lot is defined for the bridge, then we aren't allowing the bridge to be initialized. */
440 /* It doesn't need to be a reference since the bridge only lives as long as the parking lot lives. */
441 self->lot = bridge_lot;
446 struct ast_bridge *bridge_parking_new(struct parking_lot *bridge_lot)
450 bridge = ast_bridge_alloc(sizeof(struct ast_bridge_parking), &ast_bridge_parking_v_table);
451 bridge = ast_bridge_base_init(bridge, AST_BRIDGE_CAPABILITY_HOLDING,
452 AST_BRIDGE_FLAG_MERGE_INHIBIT_TO | AST_BRIDGE_FLAG_MERGE_INHIBIT_FROM
453 | AST_BRIDGE_FLAG_SWAP_INHIBIT_FROM);
454 bridge = ast_bridge_parking_init(bridge, bridge_lot);
455 bridge = ast_bridge_register(bridge);