1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312 |
- /*
- * Asterisk -- An open source telephony toolkit.
- *
- * Copyright (C) 2013, Digium, Inc.
- *
- * Joshua Colp <jcolp@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>
- <support_level>core</support_level>
- ***/
- #include "asterisk.h"
- #include <pjsip.h>
- #include <pjsip_ua.h>
- #include "asterisk/res_pjsip.h"
- #include "asterisk/res_pjsip_cli.h"
- #include "asterisk/module.h"
- #include "asterisk/taskprocessor.h"
- #include "asterisk/cli.h"
- #include "asterisk/stasis_system.h"
- #include "res_pjsip/include/res_pjsip_private.h"
- /*** DOCUMENTATION
- <configInfo name="res_pjsip_outbound_registration" language="en_US">
- <synopsis>SIP resource for outbound registrations</synopsis>
- <description><para>
- <emphasis>Outbound Registration</emphasis>
- </para>
- <para>This module allows <literal>res_pjsip</literal> to register to other SIP servers.</para>
- </description>
- <configFile name="pjsip.conf">
- <configObject name="registration">
- <synopsis>The configuration for outbound registration</synopsis>
- <description><para>
- Registration is <emphasis>COMPLETELY</emphasis> separate from the rest of
- <literal>pjsip.conf</literal>. A minimal configuration consists of
- setting a <literal>server_uri</literal> and a <literal>client_uri</literal>.
- </para></description>
- <configOption name="auth_rejection_permanent" default="yes">
- <synopsis>Determines whether failed authentication challenges are treated
- as permanent failures.</synopsis>
- <description><para>If this option is enabled and an authentication challenge fails,
- registration will not be attempted again until the configuration is reloaded.</para></description>
- </configOption>
- <configOption name="client_uri">
- <synopsis>Client SIP URI used when attemping outbound registration</synopsis>
- <description><para>
- This is the address-of-record for the outbound registration (i.e. the URI in
- the To header of the REGISTER).</para>
- <para>For registration with an ITSP, the client SIP URI may need to consist of
- an account name or number and the provider's hostname for their registrar, e.g.
- client_uri=1234567890@example.com. This may differ between providers.</para>
- <para>For registration to generic registrars, the client SIP URI will depend
- on networking specifics and configuration of the registrar.
- </para></description>
- </configOption>
- <configOption name="contact_user">
- <synopsis>Contact User to use in request</synopsis>
- </configOption>
- <configOption name="expiration" default="3600">
- <synopsis>Expiration time for registrations in seconds</synopsis>
- </configOption>
- <configOption name="max_retries" default="10">
- <synopsis>Maximum number of registration attempts.</synopsis>
- </configOption>
- <configOption name="outbound_auth" default="">
- <synopsis>Authentication object to be used for outbound registrations.</synopsis>
- </configOption>
- <configOption name="outbound_proxy" default="">
- <synopsis>Outbound Proxy used to send registrations</synopsis>
- </configOption>
- <configOption name="retry_interval" default="60">
- <synopsis>Interval in seconds between retries if outbound registration is unsuccessful</synopsis>
- </configOption>
- <configOption name="forbidden_retry_interval" default="0">
- <synopsis>Interval used when receiving a 403 Forbidden response.</synopsis>
- <description><para>
- If a 403 Forbidden is received, chan_pjsip will wait
- <replaceable>forbidden_retry_interval</replaceable> seconds before
- attempting registration again. If 0 is specified, chan_pjsip will not
- retry after receiving a 403 Forbidden response. Setting this to a non-zero
- value goes against a "SHOULD NOT" in RFC3261, but can be used to work around
- buggy registrars.
- </para></description>
- </configOption>
- <configOption name="server_uri">
- <synopsis>SIP URI of the server to register against</synopsis>
- <description><para>
- This is the URI at which to find the registrar to send the outbound REGISTER. This URI
- is used as the request URI of the outbound REGISTER request from Asterisk.</para>
- <para>For registration with an ITSP, the setting may often be just the domain of
- the registrar, e.g. sip:sip.example.com.
- </para></description>
- </configOption>
- <configOption name="transport">
- <synopsis>Transport used for outbound authentication</synopsis>
- <description>
- <note><para>A <replaceable>transport</replaceable> configured in
- <literal>pjsip.conf</literal>. As with other <literal>res_pjsip</literal> modules, this will use the first available transport of the appropriate type if unconfigured.</para></note>
- </description>
- </configOption>
- <configOption name="type">
- <synopsis>Must be of type 'registration'.</synopsis>
- </configOption>
- <configOption name="support_path">
- <synopsis>Enables Path support for outbound REGISTER requests.</synopsis>
- <description><para>
- When this option is enabled, outbound REGISTER requests will advertise
- support for Path headers so that intervening proxies can add to the Path
- header as necessary.
- </para></description>
- </configOption>
- </configObject>
- </configFile>
- </configInfo>
- <manager name="PJSIPUnregister" language="en_US">
- <synopsis>
- Unregister an outbound registration.
- </synopsis>
- <syntax>
- <xi:include xpointer="xpointer(/docs/manager[@name='Login']/syntax/parameter[@name='ActionID'])" />
- <parameter name="Registration" required="true">
- <para>The outbound registration to unregister.</para>
- </parameter>
- </syntax>
- </manager>
- <manager name="PJSIPShowRegistrationsOutbound" language="en_US">
- <synopsis>
- Lists PJSIP outbound registrations.
- </synopsis>
- <syntax />
- <description>
- <para>
- In response <literal>OutboundRegistrationDetail</literal> events showing configuration and status
- information are raised for each outbound registration object. <literal>AuthDetail</literal>
- events are raised for each associated auth object as well. Once all events are completed an
- <literal>OutboundRegistrationDetailComplete</literal> is issued.
- </para>
- </description>
- </manager>
- ***/
- /*! \brief Amount of buffer time (in seconds) before expiration that we re-register at */
- #define REREGISTER_BUFFER_TIME 10
- /*! \brief Various states that an outbound registration may be in */
- enum sip_outbound_registration_status {
- /*! \brief Currently unregistered */
- SIP_REGISTRATION_UNREGISTERED = 0,
- /*! \brief Registered, yay! */
- SIP_REGISTRATION_REGISTERED,
- /*! \brief Registration was rejected, but response was temporal */
- SIP_REGISTRATION_REJECTED_TEMPORARY,
- /*! \brief Registration was rejected, permanently */
- SIP_REGISTRATION_REJECTED_PERMANENT,
- /*! \brief Registration has been stopped */
- SIP_REGISTRATION_STOPPED,
- };
- static const char *sip_outbound_registration_status_str[] = {
- [SIP_REGISTRATION_UNREGISTERED] = "Unregistered",
- [SIP_REGISTRATION_REGISTERED] = "Registered",
- [SIP_REGISTRATION_REJECTED_TEMPORARY] = "Rejected",
- [SIP_REGISTRATION_REJECTED_PERMANENT] = "Rejected",
- [SIP_REGISTRATION_STOPPED] = "Stopped",
- };
- /*! \brief Outbound registration client state information (persists for lifetime of regc) */
- struct sip_outbound_registration_client_state {
- /*! \brief Current status of this registration */
- enum sip_outbound_registration_status status;
- /*! \brief Outbound registration client */
- pjsip_regc *client;
- /*! \brief Timer entry for retrying on temporal responses */
- pj_timer_entry timer;
- /*! \brief Current number of retries */
- unsigned int retries;
- /*! \brief Maximum number of retries permitted */
- unsigned int max_retries;
- /*! \brief Interval at which retries should occur for temporal responses */
- unsigned int retry_interval;
- /*! \brief Interval at which retries should occur for permanent responses */
- unsigned int forbidden_retry_interval;
- /*! \brief Treat authentication challenges that we cannot handle as permanent failures */
- unsigned int auth_rejection_permanent;
- /*! \brief Determines whether SIP Path support should be advertised */
- unsigned int support_path;
- /*! \brief Serializer for stuff and things */
- struct ast_taskprocessor *serializer;
- /*! \brief Configured authentication credentials */
- struct ast_sip_auth_vector outbound_auths;
- /*! \brief Registration should be destroyed after completion of transaction */
- unsigned int destroy:1;
- };
- /*! \brief Outbound registration state information (persists for lifetime that registration should exist) */
- struct sip_outbound_registration_state {
- /*! \brief Client state information */
- struct sip_outbound_registration_client_state *client_state;
- };
- /*! \brief Outbound registration information */
- struct sip_outbound_registration {
- /*! \brief Sorcery object details */
- SORCERY_OBJECT(details);
- /*! \brief Stringfields */
- AST_DECLARE_STRING_FIELDS(
- /*! \brief URI for the registrar */
- AST_STRING_FIELD(server_uri);
- /*! \brief URI for the AOR */
- AST_STRING_FIELD(client_uri);
- /*! \brief Optional user for contact header */
- AST_STRING_FIELD(contact_user);
- /*! \brief Explicit transport to use for registration */
- AST_STRING_FIELD(transport);
- /*! \brief Outbound proxy to use */
- AST_STRING_FIELD(outbound_proxy);
- );
- /*! \brief Requested expiration time */
- unsigned int expiration;
- /*! \brief Interval at which retries should occur for temporal responses */
- unsigned int retry_interval;
- /*! \brief Interval at which retries should occur for permanent responses */
- unsigned int forbidden_retry_interval;
- /*! \brief Treat authentication challenges that we cannot handle as permanent failures */
- unsigned int auth_rejection_permanent;
- /*! \brief Maximum number of retries permitted */
- unsigned int max_retries;
- /*! \brief Outbound registration state */
- struct sip_outbound_registration_state *state;
- /*! \brief Configured authentication credentials */
- struct ast_sip_auth_vector outbound_auths;
- /*! \brief Whether Path support is enabled */
- unsigned int support_path;
- };
- /*! \brief Helper function which cancels the timer on a client */
- static void cancel_registration(struct sip_outbound_registration_client_state *client_state)
- {
- if (pj_timer_heap_cancel(pjsip_endpt_get_timer_heap(ast_sip_get_pjsip_endpoint()), &client_state->timer)) {
- /* The timer was successfully cancelled, drop the refcount of client_state */
- ao2_ref(client_state, -1);
- }
- }
- static pj_str_t PATH_NAME = { "path", 4 };
- /*! \brief Callback function for registering */
- static int handle_client_registration(void *data)
- {
- RAII_VAR(struct sip_outbound_registration_client_state *, client_state, data, ao2_cleanup);
- pjsip_tx_data *tdata;
- pjsip_regc_info info;
- char server_uri[PJSIP_MAX_URL_SIZE], client_uri[PJSIP_MAX_URL_SIZE];
- cancel_registration(client_state);
- if ((client_state->status == SIP_REGISTRATION_STOPPED) ||
- (pjsip_regc_register(client_state->client, PJ_FALSE, &tdata) != PJ_SUCCESS)) {
- return 0;
- }
- pjsip_regc_get_info(client_state->client, &info);
- ast_copy_pj_str(server_uri, &info.server_uri, sizeof(server_uri));
- ast_copy_pj_str(client_uri, &info.client_uri, sizeof(client_uri));
- ast_debug(3, "REGISTER attempt %u to '%s' with client '%s'\n",
- client_state->retries + 1, server_uri, client_uri);
- if (client_state->support_path) {
- pjsip_supported_hdr *hdr;
- hdr = pjsip_msg_find_hdr(tdata->msg, PJSIP_H_SUPPORTED, NULL);
- if (!hdr) {
- /* insert a new Supported header */
- hdr = pjsip_supported_hdr_create(tdata->pool);
- if (!hdr) {
- return -1;
- }
- pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr *)hdr);
- }
- /* add on to the existing Supported header */
- pj_strassign(&hdr->values[hdr->count++], &PATH_NAME);
- }
- /* Due to the registration the callback may now get called, so bump the ref count */
- ao2_ref(client_state, +1);
- if (pjsip_regc_send(client_state->client, tdata) != PJ_SUCCESS) {
- ao2_ref(client_state, -1);
- }
- return 0;
- }
- /*! \brief Timer callback function, used just for registrations */
- static void sip_outbound_registration_timer_cb(pj_timer_heap_t *timer_heap, struct pj_timer_entry *entry)
- {
- struct sip_outbound_registration_client_state *client_state = entry->user_data;
- ao2_ref(client_state, +1);
- if (ast_sip_push_task(client_state->serializer, handle_client_registration, client_state)) {
- ast_log(LOG_WARNING, "Failed to pass outbound registration to threadpool\n");
- ao2_ref(client_state, -1);
- }
- entry->id = 0;
- }
- /*! \brief Helper function which sets up the timer to re-register in a specific amount of time */
- static void schedule_registration(struct sip_outbound_registration_client_state *client_state, unsigned int seconds)
- {
- pj_time_val delay = { .sec = seconds, };
- cancel_registration(client_state);
- ao2_ref(client_state, +1);
- if (pjsip_endpt_schedule_timer(ast_sip_get_pjsip_endpoint(), &client_state->timer, &delay) != PJ_SUCCESS) {
- ast_log(LOG_WARNING, "Failed to pass timed registration to scheduler\n");
- ao2_ref(client_state, -1);
- }
- }
- /*! \brief Callback function for unregistering (potentially) and destroying state */
- static int handle_client_state_destruction(void *data)
- {
- RAII_VAR(struct sip_outbound_registration_client_state *, client_state, data, ao2_cleanup);
- cancel_registration(client_state);
- if (client_state->client) {
- pjsip_regc_info info;
- pjsip_regc_get_info(client_state->client, &info);
- if (info.is_busy == PJ_TRUE) {
- /* If a client transaction is in progress we defer until it is complete */
- client_state->destroy = 1;
- return 0;
- }
- if (client_state->status != SIP_REGISTRATION_UNREGISTERED && client_state->status != SIP_REGISTRATION_REJECTED_PERMANENT) {
- pjsip_tx_data *tdata;
- if (pjsip_regc_unregister(client_state->client, &tdata) == PJ_SUCCESS) {
- pjsip_regc_send(client_state->client, tdata);
- }
- }
- pjsip_regc_destroy(client_state->client);
- }
- client_state->status = SIP_REGISTRATION_STOPPED;
- ast_sip_auth_vector_destroy(&client_state->outbound_auths);
- return 0;
- }
- /*! \brief Structure for registration response */
- struct registration_response {
- /*! \brief Response code for the registration attempt */
- int code;
- /*! \brief Expiration time for registration */
- int expiration;
- /*! \brief Retry-After value */
- int retry_after;
- /*! \brief Outbound registration client state */
- struct sip_outbound_registration_client_state *client_state;
- /*! \brief The response message */
- pjsip_rx_data *rdata;
- /*! \brief The response transaction */
- pjsip_transaction *tsx;
- };
- /*! \brief Registration response structure destructor */
- static void registration_response_destroy(void *obj)
- {
- struct registration_response *response = obj;
- if (response->rdata) {
- pjsip_rx_data_free_cloned(response->rdata);
- }
- ao2_cleanup(response->client_state);
- }
- /* \brief Helper funtion which determines if a response code is temporal or not */
- static int sip_outbound_registration_is_temporal(unsigned int code,
- struct sip_outbound_registration_client_state *client_state)
- {
- /* Shamelessly taken from pjsua */
- if (code == PJSIP_SC_REQUEST_TIMEOUT ||
- code == PJSIP_SC_INTERNAL_SERVER_ERROR ||
- code == PJSIP_SC_BAD_GATEWAY ||
- code == PJSIP_SC_SERVICE_UNAVAILABLE ||
- code == PJSIP_SC_SERVER_TIMEOUT ||
- ((code == PJSIP_SC_UNAUTHORIZED ||
- code == PJSIP_SC_PROXY_AUTHENTICATION_REQUIRED) &&
- !client_state->auth_rejection_permanent) ||
- PJSIP_IS_STATUS_IN_CLASS(code, 600)) {
- return 1;
- } else {
- return 0;
- }
- }
- static void schedule_retry(struct registration_response *response, unsigned int interval,
- const char *server_uri, const char *client_uri)
- {
- response->client_state->status = SIP_REGISTRATION_REJECTED_TEMPORARY;
- schedule_registration(response->client_state, interval);
- if (response->rdata) {
- ast_log(LOG_WARNING, "Temporal response '%d' received from '%s' on "
- "registration attempt to '%s', retrying in '%u'\n",
- response->code, server_uri, client_uri, interval);
- } else {
- ast_log(LOG_WARNING, "No response received from '%s' on "
- "registration attempt to '%s', retrying in '%u'\n",
- server_uri, client_uri, interval);
- }
- }
- /*! \brief Callback function for handling a response to a registration attempt */
- static int handle_registration_response(void *data)
- {
- RAII_VAR(struct registration_response *, response, data, ao2_cleanup);
- pjsip_regc_info info;
- char server_uri[PJSIP_MAX_URL_SIZE], client_uri[PJSIP_MAX_URL_SIZE];
- if (response->client_state->status == SIP_REGISTRATION_STOPPED) {
- return 0;
- }
- pjsip_regc_get_info(response->client_state->client, &info);
- ast_copy_pj_str(server_uri, &info.server_uri, sizeof(server_uri));
- ast_copy_pj_str(client_uri, &info.client_uri, sizeof(client_uri));
- if (response->code == 401 || response->code == 407) {
- pjsip_tx_data *tdata;
- if (!ast_sip_create_request_with_auth(&response->client_state->outbound_auths,
- response->rdata, response->tsx, &tdata)) {
- ao2_ref(response->client_state, +1);
- if (pjsip_regc_send(response->client_state->client, tdata) != PJ_SUCCESS) {
- ao2_cleanup(response->client_state);
- }
- return 0;
- }
- /* Otherwise, fall through so the failure is processed appropriately */
- }
- if (PJSIP_IS_STATUS_IN_CLASS(response->code, 200)) {
- /* If the registration went fine simply reschedule registration for the future */
- ast_debug(1, "Outbound registration to '%s' with client '%s' successful\n", server_uri, client_uri);
- response->client_state->status = SIP_REGISTRATION_REGISTERED;
- response->client_state->retries = 0;
- schedule_registration(response->client_state, response->expiration - REREGISTER_BUFFER_TIME);
- } else if (response->retry_after) {
- /* If we have been instructed to retry after a period of time, schedule it as such */
- schedule_retry(response, response->retry_after, server_uri, client_uri);
- } else if (response->client_state->retry_interval && sip_outbound_registration_is_temporal(response->code, response->client_state)) {
- if (response->client_state->retries == response->client_state->max_retries) {
- /* If we received enough temporal responses to exceed our maximum give up permanently */
- response->client_state->status = SIP_REGISTRATION_REJECTED_PERMANENT;
- ast_log(LOG_WARNING, "Maximum retries reached when attempting outbound registration to '%s' with client '%s', stopping registration attempt\n",
- server_uri, client_uri);
- } else {
- /* On the other hand if we can still try some more do so */
- response->client_state->retries++;
- schedule_retry(response, response->client_state->retry_interval, server_uri, client_uri);
- }
- } else {
- if (response->code == 403
- && response->client_state->forbidden_retry_interval
- && response->client_state->retries < response->client_state->max_retries) {
- /* A forbidden response retry interval is configured and there are retries remaining */
- response->client_state->status = SIP_REGISTRATION_REJECTED_TEMPORARY;
- response->client_state->retries++;
- schedule_registration(response->client_state, response->client_state->forbidden_retry_interval);
- ast_log(LOG_WARNING, "403 Forbidden fatal response received from '%s' on registration attempt to '%s', retrying in '%u' seconds\n",
- server_uri, client_uri, response->client_state->forbidden_retry_interval);
- } else {
- /* Finally if there's no hope of registering give up */
- response->client_state->status = SIP_REGISTRATION_REJECTED_PERMANENT;
- if (response->rdata) {
- ast_log(LOG_WARNING, "Fatal response '%d' received from '%s' on registration attempt to '%s', stopping outbound registration\n",
- response->code, server_uri, client_uri);
- } else {
- ast_log(LOG_WARNING, "Fatal registration attempt to '%s', stopping outbound registration\n", client_uri);
- }
- }
- }
- ast_system_publish_registry("PJSIP", client_uri, server_uri, sip_outbound_registration_status_str[response->client_state->status], NULL);
- /* If deferred destruction is in use see if we need to destroy now */
- if (response->client_state->destroy) {
- handle_client_state_destruction(response->client_state);
- }
- return 0;
- }
- /*! \brief Callback function for outbound registration client */
- static void sip_outbound_registration_response_cb(struct pjsip_regc_cbparam *param)
- {
- RAII_VAR(struct sip_outbound_registration_client_state *, client_state, param->token, ao2_cleanup);
- struct registration_response *response = ao2_alloc(sizeof(*response), registration_response_destroy);
- response->code = param->code;
- response->expiration = param->expiration;
- response->client_state = client_state;
- ao2_ref(response->client_state, +1);
- if (param->rdata) {
- struct pjsip_retry_after_hdr *retry_after = pjsip_msg_find_hdr(param->rdata->msg_info.msg, PJSIP_H_RETRY_AFTER, NULL);
- response->retry_after = retry_after ? retry_after->ivalue : 0;
- response->tsx = pjsip_rdata_get_tsx(param->rdata);
- pjsip_rx_data_clone(param->rdata, 0, &response->rdata);
- }
- if (ast_sip_push_task(client_state->serializer, handle_registration_response, response)) {
- ast_log(LOG_WARNING, "Failed to pass incoming registration response to threadpool\n");
- ao2_cleanup(response);
- }
- }
- /*! \brief Destructor function for registration state */
- static void sip_outbound_registration_state_destroy(void *obj)
- {
- struct sip_outbound_registration_state *state = obj;
- if (!state->client_state) {
- return;
- }
- if (state->client_state->serializer && ast_sip_push_task(state->client_state->serializer, handle_client_state_destruction, state->client_state)) {
- ast_log(LOG_WARNING, "Failed to pass outbound registration client destruction to threadpool\n");
- ao2_ref(state->client_state, -1);
- }
- }
- /*! \brief Destructor function for client registration state */
- static void sip_outbound_registration_client_state_destroy(void *obj)
- {
- struct sip_outbound_registration_client_state *client_state = obj;
- ast_taskprocessor_unreference(client_state->serializer);
- }
- /*! \brief Allocator function for registration state */
- static struct sip_outbound_registration_state *sip_outbound_registration_state_alloc(void)
- {
- struct sip_outbound_registration_state *state = ao2_alloc(sizeof(*state), sip_outbound_registration_state_destroy);
- if (!state || !(state->client_state = ao2_alloc(sizeof(*state->client_state), sip_outbound_registration_client_state_destroy))) {
- ao2_cleanup(state);
- return NULL;
- }
- if (!(state->client_state->serializer = ast_sip_create_serializer())) {
- ao2_cleanup(state->client_state);
- ao2_cleanup(state);
- return NULL;
- }
- state->client_state->status = SIP_REGISTRATION_UNREGISTERED;
- state->client_state->timer.user_data = state->client_state;
- state->client_state->timer.cb = sip_outbound_registration_timer_cb;
- return state;
- }
- /*! \brief Destructor function for registration information */
- static void sip_outbound_registration_destroy(void *obj)
- {
- struct sip_outbound_registration *registration = obj;
- ao2_cleanup(registration->state);
- ast_sip_auth_vector_destroy(®istration->outbound_auths);
- ast_string_field_free_memory(registration);
- }
- /*! \brief Allocator function for registration information */
- static void *sip_outbound_registration_alloc(const char *name)
- {
- struct sip_outbound_registration *registration = ast_sorcery_generic_alloc(sizeof(*registration), sip_outbound_registration_destroy);
- if (!registration || ast_string_field_init(registration, 256)) {
- ao2_cleanup(registration);
- return NULL;
- }
- return registration;
- }
- /*! \brief Helper function which populates a pj_str_t with a contact header */
- static int sip_dialog_create_contact(pj_pool_t *pool, pj_str_t *contact, const char *user, const pj_str_t *target, pjsip_tpselector *selector)
- {
- pj_str_t tmp, local_addr;
- pjsip_uri *uri;
- pjsip_sip_uri *sip_uri;
- pjsip_transport_type_e type = PJSIP_TRANSPORT_UNSPECIFIED;
- int local_port;
- pj_strdup_with_null(pool, &tmp, target);
- if (!(uri = pjsip_parse_uri(pool, tmp.ptr, tmp.slen, 0)) ||
- (!PJSIP_URI_SCHEME_IS_SIP(uri) && !PJSIP_URI_SCHEME_IS_SIPS(uri))) {
- return -1;
- }
- sip_uri = pjsip_uri_get_uri(uri);
- if (PJSIP_URI_SCHEME_IS_SIPS(sip_uri)) {
- type = PJSIP_TRANSPORT_TLS;
- } else if (!sip_uri->transport_param.slen) {
- type = PJSIP_TRANSPORT_UDP;
- } else {
- type = pjsip_transport_get_type_from_name(&sip_uri->transport_param);
- }
- if (type == PJSIP_TRANSPORT_UNSPECIFIED) {
- return -1;
- }
- if (pj_strchr(&sip_uri->host, ':')) {
- type = (pjsip_transport_type_e)(((int)type) + PJSIP_TRANSPORT_IPV6);
- }
- if (pjsip_tpmgr_find_local_addr(pjsip_endpt_get_tpmgr(ast_sip_get_pjsip_endpoint()), pool, type, selector,
- &local_addr, &local_port) != PJ_SUCCESS) {
- return -1;
- }
- if (!pj_strchr(&sip_uri->host, ':') && pj_strchr(&local_addr, ':')) {
- type = (pjsip_transport_type_e)(((int)type) + PJSIP_TRANSPORT_IPV6);
- }
- contact->ptr = pj_pool_alloc(pool, PJSIP_MAX_URL_SIZE);
- contact->slen = pj_ansi_snprintf(contact->ptr, PJSIP_MAX_URL_SIZE,
- "<%s:%s@%s%.*s%s:%d%s%s>",
- (pjsip_transport_get_flag_from_type(type) & PJSIP_TRANSPORT_SECURE) ? "sips" : "sip",
- user,
- (type & PJSIP_TRANSPORT_IPV6) ? "[" : "",
- (int)local_addr.slen,
- local_addr.ptr,
- (type & PJSIP_TRANSPORT_IPV6) ? "]" : "",
- local_port,
- (type != PJSIP_TRANSPORT_UDP && type != PJSIP_TRANSPORT_UDP6) ? ";transport=" : "",
- (type != PJSIP_TRANSPORT_UDP && type != PJSIP_TRANSPORT_UDP6) ? pjsip_transport_get_type_name(type) : "");
- return 0;
- }
- /*!
- * \internal
- * \brief Check if a registration can be reused
- *
- * This checks if the existing outbound registration's configuration differs from a newly-applied
- * outbound registration to see if the applied one.
- *
- * \param existing The pre-existing outbound registration
- * \param applied The newly-created registration
- */
- static int can_reuse_registration(struct sip_outbound_registration *existing, struct sip_outbound_registration *applied)
- {
- int i;
- if (strcmp(existing->server_uri, applied->server_uri) || strcmp(existing->client_uri, applied->client_uri) ||
- strcmp(existing->transport, applied->transport) || strcmp(existing->contact_user, applied->contact_user) ||
- strcmp(existing->outbound_proxy, applied->outbound_proxy) ||
- AST_VECTOR_SIZE(&existing->outbound_auths) != AST_VECTOR_SIZE(&applied->outbound_auths) ||
- existing->auth_rejection_permanent != applied->auth_rejection_permanent) {
- return 0;
- }
- for (i = 0; i < AST_VECTOR_SIZE(&existing->outbound_auths); ++i) {
- if (strcmp(AST_VECTOR_GET(&existing->outbound_auths, i), AST_VECTOR_GET(&applied->outbound_auths, i))) {
- return 0;
- }
- }
- return 1;
- }
- /*! \brief Helper function that allocates a pjsip registration client and configures it */
- static int sip_outbound_registration_regc_alloc(void *data)
- {
- struct sip_outbound_registration *registration = data;
- pj_pool_t *pool;
- pj_str_t tmp;
- pjsip_uri *uri;
- pj_str_t server_uri, client_uri, contact_uri;
- pjsip_tpselector selector = { .type = PJSIP_TPSELECTOR_NONE, };
- pool = pjsip_endpt_create_pool(ast_sip_get_pjsip_endpoint(), "URI Validation", 256, 256);
- if (!pool) {
- ast_log(LOG_ERROR, "Could not create pool for URI validation on outbound registration '%s'\n",
- ast_sorcery_object_get_id(registration));
- return -1;
- }
- pj_strdup2_with_null(pool, &tmp, registration->server_uri);
- uri = pjsip_parse_uri(pool, tmp.ptr, tmp.slen, 0);
- if (!uri) {
- ast_log(LOG_ERROR, "Invalid server URI '%s' specified on outbound registration '%s'\n",
- registration->server_uri, ast_sorcery_object_get_id(registration));
- pjsip_endpt_release_pool(ast_sip_get_pjsip_endpoint(), pool);
- return -1;
- }
- pj_strdup2_with_null(pool, &tmp, registration->client_uri);
- uri = pjsip_parse_uri(pool, tmp.ptr, tmp.slen, 0);
- if (!uri) {
- ast_log(LOG_ERROR, "Invalid client URI '%s' specified on outbound registration '%s'\n",
- registration->client_uri, ast_sorcery_object_get_id(registration));
- pjsip_endpt_release_pool(ast_sip_get_pjsip_endpoint(), pool);
- return -1;
- }
- pjsip_endpt_release_pool(ast_sip_get_pjsip_endpoint(), pool);
- if (!ast_strlen_zero(registration->transport)) {
- RAII_VAR(struct ast_sip_transport *, transport, ast_sorcery_retrieve_by_id(ast_sip_get_sorcery(), "transport", registration->transport), ao2_cleanup);
- if (!transport || !transport->state) {
- ast_log(LOG_ERROR, "Unable to retrieve PJSIP transport '%s' "
- " for outbound registration", registration->transport);
- return -1;
- }
- if (transport->state->transport) {
- selector.type = PJSIP_TPSELECTOR_TRANSPORT;
- selector.u.transport = transport->state->transport;
- } else if (transport->state->factory) {
- selector.type = PJSIP_TPSELECTOR_LISTENER;
- selector.u.listener = transport->state->factory;
- } else {
- return -1;
- }
- }
- if (!registration->state->client_state->client &&
- pjsip_regc_create(ast_sip_get_pjsip_endpoint(), registration->state->client_state, sip_outbound_registration_response_cb,
- ®istration->state->client_state->client) != PJ_SUCCESS) {
- return -1;
- }
- pjsip_regc_set_transport(registration->state->client_state->client, &selector);
- if (!ast_strlen_zero(registration->outbound_proxy)) {
- pjsip_route_hdr route_set, *route;
- static const pj_str_t ROUTE_HNAME = { "Route", 5 };
- pj_str_t tmp;
- pj_list_init(&route_set);
- pj_strdup2_with_null(pjsip_regc_get_pool(registration->state->client_state->client), &tmp, registration->outbound_proxy);
- if (!(route = pjsip_parse_hdr(pjsip_regc_get_pool(registration->state->client_state->client), &ROUTE_HNAME, tmp.ptr, tmp.slen, NULL))) {
- return -1;
- }
- pj_list_insert_nodes_before(&route_set, route);
- pjsip_regc_set_route_set(registration->state->client_state->client, &route_set);
- }
- pj_cstr(&server_uri, registration->server_uri);
- if (sip_dialog_create_contact(pjsip_regc_get_pool(registration->state->client_state->client), &contact_uri, S_OR(registration->contact_user, "s"), &server_uri, &selector)) {
- return -1;
- }
- pj_cstr(&client_uri, registration->client_uri);
- if (pjsip_regc_init(registration->state->client_state->client, &server_uri, &client_uri, &client_uri, 1, &contact_uri, registration->expiration) != PJ_SUCCESS) {
- return -1;
- }
- return 0;
- }
- /*! \brief Apply function which finds or allocates a state structure */
- static int sip_outbound_registration_apply(const struct ast_sorcery *sorcery, void *obj)
- {
- RAII_VAR(struct sip_outbound_registration *, existing, ast_sorcery_retrieve_by_id(sorcery, "registration", ast_sorcery_object_get_id(obj)), ao2_cleanup);
- struct sip_outbound_registration *applied = obj;
- if (ast_strlen_zero(applied->server_uri)) {
- ast_log(LOG_ERROR, "No server URI specified on outbound registration '%s'",
- ast_sorcery_object_get_id(applied));
- return -1;
- } else if (ast_strlen_zero(applied->client_uri)) {
- ast_log(LOG_ERROR, "No client URI specified on outbound registration '%s'\n",
- ast_sorcery_object_get_id(applied));
- return -1;
- }
- if (!existing) {
- /* If no existing registration exists we can just start fresh easily */
- applied->state = sip_outbound_registration_state_alloc();
- } else {
- /* If there is an existing registration things are more complicated, we can immediately reuse this state if most stuff remains unchanged */
- if (can_reuse_registration(existing, applied)) {
- applied->state = existing->state;
- ao2_ref(applied->state, +1);
- return 0;
- }
- applied->state = sip_outbound_registration_state_alloc();
- }
- if (!applied->state) {
- return -1;
- }
- return ast_sip_push_task_synchronous(NULL, sip_outbound_registration_regc_alloc, applied);
- }
- /*! \brief Helper function which performs a single registration */
- static int sip_outbound_registration_perform(void *data)
- {
- RAII_VAR(struct sip_outbound_registration *, registration, data, ao2_cleanup);
- size_t i;
- /* Just in case the client state is being reused for this registration, free the auth information */
- ast_sip_auth_vector_destroy(®istration->state->client_state->outbound_auths);
- AST_VECTOR_INIT(®istration->state->client_state->outbound_auths, AST_VECTOR_SIZE(®istration->outbound_auths));
- for (i = 0; i < AST_VECTOR_SIZE(®istration->outbound_auths); ++i) {
- const char *name = ast_strdup(AST_VECTOR_GET(®istration->outbound_auths, i));
- AST_VECTOR_APPEND(®istration->state->client_state->outbound_auths, name);
- }
- registration->state->client_state->retry_interval = registration->retry_interval;
- registration->state->client_state->forbidden_retry_interval = registration->forbidden_retry_interval;
- registration->state->client_state->max_retries = registration->max_retries;
- registration->state->client_state->retries = 0;
- registration->state->client_state->support_path = registration->support_path;
- registration->state->client_state->auth_rejection_permanent = registration->auth_rejection_permanent;
- pjsip_regc_update_expires(registration->state->client_state->client, registration->expiration);
- schedule_registration(registration->state->client_state, (ast_random() % 10) + 1);
- return 0;
- }
- /*! \brief Helper function which performs all registrations */
- static void sip_outbound_registration_perform_all(void)
- {
- RAII_VAR(struct ao2_container *, registrations, ast_sorcery_retrieve_by_fields(ast_sip_get_sorcery(), "registration", AST_RETRIEVE_FLAG_MULTIPLE | AST_RETRIEVE_FLAG_ALL, NULL), ao2_cleanup);
- struct ao2_iterator i;
- struct sip_outbound_registration *registration;
- if (!registrations) {
- return;
- }
- i = ao2_iterator_init(registrations, 0);
- while ((registration = ao2_iterator_next(&i))) {
- if (ast_sip_push_task(registration->state->client_state->serializer, sip_outbound_registration_perform, registration)) {
- ast_log(LOG_ERROR, "Failed to perform outbound registration on '%s'\n", ast_sorcery_object_get_id(registration));
- ao2_ref(registration, -1);
- }
- }
- ao2_iterator_destroy(&i);
- }
- static int outbound_auth_handler(const struct aco_option *opt, struct ast_variable *var, void *obj)
- {
- struct sip_outbound_registration *registration = obj;
- return ast_sip_auth_vector_init(®istration->outbound_auths, var->value);
- }
- static int outbound_auths_to_str(const void *obj, const intptr_t *args, char **buf)
- {
- const struct sip_outbound_registration *registration = obj;
- return ast_sip_auths_to_str(®istration->outbound_auths, buf);
- }
- static int outbound_auths_to_var_list(const void *obj, struct ast_variable **fields)
- {
- const struct sip_outbound_registration *registration = obj;
- int i;
- struct ast_variable *head = NULL;
- for (i = 0; i < AST_VECTOR_SIZE(®istration->outbound_auths) ; i++) {
- ast_variable_list_append(&head, ast_variable_new("outbound_auth",
- AST_VECTOR_GET(®istration->outbound_auths, i), ""));
- }
- if (head) {
- *fields = head;
- }
- return 0;
- }
- static struct sip_outbound_registration *retrieve_registration(const char *registration_name)
- {
- return ast_sorcery_retrieve_by_id(
- ast_sip_get_sorcery(),
- "registration",
- registration_name);
- }
- static int unregister_task(void *obj)
- {
- RAII_VAR(struct sip_outbound_registration*, registration, obj, ao2_cleanup);
- struct pjsip_regc *client = registration->state->client_state->client;
- pjsip_tx_data *tdata;
- if (pjsip_regc_unregister(client, &tdata) != PJ_SUCCESS) {
- return 0;
- }
- ao2_ref(registration->state->client_state, +1);
- if (pjsip_regc_send(client, tdata) != PJ_SUCCESS) {
- ao2_cleanup(registration->state->client_state);
- }
- return 0;
- }
- static int queue_unregister(struct sip_outbound_registration *registration)
- {
- ao2_ref(registration, +1);
- if (ast_sip_push_task(registration->state->client_state->serializer, unregister_task, registration)) {
- ao2_cleanup(registration);
- return -1;
- }
- return 0;
- }
- static char *cli_complete_registration(const char *line, const char *word,
- int pos, int state)
- {
- char *result = NULL;
- int wordlen;
- int which = 0;
- struct sip_outbound_registration *registration;
- RAII_VAR(struct ao2_container *, registrations, NULL, ao2_cleanup);
- struct ao2_iterator i;
- if (pos != 3) {
- return NULL;
- }
- wordlen = strlen(word);
- registrations = ast_sorcery_retrieve_by_fields(ast_sip_get_sorcery(), "registration",
- AST_RETRIEVE_FLAG_MULTIPLE | AST_RETRIEVE_FLAG_ALL, NULL);
- if (!registrations) {
- return NULL;
- }
- i = ao2_iterator_init(registrations, 0);
- while ((registration = ao2_iterator_next(&i))) {
- const char *name = ast_sorcery_object_get_id(registration);
- if (!strncasecmp(word, name, wordlen) && ++which > state) {
- result = ast_strdup(name);
- }
- ao2_cleanup(registration);
- if (result) {
- break;
- }
- }
- ao2_iterator_destroy(&i);
- return result;
- }
- static char *cli_unregister(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- RAII_VAR(struct sip_outbound_registration *, registration, NULL, ao2_cleanup);
- const char *registration_name;
- switch (cmd) {
- case CLI_INIT:
- e->command = "pjsip send unregister";
- e->usage =
- "Usage: pjsip send unregister <registration>\n"
- " Send a SIP REGISTER request to the specified outbound "
- "registration with an expiration of 0. This will cause the contact "
- "added by this registration to be removed on the remote system.\n";
- return NULL;
- case CLI_GENERATE:
- return cli_complete_registration(a->line, a->word, a->pos, a->n);
- }
- if (a->argc != 4) {
- return CLI_SHOWUSAGE;
- }
- registration_name = a->argv[3];
- registration = retrieve_registration(registration_name);
- if (!registration) {
- ast_cli(a->fd, "Unable to retrieve registration %s\n", registration_name);
- return CLI_FAILURE;
- }
- if (queue_unregister(registration)) {
- ast_cli(a->fd, "Failed to queue unregistration");
- return 0;
- }
- return CLI_SUCCESS;
- }
- static int ami_unregister(struct mansession *s, const struct message *m)
- {
- const char *registration_name = astman_get_header(m, "Registration");
- RAII_VAR(struct sip_outbound_registration *, registration, NULL, ao2_cleanup);
- if (ast_strlen_zero(registration_name)) {
- astman_send_error(s, m, "Registration parameter missing.");
- return 0;
- }
- registration = retrieve_registration(registration_name);
- if (!registration) {
- astman_send_error(s, m, "Unable to retrieve registration entry\n");
- return 0;
- }
- if (queue_unregister(registration)) {
- astman_send_ack(s, m, "Failed to queue unregistration");
- return 0;
- }
- astman_send_ack(s, m, "Unregistration sent");
- return 0;
- }
- struct sip_ami_outbound {
- struct ast_sip_ami *ami;
- int registered;
- int not_registered;
- struct sip_outbound_registration *registration;
- };
- static int ami_outbound_registration_task(void *obj)
- {
- struct sip_ami_outbound *ami = obj;
- RAII_VAR(struct ast_str *, buf,
- ast_sip_create_ami_event("OutboundRegistrationDetail", ami->ami), ast_free);
- if (!buf) {
- return -1;
- }
- ast_sip_sorcery_object_to_ami(ami->registration, &buf);
- if (ami->registration->state) {
- pjsip_regc_info info;
- if (ami->registration->state->client_state->status ==
- SIP_REGISTRATION_REGISTERED) {
- ++ami->registered;
- } else {
- ++ami->not_registered;
- }
- ast_str_append(&buf, 0, "Status: %s%s",
- sip_outbound_registration_status_str[
- ami->registration->state->client_state->status], "\r\n");
- pjsip_regc_get_info(ami->registration->state->client_state->client, &info);
- ast_str_append(&buf, 0, "NextReg: %d%s", info.next_reg, "\r\n");
- }
- astman_append(ami->ami->s, "%s\r\n", ast_str_buffer(buf));
- return ast_sip_format_auths_ami(&ami->registration->outbound_auths, ami->ami);
- }
- static int ami_outbound_registration_detail(void *obj, void *arg, int flags)
- {
- struct sip_ami_outbound *ami = arg;
- ami->registration = obj;
- return ast_sip_push_task_synchronous(
- NULL, ami_outbound_registration_task, ami);
- }
- static int ami_show_outbound_registrations(struct mansession *s,
- const struct message *m)
- {
- struct ast_sip_ami ami = { .s = s, .m = m, .action_id = astman_get_header(m, "ActionID"), };
- struct sip_ami_outbound ami_outbound = { .ami = &ami };
- RAII_VAR(struct ao2_container *, regs, ast_sorcery_retrieve_by_fields(
- ast_sip_get_sorcery(), "registration", AST_RETRIEVE_FLAG_MULTIPLE |
- AST_RETRIEVE_FLAG_ALL, NULL), ao2_cleanup);
- if (!regs) {
- astman_send_error(s, m, "Unable to retreive "
- "outbound registrations\n");
- return -1;
- }
- astman_send_listack(s, m, "Following are Events for each Outbound "
- "registration", "start");
- ao2_callback(regs, OBJ_NODATA, ami_outbound_registration_detail, &ami_outbound);
- astman_append(s, "Event: OutboundRegistrationDetailComplete\r\n");
- if (!ast_strlen_zero(ami.action_id)) {
- astman_append(s, "ActionID: %s\r\n", ami.action_id);
- }
- astman_append(s, "EventList: Complete\r\n"
- "Registered: %d\r\n"
- "NotRegistered: %d\r\n\r\n",
- ami_outbound.registered,
- ami_outbound.not_registered);
- return 0;
- }
- static struct ao2_container *cli_get_container(void)
- {
- RAII_VAR(struct ao2_container *, container, NULL, ao2_cleanup);
- struct ao2_container *s_container;
- container = ast_sorcery_retrieve_by_fields(ast_sip_get_sorcery(), "registration",
- AST_RETRIEVE_FLAG_MULTIPLE | AST_RETRIEVE_FLAG_ALL, NULL);
- if (!container) {
- return NULL;
- }
- s_container = ao2_container_alloc_list(AO2_ALLOC_OPT_LOCK_NOLOCK, 0,
- ast_sorcery_object_id_sort, ast_sorcery_object_id_compare);
- if (!s_container) {
- return NULL;
- }
- if (ao2_container_dup(s_container, container, 0)) {
- ao2_ref(s_container, -1);
- return NULL;
- }
- return s_container;
- }
- static int cli_iterator(void *container, ao2_callback_fn callback, void *args)
- {
- ao2_callback(container, OBJ_NODATA, callback, args);
- return 0;
- }
- static void *cli_retrieve_by_id(const char *id)
- {
- return ast_sorcery_retrieve_by_id(ast_sip_get_sorcery(), "registration", id);
- }
- static int cli_print_header(void *obj, void *arg, int flags)
- {
- struct ast_sip_cli_context *context = arg;
- ast_assert(context->output_buffer != NULL);
- ast_str_append(&context->output_buffer, 0,
- " <Registration/ServerURI..............................> <Auth..........> <Status.......>\n");
- return 0;
- }
- static int cli_print_body(void *obj, void *arg, int flags)
- {
- struct sip_outbound_registration *registration = obj;
- struct ast_sip_cli_context *context = arg;
- const char *id = ast_sorcery_object_get_id(registration);
- #define REGISTRATION_URI_FIELD_LEN 53
- ast_assert(context->output_buffer != NULL);
- ast_str_append(&context->output_buffer, 0, " %-s/%-*.*s %-16s %-16s\n",
- id,
- (int) (REGISTRATION_URI_FIELD_LEN - strlen(id)),
- (int) (REGISTRATION_URI_FIELD_LEN - strlen(id)),
- registration->server_uri,
- AST_VECTOR_SIZE(®istration->outbound_auths)
- ? AST_VECTOR_GET(®istration->outbound_auths, 0)
- : "n/a",
- sip_outbound_registration_status_str[registration->state->client_state->status]);
- if (context->show_details
- || (context->show_details_only_level_0 && context->indent_level == 0)) {
- ast_str_append(&context->output_buffer, 0, "\n");
- ast_sip_cli_print_sorcery_objectset(registration, context, 0);
- }
- return 0;
- }
- /*
- * A function pointer to callback needs to be within the
- * module in order to avoid problems with an undefined
- * symbol when the module is loaded.
- */
- static char *my_cli_traverse_objects(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- return ast_sip_cli_traverse_objects(e, cmd, a);
- }
- static struct ast_cli_entry cli_outbound_registration[] = {
- AST_CLI_DEFINE(cli_unregister, "Send a REGISTER request to an outbound registration target with a expiration of 0"),
- AST_CLI_DEFINE(my_cli_traverse_objects, "List PJSIP Registrations",
- .command = "pjsip list registrations",
- .usage = "Usage: pjsip list registrations\n"
- " List the configured PJSIP Registrations\n"),
- AST_CLI_DEFINE(my_cli_traverse_objects, "Show PJSIP Registrations",
- .command = "pjsip show registrations",
- .usage = "Usage: pjsip show registrations\n"
- " Show the configured PJSIP Registrations\n"),
- AST_CLI_DEFINE(my_cli_traverse_objects, "Show PJSIP Registration",
- .command = "pjsip show registration",
- .usage = "Usage: pjsip show registration <id>\n"
- " Show the configured PJSIP Registration\n"),
- };
- static struct ast_sip_cli_formatter_entry *cli_formatter;
- static int load_module(void)
- {
- CHECK_PJSIP_MODULE_LOADED();
- ast_sorcery_apply_default(ast_sip_get_sorcery(), "registration", "config", "pjsip.conf,criteria=type=registration");
- if (ast_sorcery_object_register(ast_sip_get_sorcery(), "registration", sip_outbound_registration_alloc, NULL, sip_outbound_registration_apply)) {
- return AST_MODULE_LOAD_DECLINE;
- }
- ast_sorcery_object_field_register(ast_sip_get_sorcery(), "registration", "type", "", OPT_NOOP_T, 0, 0);
- ast_sorcery_object_field_register(ast_sip_get_sorcery(), "registration", "server_uri", "", OPT_STRINGFIELD_T, 0, STRFLDSET(struct sip_outbound_registration, server_uri));
- ast_sorcery_object_field_register(ast_sip_get_sorcery(), "registration", "client_uri", "", OPT_STRINGFIELD_T, 0, STRFLDSET(struct sip_outbound_registration, client_uri));
- ast_sorcery_object_field_register(ast_sip_get_sorcery(), "registration", "contact_user", "", OPT_STRINGFIELD_T, 0, STRFLDSET(struct sip_outbound_registration, contact_user));
- ast_sorcery_object_field_register(ast_sip_get_sorcery(), "registration", "transport", "", OPT_STRINGFIELD_T, 0, STRFLDSET(struct sip_outbound_registration, transport));
- ast_sorcery_object_field_register(ast_sip_get_sorcery(), "registration", "outbound_proxy", "", OPT_STRINGFIELD_T, 0, STRFLDSET(struct sip_outbound_registration, outbound_proxy));
- ast_sorcery_object_field_register(ast_sip_get_sorcery(), "registration", "expiration", "3600", OPT_UINT_T, 0, FLDSET(struct sip_outbound_registration, expiration));
- ast_sorcery_object_field_register(ast_sip_get_sorcery(), "registration", "retry_interval", "60", OPT_UINT_T, 0, FLDSET(struct sip_outbound_registration, retry_interval));
- ast_sorcery_object_field_register(ast_sip_get_sorcery(), "registration", "forbidden_retry_interval", "0", OPT_UINT_T, 0, FLDSET(struct sip_outbound_registration, forbidden_retry_interval));
- ast_sorcery_object_field_register(ast_sip_get_sorcery(), "registration", "max_retries", "10", OPT_UINT_T, 0, FLDSET(struct sip_outbound_registration, max_retries));
- ast_sorcery_object_field_register(ast_sip_get_sorcery(), "registration", "auth_rejection_permanent", "yes", OPT_BOOL_T, 1, FLDSET(struct sip_outbound_registration, auth_rejection_permanent));
- ast_sorcery_object_field_register_custom(ast_sip_get_sorcery(), "registration", "outbound_auth", "", outbound_auth_handler, outbound_auths_to_str, outbound_auths_to_var_list, 0, 0);
- ast_sorcery_object_field_register(ast_sip_get_sorcery(), "registration", "support_path", "no", OPT_BOOL_T, 1, FLDSET(struct sip_outbound_registration, support_path));
- ast_sorcery_reload_object(ast_sip_get_sorcery(), "registration");
- sip_outbound_registration_perform_all();
- ast_manager_register_xml("PJSIPUnregister", EVENT_FLAG_SYSTEM | EVENT_FLAG_REPORTING, ami_unregister);
- ast_manager_register_xml("PJSIPShowRegistrationsOutbound", EVENT_FLAG_SYSTEM | EVENT_FLAG_REPORTING, ami_show_outbound_registrations);
- cli_formatter = ao2_alloc(sizeof(struct ast_sip_cli_formatter_entry), NULL);
- if (!cli_formatter) {
- ast_log(LOG_ERROR, "Unable to allocate memory for cli formatter\n");
- return -1;
- }
- cli_formatter->name = "registration";
- cli_formatter->print_header = cli_print_header;
- cli_formatter->print_body = cli_print_body;
- cli_formatter->get_container = cli_get_container;
- cli_formatter->iterate = cli_iterator;
- cli_formatter->get_id = ast_sorcery_object_get_id;
- cli_formatter->retrieve_by_id = cli_retrieve_by_id;
- ast_sip_register_cli_formatter(cli_formatter);
- ast_cli_register_multiple(cli_outbound_registration, ARRAY_LEN(cli_outbound_registration));
- return AST_MODULE_LOAD_SUCCESS;
- }
- static int reload_module(void)
- {
- ast_sorcery_reload_object(ast_sip_get_sorcery(), "registration");
- sip_outbound_registration_perform_all();
- return 0;
- }
- static int unload_module(void)
- {
- ast_cli_unregister_multiple(cli_outbound_registration, ARRAY_LEN(cli_outbound_registration));
- ast_sip_unregister_cli_formatter(cli_formatter);
- ast_manager_unregister("PJSIPShowRegistrationsOutbound");
- ast_manager_unregister("PJSIPUnregister");
- return 0;
- }
- AST_MODULE_INFO(ASTERISK_GPL_KEY, AST_MODFLAG_LOAD_ORDER, "PJSIP Outbound Registration Support",
- .support_level = AST_MODULE_SUPPORT_CORE,
- .load = load_module,
- .reload = reload_module,
- .unload = unload_module,
- .load_pri = AST_MODPRI_APP_DEPEND,
- );
|