summaryrefslogtreecommitdiff
path: root/res/res_pjsip/location.c
diff options
context:
space:
mode:
authorRichard Mudgett <rmudgett@digium.com>2014-02-06 17:55:45 +0000
committerRichard Mudgett <rmudgett@digium.com>2014-02-06 17:55:45 +0000
commitb5ca213e34055724a5cd669938e356569847edcb (patch)
treeae81ec8dc3add7c1ed3cb94c361a205e94a3bcd7 /res/res_pjsip/location.c
parent8ff02ac951c979b1e5e340b3b1cc05736780be20 (diff)
res_pjsip: Updates and adds more PJSIP CLI commands.
* Adds identify, transport, and registration support to the PJSIP CLI. * Creates three additional callbacks, one for an iterator, one for a comparator, and one for a container. This eliminates the link dependency from higher level modules to lower level ones. * Eliminates duplicate sorting in PJSIP CLI commands. * Cleans up PJSIP CLI output formatting. * Pushes CLI command registration down to the implementing source file. * Adds several ast_sip_destroy_sorcery functions to complement existing ast_sip_sorcery_initialize functions. The destroy functions unregister PJSIP CLI commands and PJSIP CLI formatters. Reported by: George Joseph Review: https://reviewboard.asterisk.org/r/3104/ ........ Merged revisions 407568 from http://svn.asterisk.org/svn/asterisk/branches/12 git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@407573 65c4cc65-6c06-0410-ace0-fbb531ad65f3
Diffstat (limited to 'res/res_pjsip/location.c')
-rw-r--r--res/res_pjsip/location.c190
1 files changed, 121 insertions, 69 deletions
diff --git a/res/res_pjsip/location.c b/res/res_pjsip/location.c
index 499ee9a6c..dd9037f60 100644
--- a/res/res_pjsip/location.c
+++ b/res/res_pjsip/location.c
@@ -97,7 +97,7 @@ static int contact_link_static(void *obj, void *arg, int flags)
{
struct ao2_container *dest = arg;
- ao2_link_flags(dest, obj, OBJ_NOLOCK);
+ ao2_link(dest, obj);
return 0;
}
@@ -117,7 +117,7 @@ struct ast_sip_contact *ast_sip_location_retrieve_first_aor_contact(const struct
return NULL;
}
- contact = ao2_callback(contacts, OBJ_NOLOCK, contact_find_first, NULL);
+ contact = ao2_callback(contacts, 0, contact_find_first, NULL);
return contact;
}
@@ -134,11 +134,11 @@ struct ao2_container *ast_sip_location_retrieve_aor_contacts(const struct ast_si
}
/* Prune any expired contacts and delete them, we do this first because static contacts can never expire */
- ao2_callback(contacts, OBJ_NOLOCK | OBJ_NODATA | OBJ_MULTIPLE | OBJ_UNLINK, contact_expire, NULL);
+ ao2_callback(contacts, OBJ_NODATA | OBJ_MULTIPLE | OBJ_UNLINK, contact_expire, NULL);
/* Add any permanent contacts from the AOR */
if (aor->permanent_contacts) {
- ao2_callback(aor->permanent_contacts, OBJ_NOLOCK | OBJ_NODATA, contact_link_static, contacts);
+ ao2_callback(aor->permanent_contacts, OBJ_NODATA, contact_link_static, contacts);
}
return contacts;
@@ -269,7 +269,7 @@ static int permanent_uri_handler(const struct aco_option *opt, struct ast_variab
}
ast_string_field_set(contact, uri, var->value);
- ao2_link_flags(aor->permanent_contacts, contact, OBJ_NOLOCK);
+ ao2_link(aor->permanent_contacts, contact);
return 0;
}
@@ -299,33 +299,7 @@ int ast_sip_for_each_aor(const char *aors, ao2_callback_fn on_aor, void *arg)
return 0;
}
-static void destroy_contact_pair(void *obj)
-{
- struct ast_sip_aor_contact_pair *pair = obj;
- ao2_cleanup(pair->aor);
- ao2_cleanup(pair->contact);
-}
-
-static struct ast_sip_aor_contact_pair *create_contact_pair(
- struct ast_sip_aor *aor, struct ast_sip_contact *contact)
-{
- struct ast_sip_aor_contact_pair *pair = ao2_alloc(
- sizeof(*pair), destroy_contact_pair);
-
- if (!pair) {
- return NULL;
- }
-
- pair->aor = aor;
- pair->contact = contact;
-
- ao2_ref(pair->aor, +1);
- ao2_ref(pair->contact, +1);
-
- return pair;
-}
-
-int ast_sip_for_each_contact(struct ast_sip_aor *aor,
+int ast_sip_for_each_contact(const struct ast_sip_aor *aor,
ao2_callback_fn on_contact, void *arg)
{
RAII_VAR(struct ao2_container *, contacts, NULL, ao2_cleanup);
@@ -340,10 +314,9 @@ int ast_sip_for_each_contact(struct ast_sip_aor *aor,
i = ao2_iterator_init(contacts, 0);
while ((contact = ao2_iterator_next(&i))) {
int res;
- RAII_VAR(struct ast_sip_aor_contact_pair *,
- acp, create_contact_pair(aor, contact), ao2_cleanup);
- if (!acp || (res = on_contact(acp, arg, 0))) {
+ ast_sorcery_object_set_extended(contact, "@aor_id", ast_sorcery_object_get_id(aor));
+ if ((res = on_contact(contact, arg, 0))) {
ao2_iterator_destroy(&i);
return -1;
}
@@ -354,11 +327,11 @@ int ast_sip_for_each_contact(struct ast_sip_aor *aor,
int ast_sip_contact_to_str(void *object, void *arg, int flags)
{
- struct ast_sip_aor_contact_pair *acp = object;
+ struct ast_sip_contact *contact = object;
struct ast_str **buf = arg;
ast_str_append(buf, 0, "%s/%s,",
- ast_sorcery_object_get_id(acp->aor), acp->contact->uri);
+ ast_sorcery_object_get_extended(contact, "aor_id"), contact->uri);
return 0;
}
@@ -413,15 +386,40 @@ struct ast_sip_endpoint_formatter endpoint_aor_formatter = {
.format_ami = format_ami_endpoint_aor
};
+static struct ao2_container *cli_get_aor_container(void)
+{
+ RAII_VAR(struct ao2_container *, container, NULL, ao2_cleanup);
+ RAII_VAR(struct ao2_container *, s_container, NULL, ao2_cleanup);
+
+ container = ast_sorcery_retrieve_by_fields(ast_sip_get_sorcery(), "aor",
+ 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_compare, NULL);
+ if (!s_container) {
+ return NULL;
+ }
+
+ if (ao2_container_dup(s_container, container, 0)) {
+ return NULL;
+ }
+ ao2_ref(s_container, +1);
+ return s_container;
+}
+
static int populate_contact_container(void *obj, void *arg, int flags)
{
- struct ast_sip_aor_contact_pair *acp = obj;
+ struct ast_sip_contact *contact = obj;
struct ao2_container *container = arg;
- ao2_link_flags(container, acp, OBJ_NOLOCK);
+
+ ao2_link(container, contact);
return 0;
}
-static int gather_aor_channels(void *obj, void *arg, int flags)
+static int gather_aor_contacts(void *obj, void *arg, int flags)
{
struct ast_sip_aor *aor = obj;
struct ao2_container *container = arg;
@@ -429,35 +427,51 @@ static int gather_aor_channels(void *obj, void *arg, int flags)
return 0;
}
-static struct ao2_container *cli_get_contact_container(struct ast_sorcery *sip_sorcery)
+static int cli_contact_compare(const void *left, const void *right, int flags)
+{
+ const struct ast_sip_contact *left_contact = left;
+ const struct ast_sip_contact *right_contact = right;
+ int rc;
+
+ if (!left_contact || !right_contact) {
+ return 0;
+ }
+ rc = strcmp(ast_sorcery_object_get_extended(left_contact, "aor_id"),
+ ast_sorcery_object_get_extended(right_contact, "aor_id"));
+ if (rc) {
+ return rc;
+ }
+ return strcmp(left_contact->uri, right_contact->uri);
+}
+
+static struct ao2_container *cli_get_contact_container(void)
{
RAII_VAR(struct ao2_container *, parent_container, NULL, ao2_cleanup);
- RAII_VAR(struct ao2_container *, s_parent_container, NULL, ao2_cleanup);
struct ao2_container *child_container;
- parent_container = ast_sorcery_retrieve_by_fields(sip_sorcery, "aor",
- AST_RETRIEVE_FLAG_MULTIPLE | AST_RETRIEVE_FLAG_ALL, NULL);
+ parent_container = cli_get_aor_container();
if (!parent_container) {
return NULL;
}
- s_parent_container = ao2_container_alloc_list(AO2_ALLOC_OPT_LOCK_NOLOCK, 0, &ast_sorcery_object_id_compare, NULL);
- if (!s_parent_container) {
- return NULL;
- }
-
- ao2_container_dup(s_parent_container, parent_container, OBJ_ORDER_ASCENDING);
-
- child_container = ao2_container_alloc_list(AO2_ALLOC_OPT_LOCK_NOLOCK, 0, NULL, NULL);
+ child_container = ao2_container_alloc_list(AO2_ALLOC_OPT_LOCK_NOLOCK, 0,
+ cli_contact_compare, NULL);
if (!child_container) {
return NULL;
}
- ao2_callback(s_parent_container, OBJ_NODATA, gather_aor_channels, child_container);
+ ao2_ref(child_container, +1);
+ ao2_callback(parent_container, OBJ_NODATA, gather_aor_contacts, child_container);
return child_container;
}
+static int cli_contact_iterator(const void *container, ao2_callback_fn callback, void *args)
+{
+ const struct ast_sip_aor *array = container;
+
+ return ast_sip_for_each_contact(array, callback, args);
+}
static int cli_print_contact_header(void *obj, void *arg, int flags)
{
@@ -477,28 +491,25 @@ static int cli_print_contact_header(void *obj, void *arg, int flags)
static int cli_print_contact_body(void *obj, void *arg, int flags)
{
- struct ast_sip_aor_contact_pair *acp = obj;
+ struct ast_sip_contact *contact = obj;
struct ast_sip_cli_context *context = arg;
char *print_name = NULL;
int print_name_len;
int indent;
int flexwidth;
+ const char *aor_id = ast_sorcery_object_get_extended(contact, "aor_id");
RAII_VAR(struct ast_sip_contact_status *, status,
- ast_sorcery_retrieve_by_id( ast_sip_get_sorcery(), CONTACT_STATUS, ast_sorcery_object_get_id(acp->contact)),
+ ast_sorcery_retrieve_by_id( ast_sip_get_sorcery(), CONTACT_STATUS, ast_sorcery_object_get_id(contact)),
ao2_cleanup);
if (!context->output_buffer) {
return -1;
}
- print_name_len = strlen(ast_sorcery_object_get_id(acp->aor))
- + strlen(acp->contact->uri) + 2;
- if (!(print_name = alloca(print_name_len))) {
- return -1;
- }
- snprintf(print_name, print_name_len, "%s/%s",
- ast_sorcery_object_get_id(acp->aor), acp->contact->uri);
+ print_name_len = strlen(aor_id) + strlen(contact->uri) + 2;
+ print_name = ast_alloca(print_name_len);
+ snprintf(print_name, print_name_len, "%s/%s", aor_id, contact->uri);
indent = CLI_INDENT_TO_SPACES(context->indent_level);
flexwidth = CLI_LAST_TABSTOP - indent - 2;
@@ -514,10 +525,11 @@ static int cli_print_contact_body(void *obj, void *arg, int flags)
return 0;
}
-static struct ao2_container *cli_get_aor_container(struct ast_sorcery *sip_sorcery)
+static int cli_aor_iterator(const void *container, ao2_callback_fn callback, void *args)
{
- return ast_sorcery_retrieve_by_fields(sip_sorcery, "aor",
- AST_RETRIEVE_FLAG_MULTIPLE | AST_RETRIEVE_FLAG_ALL, NULL);
+ const char *aor_list = container;
+
+ return ast_sip_for_each_aor(aor_list, callback, args);
}
static int cli_print_aor_header(void *obj, void *arg, int flags)
@@ -538,7 +550,7 @@ static int cli_print_aor_header(void *obj, void *arg, int flags)
if (context->recurse) {
context->indent_level++;
formatter_entry = ast_sip_lookup_cli_formatter("contact");
- if (formatter_entry) {
+ if (formatter_entry && formatter_entry->print_header) {
formatter_entry->print_header(NULL, context, 0);
}
context->indent_level--;
@@ -571,11 +583,17 @@ static int cli_print_aor_body(void *obj, void *arg, int flags)
if (context->recurse) {
context->indent_level++;
+
formatter_entry = ast_sip_lookup_cli_formatter("contact");
- if (formatter_entry) {
- ast_sip_for_each_contact(aor, formatter_entry->print_body, context);
+ if (formatter_entry && formatter_entry->print_body && formatter_entry->iterator) {
+ formatter_entry->iterator(aor, formatter_entry->print_body, context);
}
+
context->indent_level--;
+
+ if (context->indent_level == 0) {
+ ast_str_append(&context->output_buffer, 0, "\n");
+ }
}
if (context->show_details || (context->show_details_only_level_0 && context->indent_level == 0)) {
@@ -591,6 +609,8 @@ static struct ast_sip_cli_formatter_entry cli_contact_formatter = {
.print_header = cli_print_contact_header,
.print_body = cli_print_contact_body,
.get_container = cli_get_contact_container,
+ .iterator = cli_contact_iterator,
+ .comparator = cli_contact_compare,
};
static struct ast_sip_cli_formatter_entry cli_aor_formatter = {
@@ -598,11 +618,34 @@ static struct ast_sip_cli_formatter_entry cli_aor_formatter = {
.print_header = cli_print_aor_header,
.print_body = cli_print_aor_body,
.get_container = cli_get_aor_container,
+ .iterator = cli_aor_iterator,
+ .comparator = ast_sorcery_object_id_compare,
+};
+
+static struct ast_cli_entry cli_commands[] = {
+ AST_CLI_DEFINE(ast_sip_cli_traverse_objects, "List PJSIP Aors",
+ .command = "pjsip list aors",
+ .usage = "Usage: pjsip list aors\n"
+ " List the configured PJSIP Aors\n"),
+ AST_CLI_DEFINE(ast_sip_cli_traverse_objects, "Show PJSIP Aors",
+ .command = "pjsip show aors",
+ .usage = "Usage: pjsip show aors\n"
+ " Show the configured PJSIP Aors\n"),
+ AST_CLI_DEFINE(ast_sip_cli_traverse_objects, "Show PJSIP Aor",
+ .command = "pjsip show aor",
+ .usage = "Usage: pjsip show aor <id>\n"
+ " Show the configured PJSIP Aor\n"),
+
+ AST_CLI_DEFINE(ast_sip_cli_traverse_objects, "List PJSIP Contacts",
+ .command = "pjsip list contacts",
+ .usage = "Usage: pjsip list contacts\n"
+ " List the configured PJSIP contacts\n"),
};
/*! \brief Initialize sorcery with location support */
-int ast_sip_initialize_sorcery_location(struct ast_sorcery *sorcery)
+int ast_sip_initialize_sorcery_location(void)
{
+ struct ast_sorcery *sorcery = ast_sip_get_sorcery();
ast_sorcery_apply_default(sorcery, "contact", "astdb", "registrar");
ast_sorcery_apply_default(sorcery, "aor", "config", "pjsip.conf,criteria=type=aor");
@@ -635,6 +678,15 @@ int ast_sip_initialize_sorcery_location(struct ast_sorcery *sorcery)
ast_sip_register_endpoint_formatter(&endpoint_aor_formatter);
ast_sip_register_cli_formatter(&cli_contact_formatter);
ast_sip_register_cli_formatter(&cli_aor_formatter);
+ ast_cli_register_multiple(cli_commands, ARRAY_LEN(cli_commands));
+ return 0;
+}
+
+int ast_sip_destroy_sorcery_location(void)
+{
+ ast_cli_unregister_multiple(cli_commands, ARRAY_LEN(cli_commands));
+ ast_sip_unregister_cli_formatter(&cli_contact_formatter);
+ ast_sip_unregister_cli_formatter(&cli_aor_formatter);
return 0;
}