mirror of
https://github.com/asterisk/asterisk.git
synced 2025-10-06 13:07:21 +00:00
From reviewboard: "During blind transfer testing, it was noticed that tests were failing occasionally because the ARI blind transfer event was not being sent. After investigating, I detected a race condition in the blind transfer code. When blind transferring a single channel, the actual transfer operation (i.e. removing the transferee from the bridge and directing them to the proper dialplan location) is queued onto the transferee bridge channel. After queuing the transfer operation, the blind transfer Stasis message is published. At the time of publication, snapshots of the channels and bridge involved are created. The ARI subscriber to the blind transfer Stasis message then attempts to determine if the bridge or any of the involved channels are subscribed to by ARI applications. If so, then the blind transfer message is sent to the applications. The way that the ARI blind transfer message handler works is to first see if the transferer channel is subscribed to. If not, then iterate over all the channel IDs in the bridge snapshot and determine if any of those are subscribed to. In the test we were running, the lone transferee channel was subscribed to, so an ARI event should have been sent to our application. Occasionally, though, the bridge snapshot did not have any channels IDs on it at all. Why? The problem is that since the blind transfer operation is handled by a separate thread, it is possible that the transfer will have completed and the channels removed from the bridge before we publish the blind transfer Stasis message. Since the blind transfer has completed, the bridge on which the transfer occurred no longer has any channels on it, so the resulting bridge snapshot has no channels on it. Through investigation of the code, I found that attended transfers can have this issue too for the case where a transferee is transferred to an application." The fix employed here is to decouple the creation of snapshots for the transfer messages from the publication of the transfer messages. This way, snapshots can be created to reflect what they are at the time of the transfer operation. Review: https://reviewboard.asterisk.org/r/4135 ........ Merged revisions 427848 from http://svn.asterisk.org/svn/asterisk/branches/12 git-svn-id: https://origsvn.digium.com/svn/asterisk/branches/13@427870 65c4cc65-6c06-0410-ace0-fbb531ad65f3
1322 lines
34 KiB
C
1322 lines
34 KiB
C
/*
|
|
* Asterisk -- An open source telephony toolkit.
|
|
*
|
|
* Copyright (C) 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 Stasis application support.
|
|
*
|
|
* \author David M. Lee, II <dlee@digium.com>
|
|
*/
|
|
|
|
#include "asterisk.h"
|
|
|
|
ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
|
|
|
|
#include "app.h"
|
|
#include "control.h"
|
|
#include "messaging.h"
|
|
|
|
#include "asterisk/callerid.h"
|
|
#include "asterisk/stasis_app.h"
|
|
#include "asterisk/stasis_bridges.h"
|
|
#include "asterisk/stasis_channels.h"
|
|
#include "asterisk/stasis_endpoints.h"
|
|
#include "asterisk/stasis_message_router.h"
|
|
|
|
static int unsubscribe(struct stasis_app *app, const char *kind, const char *id, int terminate);
|
|
|
|
struct stasis_app {
|
|
/*! Aggregation topic for this application. */
|
|
struct stasis_topic *topic;
|
|
/*! Router for handling messages forwarded to \a topic. */
|
|
struct stasis_message_router *router;
|
|
/*! Router for handling messages to the bridge all \a topic. */
|
|
struct stasis_message_router *bridge_router;
|
|
/*! Container of the channel forwards to this app's topic. */
|
|
struct ao2_container *forwards;
|
|
/*! Callback function for this application. */
|
|
stasis_app_cb handler;
|
|
/*! Opaque data to hand to callback function. */
|
|
void *data;
|
|
/*! Name of the Stasis application */
|
|
char name[];
|
|
};
|
|
|
|
enum forward_type {
|
|
FORWARD_CHANNEL,
|
|
FORWARD_BRIDGE,
|
|
FORWARD_ENDPOINT,
|
|
};
|
|
|
|
/*! Subscription info for a particular channel/bridge. */
|
|
struct app_forwards {
|
|
/*! Count of number of times this channel/bridge has been subscribed */
|
|
int interested;
|
|
|
|
/*! Forward for the regular topic */
|
|
struct stasis_forward *topic_forward;
|
|
/*! Forward for the caching topic */
|
|
struct stasis_forward *topic_cached_forward;
|
|
|
|
/* Type of object being forwarded */
|
|
enum forward_type forward_type;
|
|
/*! Unique id of the object being forwarded */
|
|
char id[];
|
|
};
|
|
|
|
static void forwards_dtor(void *obj)
|
|
{
|
|
#ifdef AST_DEVMODE
|
|
struct app_forwards *forwards = obj;
|
|
#endif /* AST_DEVMODE */
|
|
|
|
ast_assert(forwards->topic_forward == NULL);
|
|
ast_assert(forwards->topic_cached_forward == NULL);
|
|
}
|
|
|
|
static void forwards_unsubscribe(struct app_forwards *forwards)
|
|
{
|
|
stasis_forward_cancel(forwards->topic_forward);
|
|
forwards->topic_forward = NULL;
|
|
stasis_forward_cancel(forwards->topic_cached_forward);
|
|
forwards->topic_cached_forward = NULL;
|
|
}
|
|
|
|
static struct app_forwards *forwards_create(struct stasis_app *app,
|
|
const char *id)
|
|
{
|
|
RAII_VAR(struct app_forwards *, forwards, NULL, ao2_cleanup);
|
|
|
|
if (!app || ast_strlen_zero(id)) {
|
|
return NULL;
|
|
}
|
|
|
|
forwards = ao2_alloc(sizeof(*forwards) + strlen(id) + 1, forwards_dtor);
|
|
if (!forwards) {
|
|
return NULL;
|
|
}
|
|
|
|
strcpy(forwards->id, id);
|
|
|
|
ao2_ref(forwards, +1);
|
|
return forwards;
|
|
}
|
|
|
|
/*! Forward a channel's topics to an app */
|
|
static struct app_forwards *forwards_create_channel(struct stasis_app *app,
|
|
struct ast_channel *chan)
|
|
{
|
|
RAII_VAR(struct app_forwards *, forwards, NULL, ao2_cleanup);
|
|
|
|
if (!app || !chan) {
|
|
return NULL;
|
|
}
|
|
|
|
forwards = forwards_create(app, ast_channel_uniqueid(chan));
|
|
if (!forwards) {
|
|
return NULL;
|
|
}
|
|
|
|
forwards->forward_type = FORWARD_CHANNEL;
|
|
forwards->topic_forward = stasis_forward_all(ast_channel_topic(chan),
|
|
app->topic);
|
|
if (!forwards->topic_forward) {
|
|
return NULL;
|
|
}
|
|
|
|
forwards->topic_cached_forward = stasis_forward_all(
|
|
ast_channel_topic_cached(chan), app->topic);
|
|
if (!forwards->topic_cached_forward) {
|
|
/* Half-subscribed is a bad thing */
|
|
stasis_forward_cancel(forwards->topic_forward);
|
|
forwards->topic_forward = NULL;
|
|
return NULL;
|
|
}
|
|
|
|
ao2_ref(forwards, +1);
|
|
return forwards;
|
|
}
|
|
|
|
/*! Forward a bridge's topics to an app */
|
|
static struct app_forwards *forwards_create_bridge(struct stasis_app *app,
|
|
struct ast_bridge *bridge)
|
|
{
|
|
RAII_VAR(struct app_forwards *, forwards, NULL, ao2_cleanup);
|
|
|
|
if (!app || !bridge) {
|
|
return NULL;
|
|
}
|
|
|
|
forwards = forwards_create(app, bridge->uniqueid);
|
|
if (!forwards) {
|
|
return NULL;
|
|
}
|
|
|
|
forwards->forward_type = FORWARD_BRIDGE;
|
|
forwards->topic_forward = stasis_forward_all(ast_bridge_topic(bridge),
|
|
app->topic);
|
|
if (!forwards->topic_forward) {
|
|
return NULL;
|
|
}
|
|
|
|
forwards->topic_cached_forward = stasis_forward_all(
|
|
ast_bridge_topic_cached(bridge), app->topic);
|
|
if (!forwards->topic_cached_forward) {
|
|
/* Half-subscribed is a bad thing */
|
|
stasis_forward_cancel(forwards->topic_forward);
|
|
forwards->topic_forward = NULL;
|
|
return NULL;
|
|
}
|
|
|
|
ao2_ref(forwards, +1);
|
|
return forwards;
|
|
}
|
|
|
|
/*! Forward a endpoint's topics to an app */
|
|
static struct app_forwards *forwards_create_endpoint(struct stasis_app *app,
|
|
struct ast_endpoint *endpoint)
|
|
{
|
|
RAII_VAR(struct app_forwards *, forwards, NULL, ao2_cleanup);
|
|
|
|
if (!app || !endpoint) {
|
|
return NULL;
|
|
}
|
|
|
|
forwards = forwards_create(app, ast_endpoint_get_id(endpoint));
|
|
if (!forwards) {
|
|
return NULL;
|
|
}
|
|
|
|
forwards->forward_type = FORWARD_ENDPOINT;
|
|
forwards->topic_forward = stasis_forward_all(ast_endpoint_topic(endpoint),
|
|
app->topic);
|
|
if (!forwards->topic_forward) {
|
|
return NULL;
|
|
}
|
|
|
|
forwards->topic_cached_forward = stasis_forward_all(
|
|
ast_endpoint_topic_cached(endpoint), app->topic);
|
|
if (!forwards->topic_cached_forward) {
|
|
/* Half-subscribed is a bad thing */
|
|
stasis_forward_cancel(forwards->topic_forward);
|
|
forwards->topic_forward = NULL;
|
|
return NULL;
|
|
}
|
|
|
|
ao2_ref(forwards, +1);
|
|
return forwards;
|
|
}
|
|
|
|
static int forwards_sort(const void *obj_left, const void *obj_right, int flags)
|
|
{
|
|
const struct app_forwards *object_left = obj_left;
|
|
const struct app_forwards *object_right = obj_right;
|
|
const char *right_key = obj_right;
|
|
int cmp;
|
|
|
|
switch (flags & (OBJ_POINTER | OBJ_KEY | OBJ_PARTIAL_KEY)) {
|
|
case OBJ_POINTER:
|
|
right_key = object_right->id;
|
|
/* Fall through */
|
|
case OBJ_KEY:
|
|
cmp = strcmp(object_left->id, right_key);
|
|
break;
|
|
case OBJ_PARTIAL_KEY:
|
|
/*
|
|
* We could also use a partial key struct containing a length
|
|
* so strlen() does not get called for every comparison instead.
|
|
*/
|
|
cmp = strncmp(object_left->id, right_key, strlen(right_key));
|
|
break;
|
|
default:
|
|
/* Sort can only work on something with a full or partial key. */
|
|
ast_assert(0);
|
|
cmp = 0;
|
|
break;
|
|
}
|
|
return cmp;
|
|
}
|
|
|
|
static void app_dtor(void *obj)
|
|
{
|
|
struct stasis_app *app = obj;
|
|
|
|
ast_verb(1, "Destroying Stasis app %s\n", app->name);
|
|
|
|
ast_assert(app->router == NULL);
|
|
ast_assert(app->bridge_router == NULL);
|
|
|
|
ao2_cleanup(app->topic);
|
|
app->topic = NULL;
|
|
ao2_cleanup(app->forwards);
|
|
app->forwards = NULL;
|
|
ao2_cleanup(app->data);
|
|
app->data = NULL;
|
|
}
|
|
|
|
static void call_forwarded_handler(struct stasis_app *app, struct stasis_message *message)
|
|
{
|
|
struct ast_multi_channel_blob *payload = stasis_message_data(message);
|
|
struct ast_channel_snapshot *snapshot = ast_multi_channel_blob_get_channel(payload, "forwarded");
|
|
struct ast_channel *chan;
|
|
|
|
if (!snapshot) {
|
|
return;
|
|
}
|
|
|
|
chan = ast_channel_get_by_name(snapshot->uniqueid);
|
|
if (!chan) {
|
|
return;
|
|
}
|
|
|
|
app_subscribe_channel(app, chan);
|
|
ast_channel_unref(chan);
|
|
}
|
|
|
|
static void sub_default_handler(void *data, struct stasis_subscription *sub,
|
|
struct stasis_message *message)
|
|
{
|
|
struct stasis_app *app = data;
|
|
RAII_VAR(struct ast_json *, json, NULL, ast_json_unref);
|
|
|
|
if (stasis_subscription_final_message(sub, message)) {
|
|
ao2_cleanup(app);
|
|
}
|
|
|
|
if (stasis_message_type(message) == ast_channel_dial_type()) {
|
|
call_forwarded_handler(app, message);
|
|
}
|
|
|
|
if (stasis_message_type(message) == app_end_message_type()) {
|
|
app_end_message_handler(message);
|
|
}
|
|
|
|
/* By default, send any message that has a JSON representation */
|
|
json = stasis_message_to_json(message, stasis_app_get_sanitizer());
|
|
if (!json) {
|
|
return;
|
|
}
|
|
|
|
app_send(app, json);
|
|
}
|
|
|
|
/*! \brief Typedef for callbacks that get called on channel snapshot updates */
|
|
typedef struct ast_json *(*channel_snapshot_monitor)(
|
|
struct ast_channel_snapshot *old_snapshot,
|
|
struct ast_channel_snapshot *new_snapshot,
|
|
const struct timeval *tv);
|
|
|
|
static struct ast_json *simple_channel_event(
|
|
const char *type,
|
|
struct ast_channel_snapshot *snapshot,
|
|
const struct timeval *tv)
|
|
{
|
|
struct ast_json *json_channel = ast_channel_snapshot_to_json(snapshot, stasis_app_get_sanitizer());
|
|
|
|
if (!json_channel) {
|
|
return NULL;
|
|
}
|
|
|
|
return ast_json_pack("{s: s, s: o, s: o}",
|
|
"type", type,
|
|
"timestamp", ast_json_timeval(*tv, NULL),
|
|
"channel", json_channel);
|
|
}
|
|
|
|
static struct ast_json *channel_created_event(
|
|
struct ast_channel_snapshot *snapshot,
|
|
const struct timeval *tv)
|
|
{
|
|
return simple_channel_event("ChannelCreated", snapshot, tv);
|
|
}
|
|
|
|
static struct ast_json *channel_destroyed_event(
|
|
struct ast_channel_snapshot *snapshot,
|
|
const struct timeval *tv)
|
|
{
|
|
struct ast_json *json_channel = ast_channel_snapshot_to_json(snapshot, stasis_app_get_sanitizer());
|
|
|
|
if (!json_channel) {
|
|
return NULL;
|
|
}
|
|
|
|
return ast_json_pack("{s: s, s: o, s: i, s: s, s: o}",
|
|
"type", "ChannelDestroyed",
|
|
"timestamp", ast_json_timeval(*tv, NULL),
|
|
"cause", snapshot->hangupcause,
|
|
"cause_txt", ast_cause2str(snapshot->hangupcause),
|
|
"channel", json_channel);
|
|
}
|
|
|
|
static struct ast_json *channel_state_change_event(
|
|
struct ast_channel_snapshot *snapshot,
|
|
const struct timeval *tv)
|
|
{
|
|
return simple_channel_event("ChannelStateChange", snapshot, tv);
|
|
}
|
|
|
|
/*! \brief Handle channel state changes */
|
|
static struct ast_json *channel_state(
|
|
struct ast_channel_snapshot *old_snapshot,
|
|
struct ast_channel_snapshot *new_snapshot,
|
|
const struct timeval *tv)
|
|
{
|
|
struct ast_channel_snapshot *snapshot = new_snapshot ?
|
|
new_snapshot : old_snapshot;
|
|
|
|
if (!old_snapshot) {
|
|
return channel_created_event(snapshot, tv);
|
|
} else if (!new_snapshot) {
|
|
return channel_destroyed_event(snapshot, tv);
|
|
} else if (old_snapshot->state != new_snapshot->state) {
|
|
return channel_state_change_event(snapshot, tv);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static struct ast_json *channel_dialplan(
|
|
struct ast_channel_snapshot *old_snapshot,
|
|
struct ast_channel_snapshot *new_snapshot,
|
|
const struct timeval *tv)
|
|
{
|
|
struct ast_json *json_channel;
|
|
|
|
/* No Newexten event on cache clear or first event */
|
|
if (!old_snapshot || !new_snapshot) {
|
|
return NULL;
|
|
}
|
|
|
|
/* Empty application is not valid for a Newexten event */
|
|
if (ast_strlen_zero(new_snapshot->appl)) {
|
|
return NULL;
|
|
}
|
|
|
|
if (ast_channel_snapshot_cep_equal(old_snapshot, new_snapshot)) {
|
|
return NULL;
|
|
}
|
|
|
|
json_channel = ast_channel_snapshot_to_json(new_snapshot, stasis_app_get_sanitizer());
|
|
if (!json_channel) {
|
|
return NULL;
|
|
}
|
|
|
|
return ast_json_pack("{s: s, s: o, s: s, s: s, s: o}",
|
|
"type", "ChannelDialplan",
|
|
"timestamp", ast_json_timeval(*tv, NULL),
|
|
"dialplan_app", new_snapshot->appl,
|
|
"dialplan_app_data", new_snapshot->data,
|
|
"channel", json_channel);
|
|
}
|
|
|
|
static struct ast_json *channel_callerid(
|
|
struct ast_channel_snapshot *old_snapshot,
|
|
struct ast_channel_snapshot *new_snapshot,
|
|
const struct timeval *tv)
|
|
{
|
|
struct ast_json *json_channel;
|
|
|
|
/* No NewCallerid event on cache clear or first event */
|
|
if (!old_snapshot || !new_snapshot) {
|
|
return NULL;
|
|
}
|
|
|
|
if (ast_channel_snapshot_caller_id_equal(old_snapshot, new_snapshot)) {
|
|
return NULL;
|
|
}
|
|
|
|
json_channel = ast_channel_snapshot_to_json(new_snapshot, stasis_app_get_sanitizer());
|
|
if (!json_channel) {
|
|
return NULL;
|
|
}
|
|
|
|
return ast_json_pack("{s: s, s: o, s: i, s: s, s: o}",
|
|
"type", "ChannelCallerId",
|
|
"timestamp", ast_json_timeval(*tv, NULL),
|
|
"caller_presentation", new_snapshot->caller_pres,
|
|
"caller_presentation_txt", ast_describe_caller_presentation(
|
|
new_snapshot->caller_pres),
|
|
"channel", json_channel);
|
|
}
|
|
|
|
static channel_snapshot_monitor channel_monitors[] = {
|
|
channel_state,
|
|
channel_dialplan,
|
|
channel_callerid,
|
|
};
|
|
|
|
static void sub_channel_update_handler(void *data,
|
|
struct stasis_subscription *sub,
|
|
struct stasis_message *message)
|
|
{
|
|
struct stasis_app *app = data;
|
|
struct stasis_cache_update *update;
|
|
struct ast_channel_snapshot *new_snapshot;
|
|
struct ast_channel_snapshot *old_snapshot;
|
|
const struct timeval *tv;
|
|
int i;
|
|
|
|
ast_assert(stasis_message_type(message) == stasis_cache_update_type());
|
|
|
|
update = stasis_message_data(message);
|
|
|
|
ast_assert(update->type == ast_channel_snapshot_type());
|
|
|
|
new_snapshot = stasis_message_data(update->new_snapshot);
|
|
old_snapshot = stasis_message_data(update->old_snapshot);
|
|
|
|
/* Pull timestamp from the new snapshot, or from the update message
|
|
* when there isn't one. */
|
|
tv = update->new_snapshot ?
|
|
stasis_message_timestamp(update->new_snapshot) :
|
|
stasis_message_timestamp(message);
|
|
|
|
for (i = 0; i < ARRAY_LEN(channel_monitors); ++i) {
|
|
RAII_VAR(struct ast_json *, msg, NULL, ast_json_unref);
|
|
|
|
msg = channel_monitors[i](old_snapshot, new_snapshot, tv);
|
|
if (msg) {
|
|
app_send(app, msg);
|
|
}
|
|
}
|
|
|
|
if (!new_snapshot && old_snapshot) {
|
|
unsubscribe(app, "channel", old_snapshot->uniqueid, 1);
|
|
}
|
|
}
|
|
|
|
static struct ast_json *simple_endpoint_event(
|
|
const char *type,
|
|
struct ast_endpoint_snapshot *snapshot,
|
|
const struct timeval *tv)
|
|
{
|
|
struct ast_json *json_endpoint = ast_endpoint_snapshot_to_json(snapshot, stasis_app_get_sanitizer());
|
|
|
|
if (!json_endpoint) {
|
|
return NULL;
|
|
}
|
|
|
|
return ast_json_pack("{s: s, s: o, s: o}",
|
|
"type", type,
|
|
"timestamp", ast_json_timeval(*tv, NULL),
|
|
"endpoint", json_endpoint);
|
|
}
|
|
|
|
static int message_received_handler(const char *endpoint_id, struct ast_json *json_msg, void *pvt)
|
|
{
|
|
RAII_VAR(struct ast_endpoint_snapshot *, snapshot, NULL, ao2_cleanup);
|
|
struct ast_json *json_endpoint;
|
|
struct stasis_app *app = pvt;
|
|
char *tech;
|
|
char *resource;
|
|
|
|
tech = ast_strdupa(endpoint_id);
|
|
resource = strchr(tech, '/');
|
|
if (resource) {
|
|
resource[0] = '\0';
|
|
resource++;
|
|
}
|
|
|
|
if (ast_strlen_zero(tech) || ast_strlen_zero(resource)) {
|
|
return -1;
|
|
}
|
|
|
|
snapshot = ast_endpoint_latest_snapshot(tech, resource);
|
|
if (!snapshot) {
|
|
return -1;
|
|
}
|
|
|
|
json_endpoint = ast_endpoint_snapshot_to_json(snapshot, stasis_app_get_sanitizer());
|
|
if (!json_endpoint) {
|
|
return -1;
|
|
}
|
|
|
|
app_send(app, ast_json_pack("{s: s, s: o, s: o, s: O}",
|
|
"type", "TextMessageReceived",
|
|
"timestamp", ast_json_timeval(ast_tvnow(), NULL),
|
|
"endpoint", json_endpoint,
|
|
"message", json_msg));
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void sub_endpoint_update_handler(void *data,
|
|
struct stasis_subscription *sub,
|
|
struct stasis_message *message)
|
|
{
|
|
RAII_VAR(struct ast_json *, json, NULL, ast_json_unref);
|
|
struct stasis_app *app = data;
|
|
struct stasis_cache_update *update;
|
|
struct ast_endpoint_snapshot *new_snapshot;
|
|
struct ast_endpoint_snapshot *old_snapshot;
|
|
const struct timeval *tv;
|
|
|
|
ast_assert(stasis_message_type(message) == stasis_cache_update_type());
|
|
|
|
update = stasis_message_data(message);
|
|
|
|
ast_assert(update->type == ast_endpoint_snapshot_type());
|
|
|
|
new_snapshot = stasis_message_data(update->new_snapshot);
|
|
old_snapshot = stasis_message_data(update->old_snapshot);
|
|
|
|
if (new_snapshot) {
|
|
tv = stasis_message_timestamp(update->new_snapshot);
|
|
|
|
json = simple_endpoint_event("EndpointStateChange", new_snapshot, tv);
|
|
if (!json) {
|
|
return;
|
|
}
|
|
|
|
app_send(app, json);
|
|
}
|
|
|
|
if (!new_snapshot && old_snapshot) {
|
|
unsubscribe(app, "endpoint", old_snapshot->id, 1);
|
|
}
|
|
}
|
|
|
|
static struct ast_json *simple_bridge_event(
|
|
const char *type,
|
|
struct ast_bridge_snapshot *snapshot,
|
|
const struct timeval *tv)
|
|
{
|
|
struct ast_json *json_bridge = ast_bridge_snapshot_to_json(snapshot, stasis_app_get_sanitizer());
|
|
if (!json_bridge) {
|
|
return NULL;
|
|
}
|
|
|
|
return ast_json_pack("{s: s, s: o, s: o}",
|
|
"type", type,
|
|
"timestamp", ast_json_timeval(*tv, NULL),
|
|
"bridge", json_bridge);
|
|
}
|
|
|
|
static void sub_bridge_update_handler(void *data,
|
|
struct stasis_subscription *sub,
|
|
struct stasis_message *message)
|
|
{
|
|
RAII_VAR(struct ast_json *, json, NULL, ast_json_unref);
|
|
struct stasis_app *app = data;
|
|
struct stasis_cache_update *update;
|
|
struct ast_bridge_snapshot *new_snapshot;
|
|
struct ast_bridge_snapshot *old_snapshot;
|
|
const struct timeval *tv;
|
|
|
|
ast_assert(stasis_message_type(message) == stasis_cache_update_type());
|
|
|
|
update = stasis_message_data(message);
|
|
|
|
ast_assert(update->type == ast_bridge_snapshot_type());
|
|
|
|
new_snapshot = stasis_message_data(update->new_snapshot);
|
|
old_snapshot = stasis_message_data(update->old_snapshot);
|
|
tv = update->new_snapshot ?
|
|
stasis_message_timestamp(update->new_snapshot) :
|
|
stasis_message_timestamp(message);
|
|
|
|
if (!new_snapshot) {
|
|
json = simple_bridge_event("BridgeDestroyed", old_snapshot, tv);
|
|
} else if (!old_snapshot) {
|
|
json = simple_bridge_event("BridgeCreated", new_snapshot, tv);
|
|
}
|
|
|
|
if (json) {
|
|
app_send(app, json);
|
|
}
|
|
|
|
if (!new_snapshot && old_snapshot) {
|
|
unsubscribe(app, "bridge", old_snapshot->uniqueid, 1);
|
|
}
|
|
}
|
|
|
|
|
|
/*! \brief Helper function for determining if the application is subscribed to a given entity */
|
|
static int bridge_app_subscribed(struct stasis_app *app, const char *uniqueid)
|
|
{
|
|
struct app_forwards *forwards = NULL;
|
|
|
|
forwards = ao2_find(app->forwards, uniqueid, OBJ_SEARCH_KEY);
|
|
if (!forwards) {
|
|
return 0;
|
|
}
|
|
|
|
ao2_ref(forwards, -1);
|
|
return 1;
|
|
}
|
|
|
|
static void bridge_merge_handler(void *data, struct stasis_subscription *sub,
|
|
struct stasis_message *message)
|
|
{
|
|
struct stasis_app *app = data;
|
|
struct ast_bridge_merge_message *merge;
|
|
|
|
merge = stasis_message_data(message);
|
|
|
|
/* Find out if we're subscribed to either bridge */
|
|
if (bridge_app_subscribed(app, merge->from->uniqueid) ||
|
|
bridge_app_subscribed(app, merge->to->uniqueid)) {
|
|
/* Forward the message to the app */
|
|
stasis_publish(app->topic, message);
|
|
}
|
|
}
|
|
|
|
/*! \brief Callback function for checking if channels in a bridge are subscribed to */
|
|
static int bridge_app_subscribed_involved(struct stasis_app *app, struct ast_bridge_snapshot *snapshot)
|
|
{
|
|
int subscribed = 0;
|
|
struct ao2_iterator iter;
|
|
char *uniqueid;
|
|
|
|
if (bridge_app_subscribed(app, snapshot->uniqueid)) {
|
|
return 1;
|
|
}
|
|
|
|
iter = ao2_iterator_init(snapshot->channels, 0);
|
|
for (; (uniqueid = ao2_iterator_next(&iter)); ao2_ref(uniqueid, -1)) {
|
|
if (bridge_app_subscribed(app, uniqueid)) {
|
|
subscribed = 1;
|
|
ao2_ref(uniqueid, -1);
|
|
break;
|
|
}
|
|
}
|
|
ao2_iterator_destroy(&iter);
|
|
|
|
return subscribed;
|
|
}
|
|
|
|
static void set_replacement_channel(struct ast_channel_snapshot *to_be_replaced,
|
|
struct ast_channel_snapshot *replacing)
|
|
{
|
|
struct stasis_app_control *control = stasis_app_control_find_by_channel_id(
|
|
to_be_replaced->uniqueid);
|
|
struct ast_channel *chan = ast_channel_get_by_name(replacing->uniqueid);
|
|
|
|
if (control && chan) {
|
|
ast_channel_lock(chan);
|
|
app_set_replace_channel_app(chan, app_name(control_app(control)));
|
|
app_set_replace_channel_snapshot(chan, to_be_replaced);
|
|
ast_channel_unlock(chan);
|
|
}
|
|
ast_channel_cleanup(chan);
|
|
ao2_cleanup(control);
|
|
}
|
|
|
|
static void bridge_blind_transfer_handler(void *data, struct stasis_subscription *sub,
|
|
struct stasis_message *message)
|
|
{
|
|
struct stasis_app *app = data;
|
|
struct ast_blind_transfer_message *transfer_msg = stasis_message_data(message);
|
|
struct ast_bridge_snapshot *bridge = transfer_msg->bridge;
|
|
|
|
if (transfer_msg->replace_channel) {
|
|
set_replacement_channel(transfer_msg->transferer, transfer_msg->replace_channel);
|
|
}
|
|
|
|
if (bridge_app_subscribed(app, transfer_msg->transferer->uniqueid) ||
|
|
(bridge && bridge_app_subscribed_involved(app, bridge))) {
|
|
stasis_publish(app->topic, message);
|
|
}
|
|
}
|
|
|
|
static void bridge_attended_transfer_handler(void *data, struct stasis_subscription *sub,
|
|
struct stasis_message *message)
|
|
{
|
|
struct stasis_app *app = data;
|
|
struct ast_attended_transfer_message *transfer_msg = stasis_message_data(message);
|
|
int subscribed = 0;
|
|
|
|
subscribed = bridge_app_subscribed(app, transfer_msg->to_transferee.channel_snapshot->uniqueid);
|
|
if (!subscribed) {
|
|
subscribed = bridge_app_subscribed(app, transfer_msg->to_transfer_target.channel_snapshot->uniqueid);
|
|
}
|
|
if (!subscribed && transfer_msg->to_transferee.bridge_snapshot) {
|
|
subscribed = bridge_app_subscribed_involved(app, transfer_msg->to_transferee.bridge_snapshot);
|
|
}
|
|
if (!subscribed && transfer_msg->to_transfer_target.bridge_snapshot) {
|
|
subscribed = bridge_app_subscribed_involved(app, transfer_msg->to_transfer_target.bridge_snapshot);
|
|
}
|
|
|
|
if (!subscribed) {
|
|
switch (transfer_msg->dest_type) {
|
|
case AST_ATTENDED_TRANSFER_DEST_BRIDGE_MERGE:
|
|
subscribed = bridge_app_subscribed(app, transfer_msg->dest.bridge);
|
|
break;
|
|
case AST_ATTENDED_TRANSFER_DEST_LINK:
|
|
subscribed = bridge_app_subscribed(app, transfer_msg->dest.links[0]->uniqueid);
|
|
if (!subscribed) {
|
|
subscribed = bridge_app_subscribed(app, transfer_msg->dest.links[1]->uniqueid);
|
|
}
|
|
break;
|
|
break;
|
|
case AST_ATTENDED_TRANSFER_DEST_THREEWAY:
|
|
subscribed = bridge_app_subscribed_involved(app, transfer_msg->dest.threeway.bridge_snapshot);
|
|
if (!subscribed) {
|
|
subscribed = bridge_app_subscribed(app, transfer_msg->dest.threeway.channel_snapshot->uniqueid);
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (subscribed) {
|
|
stasis_publish(app->topic, message);
|
|
}
|
|
|
|
if (transfer_msg->replace_channel) {
|
|
set_replacement_channel(transfer_msg->to_transferee.channel_snapshot,
|
|
transfer_msg->replace_channel);
|
|
}
|
|
|
|
if (transfer_msg->dest_type == AST_ATTENDED_TRANSFER_DEST_LINK) {
|
|
set_replacement_channel(transfer_msg->to_transferee.channel_snapshot,
|
|
transfer_msg->dest.links[0]);
|
|
set_replacement_channel(transfer_msg->to_transfer_target.channel_snapshot,
|
|
transfer_msg->dest.links[1]);
|
|
}
|
|
}
|
|
|
|
static void bridge_default_handler(void *data, struct stasis_subscription *sub,
|
|
struct stasis_message *message)
|
|
{
|
|
struct stasis_app *app = data;
|
|
|
|
if (stasis_subscription_final_message(sub, message)) {
|
|
ao2_cleanup(app);
|
|
}
|
|
}
|
|
|
|
struct stasis_app *app_create(const char *name, stasis_app_cb handler, void *data)
|
|
{
|
|
RAII_VAR(struct stasis_app *, app, NULL, ao2_cleanup);
|
|
size_t size;
|
|
int res = 0;
|
|
|
|
ast_assert(name != NULL);
|
|
ast_assert(handler != NULL);
|
|
|
|
ast_verb(1, "Creating Stasis app '%s'\n", name);
|
|
|
|
size = sizeof(*app) + strlen(name) + 1;
|
|
app = ao2_alloc_options(size, app_dtor, AO2_ALLOC_OPT_LOCK_MUTEX);
|
|
|
|
if (!app) {
|
|
return NULL;
|
|
}
|
|
|
|
app->forwards = ao2_container_alloc_rbtree(AO2_ALLOC_OPT_LOCK_MUTEX,
|
|
AO2_CONTAINER_ALLOC_OPT_DUPS_OBJ_REJECT,
|
|
forwards_sort, NULL);
|
|
if (!app->forwards) {
|
|
return NULL;
|
|
}
|
|
|
|
app->topic = stasis_topic_create(name);
|
|
if (!app->topic) {
|
|
return NULL;
|
|
}
|
|
|
|
app->bridge_router = stasis_message_router_create(ast_bridge_topic_all());
|
|
if (!app->bridge_router) {
|
|
return NULL;
|
|
}
|
|
|
|
res |= stasis_message_router_add(app->bridge_router,
|
|
ast_bridge_merge_message_type(), bridge_merge_handler, app);
|
|
|
|
res |= stasis_message_router_add(app->bridge_router,
|
|
ast_blind_transfer_type(), bridge_blind_transfer_handler, app);
|
|
|
|
res |= stasis_message_router_add(app->bridge_router,
|
|
ast_attended_transfer_type(), bridge_attended_transfer_handler, app);
|
|
|
|
res |= stasis_message_router_set_default(app->bridge_router,
|
|
bridge_default_handler, app);
|
|
|
|
if (res != 0) {
|
|
return NULL;
|
|
}
|
|
/* Bridge router holds a reference */
|
|
ao2_ref(app, +1);
|
|
|
|
app->router = stasis_message_router_create(app->topic);
|
|
if (!app->router) {
|
|
return NULL;
|
|
}
|
|
|
|
res |= stasis_message_router_add_cache_update(app->router,
|
|
ast_bridge_snapshot_type(), sub_bridge_update_handler, app);
|
|
|
|
res |= stasis_message_router_add_cache_update(app->router,
|
|
ast_channel_snapshot_type(), sub_channel_update_handler, app);
|
|
|
|
res |= stasis_message_router_add_cache_update(app->router,
|
|
ast_endpoint_snapshot_type(), sub_endpoint_update_handler, app);
|
|
|
|
res |= stasis_message_router_set_default(app->router,
|
|
sub_default_handler, app);
|
|
|
|
if (res != 0) {
|
|
return NULL;
|
|
}
|
|
/* Router holds a reference */
|
|
ao2_ref(app, +1);
|
|
|
|
strncpy(app->name, name, size - sizeof(*app));
|
|
app->handler = handler;
|
|
ao2_ref(data, +1);
|
|
app->data = data;
|
|
|
|
ao2_ref(app, +1);
|
|
return app;
|
|
}
|
|
|
|
struct stasis_topic *ast_app_get_topic(struct stasis_app *app) {
|
|
return app->topic;
|
|
}
|
|
|
|
/*!
|
|
* \brief Send a message to the given application.
|
|
* \param app App to send the message to.
|
|
* \param message Message to send.
|
|
*/
|
|
void app_send(struct stasis_app *app, struct ast_json *message)
|
|
{
|
|
stasis_app_cb handler;
|
|
RAII_VAR(void *, data, NULL, ao2_cleanup);
|
|
|
|
/* Copy off mutable state with lock held */
|
|
{
|
|
SCOPED_AO2LOCK(lock, app);
|
|
handler = app->handler;
|
|
if (app->data) {
|
|
ao2_ref(app->data, +1);
|
|
data = app->data;
|
|
}
|
|
/* Name is immutable; no need to copy */
|
|
}
|
|
|
|
if (!handler) {
|
|
ast_verb(3,
|
|
"Inactive Stasis app '%s' missed message\n", app->name);
|
|
return;
|
|
}
|
|
|
|
handler(data, app->name, message);
|
|
}
|
|
|
|
void app_deactivate(struct stasis_app *app)
|
|
{
|
|
SCOPED_AO2LOCK(lock, app);
|
|
ast_verb(1, "Deactivating Stasis app '%s'\n", app->name);
|
|
app->handler = NULL;
|
|
ao2_cleanup(app->data);
|
|
app->data = NULL;
|
|
}
|
|
|
|
void app_shutdown(struct stasis_app *app)
|
|
{
|
|
SCOPED_AO2LOCK(lock, app);
|
|
|
|
ast_assert(app_is_finished(app));
|
|
|
|
stasis_message_router_unsubscribe(app->router);
|
|
app->router = NULL;
|
|
stasis_message_router_unsubscribe(app->bridge_router);
|
|
app->bridge_router = NULL;
|
|
}
|
|
|
|
int app_is_active(struct stasis_app *app)
|
|
{
|
|
SCOPED_AO2LOCK(lock, app);
|
|
return app->handler != NULL;
|
|
}
|
|
|
|
int app_is_finished(struct stasis_app *app)
|
|
{
|
|
SCOPED_AO2LOCK(lock, app);
|
|
|
|
return app->handler == NULL && ao2_container_count(app->forwards) == 0;
|
|
}
|
|
|
|
void app_update(struct stasis_app *app, stasis_app_cb handler, void *data)
|
|
{
|
|
SCOPED_AO2LOCK(lock, app);
|
|
|
|
if (app->handler) {
|
|
RAII_VAR(struct ast_json *, msg, NULL, ast_json_unref);
|
|
|
|
ast_verb(1, "Replacing Stasis app '%s'\n", app->name);
|
|
|
|
msg = ast_json_pack("{s: s, s: s}",
|
|
"type", "ApplicationReplaced",
|
|
"application", app->name);
|
|
if (msg) {
|
|
app_send(app, msg);
|
|
}
|
|
} else {
|
|
ast_verb(1, "Activating Stasis app '%s'\n", app->name);
|
|
}
|
|
|
|
app->handler = handler;
|
|
ao2_cleanup(app->data);
|
|
if (data) {
|
|
ao2_ref(data, +1);
|
|
}
|
|
app->data = data;
|
|
}
|
|
|
|
const char *app_name(const struct stasis_app *app)
|
|
{
|
|
return app->name;
|
|
}
|
|
|
|
struct ast_json *app_to_json(const struct stasis_app *app)
|
|
{
|
|
RAII_VAR(struct ast_json *, json, NULL, ast_json_unref);
|
|
struct ast_json *channels;
|
|
struct ast_json *bridges;
|
|
struct ast_json *endpoints;
|
|
struct ao2_iterator i;
|
|
void *obj;
|
|
|
|
json = ast_json_pack("{s: s, s: [], s: [], s: []}",
|
|
"name", app->name,
|
|
"channel_ids", "bridge_ids", "endpoint_ids");
|
|
channels = ast_json_object_get(json, "channel_ids");
|
|
bridges = ast_json_object_get(json, "bridge_ids");
|
|
endpoints = ast_json_object_get(json, "endpoint_ids");
|
|
|
|
i = ao2_iterator_init(app->forwards, 0);
|
|
while ((obj = ao2_iterator_next(&i))) {
|
|
RAII_VAR(struct app_forwards *, forwards, obj, ao2_cleanup);
|
|
RAII_VAR(struct ast_json *, id, NULL, ast_json_unref);
|
|
int append_res = -1;
|
|
|
|
id = ast_json_string_create(forwards->id);
|
|
|
|
switch (forwards->forward_type) {
|
|
case FORWARD_CHANNEL:
|
|
append_res = ast_json_array_append(channels,
|
|
ast_json_ref(id));
|
|
break;
|
|
case FORWARD_BRIDGE:
|
|
append_res = ast_json_array_append(bridges,
|
|
ast_json_ref(id));
|
|
break;
|
|
case FORWARD_ENDPOINT:
|
|
append_res = ast_json_array_append(endpoints,
|
|
ast_json_ref(id));
|
|
break;
|
|
}
|
|
|
|
if (append_res != 0) {
|
|
ast_log(LOG_ERROR, "Error building response\n");
|
|
ao2_iterator_destroy(&i);
|
|
return NULL;
|
|
}
|
|
}
|
|
ao2_iterator_destroy(&i);
|
|
|
|
return ast_json_ref(json);
|
|
}
|
|
|
|
int app_subscribe_channel(struct stasis_app *app, struct ast_channel *chan)
|
|
{
|
|
int res;
|
|
|
|
if (!app || !chan) {
|
|
return -1;
|
|
} else {
|
|
RAII_VAR(struct app_forwards *, forwards, NULL, ao2_cleanup);
|
|
SCOPED_AO2LOCK(lock, app->forwards);
|
|
|
|
forwards = ao2_find(app->forwards, ast_channel_uniqueid(chan),
|
|
OBJ_SEARCH_KEY | OBJ_NOLOCK);
|
|
if (!forwards) {
|
|
/* Forwards not found, create one */
|
|
forwards = forwards_create_channel(app, chan);
|
|
if (!forwards) {
|
|
return -1;
|
|
}
|
|
|
|
res = ao2_link_flags(app->forwards, forwards,
|
|
OBJ_NOLOCK);
|
|
if (!res) {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
++forwards->interested;
|
|
ast_debug(3, "Channel '%s' is %d interested in %s\n", ast_channel_uniqueid(chan), forwards->interested, app->name);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
static int subscribe_channel(struct stasis_app *app, void *obj)
|
|
{
|
|
return app_subscribe_channel(app, obj);
|
|
}
|
|
|
|
static int unsubscribe(struct stasis_app *app, const char *kind, const char *id, int terminate)
|
|
{
|
|
RAII_VAR(struct app_forwards *, forwards, NULL, ao2_cleanup);
|
|
SCOPED_AO2LOCK(lock, app->forwards);
|
|
|
|
forwards = ao2_find(app->forwards, id, OBJ_SEARCH_KEY | OBJ_NOLOCK);
|
|
if (!forwards) {
|
|
ast_debug(3, "App '%s' not subscribed to %s '%s'\n", app->name, kind, id);
|
|
return -1;
|
|
}
|
|
forwards->interested--;
|
|
|
|
ast_debug(3, "%s '%s': is %d interested in %s\n", kind, id, forwards->interested, app->name);
|
|
if (forwards->interested == 0 || terminate) {
|
|
/* No one is interested any more; unsubscribe */
|
|
ast_debug(3, "%s '%s' unsubscribed from %s\n", kind, id, app->name);
|
|
forwards_unsubscribe(forwards);
|
|
ao2_find(app->forwards, forwards,
|
|
OBJ_POINTER | OBJ_NOLOCK | OBJ_UNLINK |
|
|
OBJ_NODATA);
|
|
|
|
if (!strcmp(kind, "endpoint")) {
|
|
messaging_app_unsubscribe_endpoint(app->name, id);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int app_unsubscribe_channel(struct stasis_app *app, struct ast_channel *chan)
|
|
{
|
|
if (!app || !chan) {
|
|
return -1;
|
|
}
|
|
|
|
return app_unsubscribe_channel_id(app, ast_channel_uniqueid(chan));
|
|
}
|
|
|
|
int app_unsubscribe_channel_id(struct stasis_app *app, const char *channel_id)
|
|
{
|
|
if (!app || !channel_id) {
|
|
return -1;
|
|
}
|
|
|
|
return unsubscribe(app, "channel", channel_id, 0);
|
|
}
|
|
|
|
int app_is_subscribed_channel_id(struct stasis_app *app, const char *channel_id)
|
|
{
|
|
RAII_VAR(struct app_forwards *, forwards, NULL, ao2_cleanup);
|
|
forwards = ao2_find(app->forwards, channel_id, OBJ_SEARCH_KEY);
|
|
return forwards != NULL;
|
|
}
|
|
|
|
int app_replace_channel_forwards(struct stasis_app *app, const char *old_id, struct ast_channel *new_chan)
|
|
{
|
|
RAII_VAR(struct app_forwards *, old_forwards, NULL, ao2_cleanup);
|
|
struct app_forwards *new_forwards;
|
|
|
|
old_forwards = ao2_find(app->forwards, old_id, OBJ_SEARCH_KEY | OBJ_UNLINK);
|
|
if (!old_forwards) {
|
|
return -1;
|
|
}
|
|
|
|
new_forwards = forwards_create_channel(app, new_chan);
|
|
if (!new_forwards) {
|
|
return -1;
|
|
}
|
|
|
|
new_forwards->interested = old_forwards->interested;
|
|
ao2_link_flags(app->forwards, new_forwards, 0);
|
|
ao2_cleanup(new_forwards);
|
|
|
|
/* Clean up old forwards */
|
|
forwards_unsubscribe(old_forwards);
|
|
return 0;
|
|
}
|
|
|
|
static void *channel_find(const struct stasis_app *app, const char *id)
|
|
{
|
|
return ast_channel_get_by_name(id);
|
|
}
|
|
|
|
struct stasis_app_event_source channel_event_source = {
|
|
.scheme = "channel:",
|
|
.find = channel_find,
|
|
.subscribe = subscribe_channel,
|
|
.unsubscribe = app_unsubscribe_channel_id,
|
|
.is_subscribed = app_is_subscribed_channel_id
|
|
};
|
|
|
|
int app_subscribe_bridge(struct stasis_app *app, struct ast_bridge *bridge)
|
|
{
|
|
if (!app || !bridge) {
|
|
return -1;
|
|
} else {
|
|
RAII_VAR(struct app_forwards *, forwards, NULL, ao2_cleanup);
|
|
SCOPED_AO2LOCK(lock, app->forwards);
|
|
|
|
forwards = ao2_find(app->forwards, bridge->uniqueid,
|
|
OBJ_SEARCH_KEY | OBJ_NOLOCK);
|
|
|
|
if (!forwards) {
|
|
/* Forwards not found, create one */
|
|
forwards = forwards_create_bridge(app, bridge);
|
|
if (!forwards) {
|
|
return -1;
|
|
}
|
|
ao2_link_flags(app->forwards, forwards, OBJ_NOLOCK);
|
|
}
|
|
|
|
++forwards->interested;
|
|
ast_debug(3, "Bridge '%s' is %d interested in %s\n", bridge->uniqueid, forwards->interested, app->name);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
static int subscribe_bridge(struct stasis_app *app, void *obj)
|
|
{
|
|
return app_subscribe_bridge(app, obj);
|
|
}
|
|
|
|
int app_unsubscribe_bridge(struct stasis_app *app, struct ast_bridge *bridge)
|
|
{
|
|
if (!app || !bridge) {
|
|
return -1;
|
|
}
|
|
|
|
return app_unsubscribe_bridge_id(app, bridge->uniqueid);
|
|
}
|
|
|
|
int app_unsubscribe_bridge_id(struct stasis_app *app, const char *bridge_id)
|
|
{
|
|
if (!app || !bridge_id) {
|
|
return -1;
|
|
}
|
|
|
|
return unsubscribe(app, "bridge", bridge_id, 0);
|
|
}
|
|
|
|
int app_is_subscribed_bridge_id(struct stasis_app *app, const char *bridge_id)
|
|
{
|
|
RAII_VAR(struct app_forwards *, forwards, NULL, ao2_cleanup);
|
|
forwards = ao2_find(app->forwards, bridge_id, OBJ_SEARCH_KEY);
|
|
return forwards != NULL;
|
|
}
|
|
|
|
static void *bridge_find(const struct stasis_app *app, const char *id)
|
|
{
|
|
return stasis_app_bridge_find_by_id(id);
|
|
}
|
|
|
|
struct stasis_app_event_source bridge_event_source = {
|
|
.scheme = "bridge:",
|
|
.find = bridge_find,
|
|
.subscribe = subscribe_bridge,
|
|
.unsubscribe = app_unsubscribe_bridge_id,
|
|
.is_subscribed = app_is_subscribed_bridge_id
|
|
};
|
|
|
|
int app_subscribe_endpoint(struct stasis_app *app, struct ast_endpoint *endpoint)
|
|
{
|
|
if (!app || !endpoint) {
|
|
return -1;
|
|
} else {
|
|
RAII_VAR(struct app_forwards *, forwards, NULL, ao2_cleanup);
|
|
SCOPED_AO2LOCK(lock, app->forwards);
|
|
|
|
forwards = ao2_find(app->forwards, ast_endpoint_get_id(endpoint),
|
|
OBJ_SEARCH_KEY | OBJ_NOLOCK);
|
|
|
|
if (!forwards) {
|
|
/* Forwards not found, create one */
|
|
forwards = forwards_create_endpoint(app, endpoint);
|
|
if (!forwards) {
|
|
return -1;
|
|
}
|
|
ao2_link_flags(app->forwards, forwards, OBJ_NOLOCK);
|
|
|
|
/* Subscribe for messages */
|
|
messaging_app_subscribe_endpoint(app->name, endpoint, &message_received_handler, app);
|
|
}
|
|
|
|
++forwards->interested;
|
|
ast_debug(3, "Endpoint '%s' is %d interested in %s\n", ast_endpoint_get_id(endpoint), forwards->interested, app->name);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
static int subscribe_endpoint(struct stasis_app *app, void *obj)
|
|
{
|
|
return app_subscribe_endpoint(app, obj);
|
|
}
|
|
|
|
int app_unsubscribe_endpoint_id(struct stasis_app *app, const char *endpoint_id)
|
|
{
|
|
if (!app || !endpoint_id) {
|
|
return -1;
|
|
}
|
|
|
|
return unsubscribe(app, "endpoint", endpoint_id, 0);
|
|
}
|
|
|
|
int app_is_subscribed_endpoint_id(struct stasis_app *app, const char *endpoint_id)
|
|
{
|
|
RAII_VAR(struct app_forwards *, forwards, NULL, ao2_cleanup);
|
|
forwards = ao2_find(app->forwards, endpoint_id, OBJ_SEARCH_KEY);
|
|
return forwards != NULL;
|
|
}
|
|
|
|
static void *endpoint_find(const struct stasis_app *app, const char *id)
|
|
{
|
|
return ast_endpoint_find_by_id(id);
|
|
}
|
|
|
|
struct stasis_app_event_source endpoint_event_source = {
|
|
.scheme = "endpoint:",
|
|
.find = endpoint_find,
|
|
.subscribe = subscribe_endpoint,
|
|
.unsubscribe = app_unsubscribe_endpoint_id,
|
|
.is_subscribed = app_is_subscribed_endpoint_id
|
|
};
|
|
|
|
void stasis_app_register_event_sources(void)
|
|
{
|
|
stasis_app_register_event_source(&channel_event_source);
|
|
stasis_app_register_event_source(&bridge_event_source);
|
|
stasis_app_register_event_source(&endpoint_event_source);
|
|
}
|
|
|
|
int stasis_app_is_core_event_source(struct stasis_app_event_source *obj)
|
|
{
|
|
return obj == &endpoint_event_source ||
|
|
obj == &bridge_event_source ||
|
|
obj == &channel_event_source;
|
|
}
|
|
|
|
void stasis_app_unregister_event_sources(void)
|
|
{
|
|
stasis_app_unregister_event_source(&endpoint_event_source);
|
|
stasis_app_unregister_event_source(&bridge_event_source);
|
|
stasis_app_unregister_event_source(&channel_event_source);
|
|
}
|
|
|
|
|