123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065 |
- /*
- * Asterisk -- An open source telephony toolkit.
- *
- * Copyright (C) 2012 - 2013, Digium, Inc.
- *
- * David M. Lee, II <dlee@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 Implementation for ARI stubs.
- *
- * \author David M. Lee, II <dlee@digium.com>
- */
- /*** MODULEINFO
- <depend type="module">res_stasis_recording</depend>
- <depend type="module">res_stasis_playback</depend>
- <support_level>core</support_level>
- ***/
- #include "asterisk.h"
- ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
- #include "resource_bridges.h"
- #include "asterisk/stasis.h"
- #include "asterisk/stasis_bridges.h"
- #include "asterisk/stasis_app.h"
- #include "asterisk/stasis_app_impl.h"
- #include "asterisk/stasis_app_playback.h"
- #include "asterisk/stasis_app_recording.h"
- #include "asterisk/stasis_channels.h"
- #include "asterisk/core_unreal.h"
- #include "asterisk/channel.h"
- #include "asterisk/bridge.h"
- #include "asterisk/format_cap.h"
- #include "asterisk/file.h"
- #include "asterisk/musiconhold.h"
- #include "asterisk/format_cache.h"
- /*!
- * \brief Finds a bridge, filling the response with an error, if appropriate.
- *
- * \param[out] response Response to fill with an error if control is not found.
- * \param bridge_id ID of the bridge to lookup.
- *
- * \return Bridget.
- * \return \c NULL if bridge does not exist.
- */
- static struct ast_bridge *find_bridge(
- struct ast_ari_response *response,
- const char *bridge_id)
- {
- RAII_VAR(struct ast_bridge *, bridge, NULL, ao2_cleanup);
- ast_assert(response != NULL);
- bridge = stasis_app_bridge_find_by_id(bridge_id);
- if (bridge == NULL) {
- RAII_VAR(struct ast_bridge_snapshot *, snapshot,
- ast_bridge_snapshot_get_latest(bridge_id), ao2_cleanup);
- if (!snapshot) {
- ast_ari_response_error(response, 404, "Not found",
- "Bridge not found");
- return NULL;
- }
- ast_ari_response_error(response, 409, "Conflict",
- "Bridge not in Stasis application");
- return NULL;
- }
- ao2_ref(bridge, +1);
- return bridge;
- }
- /*!
- * \brief Finds the control object for a channel, filling the response with an
- * error, if appropriate.
- * \param[out] response Response to fill with an error if control is not found.
- * \param channel_id ID of the channel to lookup.
- * \return Channel control object.
- * \return \c NULL if control object does not exist.
- */
- static struct stasis_app_control *find_channel_control(
- struct ast_ari_response *response,
- const char *channel_id)
- {
- RAII_VAR(struct stasis_app_control *, control, NULL, ao2_cleanup);
- ast_assert(response != NULL);
- control = stasis_app_control_find_by_channel_id(channel_id);
- if (control == NULL) {
- /* Distinguish between 400 and 422 errors */
- RAII_VAR(struct ast_channel_snapshot *, snapshot, NULL,
- ao2_cleanup);
- snapshot = ast_channel_snapshot_get_latest(channel_id);
- if (snapshot == NULL) {
- ast_log(LOG_DEBUG, "Couldn't find '%s'\n", channel_id);
- ast_ari_response_error(response, 400, "Bad Request",
- "Channel not found");
- return NULL;
- }
- ast_log(LOG_DEBUG, "Found non-stasis '%s'\n", channel_id);
- ast_ari_response_error(response, 422, "Unprocessable Entity",
- "Channel not in Stasis application");
- return NULL;
- }
- ao2_ref(control, +1);
- return control;
- }
- struct control_list {
- size_t count;
- struct stasis_app_control *controls[];
- };
- static void control_list_dtor(void *obj) {
- struct control_list *list = obj;
- size_t i;
- for (i = 0; i < list->count; ++i) {
- ao2_cleanup(list->controls[i]);
- list->controls[i] = NULL;
- }
- }
- static struct control_list *control_list_create(struct ast_ari_response *response, size_t count, const char **channels) {
- RAII_VAR(struct control_list *, list, NULL, ao2_cleanup);
- size_t i;
- if (count == 0 || !channels) {
- ast_ari_response_error(response, 400, "Bad Request", "Missing parameter channel");
- return NULL;
- }
- list = ao2_alloc(sizeof(*list) + count * sizeof(list->controls[0]), control_list_dtor);
- if (!list) {
- ast_ari_response_alloc_failed(response);
- return NULL;
- }
- for (i = 0; i < count; ++i) {
- if (ast_strlen_zero(channels[i])) {
- continue;
- }
- list->controls[list->count] =
- find_channel_control(response, channels[i]);
- if (!list->controls[list->count]) {
- /* response filled in by find_channel_control() */
- return NULL;
- }
- ++list->count;
- }
- if (list->count == 0) {
- ast_ari_response_error(response, 400, "Bad Request", "Missing parameter channel");
- return NULL;
- }
- ao2_ref(list, +1);
- return list;
- }
- static int check_add_remove_channel(struct ast_ari_response *response,
- struct stasis_app_control *control,
- enum stasis_app_control_channel_result result)
- {
- switch (result) {
- case STASIS_APP_CHANNEL_RECORDING :
- ast_ari_response_error(
- response, 409, "Conflict", "Channel %s currently recording",
- stasis_app_control_get_channel_id(control));
- return -1;
- case STASIS_APP_CHANNEL_OKAY:
- return 0;
- }
- return 0;
- }
- void ast_ari_bridges_add_channel(struct ast_variable *headers,
- struct ast_ari_bridges_add_channel_args *args,
- struct ast_ari_response *response)
- {
- RAII_VAR(struct ast_bridge *, bridge, find_bridge(response, args->bridge_id), ao2_cleanup);
- RAII_VAR(struct control_list *, list, NULL, ao2_cleanup);
- size_t i;
- int has_error = 0;
- if (!bridge) {
- /* Response filled in by find_bridge() */
- return;
- }
- list = control_list_create(response, args->channel_count, args->channel);
- if (!list) {
- /* Response filled in by control_list_create() */
- return;
- }
- for (i = 0; i < list->count; ++i) {
- stasis_app_control_clear_roles(list->controls[i]);
- if (!ast_strlen_zero(args->role)) {
- if (stasis_app_control_add_role(list->controls[i], args->role)) {
- ast_ari_response_alloc_failed(response);
- return;
- }
- }
- }
- for (i = 0; i < list->count; ++i) {
- if ((has_error = check_add_remove_channel(response, list->controls[i],
- stasis_app_control_add_channel_to_bridge(
- list->controls[i], bridge)))) {
- break;
- }
- }
- if (!has_error) {
- ast_ari_response_no_content(response);
- }
- }
- void ast_ari_bridges_remove_channel(struct ast_variable *headers,
- struct ast_ari_bridges_remove_channel_args *args,
- struct ast_ari_response *response)
- {
- RAII_VAR(struct ast_bridge *, bridge, find_bridge(response, args->bridge_id), ao2_cleanup);
- RAII_VAR(struct control_list *, list, NULL, ao2_cleanup);
- size_t i;
- if (!bridge) {
- /* Response filled in by find_bridge() */
- return;
- }
- list = control_list_create(response, args->channel_count, args->channel);
- if (!list) {
- /* Response filled in by control_list_create() */
- return;
- }
- /* Make sure all of the channels are in this bridge */
- for (i = 0; i < list->count; ++i) {
- if (stasis_app_get_bridge(list->controls[i]) != bridge) {
- ast_log(LOG_WARNING, "Channel %s not in bridge %s\n",
- args->channel[i], args->bridge_id);
- ast_ari_response_error(response, 422,
- "Unprocessable Entity",
- "Channel not in this bridge");
- return;
- }
- }
- /* Now actually remove it */
- for (i = 0; i < list->count; ++i) {
- stasis_app_control_remove_channel_from_bridge(list->controls[i],
- bridge);
- }
- ast_ari_response_no_content(response);
- }
- struct bridge_channel_control_thread_data {
- struct ast_channel *bridge_channel;
- struct stasis_app_control *control;
- struct stasis_forward *forward;
- char bridge_id[0];
- };
- static void *bridge_channel_control_thread(void *data)
- {
- struct bridge_channel_control_thread_data *thread_data = data;
- struct ast_channel *bridge_channel = thread_data->bridge_channel;
- struct stasis_app_control *control = thread_data->control;
- struct stasis_forward *forward = thread_data->forward;
- char *bridge_id = ast_strdupa(thread_data->bridge_id);
- RAII_VAR(struct ast_callid *, callid, ast_channel_callid(bridge_channel), ast_callid_cleanup);
- if (callid) {
- ast_callid_threadassoc_add(callid);
- }
- ast_free(thread_data);
- thread_data = NULL;
- stasis_app_control_execute_until_exhausted(bridge_channel, control);
- stasis_app_control_flush_queue(control);
- stasis_app_bridge_playback_channel_remove(bridge_id, control);
- stasis_forward_cancel(forward);
- ao2_cleanup(control);
- ast_hangup(bridge_channel);
- return NULL;
- }
- static struct ast_channel *prepare_bridge_media_channel(const char *type)
- {
- RAII_VAR(struct ast_format_cap *, cap, NULL, ao2_cleanup);
- struct ast_channel *chan;
- cap = ast_format_cap_alloc(AST_FORMAT_CAP_FLAG_DEFAULT);
- if (!cap) {
- return NULL;
- }
- ast_format_cap_append(cap, ast_format_slin, 0);
- chan = ast_request(type, cap, NULL, NULL, "ARI", NULL);
- if (!chan) {
- return NULL;
- }
- if (stasis_app_channel_unreal_set_internal(chan)) {
- ast_channel_cleanup(chan);
- return NULL;
- }
- return chan;
- }
- /*!
- * \brief Performs common setup for a bridge playback operation
- * with both new controls and when existing controls are found.
- *
- * \param args_media media string split from arguments
- * \param args_lang language string split from arguments
- * \param args_offset_ms milliseconds offset split from arguments
- * \param args_playback_id string to use for playback split from
- * arguments (null valid)
- * \param response ARI response being built
- * \param bridge Bridge the playback is being peformed on
- * \param control Control being used for the playback channel
- * \param json contents of the response to ARI
- * \param playback_url stores playback URL for use with response
- *
- * \retval -1 operation failed
- * \retval operation was successful
- */
- static int ari_bridges_play_helper(const char *args_media,
- const char *args_lang,
- int args_offset_ms,
- int args_skipms,
- const char *args_playback_id,
- struct ast_ari_response *response,
- struct ast_bridge *bridge,
- struct stasis_app_control *control,
- struct ast_json **json,
- char **playback_url)
- {
- RAII_VAR(struct ast_channel_snapshot *, snapshot, NULL, ao2_cleanup);
- RAII_VAR(struct stasis_app_playback *, playback, NULL, ao2_cleanup);
- const char *language;
- snapshot = stasis_app_control_get_snapshot(control);
- if (!snapshot) {
- ast_ari_response_error(
- response, 500, "Internal Error", "Failed to get control snapshot");
- return -1;
- }
- language = S_OR(args_lang, snapshot->language);
- playback = stasis_app_control_play_uri(control, args_media, language,
- bridge->uniqueid, STASIS_PLAYBACK_TARGET_BRIDGE, args_skipms,
- args_offset_ms, args_playback_id);
- if (!playback) {
- ast_ari_response_alloc_failed(response);
- return -1;
- }
- if (ast_asprintf(playback_url, "/playbacks/%s",
- stasis_app_playback_get_id(playback)) == -1) {
- ast_ari_response_alloc_failed(response);
- return -1;
- }
- *json = stasis_app_playback_to_json(playback);
- if (!*json) {
- ast_ari_response_alloc_failed(response);
- return -1;
- }
- return 0;
- }
- static void ari_bridges_play_new(const char *args_media,
- const char *args_lang,
- int args_offset_ms,
- int args_skipms,
- const char *args_playback_id,
- struct ast_ari_response *response,
- struct ast_bridge *bridge)
- {
- RAII_VAR(struct ast_channel *, play_channel, NULL, ast_hangup);
- RAII_VAR(struct stasis_app_control *, control, NULL, ao2_cleanup);
- RAII_VAR(struct ast_json *, json, NULL, ast_json_unref);
- RAII_VAR(struct stasis_forward *, channel_forward, NULL, stasis_forward_cancel);
- RAII_VAR(char *, playback_url, NULL, ast_free);
- struct stasis_topic *channel_topic;
- struct stasis_topic *bridge_topic;
- struct bridge_channel_control_thread_data *thread_data;
- pthread_t threadid;
- if (!(play_channel = prepare_bridge_media_channel("Announcer"))) {
- ast_ari_response_error(
- response, 500, "Internal Error", "Could not create playback channel");
- return;
- }
- ast_debug(1, "Created announcer channel '%s'\n", ast_channel_name(play_channel));
- bridge_topic = ast_bridge_topic(bridge);
- channel_topic = ast_channel_topic(play_channel);
- /* Forward messages from the playback channel topic to the bridge topic so that anything listening for
- * messages on the bridge topic will receive the playback start/stop messages. Other messages that would
- * go to this channel will be suppressed since the channel is marked as internal.
- */
- if (!bridge_topic || !channel_topic || !(channel_forward = stasis_forward_all(channel_topic, bridge_topic))) {
- ast_ari_response_error(
- response, 500, "Internal Error", "Could not forward playback channel stasis messages to bridge topic");
- return;
- }
- if (ast_unreal_channel_push_to_bridge(play_channel, bridge,
- AST_BRIDGE_CHANNEL_FLAG_IMMOVABLE | AST_BRIDGE_CHANNEL_FLAG_LONELY)) {
- ast_ari_response_error(
- response, 500, "Internal Error", "Failed to put playback channel into the bridge");
- return;
- }
- control = stasis_app_control_create(play_channel);
- if (control == NULL) {
- ast_ari_response_alloc_failed(response);
- return;
- }
- ao2_lock(control);
- if (ari_bridges_play_helper(args_media, args_lang, args_offset_ms,
- args_skipms, args_playback_id, response, bridge, control,
- &json, &playback_url)) {
- ao2_unlock(control);
- return;
- }
- ao2_unlock(control);
- if (stasis_app_bridge_playback_channel_add(bridge, play_channel, control)) {
- ast_ari_response_alloc_failed(response);
- return;
- }
- /* Give play_channel and control reference to the thread data */
- thread_data = ast_malloc(sizeof(*thread_data) + strlen(bridge->uniqueid) + 1);
- if (!thread_data) {
- stasis_app_bridge_playback_channel_remove((char *)bridge->uniqueid, control);
- ast_ari_response_alloc_failed(response);
- return;
- }
- thread_data->bridge_channel = play_channel;
- thread_data->control = control;
- thread_data->forward = channel_forward;
- /* Safe */
- strcpy(thread_data->bridge_id, bridge->uniqueid);
- if (ast_pthread_create_detached(&threadid, NULL, bridge_channel_control_thread, thread_data)) {
- stasis_app_bridge_playback_channel_remove((char *)bridge->uniqueid, control);
- ast_ari_response_alloc_failed(response);
- ast_free(thread_data);
- return;
- }
- /* These are owned by the other thread now, so we don't want RAII_VAR disposing of them. */
- play_channel = NULL;
- control = NULL;
- channel_forward = NULL;
- ast_ari_response_created(response, playback_url, ast_json_ref(json));
- }
- enum play_found_result {
- PLAY_FOUND_SUCCESS,
- PLAY_FOUND_FAILURE,
- PLAY_FOUND_CHANNEL_UNAVAILABLE,
- };
- /*!
- * \brief Performs common setup for a bridge playback operation
- * with both new controls and when existing controls are found.
- *
- * \param args_media media string split from arguments
- * \param args_lang language string split from arguments
- * \param args_offset_ms milliseconds offset split from arguments
- * \param args_playback_id string to use for playback split from
- * arguments (null valid)
- * \param response ARI response being built
- * \param bridge Bridge the playback is being peformed on
- * \param found_channel The channel that was found controlling playback
- *
- * \retval PLAY_FOUND_SUCCESS The operation was successful
- * \retval PLAY_FOUND_FAILURE The operation failed (terminal failure)
- * \retval PLAY_FOUND_CHANNEL_UNAVAILABLE The operation failed because
- * the channel requested to playback with is breaking down.
- */
- static enum play_found_result ari_bridges_play_found(const char *args_media,
- const char *args_lang,
- int args_offset_ms,
- int args_skipms,
- const char *args_playback_id,
- struct ast_ari_response *response,
- struct ast_bridge *bridge,
- struct ast_channel *found_channel)
- {
- RAII_VAR(struct ast_channel *, play_channel, found_channel, ao2_cleanup);
- RAII_VAR(struct stasis_app_control *, control, NULL, ao2_cleanup);
- RAII_VAR(char *, playback_url, NULL, ast_free);
- RAII_VAR(struct ast_json *, json, NULL, ast_json_unref);
- control = stasis_app_control_find_by_channel(play_channel);
- if (!control) {
- return PLAY_FOUND_CHANNEL_UNAVAILABLE;
- }
- ao2_lock(control);
- if (stasis_app_control_is_done(control)) {
- /* We failed to queue the action. Bailout and return that we aren't terminal. */
- ao2_unlock(control);
- return PLAY_FOUND_CHANNEL_UNAVAILABLE;
- }
- if (ari_bridges_play_helper(args_media, args_lang, args_offset_ms,
- args_skipms, args_playback_id, response, bridge, control,
- &json, &playback_url)) {
- ao2_unlock(control);
- return PLAY_FOUND_FAILURE;
- }
- ao2_unlock(control);
- ast_ari_response_created(response, playback_url, ast_json_ref(json));
- return PLAY_FOUND_SUCCESS;
- }
- static void ari_bridges_handle_play(
- const char *args_bridge_id,
- const char *args_media,
- const char *args_lang,
- int args_offset_ms,
- int args_skipms,
- const char *args_playback_id,
- struct ast_ari_response *response)
- {
- RAII_VAR(struct ast_bridge *, bridge, find_bridge(response, args_bridge_id), ao2_cleanup);
- struct ast_channel *play_channel;
- ast_assert(response != NULL);
- if (!bridge) {
- return;
- }
- while ((play_channel = stasis_app_bridge_playback_channel_find(bridge))) {
- /* If ari_bridges_play_found fails because the channel is unavailable for
- * playback, The channel will be removed from the playback list soon. We
- * can keep trying to get channels from the list until we either get one
- * that will work or else there isn't a channel for this bridge anymore,
- * in which case we'll revert to ari_bridges_play_new.
- */
- if (ari_bridges_play_found(args_media, args_lang, args_offset_ms,
- args_skipms, args_playback_id, response,bridge,
- play_channel) == PLAY_FOUND_CHANNEL_UNAVAILABLE) {
- continue;
- }
- return;
- }
- ari_bridges_play_new(args_media, args_lang, args_offset_ms,
- args_skipms, args_playback_id, response, bridge);
- }
- void ast_ari_bridges_play(struct ast_variable *headers,
- struct ast_ari_bridges_play_args *args,
- struct ast_ari_response *response)
- {
- ari_bridges_handle_play(args->bridge_id,
- args->media,
- args->lang,
- args->offsetms,
- args->skipms,
- args->playback_id,
- response);
- }
- void ast_ari_bridges_play_with_id(struct ast_variable *headers,
- struct ast_ari_bridges_play_with_id_args *args,
- struct ast_ari_response *response)
- {
- ari_bridges_handle_play(args->bridge_id,
- args->media,
- args->lang,
- args->offsetms,
- args->skipms,
- args->playback_id,
- response);
- }
- void ast_ari_bridges_record(struct ast_variable *headers,
- struct ast_ari_bridges_record_args *args,
- struct ast_ari_response *response)
- {
- RAII_VAR(struct ast_bridge *, bridge, find_bridge(response, args->bridge_id), ao2_cleanup);
- RAII_VAR(struct ast_channel *, record_channel, NULL, ast_hangup);
- RAII_VAR(struct stasis_app_control *, control, NULL, ao2_cleanup);
- RAII_VAR(struct stasis_app_recording *, recording, NULL, ao2_cleanup);
- RAII_VAR(char *, recording_url, NULL, ast_free);
- RAII_VAR(struct ast_json *, json, NULL, ast_json_unref);
- RAII_VAR(struct stasis_app_recording_options *, options, NULL, ao2_cleanup);
- RAII_VAR(char *, uri_encoded_name, NULL, ast_free);
- RAII_VAR(struct stasis_forward *, channel_forward, NULL, stasis_forward_cancel);
- struct stasis_topic *channel_topic;
- struct stasis_topic *bridge_topic;
- size_t uri_name_maxlen;
- struct bridge_channel_control_thread_data *thread_data;
- pthread_t threadid;
- ast_assert(response != NULL);
- if (bridge == NULL) {
- return;
- }
- if (!(record_channel = prepare_bridge_media_channel("Recorder"))) {
- ast_ari_response_error(
- response, 500, "Internal Server Error", "Failed to create recording channel");
- return;
- }
- bridge_topic = ast_bridge_topic(bridge);
- channel_topic = ast_channel_topic(record_channel);
- /* Forward messages from the recording channel topic to the bridge topic so that anything listening for
- * messages on the bridge topic will receive the recording start/stop messages. Other messages that would
- * go to this channel will be suppressed since the channel is marked as internal.
- */
- if (!bridge_topic || !channel_topic || !(channel_forward = stasis_forward_all(channel_topic, bridge_topic))) {
- ast_ari_response_error(
- response, 500, "Internal Error", "Could not forward record channel stasis messages to bridge topic");
- return;
- }
- if (ast_unreal_channel_push_to_bridge(record_channel, bridge,
- AST_BRIDGE_CHANNEL_FLAG_IMMOVABLE | AST_BRIDGE_CHANNEL_FLAG_LONELY)) {
- ast_ari_response_error(
- response, 500, "Internal Error", "Failed to put recording channel into the bridge");
- return;
- }
- control = stasis_app_control_create(record_channel);
- if (control == NULL) {
- ast_ari_response_alloc_failed(response);
- return;
- }
- options = stasis_app_recording_options_create(args->name, args->format);
- if (options == NULL) {
- ast_ari_response_alloc_failed(response);
- return;
- }
- ast_string_field_build(options, target, "bridge:%s", args->bridge_id);
- options->max_silence_seconds = args->max_silence_seconds;
- options->max_duration_seconds = args->max_duration_seconds;
- options->terminate_on =
- stasis_app_recording_termination_parse(args->terminate_on);
- options->if_exists =
- stasis_app_recording_if_exists_parse(args->if_exists);
- options->beep = args->beep;
- if (options->terminate_on == STASIS_APP_RECORDING_TERMINATE_INVALID) {
- ast_ari_response_error(
- response, 400, "Bad Request",
- "terminateOn invalid");
- return;
- }
- if (options->if_exists == AST_RECORD_IF_EXISTS_ERROR) {
- ast_ari_response_error(
- response, 400, "Bad Request",
- "ifExists invalid");
- return;
- }
- if (!ast_get_format_for_file_ext(options->format)) {
- ast_ari_response_error(
- response, 422, "Unprocessable Entity",
- "specified format is unknown on this system");
- return;
- }
- recording = stasis_app_control_record(control, options);
- if (recording == NULL) {
- switch(errno) {
- case EINVAL:
- /* While the arguments are invalid, we should have
- * caught them prior to calling record.
- */
- ast_ari_response_error(
- response, 500, "Internal Server Error",
- "Error parsing request");
- break;
- case EEXIST:
- ast_ari_response_error(response, 409, "Conflict",
- "Recording '%s' already exists and can not be overwritten",
- args->name);
- break;
- case ENOMEM:
- ast_ari_response_alloc_failed(response);
- break;
- case EPERM:
- ast_ari_response_error(
- response, 400, "Bad Request",
- "Recording name invalid");
- break;
- default:
- ast_log(LOG_WARNING,
- "Unrecognized recording error: %s\n",
- strerror(errno));
- ast_ari_response_error(
- response, 500, "Internal Server Error",
- "Internal Server Error");
- break;
- }
- return;
- }
- uri_name_maxlen = strlen(args->name) * 3;
- uri_encoded_name = ast_malloc(uri_name_maxlen);
- if (!uri_encoded_name) {
- ast_ari_response_alloc_failed(response);
- return;
- }
- ast_uri_encode(args->name, uri_encoded_name, uri_name_maxlen, ast_uri_http);
- if (ast_asprintf(&recording_url, "/recordings/live/%s",
- uri_encoded_name) == -1) {
- recording_url = NULL;
- ast_ari_response_alloc_failed(response);
- return;
- }
- json = stasis_app_recording_to_json(recording);
- if (!json) {
- ast_ari_response_alloc_failed(response);
- return;
- }
- thread_data = ast_calloc(1, sizeof(*thread_data));
- if (!thread_data) {
- ast_ari_response_alloc_failed(response);
- return;
- }
- thread_data->bridge_channel = record_channel;
- thread_data->control = control;
- thread_data->forward = channel_forward;
- if (ast_pthread_create_detached(&threadid, NULL, bridge_channel_control_thread, thread_data)) {
- ast_ari_response_alloc_failed(response);
- ast_free(thread_data);
- return;
- }
- /* These are owned by the other thread now, so we don't want RAII_VAR disposing of them. */
- record_channel = NULL;
- control = NULL;
- channel_forward = NULL;
- ast_ari_response_created(response, recording_url, ast_json_ref(json));
- }
- void ast_ari_bridges_start_moh(struct ast_variable *headers,
- struct ast_ari_bridges_start_moh_args *args,
- struct ast_ari_response *response)
- {
- RAII_VAR(struct ast_bridge *, bridge, find_bridge(response, args->bridge_id), ao2_cleanup);
- struct ast_channel *moh_channel;
- const char *moh_class = args->moh_class;
- if (!bridge) {
- /* The response is provided by find_bridge() */
- return;
- }
- moh_channel = stasis_app_bridge_moh_channel(bridge);
- if (!moh_channel) {
- ast_ari_response_alloc_failed(response);
- return;
- }
- ast_moh_start(moh_channel, moh_class, NULL);
- ast_channel_cleanup(moh_channel);
- ast_ari_response_no_content(response);
- }
- void ast_ari_bridges_stop_moh(struct ast_variable *headers,
- struct ast_ari_bridges_stop_moh_args *args,
- struct ast_ari_response *response)
- {
- RAII_VAR(struct ast_bridge *, bridge, find_bridge(response, args->bridge_id), ao2_cleanup);
- if (!bridge) {
- /* the response is provided by find_bridge() */
- return;
- }
- if (stasis_app_bridge_moh_stop(bridge)) {
- ast_ari_response_error(
- response, 409, "Conflict",
- "Bridge isn't playing music");
- return;
- }
- ast_ari_response_no_content(response);
- }
- void ast_ari_bridges_get(struct ast_variable *headers,
- struct ast_ari_bridges_get_args *args,
- struct ast_ari_response *response)
- {
- RAII_VAR(struct ast_bridge_snapshot *, snapshot, ast_bridge_snapshot_get_latest(args->bridge_id), ao2_cleanup);
- if (!snapshot) {
- ast_ari_response_error(
- response, 404, "Not Found",
- "Bridge not found");
- return;
- }
- ast_ari_response_ok(response,
- ast_bridge_snapshot_to_json(snapshot, stasis_app_get_sanitizer()));
- }
- void ast_ari_bridges_destroy(struct ast_variable *headers,
- struct ast_ari_bridges_destroy_args *args,
- struct ast_ari_response *response)
- {
- RAII_VAR(struct ast_bridge *, bridge, find_bridge(response, args->bridge_id), ao2_cleanup);
- if (!bridge) {
- return;
- }
- stasis_app_bridge_destroy(args->bridge_id);
- ast_ari_response_no_content(response);
- }
- void ast_ari_bridges_list(struct ast_variable *headers,
- struct ast_ari_bridges_list_args *args,
- struct ast_ari_response *response)
- {
- RAII_VAR(struct stasis_cache *, cache, NULL, ao2_cleanup);
- RAII_VAR(struct ao2_container *, snapshots, NULL, ao2_cleanup);
- RAII_VAR(struct ast_json *, json, NULL, ast_json_unref);
- struct ao2_iterator i;
- void *obj;
- cache = ast_bridge_cache();
- if (!cache) {
- ast_ari_response_error(
- response, 500, "Internal Server Error",
- "Message bus not initialized");
- return;
- }
- ao2_ref(cache, +1);
- snapshots = stasis_cache_dump(cache, ast_bridge_snapshot_type());
- if (!snapshots) {
- ast_ari_response_alloc_failed(response);
- return;
- }
- json = ast_json_array_create();
- if (!json) {
- ast_ari_response_alloc_failed(response);
- return;
- }
- i = ao2_iterator_init(snapshots, 0);
- while ((obj = ao2_iterator_next(&i))) {
- RAII_VAR(struct stasis_message *, msg, obj, ao2_cleanup);
- struct ast_bridge_snapshot *snapshot = stasis_message_data(msg);
- struct ast_json *json_bridge = ast_bridge_snapshot_to_json(snapshot, stasis_app_get_sanitizer());
- if (!json_bridge || ast_json_array_append(json, json_bridge)) {
- ao2_iterator_destroy(&i);
- ast_ari_response_alloc_failed(response);
- return;
- }
- }
- ao2_iterator_destroy(&i);
- ast_ari_response_ok(response, ast_json_ref(json));
- }
- void ast_ari_bridges_create(struct ast_variable *headers,
- struct ast_ari_bridges_create_args *args,
- struct ast_ari_response *response)
- {
- RAII_VAR(struct ast_bridge *, bridge, stasis_app_bridge_create(args->type, args->name, args->bridge_id), ao2_cleanup);
- RAII_VAR(struct ast_bridge_snapshot *, snapshot, NULL, ao2_cleanup);
- if (!bridge) {
- ast_ari_response_error(
- response, 500, "Internal Error",
- "Unable to create bridge");
- return;
- }
- ast_bridge_lock(bridge);
- snapshot = ast_bridge_snapshot_create(bridge);
- ast_bridge_unlock(bridge);
- if (!snapshot) {
- ast_ari_response_error(
- response, 500, "Internal Error",
- "Unable to create snapshot for new bridge");
- return;
- }
- ast_ari_response_ok(response,
- ast_bridge_snapshot_to_json(snapshot, stasis_app_get_sanitizer()));
- }
- void ast_ari_bridges_create_with_id(struct ast_variable *headers,
- struct ast_ari_bridges_create_with_id_args *args,
- struct ast_ari_response *response)
- {
- RAII_VAR(struct ast_bridge *, bridge, find_bridge(response, args->bridge_id), ao2_cleanup);
- RAII_VAR(struct ast_bridge_snapshot *, snapshot, NULL, ao2_cleanup);
- if (bridge) {
- /* update */
- if (!ast_strlen_zero(args->name)
- && strcmp(args->name, bridge->name)) {
- ast_ari_response_error(
- response, 500, "Internal Error",
- "Changing bridge name is not implemented");
- return;
- }
- if (!ast_strlen_zero(args->type)) {
- ast_ari_response_error(
- response, 500, "Internal Error",
- "Supplying a bridge type when updating a bridge is not allowed.");
- return;
- }
- ast_ari_response_ok(response,
- ast_bridge_snapshot_to_json(snapshot, stasis_app_get_sanitizer()));
- return;
- }
- bridge = stasis_app_bridge_create(args->type, args->name, args->bridge_id);
- if (!bridge) {
- ast_ari_response_error(
- response, 500, "Internal Error",
- "Unable to create bridge");
- return;
- }
- ast_bridge_lock(bridge);
- snapshot = ast_bridge_snapshot_create(bridge);
- ast_bridge_unlock(bridge);
- if (!snapshot) {
- ast_ari_response_error(
- response, 500, "Internal Error",
- "Unable to create snapshot for new bridge");
- return;
- }
- ast_ari_response_ok(response,
- ast_bridge_snapshot_to_json(snapshot, stasis_app_get_sanitizer()));
- }
- static int bridge_set_video_source_cb(struct stasis_app_control *control,
- struct ast_channel *chan, void *data)
- {
- struct ast_bridge *bridge = data;
- ast_bridge_lock(bridge);
- ast_bridge_set_single_src_video_mode(bridge, chan);
- ast_bridge_unlock(bridge);
- return 0;
- }
- void ast_ari_bridges_set_video_source(struct ast_variable *headers,
- struct ast_ari_bridges_set_video_source_args *args, struct ast_ari_response *response)
- {
- struct ast_bridge *bridge;
- struct stasis_app_control *control;
- bridge = find_bridge(response, args->bridge_id);
- if (!bridge) {
- return;
- }
- control = find_channel_control(response, args->channel_id);
- if (!control) {
- ao2_ref(bridge, -1);
- return;
- }
- if (stasis_app_get_bridge(control) != bridge) {
- ast_ari_response_error(response, 422,
- "Unprocessable Entity",
- "Channel not in this bridge");
- ao2_ref(bridge, -1);
- ao2_ref(control, -1);
- return;
- }
- stasis_app_send_command(control, bridge_set_video_source_cb,
- ao2_bump(bridge), __ao2_cleanup);
- ao2_ref(bridge, -1);
- ao2_ref(control, -1);
- ast_ari_response_no_content(response);
- }
- void ast_ari_bridges_clear_video_source(struct ast_variable *headers,
- struct ast_ari_bridges_clear_video_source_args *args, struct ast_ari_response *response)
- {
- struct ast_bridge *bridge;
- bridge = find_bridge(response, args->bridge_id);
- if (!bridge) {
- return;
- }
- ast_bridge_lock(bridge);
- ast_bridge_set_talker_src_video_mode(bridge);
- ast_bridge_unlock(bridge);
- ao2_ref(bridge, -1);
- ast_ari_response_no_content(response);
- }
|