mirror of
https://github.com/asterisk/asterisk.git
synced 2025-09-03 11:25:35 +00:00
There were a number of places in the res_pjsip stack that were getting all endpoints or all aors, and then filtering them locally. A good example is pjsip_options which, on startup, retrieves all endpoints, then the aors for those endpoints, then tests the aors to see if the qualify_frequency is > 0. One issue was that it never did anything with the endpoints other than retrieve the aors so we probably could have skipped a step and just retrieved all aors. But nevermind. This worked reasonably well with local config files but with a realtime backend and thousands of objects, this was a nightmare. The issue really boiled down to the fact that while realtime supports predicates that are passed to the database engine, the non-realtime sorcery backends didn't. They do now. The realtime engines have a scheme for doing simple comparisons. They take in an ast_variable (or list) for matching, and the name of each variable can contain an operator. For instance, a name of "qualify_frequency >" and a value of "0" would create a SQL predicate that looks like "where qualify_frequency > '0'". If there's no operator after the name, the engines add an '=' so a simple name of "qualify_frequency" and a value of "10" would return exact matches. The non-realtime backends decide whether to include an object in a result set by calling ast_sorcery_changeset_create on every object in the internal container. However, ast_sorcery_changeset_create only does exact string matches though so a name of "qualify_frequency >" and a value of "0" returns nothing because the literal "qualify_frequency >" doesn't match any name in the objset set. So, the real task was to create a generic string matcher that can take a left value, operator and a right value and perform the match. To that end, strings.c has a new ast_strings_match(left, operator, right) function. Left and right are the strings to operate on and the operator can be a string containing any of the following: = (or NULL or ""), !=, >, >=, <, <=, like or regex. If the operator is like or regex, the right string should be a %-pattern or a regex expression. If both left and right can be converted to float, then a numeric comparison is performed, otherwise a string comparison is performed. To use this new function on ast_variables, 2 new functions were added to config.c. One that compares 2 ast_variables, and one that compares 2 ast_variable lists. The former is useful when you want to compare 2 ast_variables that happen to be in a list but don't want to traverse the list. The latter will traverse the right list and return true if all the variables in it match the left list. Now, the backends' fields_cmp functions call ast_variable_lists_match instead of ast_sorcery_changeset_create and they can now process the same syntax as the realtime engines. The realtime backend just passes the variable list unaltered to the engine. The only gotcha is that there's no common realtime engine support for regex so that's been noted in the api docs for ast_sorcery_retrieve_by_fields. Only one more change to sorcery was done... A new config flag "allow_unqualified_fetch" was added to reg_sorcery_realtime. "no": ignore fetches if no predicate fields were supplied. "error": same as no but emit an error. (good for testing) "yes": allow (the default); "warn": allow but emit a warning. (good for testing) Now on to res_pjsip... pjsip_options was modified to retrieve aors with qualify_frequency > 0 rather than all endpoints then all aors. Not only was this a big improvement in realtime retrieval but even for config files there's an improvement because we're not going through endpoints anymore. res_pjsip_mwi was modified to retieve only endpoints with something in the mailboxes field instead of all endpoints then testing mailboxes. res_pjsip_registrar_expire was completely refactored. It was retrieving all contacts then setting up scheduler entries to check for expiration. Now, it's a single thread (like keepalive) that periodically retrieves only contacts whose expiration time is < now and deletes them. A new contact_expiration_check_interval was added to global with a default of 30 seconds. Ross Beer reports that with this patch, his Asterisk startup time dropped from around an hour to under 30 seconds. There are still objects that can't be filtered at the database like identifies, transports, and registrations. These are not going to be anywhere near as numerous as endpoints, aors, auths, contacts however. Back to allow_unqualified_fetch. If this is set to yes and you have a very large number of objects in the database, the pjsip CLI commands will attempt to retrive ALL of them if not qualified with a LIKE. Worse, if you type "pjsip show endpoint <tab>" guess what's going to happen? :) Having a cache helps but all the objects will have to be retrieved at least once to fill the cache. Setting allow_unqualified_fetch=no prevents the mass retrieve and should be used on endpoints, auths, aors, and contacts. It should NOT be used for identifies, registrations and transports since these MUST be retrieved in bulk. Example sorcery.conf: [res_pjsip] endpoint=config,pjsip.conf,criteria=type=endpoint endpoint=realtime,ps_endpoints,allow_unqualified_fetch=error ASTERISK-25826 #close Reported-by: Ross Beer Tested-by: Ross Beer Change-Id: Id2691e447db90892890036e663aaf907b2dc1c67
1135 lines
30 KiB
C
1135 lines
30 KiB
C
/*
|
|
* Asterisk -- An open source telephony toolkit.
|
|
*
|
|
* Copyright (C) 2013, Digium, Inc.
|
|
*
|
|
* Mark Michelson <mmichelson@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.
|
|
*/
|
|
|
|
/*** MODULEINFO
|
|
<depend>pjproject</depend>
|
|
<depend>res_pjsip</depend>
|
|
<depend>res_pjsip_pubsub</depend>
|
|
<support_level>core</support_level>
|
|
***/
|
|
|
|
#include "asterisk.h"
|
|
|
|
#include <pjsip.h>
|
|
#include <pjsip_simple.h>
|
|
#include <pjlib.h>
|
|
|
|
#include "asterisk/res_pjsip.h"
|
|
#include "asterisk/res_pjsip_pubsub.h"
|
|
#include "asterisk/res_pjsip_body_generator_types.h"
|
|
#include "asterisk/module.h"
|
|
#include "asterisk/logger.h"
|
|
#include "asterisk/astobj2.h"
|
|
#include "asterisk/sorcery.h"
|
|
#include "asterisk/stasis.h"
|
|
#include "asterisk/app.h"
|
|
|
|
struct mwi_subscription;
|
|
static struct ao2_container *unsolicited_mwi;
|
|
|
|
#define STASIS_BUCKETS 13
|
|
#define MWI_BUCKETS 53
|
|
|
|
#define MWI_TYPE "application"
|
|
#define MWI_SUBTYPE "simple-message-summary"
|
|
|
|
#define MWI_DATASTORE "MWI datastore"
|
|
|
|
static void mwi_subscription_shutdown(struct ast_sip_subscription *sub);
|
|
static void mwi_to_ami(struct ast_sip_subscription *sub, struct ast_str **buf);
|
|
static int mwi_new_subscribe(struct ast_sip_endpoint *endpoint,
|
|
const char *resource);
|
|
static int mwi_subscription_established(struct ast_sip_subscription *sub);
|
|
static void *mwi_get_notify_data(struct ast_sip_subscription *sub);
|
|
|
|
static struct ast_sip_notifier mwi_notifier = {
|
|
.default_accept = MWI_TYPE"/"MWI_SUBTYPE,
|
|
.new_subscribe = mwi_new_subscribe,
|
|
.subscription_established = mwi_subscription_established,
|
|
.get_notify_data = mwi_get_notify_data,
|
|
};
|
|
|
|
static struct ast_sip_subscription_handler mwi_handler = {
|
|
.event_name = "message-summary",
|
|
.body_type = AST_SIP_MESSAGE_ACCUMULATOR,
|
|
.accept = { MWI_TYPE"/"MWI_SUBTYPE, },
|
|
.subscription_shutdown = mwi_subscription_shutdown,
|
|
.to_ami = mwi_to_ami,
|
|
.notifier = &mwi_notifier,
|
|
};
|
|
|
|
/*!
|
|
* \brief Wrapper for stasis subscription
|
|
*
|
|
* An MWI subscription has a container of these. This
|
|
* represents a stasis subscription for MWI state.
|
|
*/
|
|
struct mwi_stasis_subscription {
|
|
/*! The MWI stasis subscription */
|
|
struct stasis_subscription *stasis_sub;
|
|
/*! The mailbox corresponding with the MWI subscription. Used as a hash key */
|
|
char mailbox[1];
|
|
};
|
|
|
|
/*!
|
|
* \brief A subscription for MWI
|
|
*
|
|
* This subscription is the basis for MWI for an endpoint. Each
|
|
* endpoint that uses MWI will have a corresponding mwi_subscription.
|
|
*
|
|
* This structure acts as the owner for the underlying SIP subscription.
|
|
* When the mwi_subscription is destroyed, the SIP subscription dies, too.
|
|
* The mwi_subscription's lifetime is governed by its underlying stasis
|
|
* subscriptions. When all stasis subscriptions are destroyed, the
|
|
* mwi_subscription is destroyed as well.
|
|
*/
|
|
struct mwi_subscription {
|
|
/*! Container of \ref mwi_stasis_subscription structures.
|
|
* A single MWI subscription may be for multiple mailboxes, thus
|
|
* requiring multiple stasis subscriptions
|
|
*/
|
|
struct ao2_container *stasis_subs;
|
|
/*! The SIP subscription. Unsolicited MWI does not use this */
|
|
struct ast_sip_subscription *sip_sub;
|
|
/*! AORs we should react to for unsolicited MWI NOTIFY */
|
|
char *aors;
|
|
/*! Is the MWI solicited (i.e. Initiated with an external SUBSCRIBE) ? */
|
|
unsigned int is_solicited;
|
|
/*! Identifier for the subscription.
|
|
* The identifier is the same as the corresponding endpoint's stasis ID.
|
|
* Used as a hash key
|
|
*/
|
|
char id[1];
|
|
};
|
|
|
|
static void mwi_stasis_cb(void *userdata, struct stasis_subscription *sub,
|
|
struct stasis_message *msg);
|
|
|
|
static struct mwi_stasis_subscription *mwi_stasis_subscription_alloc(const char *mailbox, struct mwi_subscription *mwi_sub)
|
|
{
|
|
struct mwi_stasis_subscription *mwi_stasis_sub;
|
|
struct stasis_topic *topic;
|
|
|
|
if (!mwi_sub) {
|
|
return NULL;
|
|
}
|
|
|
|
mwi_stasis_sub = ao2_alloc(sizeof(*mwi_stasis_sub) + strlen(mailbox), NULL);
|
|
if (!mwi_stasis_sub) {
|
|
return NULL;
|
|
}
|
|
|
|
topic = ast_mwi_topic(mailbox);
|
|
|
|
/* Safe strcpy */
|
|
strcpy(mwi_stasis_sub->mailbox, mailbox);
|
|
|
|
ast_debug(3, "Creating stasis MWI subscription to mailbox %s for endpoint %s\n",
|
|
mailbox, mwi_sub->id);
|
|
ao2_ref(mwi_sub, +1);
|
|
mwi_stasis_sub->stasis_sub = stasis_subscribe_pool(topic, mwi_stasis_cb, mwi_sub);
|
|
if (!mwi_stasis_sub->stasis_sub) {
|
|
/* Failed to subscribe. */
|
|
ao2_ref(mwi_stasis_sub, -1);
|
|
ao2_ref(mwi_sub, -1);
|
|
mwi_stasis_sub = NULL;
|
|
}
|
|
return mwi_stasis_sub;
|
|
}
|
|
|
|
static int stasis_sub_hash(const void *obj, const int flags)
|
|
{
|
|
const struct mwi_stasis_subscription *object;
|
|
const char *key;
|
|
|
|
switch (flags & OBJ_SEARCH_MASK) {
|
|
case OBJ_SEARCH_KEY:
|
|
key = obj;
|
|
break;
|
|
case OBJ_SEARCH_OBJECT:
|
|
object = obj;
|
|
key = object->mailbox;
|
|
break;
|
|
default:
|
|
ast_assert(0);
|
|
return 0;
|
|
}
|
|
return ast_str_hash(key);
|
|
}
|
|
|
|
static int stasis_sub_cmp(void *obj, void *arg, int flags)
|
|
{
|
|
const struct mwi_stasis_subscription *sub_left = obj;
|
|
const struct mwi_stasis_subscription *sub_right = arg;
|
|
const char *right_key = arg;
|
|
int cmp;
|
|
|
|
switch (flags & OBJ_SEARCH_MASK) {
|
|
case OBJ_SEARCH_OBJECT:
|
|
right_key = sub_right->mailbox;
|
|
/* Fall through */
|
|
case OBJ_SEARCH_KEY:
|
|
cmp = strcmp(sub_left->mailbox, right_key);
|
|
break;
|
|
case OBJ_SEARCH_PARTIAL_KEY:
|
|
cmp = strncmp(sub_left->mailbox, right_key, strlen(right_key));
|
|
break;
|
|
default:
|
|
cmp = 0;
|
|
break;
|
|
}
|
|
if (cmp) {
|
|
return 0;
|
|
}
|
|
return CMP_MATCH;
|
|
}
|
|
|
|
static void mwi_subscription_destructor(void *obj)
|
|
{
|
|
struct mwi_subscription *sub = obj;
|
|
|
|
ast_debug(3, "Destroying MWI subscription for endpoint %s\n", sub->id);
|
|
if (sub->is_solicited) {
|
|
ast_sip_subscription_destroy(sub->sip_sub);
|
|
}
|
|
ao2_cleanup(sub->stasis_subs);
|
|
ast_free(sub->aors);
|
|
}
|
|
|
|
static struct mwi_subscription *mwi_subscription_alloc(struct ast_sip_endpoint *endpoint,
|
|
unsigned int is_solicited, struct ast_sip_subscription *sip_sub)
|
|
{
|
|
struct mwi_subscription *sub;
|
|
const char *endpoint_id = ast_sorcery_object_get_id(endpoint);
|
|
|
|
sub = ao2_alloc(sizeof(*sub) + strlen(endpoint_id),
|
|
mwi_subscription_destructor);
|
|
|
|
if (!sub) {
|
|
return NULL;
|
|
}
|
|
|
|
/* Safe strcpy */
|
|
strcpy(sub->id, endpoint_id);
|
|
|
|
/* Unsolicited MWI doesn't actually result in a SIP subscription being
|
|
* created. This is because a SIP subscription associates with a dialog.
|
|
* Most devices expect unsolicited MWI NOTIFYs to appear out of dialog. If
|
|
* they receive an in-dialog MWI NOTIFY (i.e. with a to-tag), then they
|
|
* will reject the NOTIFY with a 481, thus resulting in message-waiting
|
|
* state not being updated on the device
|
|
*/
|
|
if (is_solicited) {
|
|
sub->sip_sub = sip_sub;
|
|
}
|
|
|
|
sub->stasis_subs = ao2_container_alloc(STASIS_BUCKETS, stasis_sub_hash, stasis_sub_cmp);
|
|
if (!sub->stasis_subs) {
|
|
ao2_cleanup(sub);
|
|
return NULL;
|
|
}
|
|
sub->is_solicited = is_solicited;
|
|
|
|
if (!is_solicited && !ast_strlen_zero(endpoint->aors)) {
|
|
sub->aors = ast_strdup(endpoint->aors);
|
|
if (!sub->aors) {
|
|
ao2_ref(sub, -1);
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
ast_debug(3, "Created %s MWI subscription for endpoint %s\n", is_solicited ? "solicited" : "unsolicited", sub->id);
|
|
|
|
return sub;
|
|
}
|
|
|
|
static int mwi_sub_hash(const void *obj, const int flags)
|
|
{
|
|
const struct mwi_subscription *object;
|
|
const char *key;
|
|
|
|
switch (flags & OBJ_SEARCH_MASK) {
|
|
case OBJ_SEARCH_KEY:
|
|
key = obj;
|
|
break;
|
|
case OBJ_SEARCH_OBJECT:
|
|
object = obj;
|
|
key = object->id;
|
|
break;
|
|
default:
|
|
ast_assert(0);
|
|
return 0;
|
|
}
|
|
return ast_str_hash(key);
|
|
}
|
|
|
|
static int mwi_sub_cmp(void *obj, void *arg, int flags)
|
|
{
|
|
const struct mwi_subscription *sub_left = obj;
|
|
const struct mwi_subscription *sub_right = arg;
|
|
const char *right_key = arg;
|
|
int cmp;
|
|
|
|
switch (flags & OBJ_SEARCH_MASK) {
|
|
case OBJ_SEARCH_OBJECT:
|
|
right_key = sub_right->id;
|
|
/* Fall through */
|
|
case OBJ_SEARCH_KEY:
|
|
cmp = strcmp(sub_left->id, right_key);
|
|
break;
|
|
case OBJ_SEARCH_PARTIAL_KEY:
|
|
cmp = strncmp(sub_left->id, right_key, strlen(right_key));
|
|
break;
|
|
default:
|
|
cmp = 0;
|
|
break;
|
|
}
|
|
if (cmp) {
|
|
return 0;
|
|
}
|
|
return CMP_MATCH;
|
|
}
|
|
|
|
static int get_message_count(void *obj, void *arg, int flags)
|
|
{
|
|
struct stasis_message *msg;
|
|
struct mwi_stasis_subscription *mwi_stasis = obj;
|
|
struct ast_sip_message_accumulator *counter = arg;
|
|
struct ast_mwi_state *mwi_state;
|
|
|
|
msg = stasis_cache_get(ast_mwi_state_cache(), ast_mwi_state_type(), mwi_stasis->mailbox);
|
|
if (!msg) {
|
|
return 0;
|
|
}
|
|
|
|
mwi_state = stasis_message_data(msg);
|
|
counter->old_msgs += mwi_state->old_msgs;
|
|
counter->new_msgs += mwi_state->new_msgs;
|
|
|
|
ao2_ref(msg, -1);
|
|
|
|
return 0;
|
|
}
|
|
|
|
struct unsolicited_mwi_data {
|
|
struct mwi_subscription *sub;
|
|
struct ast_sip_endpoint *endpoint;
|
|
pjsip_evsub_state state;
|
|
const struct ast_sip_body *body;
|
|
};
|
|
|
|
static int send_unsolicited_mwi_notify_to_contact(void *obj, void *arg, int flags)
|
|
{
|
|
struct unsolicited_mwi_data *mwi_data = arg;
|
|
struct mwi_subscription *sub = mwi_data->sub;
|
|
struct ast_sip_endpoint *endpoint = mwi_data->endpoint;
|
|
pjsip_evsub_state state = mwi_data->state;
|
|
const struct ast_sip_body *body = mwi_data->body;
|
|
struct ast_sip_contact *contact = obj;
|
|
const char *state_name;
|
|
pjsip_tx_data *tdata;
|
|
pjsip_sub_state_hdr *sub_state;
|
|
pjsip_event_hdr *event;
|
|
const pjsip_hdr *allow_events = pjsip_evsub_get_allow_events_hdr(NULL);
|
|
|
|
if (ast_sip_create_request("NOTIFY", NULL, endpoint, NULL, contact, &tdata)) {
|
|
ast_log(LOG_WARNING, "Unable to create unsolicited NOTIFY request to endpoint %s URI %s\n", sub->id, contact->uri);
|
|
return 0;
|
|
}
|
|
|
|
if (!ast_strlen_zero(endpoint->subscription.mwi.fromuser)) {
|
|
pjsip_fromto_hdr *from = pjsip_msg_find_hdr(tdata->msg, PJSIP_H_FROM, NULL);
|
|
pjsip_name_addr *from_name_addr = (pjsip_name_addr *) from->uri;
|
|
pjsip_sip_uri *from_uri = pjsip_uri_get_uri(from_name_addr->uri);
|
|
|
|
pj_strdup2(tdata->pool, &from_uri->user, endpoint->subscription.mwi.fromuser);
|
|
}
|
|
|
|
switch (state) {
|
|
case PJSIP_EVSUB_STATE_ACTIVE:
|
|
state_name = "active";
|
|
break;
|
|
case PJSIP_EVSUB_STATE_TERMINATED:
|
|
default:
|
|
state_name = "terminated";
|
|
break;
|
|
}
|
|
|
|
sub_state = pjsip_sub_state_hdr_create(tdata->pool);
|
|
pj_cstr(&sub_state->sub_state, state_name);
|
|
pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr *) sub_state);
|
|
|
|
event = pjsip_event_hdr_create(tdata->pool);
|
|
pj_cstr(&event->event_type, "message-summary");
|
|
pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr *) event);
|
|
|
|
pjsip_msg_add_hdr(tdata->msg, pjsip_hdr_shallow_clone(tdata->pool, allow_events));
|
|
ast_sip_add_body(tdata, body);
|
|
ast_sip_send_request(tdata, NULL, endpoint, NULL, NULL);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void send_unsolicited_mwi_notify(struct mwi_subscription *sub,
|
|
struct ast_sip_message_accumulator *counter)
|
|
{
|
|
RAII_VAR(struct ast_sip_endpoint *, endpoint, ast_sorcery_retrieve_by_id(ast_sip_get_sorcery(),
|
|
"endpoint", sub->id), ao2_cleanup);
|
|
char *endpoint_aors;
|
|
char *aor_name;
|
|
struct ast_sip_body body;
|
|
struct ast_str *body_text;
|
|
struct ast_sip_body_data body_data = {
|
|
.body_type = AST_SIP_MESSAGE_ACCUMULATOR,
|
|
.body_data = counter,
|
|
};
|
|
|
|
if (!endpoint) {
|
|
ast_log(LOG_WARNING, "Unable to send unsolicited MWI to %s because endpoint does not exist\n",
|
|
sub->id);
|
|
return;
|
|
}
|
|
if (ast_strlen_zero(endpoint->aors)) {
|
|
ast_log(LOG_WARNING, "Unable to send unsolicited MWI to %s because the endpoint has no"
|
|
" configured AORs\n", sub->id);
|
|
return;
|
|
}
|
|
|
|
body.type = MWI_TYPE;
|
|
body.subtype = MWI_SUBTYPE;
|
|
|
|
body_text = ast_str_create(64);
|
|
|
|
if (!body_text) {
|
|
return;
|
|
}
|
|
|
|
if (ast_sip_pubsub_generate_body_content(body.type, body.subtype, &body_data, &body_text)) {
|
|
ast_log(LOG_WARNING, "Unable to generate SIP MWI NOTIFY body.\n");
|
|
ast_free(body_text);
|
|
return;
|
|
}
|
|
|
|
body.body_text = ast_str_buffer(body_text);
|
|
|
|
endpoint_aors = ast_strdupa(endpoint->aors);
|
|
|
|
ast_debug(5, "Sending unsolicited MWI NOTIFY to endpoint %s, new messages: %d, old messages: %d\n",
|
|
sub->id, counter->new_msgs, counter->old_msgs);
|
|
|
|
while ((aor_name = ast_strip(strsep(&endpoint_aors, ",")))) {
|
|
RAII_VAR(struct ast_sip_aor *, aor, ast_sip_location_retrieve_aor(aor_name), ao2_cleanup);
|
|
RAII_VAR(struct ao2_container *, contacts, NULL, ao2_cleanup);
|
|
struct unsolicited_mwi_data mwi_data = {
|
|
.sub = sub,
|
|
.endpoint = endpoint,
|
|
.body = &body,
|
|
};
|
|
|
|
if (!aor) {
|
|
ast_log(LOG_WARNING, "Unable to locate AOR %s for unsolicited MWI\n", aor_name);
|
|
continue;
|
|
}
|
|
|
|
contacts = ast_sip_location_retrieve_aor_contacts(aor);
|
|
if (!contacts || (ao2_container_count(contacts) == 0)) {
|
|
ast_debug(1, "No contacts bound to AOR %s. Cannot send unsolicited MWI until a contact registers.\n", aor_name);
|
|
continue;
|
|
}
|
|
|
|
ao2_callback(contacts, OBJ_NODATA, send_unsolicited_mwi_notify_to_contact, &mwi_data);
|
|
}
|
|
|
|
ast_free(body_text);
|
|
}
|
|
|
|
static void send_mwi_notify(struct mwi_subscription *sub)
|
|
{
|
|
struct ast_sip_message_accumulator counter = {
|
|
.old_msgs = 0,
|
|
.new_msgs = 0,
|
|
};
|
|
struct ast_sip_body_data data = {
|
|
.body_type = AST_SIP_MESSAGE_ACCUMULATOR,
|
|
.body_data = &counter,
|
|
};
|
|
|
|
ao2_callback(sub->stasis_subs, OBJ_NODATA, get_message_count, &counter);
|
|
|
|
if (sub->is_solicited) {
|
|
ast_sip_subscription_notify(sub->sip_sub, &data, 0);
|
|
return;
|
|
}
|
|
|
|
send_unsolicited_mwi_notify(sub, &counter);
|
|
}
|
|
|
|
static int unsubscribe_stasis(void *obj, void *arg, int flags)
|
|
{
|
|
struct mwi_stasis_subscription *mwi_stasis = obj;
|
|
if (mwi_stasis->stasis_sub) {
|
|
ast_debug(3, "Removing stasis subscription to mailbox %s\n", mwi_stasis->mailbox);
|
|
mwi_stasis->stasis_sub = stasis_unsubscribe_and_join(mwi_stasis->stasis_sub);
|
|
}
|
|
return CMP_MATCH;
|
|
}
|
|
|
|
static void mwi_subscription_shutdown(struct ast_sip_subscription *sub)
|
|
{
|
|
struct mwi_subscription *mwi_sub;
|
|
struct ast_datastore *mwi_datastore;
|
|
|
|
mwi_datastore = ast_sip_subscription_get_datastore(sub, MWI_DATASTORE);
|
|
if (!mwi_datastore) {
|
|
return;
|
|
}
|
|
|
|
mwi_sub = mwi_datastore->data;
|
|
ao2_callback(mwi_sub->stasis_subs, OBJ_UNLINK | OBJ_NODATA | OBJ_MULTIPLE, unsubscribe_stasis, NULL);
|
|
ast_sip_subscription_remove_datastore(sub, MWI_DATASTORE);
|
|
|
|
ao2_ref(mwi_datastore, -1);
|
|
}
|
|
|
|
static void mwi_ds_destroy(void *data)
|
|
{
|
|
struct mwi_subscription *sub = data;
|
|
|
|
ao2_ref(sub, -1);
|
|
}
|
|
|
|
static struct ast_datastore_info mwi_ds_info = {
|
|
.destroy = mwi_ds_destroy,
|
|
};
|
|
|
|
static int add_mwi_datastore(struct mwi_subscription *sub)
|
|
{
|
|
struct ast_datastore *mwi_datastore;
|
|
int res;
|
|
|
|
mwi_datastore = ast_sip_subscription_alloc_datastore(&mwi_ds_info, MWI_DATASTORE);
|
|
if (!mwi_datastore) {
|
|
return -1;
|
|
}
|
|
ao2_ref(sub, +1);
|
|
mwi_datastore->data = sub;
|
|
|
|
/*
|
|
* NOTE: Adding the datastore to the subscription creates a ref loop
|
|
* that must be manually broken.
|
|
*/
|
|
res = ast_sip_subscription_add_datastore(sub->sip_sub, mwi_datastore);
|
|
ao2_ref(mwi_datastore, -1);
|
|
return res;
|
|
}
|
|
|
|
/*!
|
|
* \brief Determines if an endpoint is receiving unsolicited MWI for a particular mailbox.
|
|
*
|
|
* \param endpoint The endpoint to check
|
|
* \param mailbox The candidate mailbox
|
|
* \retval 0 The endpoint does not receive unsolicited MWI for this mailbox
|
|
* \retval 1 The endpoint receives unsolicited MWI for this mailbox
|
|
*/
|
|
static int endpoint_receives_unsolicited_mwi_for_mailbox(struct ast_sip_endpoint *endpoint,
|
|
const char *mailbox)
|
|
{
|
|
struct ao2_iterator *mwi_subs;
|
|
struct mwi_subscription *mwi_sub;
|
|
const char *endpoint_id = ast_sorcery_object_get_id(endpoint);
|
|
int ret = 0;
|
|
|
|
mwi_subs = ao2_find(unsolicited_mwi, endpoint_id, OBJ_SEARCH_KEY | OBJ_MULTIPLE);
|
|
|
|
if (!mwi_subs) {
|
|
return 0;
|
|
}
|
|
|
|
for (; (mwi_sub = ao2_iterator_next(mwi_subs)) && !ret; ao2_cleanup(mwi_sub)) {
|
|
struct mwi_stasis_subscription *mwi_stasis;
|
|
|
|
mwi_stasis = ao2_find(mwi_sub->stasis_subs, mailbox, OBJ_SEARCH_KEY);
|
|
if (mwi_stasis) {
|
|
ret = 1;
|
|
ao2_cleanup(mwi_stasis);
|
|
}
|
|
}
|
|
|
|
ao2_iterator_destroy(mwi_subs);
|
|
return ret;
|
|
}
|
|
|
|
/*!
|
|
* \brief Determine if an endpoint is a candidate to be able to subscribe for MWI
|
|
*
|
|
* Currently, this just makes sure that the endpoint is not already receiving unsolicted
|
|
* MWI for any of an AOR's configured mailboxes.
|
|
*
|
|
* \param obj The AOR to which the endpoint is subscribing.
|
|
* \param arg The endpoint that is attempting to subscribe.
|
|
* \param flags Unused.
|
|
* \retval 0 Endpoint is a candidate to subscribe to MWI on the AOR.
|
|
* \retval -1 The endpoint cannot subscribe to MWI on the AOR.
|
|
*/
|
|
static int mwi_validate_for_aor(void *obj, void *arg, int flags)
|
|
{
|
|
struct ast_sip_aor *aor = obj;
|
|
struct ast_sip_endpoint *endpoint = arg;
|
|
char *mailboxes;
|
|
char *mailbox;
|
|
|
|
if (ast_strlen_zero(aor->mailboxes)) {
|
|
return 0;
|
|
}
|
|
|
|
mailboxes = ast_strdupa(aor->mailboxes);
|
|
while ((mailbox = ast_strip(strsep(&mailboxes, ",")))) {
|
|
if (ast_strlen_zero(mailbox)) {
|
|
continue;
|
|
}
|
|
|
|
if (endpoint_receives_unsolicited_mwi_for_mailbox(endpoint, mailbox)) {
|
|
ast_debug(1, "Endpoint '%s' already configured for unsolicited MWI for mailbox '%s'. "
|
|
"Denying MWI subscription to %s\n", ast_sorcery_object_get_id(endpoint), mailbox,
|
|
ast_sorcery_object_get_id(aor));
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int mwi_on_aor(void *obj, void *arg, int flags)
|
|
{
|
|
struct ast_sip_aor *aor = obj;
|
|
struct mwi_subscription *sub = arg;
|
|
char *mailboxes;
|
|
char *mailbox;
|
|
|
|
if (ast_strlen_zero(aor->mailboxes)) {
|
|
return 0;
|
|
}
|
|
|
|
mailboxes = ast_strdupa(aor->mailboxes);
|
|
while ((mailbox = ast_strip(strsep(&mailboxes, ",")))) {
|
|
struct mwi_stasis_subscription *mwi_stasis_sub;
|
|
|
|
if (ast_strlen_zero(mailbox)) {
|
|
continue;
|
|
}
|
|
|
|
mwi_stasis_sub = mwi_stasis_subscription_alloc(mailbox, sub);
|
|
if (!mwi_stasis_sub) {
|
|
continue;
|
|
}
|
|
|
|
ao2_link(sub->stasis_subs, mwi_stasis_sub);
|
|
ao2_ref(mwi_stasis_sub, -1);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct mwi_subscription *mwi_create_subscription(
|
|
struct ast_sip_endpoint *endpoint, struct ast_sip_subscription *sip_sub)
|
|
{
|
|
struct mwi_subscription *sub = mwi_subscription_alloc(endpoint, 1, sip_sub);
|
|
|
|
if (!sub) {
|
|
return NULL;
|
|
}
|
|
|
|
if (add_mwi_datastore(sub)) {
|
|
ast_log(LOG_WARNING, "Unable to add datastore for MWI subscription to %s\n",
|
|
sub->id);
|
|
ao2_ref(sub, -1);
|
|
return NULL;
|
|
}
|
|
|
|
return sub;
|
|
}
|
|
|
|
static struct mwi_subscription *mwi_subscribe_single(
|
|
struct ast_sip_endpoint *endpoint, struct ast_sip_subscription *sip_sub, const char *name)
|
|
{
|
|
struct ast_sip_aor *aor;
|
|
struct mwi_subscription *sub;
|
|
|
|
aor = ast_sip_location_retrieve_aor(name);
|
|
if (!aor) {
|
|
/*! I suppose it's possible for the AOR to disappear on us
|
|
* between accepting the subscription and sending the first
|
|
* NOTIFY...
|
|
*/
|
|
ast_log(LOG_WARNING, "Unable to locate aor %s. MWI subscription failed.\n",
|
|
name);
|
|
return NULL;
|
|
}
|
|
|
|
sub = mwi_create_subscription(endpoint, sip_sub);
|
|
if (sub) {
|
|
mwi_on_aor(aor, sub, 0);
|
|
}
|
|
|
|
ao2_ref(aor, -1);
|
|
return sub;
|
|
}
|
|
|
|
static struct mwi_subscription *mwi_subscribe_all(
|
|
struct ast_sip_endpoint *endpoint, struct ast_sip_subscription *sip_sub)
|
|
{
|
|
struct mwi_subscription *sub;
|
|
|
|
sub = mwi_create_subscription(endpoint, sip_sub);
|
|
if (!sub) {
|
|
return NULL;
|
|
}
|
|
|
|
ast_sip_for_each_aor(endpoint->aors, mwi_on_aor, sub);
|
|
return sub;
|
|
}
|
|
|
|
static int mwi_new_subscribe(struct ast_sip_endpoint *endpoint,
|
|
const char *resource)
|
|
{
|
|
RAII_VAR(struct ast_sip_aor *, aor, NULL, ao2_cleanup);
|
|
|
|
if (ast_strlen_zero(resource)) {
|
|
if (ast_sip_for_each_aor(endpoint->aors, mwi_validate_for_aor, endpoint)) {
|
|
return 500;
|
|
}
|
|
return 200;
|
|
}
|
|
|
|
aor = ast_sip_location_retrieve_aor(resource);
|
|
if (!aor) {
|
|
ast_debug(1, "Unable to locate aor %s. MWI subscription failed.\n",
|
|
resource);
|
|
return 404;
|
|
}
|
|
|
|
if (ast_strlen_zero(aor->mailboxes)) {
|
|
ast_debug(1, "AOR %s has no configured mailboxes. MWI subscription failed.\n",
|
|
resource);
|
|
return 404;
|
|
}
|
|
|
|
if (mwi_validate_for_aor(aor, endpoint, 0)) {
|
|
return 500;
|
|
}
|
|
|
|
return 200;
|
|
}
|
|
|
|
static int mwi_subscription_established(struct ast_sip_subscription *sip_sub)
|
|
{
|
|
const char *resource = ast_sip_subscription_get_resource_name(sip_sub);
|
|
struct mwi_subscription *sub;
|
|
struct ast_sip_endpoint *endpoint = ast_sip_subscription_get_endpoint(sip_sub);
|
|
|
|
/* no aor in uri? subscribe to all on endpoint */
|
|
if (ast_strlen_zero(resource)) {
|
|
sub = mwi_subscribe_all(endpoint, sip_sub);
|
|
} else {
|
|
sub = mwi_subscribe_single(endpoint, sip_sub, resource);
|
|
}
|
|
if (!sub) {
|
|
ao2_cleanup(endpoint);
|
|
return -1;
|
|
}
|
|
|
|
if (!ao2_container_count(sub->stasis_subs)) {
|
|
/*
|
|
* We setup no MWI subscriptions so remove the MWI datastore
|
|
* to break the ref loop.
|
|
*/
|
|
ast_sip_subscription_remove_datastore(sip_sub, MWI_DATASTORE);
|
|
}
|
|
|
|
ao2_cleanup(sub);
|
|
ao2_cleanup(endpoint);
|
|
return 0;
|
|
}
|
|
|
|
static void *mwi_get_notify_data(struct ast_sip_subscription *sub)
|
|
{
|
|
struct ast_sip_message_accumulator *counter;
|
|
struct mwi_subscription *mwi_sub;
|
|
struct ast_datastore *mwi_datastore;
|
|
|
|
mwi_datastore = ast_sip_subscription_get_datastore(sub, MWI_DATASTORE);
|
|
if (!mwi_datastore) {
|
|
return NULL;
|
|
}
|
|
mwi_sub = mwi_datastore->data;
|
|
|
|
counter = ao2_alloc(sizeof(*counter), NULL);
|
|
if (!counter) {
|
|
ao2_cleanup(mwi_datastore);
|
|
return NULL;
|
|
}
|
|
|
|
ao2_callback(mwi_sub->stasis_subs, OBJ_NODATA, get_message_count, counter);
|
|
ao2_cleanup(mwi_datastore);
|
|
return counter;
|
|
}
|
|
|
|
static void mwi_subscription_mailboxes_str(struct ao2_container *stasis_subs,
|
|
struct ast_str **str)
|
|
{
|
|
int is_first = 1;
|
|
struct mwi_stasis_subscription *node;
|
|
struct ao2_iterator i = ao2_iterator_init(stasis_subs, 0);
|
|
|
|
while ((node = ao2_iterator_next(&i))) {
|
|
if (is_first) {
|
|
is_first = 0;
|
|
ast_str_append(str, 0, "%s", node->mailbox);
|
|
} else {
|
|
ast_str_append(str, 0, ",%s", node->mailbox);
|
|
}
|
|
ao2_ref(node, -1);
|
|
}
|
|
ao2_iterator_destroy(&i);
|
|
}
|
|
|
|
static void mwi_to_ami(struct ast_sip_subscription *sub,
|
|
struct ast_str **buf)
|
|
{
|
|
struct mwi_subscription *mwi_sub;
|
|
struct ast_datastore *mwi_datastore;
|
|
|
|
mwi_datastore = ast_sip_subscription_get_datastore(sub, MWI_DATASTORE);
|
|
if (!mwi_datastore) {
|
|
return;
|
|
}
|
|
|
|
mwi_sub = mwi_datastore->data;
|
|
|
|
ast_str_append(buf, 0, "SubscriptionType: mwi\r\n");
|
|
ast_str_append(buf, 0, "Mailboxes: ");
|
|
mwi_subscription_mailboxes_str(mwi_sub->stasis_subs, buf);
|
|
ast_str_append(buf, 0, "\r\n");
|
|
|
|
ao2_ref(mwi_datastore, -1);
|
|
}
|
|
|
|
static int serialized_notify(void *userdata)
|
|
{
|
|
struct mwi_subscription *mwi_sub = userdata;
|
|
|
|
send_mwi_notify(mwi_sub);
|
|
ao2_ref(mwi_sub, -1);
|
|
return 0;
|
|
}
|
|
|
|
static int serialized_cleanup(void *userdata)
|
|
{
|
|
struct mwi_subscription *mwi_sub = userdata;
|
|
|
|
/* This is getting rid of the reference that was added
|
|
* just before this serialized task was pushed.
|
|
*/
|
|
ao2_cleanup(mwi_sub);
|
|
/* This is getting rid of the reference held by the
|
|
* stasis subscription
|
|
*/
|
|
ao2_cleanup(mwi_sub);
|
|
return 0;
|
|
}
|
|
|
|
static int send_notify(void *obj, void *arg, int flags)
|
|
{
|
|
struct mwi_subscription *mwi_sub = obj;
|
|
struct ast_taskprocessor *serializer = mwi_sub->is_solicited
|
|
? ast_sip_subscription_get_serializer(mwi_sub->sip_sub)
|
|
: NULL;
|
|
|
|
if (ast_sip_push_task(serializer, serialized_notify, ao2_bump(mwi_sub))) {
|
|
ao2_ref(mwi_sub, -1);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void mwi_stasis_cb(void *userdata, struct stasis_subscription *sub,
|
|
struct stasis_message *msg)
|
|
{
|
|
struct mwi_subscription *mwi_sub = userdata;
|
|
|
|
if (stasis_subscription_final_message(sub, msg)) {
|
|
if (ast_sip_push_task(NULL, serialized_cleanup, ao2_bump(mwi_sub))) {
|
|
ao2_ref(mwi_sub, -1);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (ast_mwi_state_type() == stasis_message_type(msg)) {
|
|
send_notify(mwi_sub, NULL, 0);
|
|
}
|
|
}
|
|
|
|
/*! \note Called with the unsolicited_mwi conainer lock held. */
|
|
static int create_mwi_subscriptions_for_endpoint(void *obj, void *arg, int flags)
|
|
{
|
|
RAII_VAR(struct mwi_subscription *, aggregate_sub, NULL, ao2_cleanup);
|
|
struct ast_sip_endpoint *endpoint = obj;
|
|
char *endpoint_aors, *aor_name, *mailboxes, *mailbox;
|
|
struct ao2_container *contacts = NULL;
|
|
|
|
if (ast_strlen_zero(endpoint->subscription.mwi.mailboxes)) {
|
|
return 0;
|
|
}
|
|
|
|
endpoint_aors = ast_strdupa(endpoint->aors);
|
|
|
|
while ((aor_name = ast_strip(strsep(&endpoint_aors, ",")))) {
|
|
RAII_VAR(struct ast_sip_aor *, aor, ast_sip_location_retrieve_aor(aor_name), ao2_cleanup);
|
|
|
|
if (!aor) {
|
|
continue;
|
|
}
|
|
|
|
contacts = ast_sip_location_retrieve_aor_contacts(aor);
|
|
if (!contacts || (ao2_container_count(contacts) == 0)) {
|
|
ao2_cleanup(contacts);
|
|
contacts = NULL;
|
|
continue;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
if (!contacts) {
|
|
return 0;
|
|
}
|
|
|
|
ao2_ref(contacts, -1);
|
|
|
|
if (endpoint->subscription.mwi.aggregate) {
|
|
aggregate_sub = mwi_subscription_alloc(endpoint, 0, NULL);
|
|
if (!aggregate_sub) {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
mailboxes = ast_strdupa(endpoint->subscription.mwi.mailboxes);
|
|
while ((mailbox = ast_strip(strsep(&mailboxes, ",")))) {
|
|
struct mwi_subscription *sub;
|
|
struct mwi_stasis_subscription *mwi_stasis_sub;
|
|
|
|
if (ast_strlen_zero(mailbox)) {
|
|
continue;
|
|
}
|
|
|
|
sub = aggregate_sub ?: mwi_subscription_alloc(endpoint, 0, NULL);
|
|
mwi_stasis_sub = mwi_stasis_subscription_alloc(mailbox, sub);
|
|
if (mwi_stasis_sub) {
|
|
ao2_link(sub->stasis_subs, mwi_stasis_sub);
|
|
ao2_ref(mwi_stasis_sub, -1);
|
|
}
|
|
if (!aggregate_sub && sub) {
|
|
ao2_link_flags(unsolicited_mwi, sub, OBJ_NOLOCK);
|
|
ao2_ref(sub, -1);
|
|
}
|
|
}
|
|
if (aggregate_sub) {
|
|
ao2_link_flags(unsolicited_mwi, aggregate_sub, OBJ_NOLOCK);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int unsubscribe(void *obj, void *arg, int flags)
|
|
{
|
|
struct mwi_subscription *mwi_sub = obj;
|
|
|
|
ao2_callback(mwi_sub->stasis_subs, OBJ_UNLINK | OBJ_NODATA | OBJ_MULTIPLE, unsubscribe_stasis, NULL);
|
|
|
|
return CMP_MATCH;
|
|
}
|
|
|
|
static void create_mwi_subscriptions(void)
|
|
{
|
|
struct ao2_container *endpoints;
|
|
struct ast_variable *var;
|
|
|
|
var = ast_variable_new("mailboxes !=", "", "");
|
|
|
|
endpoints = ast_sorcery_retrieve_by_fields(ast_sip_get_sorcery(), "endpoint",
|
|
AST_RETRIEVE_FLAG_MULTIPLE, var);
|
|
|
|
ast_variables_destroy(var);
|
|
if (!endpoints) {
|
|
return;
|
|
}
|
|
|
|
/* We remove all the old stasis subscriptions first before applying the new configuration. This
|
|
* prevents a situation where there might be multiple overlapping stasis subscriptions for an
|
|
* endpoint for mailboxes. Though there may be mailbox changes during the gap between unsubscribing
|
|
* and resubscribing, up-to-date mailbox state will be sent out to the endpoint when the
|
|
* new stasis subscription is established
|
|
*/
|
|
ao2_lock(unsolicited_mwi);
|
|
ao2_callback(unsolicited_mwi, OBJ_NOLOCK | OBJ_UNLINK | OBJ_NODATA | OBJ_MULTIPLE, unsubscribe, NULL);
|
|
ao2_callback(endpoints, OBJ_NODATA, create_mwi_subscriptions_for_endpoint, NULL);
|
|
ao2_unlock(unsolicited_mwi);
|
|
|
|
ao2_ref(endpoints, -1);
|
|
}
|
|
|
|
/*! \brief Function called to send MWI NOTIFY on any unsolicited mailboxes relating to this AOR */
|
|
static int send_contact_notify(void *obj, void *arg, int flags)
|
|
{
|
|
struct mwi_subscription *mwi_sub = obj;
|
|
const char *aor = arg;
|
|
|
|
if (!mwi_sub->aors || !strstr(mwi_sub->aors, aor)) {
|
|
return 0;
|
|
}
|
|
|
|
if (ast_sip_push_task(NULL, serialized_notify, ao2_bump(mwi_sub))) {
|
|
ao2_ref(mwi_sub, -1);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*! \brief Function called when a contact is updated */
|
|
static void mwi_contact_updated(const void *object)
|
|
{
|
|
char *id = ast_strdupa(ast_sorcery_object_get_id(object)), *aor = NULL;
|
|
|
|
aor = strsep(&id, ";@");
|
|
|
|
ao2_callback(unsolicited_mwi, OBJ_NODATA, send_contact_notify, aor);
|
|
}
|
|
|
|
/*! \brief Function called when a contact is added */
|
|
static void mwi_contact_added(const void *object)
|
|
{
|
|
const struct ast_sip_contact *contact = object;
|
|
struct ao2_iterator *mwi_subs;
|
|
struct mwi_subscription *mwi_sub;
|
|
const char *endpoint_id = ast_sorcery_object_get_id(contact->endpoint);
|
|
|
|
if (ast_strlen_zero(contact->endpoint->subscription.mwi.mailboxes)) {
|
|
return;
|
|
}
|
|
|
|
ao2_lock(unsolicited_mwi);
|
|
|
|
mwi_subs = ao2_find(unsolicited_mwi, endpoint_id, OBJ_SEARCH_KEY | OBJ_MULTIPLE | OBJ_NOLOCK | OBJ_UNLINK);
|
|
if (mwi_subs) {
|
|
for (; (mwi_sub = ao2_iterator_next(mwi_subs)); ao2_cleanup(mwi_sub)) {
|
|
unsubscribe(mwi_sub, NULL, 0);
|
|
}
|
|
ao2_iterator_destroy(mwi_subs);
|
|
}
|
|
|
|
create_mwi_subscriptions_for_endpoint(contact->endpoint, NULL, 0);
|
|
|
|
ao2_unlock(unsolicited_mwi);
|
|
|
|
mwi_contact_updated(object);
|
|
}
|
|
|
|
/*! \brief Observer for contacts so unsolicited MWI is sent when a contact changes */
|
|
static const struct ast_sorcery_observer mwi_contact_observer = {
|
|
.created = mwi_contact_added,
|
|
.updated = mwi_contact_updated,
|
|
};
|
|
|
|
/*! \brief Task invoked to send initial MWI NOTIFY for unsolicited */
|
|
static int send_initial_notify_all(void *obj)
|
|
{
|
|
ao2_callback(unsolicited_mwi, OBJ_NODATA, send_notify, NULL);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*! \brief Event callback which fires initial unsolicited MWI NOTIFY messages when we're fully booted */
|
|
static void mwi_startup_event_cb(void *data, struct stasis_subscription *sub, struct stasis_message *message)
|
|
{
|
|
struct ast_json_payload *payload;
|
|
const char *type;
|
|
|
|
if (stasis_message_type(message) != ast_manager_get_generic_type()) {
|
|
return;
|
|
}
|
|
|
|
payload = stasis_message_data(message);
|
|
type = ast_json_string_get(ast_json_object_get(payload->json, "type"));
|
|
|
|
if (strcmp(type, "FullyBooted")) {
|
|
return;
|
|
}
|
|
|
|
ast_sip_push_task(NULL, send_initial_notify_all, NULL);
|
|
|
|
stasis_unsubscribe(sub);
|
|
}
|
|
|
|
static int reload(void)
|
|
{
|
|
create_mwi_subscriptions();
|
|
return 0;
|
|
}
|
|
|
|
static int load_module(void)
|
|
{
|
|
CHECK_PJSIP_MODULE_LOADED();
|
|
|
|
if (ast_sip_register_subscription_handler(&mwi_handler)) {
|
|
return AST_MODULE_LOAD_DECLINE;
|
|
}
|
|
|
|
unsolicited_mwi = ao2_container_alloc(MWI_BUCKETS, mwi_sub_hash, mwi_sub_cmp);
|
|
if (!unsolicited_mwi) {
|
|
ast_sip_unregister_subscription_handler(&mwi_handler);
|
|
return AST_MODULE_LOAD_DECLINE;
|
|
}
|
|
|
|
create_mwi_subscriptions();
|
|
ast_sorcery_observer_add(ast_sip_get_sorcery(), "contact", &mwi_contact_observer);
|
|
|
|
if (ast_test_flag(&ast_options, AST_OPT_FLAG_FULLY_BOOTED)) {
|
|
ast_sip_push_task(NULL, send_initial_notify_all, NULL);
|
|
} else {
|
|
stasis_subscribe_pool(ast_manager_get_topic(), mwi_startup_event_cb, NULL);
|
|
}
|
|
|
|
return AST_MODULE_LOAD_SUCCESS;
|
|
}
|
|
|
|
static int unload_module(void)
|
|
{
|
|
ao2_callback(unsolicited_mwi, OBJ_UNLINK | OBJ_NODATA | OBJ_MULTIPLE, unsubscribe, NULL);
|
|
ao2_ref(unsolicited_mwi, -1);
|
|
ast_sorcery_observer_remove(ast_sip_get_sorcery(), "contact", &mwi_contact_observer);
|
|
ast_sip_unregister_subscription_handler(&mwi_handler);
|
|
return 0;
|
|
}
|
|
|
|
AST_MODULE_INFO(ASTERISK_GPL_KEY, AST_MODFLAG_LOAD_ORDER, "PJSIP MWI resource",
|
|
.support_level = AST_MODULE_SUPPORT_CORE,
|
|
.load = load_module,
|
|
.unload = unload_module,
|
|
.reload = reload,
|
|
.load_pri = AST_MODPRI_CHANNEL_DEPEND,
|
|
);
|