freeswitch/libs/sofia-sip/libsofia-sip-ua/nta/check_nta.c

484 lines
10 KiB
C

/*
* This file is part of the Sofia-SIP package
*
* Copyright (C) 2008 Nokia Corporation.
*
* Contact: Pekka Pessi <pekka.pessi@nokia.com>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public License
* as published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA
*
*/
/**@CFILE check_nta.c
* @brief 2nd test Suite for Sofia SIP Transaction Engine
*
* @author Pekka Pessi <Pekka.Pessi@nokia.com>
*
* @date Created: Wed Apr 30 12:48:27 EEST 2008 ppessi
*/
#include "config.h"
#undef NDEBUG
#include "check_nta.h"
#include "s2dns.h"
#include "s2base.h"
#include "s2sip.h"
#include <sofia-sip/sip_header.h>
#include <sofia-sip/sip_status.h>
#include <sofia-sip/msg_addr.h>
#include <sofia-sip/su_log.h>
#include <sofia-sip/su_tagarg.h>
#include <sofia-sip/su_alloc.h>
#include <sofia-sip/su_string.h>
#include <sofia-sip/sresolv.h>
#include <stdlib.h>
#include <string.h>
#include <assert.h>
#include <limits.h>
#include <time.h>
/* -- Globals -------------------------------------------------------------- */
struct s2nta *s2;
/* -- main ----------------------------------------------------------------- */
static void usage(int exitcode)
{
fprintf(exitcode ? stderr : stdout,
"usage: %s [--xml=logfile] case,...\n", s2_tester);
exit(exitcode);
}
int main(int argc, char *argv[])
{
int i, failed = 0, selected = 0;
char const *xml = NULL;
Suite *suite;
SRunner *runner;
s2_tester = "check_nta";
s2_suite("NTA");
if (getenv("CHECK_NTA_VERBOSE"))
s2_start_stop = strtoul(getenv("CHECK_NTA_VERBOSE"), NULL, 10);
for (i = 1; argv[i]; i++) {
if (su_strnmatch(argv[i], "--xml=", strlen("--xml="))) {
xml = argv[i] + strlen("--xml=");
}
else if (su_strmatch(argv[i], "--xml")) {
if (!(xml = argv[++i]))
usage(2);
}
else if (su_strmatch(argv[i], "-v")) {
s2_start_stop = 1;
}
else if (su_strmatch(argv[i], "-?") ||
su_strmatch(argv[i], "-h") ||
su_strmatch(argv[i], "--help"))
usage(0);
else {
s2_select_tests(argv[i]);
selected = 1;
}
}
if (!selected)
s2_select_tests(getenv("CHECK_NTA_CASES"));
suite = suite_create("Unit tests for nta (Sofia-SIP Transaction Engine)");
suite_add_tcase(suite, check_nta_api_1_0());
suite_add_tcase(suite, check_nta_client_2_0());
suite_add_tcase(suite, check_nta_client_2_1());
suite_add_tcase(suite, check_nta_client_2_2());
runner = srunner_create(suite);
if (xml)
srunner_set_xml(runner, argv[1]);
srunner_run_all(runner, CK_ENV);
failed = srunner_ntests_failed(runner);
srunner_free(runner);
exit(failed ? EXIT_FAILURE : EXIT_SUCCESS);
}
/* -- NTA callbacks -------------------------------------------------------- */
struct event *
s2_nta_remove_event(struct event *e)
{
if ((*e->prev = e->next))
e->next->prev = e->prev;
e->prev = NULL, e->next = NULL;
return e;
}
void
s2_nta_free_event(struct event *e)
{
if (e) {
if (e->prev) {
if ((*e->prev = e->next))
e->next->prev = e->prev;
}
if (e->msg)
msg_destroy(e->msg);
free(e);
}
}
void
s2_nta_flush_events(void)
{
while (s2->events) {
s2_nta_free_event(s2->events);
}
}
struct event *
s2_nta_next_event(void)
{
for (;;) {
if (s2->events)
return s2_nta_remove_event(s2->events);
su_root_step(s2->root, 1);
}
}
struct event *
s2_nta_vwait_for(enum wait_for wait_for0,
void const *value0,
va_list va0)
{
struct event *e;
for (;;) {
for (e = s2->events; e; e = e->next) {
va_list va;
enum wait_for wait_for;
void const *value;
va_copy(va, va0);
for (wait_for = wait_for0, value = value0;
wait_for;
wait_for = va_arg(va, enum wait_for),
value = va_arg(va, void const *)) {
switch (wait_for) {
case wait_for_amagic:
if (value != e->amagic)
goto next;
break;
case wait_for_omagic:
if (value != e->omagic)
goto next;
break;
case wait_for_orq:
if (value != e->orq)
goto next;
break;
case wait_for_lmagic:
if (value != e->lmagic)
goto next;
break;
case wait_for_leg:
if (value != e->leg)
goto next;
break;
case wait_for_imagic:
if (value != e->imagic)
goto next;
break;
case wait_for_irq:
if (value != e->irq)
goto next;
break;
case wait_for_method:
if ((sip_method_t)value != e->method)
goto next;
break;
case wait_for_method_name:
if (!su_strmatch(value, e->method_name))
goto next;
break;
case wait_for_status:
if ((int)value != e->status)
goto next;
break;
case wait_for_phrase:
if (!su_casematch(value, e->phrase))
goto next;
break;
}
}
next:
va_end(va);
if (!wait_for)
return s2_nta_remove_event(e);
}
su_root_step(s2->root, 1);
}
}
struct event *
s2_nta_wait_for(enum wait_for wait_for,
void const *value,
...)
{
struct event *e;
va_list va;
va_start(va, value);
e = s2_nta_vwait_for(wait_for, value, va);
va_end(va);
return e;
}
int
s2_nta_check_request(enum wait_for wait_for,
void const *value,
...)
{
struct event *e;
va_list va;
va_start(va, value);
e = s2_nta_vwait_for(wait_for, value, va);
va_end(va);
s2_nta_free_event(e);
return e != NULL;
}
int
s2_nta_msg_callback(nta_agent_magic_t *magic,
nta_agent_t *nta,
msg_t *msg,
sip_t *sip)
{
struct event *e, **prev;
e = calloc(1, sizeof *e);
e->amagic = magic;
e->msg = msg;
e->sip = sip;
if (sip->sip_request) {
e->method = sip->sip_request->rq_method;
e->method_name = sip->sip_request->rq_method_name;
}
else {
e->status = sip->sip_status->st_status;
e->phrase = sip->sip_status->st_phrase;
}
for (prev = &s2->events; *prev; prev = &(*prev)->next)
;
*prev = e, e->prev = prev;
return 0;
}
int
s2_nta_orq_callback(nta_outgoing_magic_t *magic,
nta_outgoing_t *orq,
sip_t const *sip)
{
struct event *e, **prev;
e = calloc(1, sizeof *e);
e->omagic = magic;
e->orq = orq;
e->msg = nta_outgoing_getresponse(orq);
e->sip = sip_object(e->msg);
e->status = nta_outgoing_status(orq);
e->phrase = sip ? sip->sip_status->st_phrase : "";
for (prev = &s2->events; *prev; prev = &(*prev)->next)
;
*prev = e, e->prev = prev;
return 0;
}
int
s2_nta_leg_callback(nta_leg_magic_t *magic,
nta_leg_t *leg,
nta_incoming_t *irq,
sip_t const *sip)
{
struct event *e, **prev;
e = calloc(1, sizeof *e);
e->lmagic = magic;
e->leg = leg;
e->irq = irq;
e->msg = nta_incoming_getrequest(irq);
e->sip = sip_object(e->msg);
e->method = e->sip->sip_request->rq_method;
e->method_name = e->sip->sip_request->rq_method_name;
for (prev = &s2->events; *prev; prev = &(*prev)->next)
;
*prev = e, e->prev = prev;
return 0;
}
int
s2_nta_irq_callback(nta_incoming_magic_t *magic,
nta_incoming_t *irq,
sip_t const *sip)
{
struct event *e, **prev;
e = calloc(1, sizeof *e);
e->imagic = magic;
e->irq = irq;
e->msg = nta_incoming_getrequest_ackcancel(irq);
e->sip = sip_object(e->msg);
e->method = e->sip ? e->sip->sip_request->rq_method : 0;
e->method_name = e->sip ? e->sip->sip_request->rq_method_name : NULL;
for (prev = &s2->events; *prev; prev = &(*prev)->next)
;
*prev = e, e->prev = prev;
return 0;
}
/* ====================================================================== */
SOFIAPUBVAR su_log_t nta_log[];
SOFIAPUBVAR su_log_t sresolv_log[];
SOFIAPUBVAR su_log_t tport_log[];
SOFIAPUBVAR su_log_t su_log_default[];
void
s2_nta_setup_logs(int level)
{
su_log_soft_set_level(su_log_default, level);
su_log_soft_set_level(tport_log, level);
su_log_soft_set_level(nta_log, level);
su_log_soft_set_level(sresolv_log, level);
if (getenv("TPORT_LOG") == NULL && getenv("S2_TPORT_LOG") == NULL) {
if (s2sip)
tport_set_params(s2sip->master, TPTAG_LOG(level > 1), TAG_END());
}
}
void
s2_nta_setup(char const *label,
char const * const *transports,
tag_type_t tag, tag_value_t value, ...)
{
ta_list ta;
s2_setup(label);
s2_nta_setup_logs(0);
s2_dns_setup(s2base->root);
ta_start(ta, tag, value);
s2_sip_setup("example.org", transports, ta_tags(ta));
ta_end(ta);
assert(s2sip->contact);
s2_dns_domain("example.org", 1,
"s2", 1, s2sip->udp.contact ? s2sip->udp.contact->m_url : NULL,
"s2", 1, s2sip->tcp.contact ? s2sip->tcp.contact->m_url : NULL,
"s2", 1, s2sip->tls.contact ? s2sip->tls.contact->m_url : NULL,
NULL);
}
nta_agent_t *
s2_nta_agent_setup(url_string_t const *bind_url,
nta_message_f *callback,
nta_agent_magic_t *magic,
tag_type_t tag, tag_value_t value, ...)
{
ta_list ta;
assert(s2base);
s2 = su_home_new(sizeof *s2); assert(s2);
s2->root = su_root_clone(s2base->root, s2); assert(s2->root);
fail_unless(s2->root != NULL);
ta_start(ta, tag, value);
s2->nta =
nta_agent_create(s2->root,
bind_url ? bind_url : URL_STRING_MAKE("sip:*:*"),
callback,
magic,
/* Use internal DNS server */
/* Force sresolv to use localhost and s2dns as DNS server */
#if HAVE_WIN32
SRESTAG_RESOLV_CONF("NUL"),
#else
SRESTAG_RESOLV_CONF("/dev/null"),
#endif
ta_tags(ta));
ta_end(ta);
assert(s2->nta);
if (callback == NULL)
s2->default_leg = nta_leg_tcreate(s2->nta, s2_nta_leg_callback, NULL,
NTATAG_NO_DIALOG(1),
TAG_END());
return s2->nta;
}
void s2_nta_teardown(void)
{
if (s2) {
s2_nta_flush_events();
if (s2->default_leg)
nta_leg_destroy(s2->default_leg), s2->default_leg = NULL;
nta_agent_destroy(s2->nta), s2->nta = NULL;
su_root_destroy(s2->root), s2->root = NULL;
su_home_unref(s2->home);
s2 = NULL;
}
s2_dns_teardown();
s2_sip_teardown();
s2_teardown();
}