update dingaling to be able to use TLS jabber servers such as googletalk

see sample config for new options.

the dingaling library has changed so you must rebuild it 
rm libs/libdingaling/.complete
make installall



git-svn-id: http://svn.freeswitch.org/svn/freeswitch/trunk@2251 d0543943-73ff-0310-b7d9-9358b9ac24b2
This commit is contained in:
Anthony Minessale 2006-08-11 00:24:38 +00:00
parent a53f2b0c39
commit 07a7b6e54a
7 changed files with 1597 additions and 1923 deletions

View File

@ -254,6 +254,13 @@
<param name="message" value="Jingle all the way"/> <param name="message" value="Jingle all the way"/>
<param name="rtp-ip" value="10.0.0.1"/> <param name="rtp-ip" value="10.0.0.1"/>
<param name="auto-login" value="true"/> <param name="auto-login" value="true"/>
<!-- SASL "plain" or "md5" -->
<param name="sasl" value="plain"/>
<!-- if the server where the jabber is hosted is not the same
as the one in the jid -->
<!--<param name="server" value="alternate.server.com"/>-->
<!-- Enable TLS or not -->
<param name="tls" value="true"/>
<!-- disable to trade async for more calls --> <!-- disable to trade async for more calls -->
<param name="use-rtp-timer" value="true"/> <param name="use-rtp-timer" value="true"/>
<!-- or --> <!-- or -->

View File

@ -1,8 +1,8 @@
# Makefile.in generated by automake 1.9.6 from Makefile.am. # Makefile.in generated by automake 1.9.2 from Makefile.am.
# @configure_input@ # @configure_input@
# Copyright (C) 1994, 1995, 1996, 1997, 1998, 1999, 2000, 2001, 2002, # Copyright (C) 1994, 1995, 1996, 1997, 1998, 1999, 2000, 2001, 2002,
# 2003, 2004, 2005 Free Software Foundation, Inc. # 2003, 2004 Free Software Foundation, Inc.
# This Makefile.in is free software; the Free Software Foundation # This Makefile.in is free software; the Free Software Foundation
# gives unlimited permission to copy and/or distribute it, # gives unlimited permission to copy and/or distribute it,
# with or without modifications, as long as this notice is preserved. # with or without modifications, as long as this notice is preserved.
@ -15,6 +15,8 @@
@SET_MAKE@ @SET_MAKE@
SOURCES = $(libdingaling_la_SOURCES)
srcdir = @srcdir@ srcdir = @srcdir@
top_srcdir = @top_srcdir@ top_srcdir = @top_srcdir@
VPATH = @srcdir@ VPATH = @srcdir@
@ -69,11 +71,11 @@ depcomp = $(SHELL) $(top_srcdir)/depcomp
am__depfiles_maybe = depfiles am__depfiles_maybe = depfiles
COMPILE = $(CC) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) \ COMPILE = $(CC) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) \
$(CPPFLAGS) $(AM_CFLAGS) $(CFLAGS) $(CPPFLAGS) $(AM_CFLAGS) $(CFLAGS)
LTCOMPILE = $(LIBTOOL) --tag=CC --mode=compile $(CC) $(DEFS) \ LTCOMPILE = $(LIBTOOL) --mode=compile --tag=CC $(CC) $(DEFS) \
$(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) \ $(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) \
$(AM_CFLAGS) $(CFLAGS) $(AM_CFLAGS) $(CFLAGS)
CCLD = $(CC) CCLD = $(CC)
LINK = $(LIBTOOL) --tag=CC --mode=link $(CCLD) $(AM_CFLAGS) $(CFLAGS) \ LINK = $(LIBTOOL) --mode=link --tag=CC $(CCLD) $(AM_CFLAGS) $(CFLAGS) \
$(AM_LDFLAGS) $(LDFLAGS) -o $@ $(AM_LDFLAGS) $(LDFLAGS) -o $@
SOURCES = $(libdingaling_la_SOURCES) SOURCES = $(libdingaling_la_SOURCES)
DIST_SOURCES = $(libdingaling_la_SOURCES) DIST_SOURCES = $(libdingaling_la_SOURCES)
@ -313,11 +315,11 @@ distclean-compile:
@am__fastdepCC_FALSE@ $(LTCOMPILE) -c -o $@ $< @am__fastdepCC_FALSE@ $(LTCOMPILE) -c -o $@ $<
libdingaling_la-libdingaling.lo: src/libdingaling.c libdingaling_la-libdingaling.lo: src/libdingaling.c
@am__fastdepCC_TRUE@ if $(LIBTOOL) --tag=CC --mode=compile $(CC) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) $(libdingaling_la_CFLAGS) $(CFLAGS) -MT libdingaling_la-libdingaling.lo -MD -MP -MF "$(DEPDIR)/libdingaling_la-libdingaling.Tpo" -c -o libdingaling_la-libdingaling.lo `test -f 'src/libdingaling.c' || echo '$(srcdir)/'`src/libdingaling.c; \ @am__fastdepCC_TRUE@ if $(LIBTOOL) --mode=compile --tag=CC $(CC) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) $(libdingaling_la_CFLAGS) $(CFLAGS) -MT libdingaling_la-libdingaling.lo -MD -MP -MF "$(DEPDIR)/libdingaling_la-libdingaling.Tpo" -c -o libdingaling_la-libdingaling.lo `test -f 'src/libdingaling.c' || echo '$(srcdir)/'`src/libdingaling.c; \
@am__fastdepCC_TRUE@ then mv -f "$(DEPDIR)/libdingaling_la-libdingaling.Tpo" "$(DEPDIR)/libdingaling_la-libdingaling.Plo"; else rm -f "$(DEPDIR)/libdingaling_la-libdingaling.Tpo"; exit 1; fi @am__fastdepCC_TRUE@ then mv -f "$(DEPDIR)/libdingaling_la-libdingaling.Tpo" "$(DEPDIR)/libdingaling_la-libdingaling.Plo"; else rm -f "$(DEPDIR)/libdingaling_la-libdingaling.Tpo"; exit 1; fi
@AMDEP_TRUE@@am__fastdepCC_FALSE@ source='src/libdingaling.c' object='libdingaling_la-libdingaling.lo' libtool=yes @AMDEPBACKSLASH@ @AMDEP_TRUE@@am__fastdepCC_FALSE@ source='src/libdingaling.c' object='libdingaling_la-libdingaling.lo' libtool=yes @AMDEPBACKSLASH@
@AMDEP_TRUE@@am__fastdepCC_FALSE@ DEPDIR=$(DEPDIR) $(CCDEPMODE) $(depcomp) @AMDEPBACKSLASH@ @AMDEP_TRUE@@am__fastdepCC_FALSE@ DEPDIR=$(DEPDIR) $(CCDEPMODE) $(depcomp) @AMDEPBACKSLASH@
@am__fastdepCC_FALSE@ $(LIBTOOL) --tag=CC --mode=compile $(CC) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) $(libdingaling_la_CFLAGS) $(CFLAGS) -c -o libdingaling_la-libdingaling.lo `test -f 'src/libdingaling.c' || echo '$(srcdir)/'`src/libdingaling.c @am__fastdepCC_FALSE@ $(LIBTOOL) --mode=compile --tag=CC $(CC) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) $(libdingaling_la_CFLAGS) $(CFLAGS) -c -o libdingaling_la-libdingaling.lo `test -f 'src/libdingaling.c' || echo '$(srcdir)/'`src/libdingaling.c
mostlyclean-libtool: mostlyclean-libtool:
-rm -f *.lo -rm -f *.lo
@ -353,13 +355,7 @@ uninstall-library_includeHEADERS:
# (which will cause the Makefiles to be regenerated when you run `make'); # (which will cause the Makefiles to be regenerated when you run `make');
# (2) otherwise, pass the desired values on the `make' command line. # (2) otherwise, pass the desired values on the `make' command line.
$(RECURSIVE_TARGETS): $(RECURSIVE_TARGETS):
@failcom='exit 1'; \ @set fnord $$MAKEFLAGS; amf=$$2; \
for f in x $$MAKEFLAGS; do \
case $$f in \
*=* | --[!k]*);; \
*k*) failcom='fail=yes';; \
esac; \
done; \
dot_seen=no; \ dot_seen=no; \
target=`echo $@ | sed s/-recursive//`; \ target=`echo $@ | sed s/-recursive//`; \
list='$(SUBDIRS)'; for subdir in $$list; do \ list='$(SUBDIRS)'; for subdir in $$list; do \
@ -371,7 +367,7 @@ $(RECURSIVE_TARGETS):
local_target="$$target"; \ local_target="$$target"; \
fi; \ fi; \
(cd $$subdir && $(MAKE) $(AM_MAKEFLAGS) $$local_target) \ (cd $$subdir && $(MAKE) $(AM_MAKEFLAGS) $$local_target) \
|| eval $$failcom; \ || case "$$amf" in *=*) exit 1;; *k*) fail=yes;; *) exit 1;; esac; \
done; \ done; \
if test "$$dot_seen" = "no"; then \ if test "$$dot_seen" = "no"; then \
$(MAKE) $(AM_MAKEFLAGS) "$$target-am" || exit 1; \ $(MAKE) $(AM_MAKEFLAGS) "$$target-am" || exit 1; \
@ -379,13 +375,7 @@ $(RECURSIVE_TARGETS):
mostlyclean-recursive clean-recursive distclean-recursive \ mostlyclean-recursive clean-recursive distclean-recursive \
maintainer-clean-recursive: maintainer-clean-recursive:
@failcom='exit 1'; \ @set fnord $$MAKEFLAGS; amf=$$2; \
for f in x $$MAKEFLAGS; do \
case $$f in \
*=* | --[!k]*);; \
*k*) failcom='fail=yes';; \
esac; \
done; \
dot_seen=no; \ dot_seen=no; \
case "$@" in \ case "$@" in \
distclean-* | maintainer-clean-*) list='$(DIST_SUBDIRS)' ;; \ distclean-* | maintainer-clean-*) list='$(DIST_SUBDIRS)' ;; \
@ -406,7 +396,7 @@ maintainer-clean-recursive:
local_target="$$target"; \ local_target="$$target"; \
fi; \ fi; \
(cd $$subdir && $(MAKE) $(AM_MAKEFLAGS) $$local_target) \ (cd $$subdir && $(MAKE) $(AM_MAKEFLAGS) $$local_target) \
|| eval $$failcom; \ || case "$$amf" in *=*) exit 1;; *k*) fail=yes;; *) exit 1;; esac; \
done && test -z "$$fail" done && test -z "$$fail"
tags-recursive: tags-recursive:
list='$(SUBDIRS)'; for subdir in $$list; do \ list='$(SUBDIRS)'; for subdir in $$list; do \

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

View File

@ -64,7 +64,7 @@
#define microsleep(x) apr_sleep(x * 1000) #define microsleep(x) apr_sleep(x * 1000)
static int opt_timeout = 30; static int opt_timeout = 30;
static int opt_use_tls = 0;
static struct { static struct {
unsigned int flags; unsigned int flags;
@ -73,6 +73,7 @@ static struct {
apr_pool_t *memory_pool; apr_pool_t *memory_pool;
unsigned int id; unsigned int id;
ldl_logger_t logger; ldl_logger_t logger;
apr_thread_mutex_t *flag_mutex;
} globals; } globals;
struct packet_node { struct packet_node {
@ -94,7 +95,9 @@ struct ldl_handle {
iksfilter *filter; iksfilter *filter;
char *login; char *login;
char *password; char *password;
char *server;
char *status_msg; char *status_msg;
uint16_t port;
int features; int features;
int counter; int counter;
int job_done; int job_done;
@ -105,6 +108,7 @@ struct ldl_handle {
apr_hash_t *retry_hash; apr_hash_t *retry_hash;
apr_hash_t *probe_hash; apr_hash_t *probe_hash;
apr_thread_mutex_t *lock; apr_thread_mutex_t *lock;
apr_thread_mutex_t *flag_mutex;
ldl_loop_callback_t loop_callback; ldl_loop_callback_t loop_callback;
ldl_session_callback_t session_callback; ldl_session_callback_t session_callback;
ldl_response_callback_t response_callback; ldl_response_callback_t response_callback;
@ -477,13 +481,26 @@ static int on_commands(void *user_data, ikspak *pak)
//char *to = iks_find_attrib(pak->x, "to"); //char *to = iks_find_attrib(pak->x, "to");
char *iqid = iks_find_attrib(pak->x, "id"); char *iqid = iks_find_attrib(pak->x, "id");
char *type = iks_find_attrib(pak->x, "type"); char *type = iks_find_attrib(pak->x, "type");
uint8_t is_result = strcasecmp(type, "result") ? 0 : 1;
iks *xml; iks *xml;
//printf("XXXXX from=%s to=%s type=%s\n", from, to, type); if (is_result) {
iks *tag = iks_child (pak->x);
while(tag) {
if (!strcasecmp(iks_name(tag), "bind")) {
char *jid = iks_find_cdata(tag, "jid");
char *resource = strchr(jid, '/');
handle->acc->resource = apr_pstrdup(handle->pool, resource);
handle->login = apr_pstrdup(handle->pool, jid);
break;
}
tag = iks_next_tag(tag);
}
}
if ((!strcasecmp(type, "result") || !strcasecmp(type, "error")) && iqid && from) { if ((is_result || !strcasecmp(type, "error")) && iqid && from) {
cancel_retry(handle, iqid); cancel_retry(handle, iqid);
if (!strcasecmp(type, "result")) { if (is_result) {
if (handle->response_callback) { if (handle->response_callback) {
handle->response_callback(handle, iqid); handle->response_callback(handle, iqid);
} }
@ -530,24 +547,60 @@ static int on_result(void *user_data, ikspak *pak)
iks_insert_attrib(ctag, "xmlns", "http://jabber.org/protocol/caps"); iks_insert_attrib(ctag, "xmlns", "http://jabber.org/protocol/caps");
apr_queue_push(handle->queue, msg); apr_queue_push(handle->queue, msg);
ldl_set_flag(handle, LDL_FLAG_READY); ldl_set_flag_locked(handle, LDL_FLAG_READY);
return IKS_FILTER_EAT; return IKS_FILTER_EAT;
} }
static const char c64[65] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
#define B64BUFFLEN 1024
static int b64encode(unsigned char *in, uint32_t ilen, unsigned char *out, uint32_t olen) {
int x=0,y=0,bytes=0;
unsigned int b=0,l=0;
for(x=0;x<ilen;x++) {
b = (b<<8) + in[x];
l += 8;
while (l >= 6) {
out[bytes++] = c64[(b>>(l-=6))%64];
if(++y!=72) {
continue;
}
out[bytes++] = '\n';
y=0;
}
}
if (l > 0) {
out[bytes++] = c64[((b%16)<<(6-l))%64];
}
if (l != 0) while (l < 6) {
out[bytes++] = '=', l += 2;
}
return 0;
}
static int on_stream(ldl_handle_t *handle, int type, iks * node) static int on_stream(ldl_handle_t *handle, int type, iks * node)
{ {
handle->counter = opt_timeout; handle->counter = opt_timeout;
switch (type) { switch (type) {
case IKS_NODE_START: case IKS_NODE_START:
if (opt_use_tls && !iks_is_secure(handle->parser)) { if (ldl_test_flag(handle, LDL_FLAG_TLS) && !iks_is_secure(handle->parser)) {
if (iks_has_tls()) {
iks_start_tls(handle->parser); iks_start_tls(handle->parser);
} else {
globals.logger(DL_LOG_DEBUG, "TLS NOT SUPPORTED IN THIS BUILD!\n");
}
} }
break; break;
case IKS_NODE_NORMAL: case IKS_NODE_NORMAL:
if (strcmp("stream:features", iks_name(node)) == 0) { if (strcmp("stream:features", iks_name(node)) == 0) {
handle->features = iks_stream_features(node); handle->features = iks_stream_features(node);
if (opt_use_tls && !iks_is_secure(handle->parser)) if (ldl_test_flag(handle, LDL_FLAG_TLS) && !iks_is_secure(handle->parser))
break; break;
if (ldl_test_flag(handle, LDL_FLAG_CONNECTED)) { if (ldl_test_flag(handle, LDL_FLAG_CONNECTED)) {
iks *t; iks *t;
@ -566,7 +619,27 @@ static int on_stream(ldl_handle_t *handle, int type, iks * node)
if (handle->features & IKS_STREAM_SASL_MD5) { if (handle->features & IKS_STREAM_SASL_MD5) {
iks_start_sasl(handle->parser, IKS_SASL_DIGEST_MD5, handle->acc->user, handle->password); iks_start_sasl(handle->parser, IKS_SASL_DIGEST_MD5, handle->acc->user, handle->password);
} else if (handle->features & IKS_STREAM_SASL_PLAIN) { } else if (handle->features & IKS_STREAM_SASL_PLAIN) {
iks_start_sasl(handle->parser, IKS_SASL_PLAIN, handle->acc->user, handle->password); iks *x = NULL;
if ((x = iks_new("auth"))) {
char s[512] = "";
char base64[1024] = "";
uint32_t slen;
iks_insert_attrib(x, "xmlns", IKS_NS_XMPP_SASL);
iks_insert_attrib(x, "mechanism", "PLAIN");
iks_insert_attrib(x, "encoding", "UTF-8");
snprintf(s, sizeof(s), "%c%s%c%s", 0, handle->acc->user, 0, handle->password);
slen = strlen(handle->acc->user) + strlen(handle->password) + 2;
b64encode((unsigned char *)s, slen, (unsigned char *) base64, sizeof(base64));
iks_insert_cdata(x, base64, 0);
iks_send(handle->parser, x);
iks_delete(x);
} else {
globals.logger(DL_LOG_DEBUG, "Memory ERROR!\n");
break;
}
} }
} }
} else if (strcmp("failure", iks_name(node)) == 0) { } else if (strcmp("failure", iks_name(node)) == 0) {
@ -577,26 +650,26 @@ static int on_stream(ldl_handle_t *handle, int type, iks * node)
} else if (strcmp("success", iks_name(node)) == 0) { } else if (strcmp("success", iks_name(node)) == 0) {
globals.logger(DL_LOG_DEBUG, "XMPP server connected\n"); globals.logger(DL_LOG_DEBUG, "XMPP server connected\n");
iks_send_header(handle->parser, handle->acc->server); iks_send_header(handle->parser, handle->acc->server);
ldl_set_flag(handle, LDL_FLAG_CONNECTED); ldl_set_flag_locked(handle, LDL_FLAG_CONNECTED);
if (handle->session_callback) { if (handle->session_callback) {
handle->session_callback(handle, NULL, LDL_SIGNAL_CONNECTED, "core", "Server Connected", handle->login); handle->session_callback(handle, NULL, LDL_SIGNAL_CONNECTED, "core", "Server Connected", handle->login);
} }
} else { } else {
ikspak *pak; ikspak *pak;
if (!ldl_test_flag(handle, LDL_FLAG_AUTHORIZED)) { if (!ldl_test_flag(handle, LDL_FLAG_AUTHORIZED)) {
if (handle->session_callback) { if (handle->session_callback) {
handle->session_callback(handle, NULL, LDL_SIGNAL_LOGIN_SUCCESS, "core", "Login Success", handle->login); handle->session_callback(handle, NULL, LDL_SIGNAL_LOGIN_SUCCESS, "core", "Login Success", handle->login);
} }
globals.logger(DL_LOG_DEBUG, "XMPP authenticated\n"); globals.logger(DL_LOG_DEBUG, "XMPP authenticated\n");
ldl_set_flag(handle, LDL_FLAG_AUTHORIZED); ldl_set_flag_locked(handle, LDL_FLAG_AUTHORIZED);
} }
pak = iks_packet(node); pak = iks_packet(node);
iks_filter_packet(handle->filter, pak); iks_filter_packet(handle->filter, pak);
if (handle->job_done == 1) if (handle->job_done == 1) {
return IKS_HOOK; return IKS_HOOK;
} }
}
break; break;
#if 0 #if 0
case IKS_NODE_STOP: case IKS_NODE_STOP:
@ -724,7 +797,6 @@ static void ldl_flush_queue(ldl_handle_t *handle)
apr_time_t now = apr_time_now(); apr_time_t now = apr_time_now();
x++; x++;
//printf("%s %lld %lld %u\n", packet_node->id, packet_node->next, now, packet_node->retries);
if (packet_node->next <= now) { if (packet_node->next <= now) {
if (packet_node->retries > 0) { if (packet_node->retries > 0) {
packet_node->retries--; packet_node->retries--;
@ -749,6 +821,46 @@ static void ldl_flush_queue(ldl_handle_t *handle)
apr_thread_mutex_unlock(handle->lock); apr_thread_mutex_unlock(handle->lock);
} }
static void *APR_THREAD_FUNC queue_thread(apr_thread_t *thread, void *obj)
{
ldl_handle_t *handle = (ldl_handle_t *) obj;
ldl_set_flag_locked(handle, LDL_FLAG_QUEUE_RUNNING);
while (ldl_test_flag(handle, LDL_FLAG_RUNNING)) {
ldl_flush_queue(handle);
if (handle->loop_callback(handle) != LDL_STATUS_SUCCESS) {
int fd;
if ((fd = iks_fd(handle->parser)) > -1) {
shutdown(fd, 0x02);
}
ldl_clear_flag_locked(handle, LDL_FLAG_RUNNING);
break;
}
microsleep(100);
}
ldl_clear_flag_locked(handle, LDL_FLAG_QUEUE_RUNNING);
return NULL;
}
static void launch_queue_thread(ldl_handle_t *handle)
{
apr_thread_t *thread;
apr_threadattr_t *thd_attr;;
apr_threadattr_create(&thd_attr, handle->pool);
apr_threadattr_detach_set(thd_attr, 1);
apr_threadattr_stacksize_set(thd_attr, 512 * 1024);
apr_thread_create(&thread, thd_attr, queue_thread, handle, handle->pool);
}
static void xmpp_connect(ldl_handle_t *handle, char *jabber_id, char *pass) static void xmpp_connect(ldl_handle_t *handle, char *jabber_id, char *pass)
{ {
while (ldl_test_flag(handle, LDL_FLAG_RUNNING)) { while (ldl_test_flag(handle, LDL_FLAG_RUNNING)) {
@ -769,7 +881,12 @@ static void xmpp_connect(ldl_handle_t *handle, char *jabber_id, char *pass)
j_setup_filter(handle, "fixme"); j_setup_filter(handle, "fixme");
e = iks_connect_tcp(handle->parser, handle->acc->server, IKS_JABBER_PORT); e = iks_connect_via(handle->parser,
handle->server ? handle->server : handle->acc->server,
handle->port ? handle->port : IKS_JABBER_PORT,
//handle->server ? handle->server : handle->acc->server);
handle->acc->server);
switch (e) { switch (e) {
case IKS_OK: case IKS_OK:
break; break;
@ -783,12 +900,18 @@ static void xmpp_connect(ldl_handle_t *handle, char *jabber_id, char *pass)
continue; continue;
} }
handle->counter = opt_timeout; handle->counter = opt_timeout;
if (ldl_test_flag(handle, LDL_FLAG_TLS)) {
launch_queue_thread(handle);
}
while (ldl_test_flag(handle, LDL_FLAG_RUNNING)) { while (ldl_test_flag(handle, LDL_FLAG_RUNNING)) {
e = iks_recv(handle->parser, 1); e = iks_recv(handle->parser, 1);
if (handle->loop_callback) { if (!ldl_test_flag(handle, LDL_FLAG_TLS) && handle->loop_callback) {
if (handle->loop_callback(handle) != LDL_STATUS_SUCCESS) { if (handle->loop_callback(handle) != LDL_STATUS_SUCCESS) {
ldl_clear_flag(handle, LDL_FLAG_RUNNING); ldl_clear_flag_locked(handle, LDL_FLAG_RUNNING);
break; break;
} }
} }
@ -807,7 +930,8 @@ static void xmpp_connect(ldl_handle_t *handle, char *jabber_id, char *pass)
break; break;
} }
if (ldl_test_flag(handle, LDL_FLAG_READY)) {
if (!ldl_test_flag(handle, LDL_FLAG_TLS) && ldl_test_flag(handle, LDL_FLAG_READY)) {
ldl_flush_queue(handle); ldl_flush_queue(handle);
} }
@ -828,11 +952,16 @@ static void xmpp_connect(ldl_handle_t *handle, char *jabber_id, char *pass)
iks_disconnect(handle->parser); iks_disconnect(handle->parser);
iks_parser_delete(handle->parser); iks_parser_delete(handle->parser);
ldl_clear_flag(handle, LDL_FLAG_CONNECTED); ldl_clear_flag_locked(handle, LDL_FLAG_CONNECTED);
ldl_clear_flag(handle, LDL_FLAG_AUTHORIZED); ldl_clear_flag_locked(handle, LDL_FLAG_AUTHORIZED);
}
ldl_clear_flag_locked(handle, LDL_FLAG_RUNNING);
while(ldl_test_flag(handle, LDL_FLAG_QUEUE_RUNNING)) {
microsleep(100);
} }
ldl_clear_flag(handle, LDL_FLAG_RUNNING);
} }
@ -1134,16 +1263,18 @@ ldl_status ldl_global_init(int debug)
} }
memset(&globals, 0, sizeof(globals)); memset(&globals, 0, sizeof(globals));
if (apr_pool_create(&globals.memory_pool, NULL) != LDL_STATUS_SUCCESS) { if (apr_pool_create(&globals.memory_pool, NULL) != LDL_STATUS_SUCCESS) {
globals.logger(DL_LOG_DEBUG, "Could not allocate memory pool\n"); globals.logger(DL_LOG_DEBUG, "Could not allocate memory pool\n");
return LDL_STATUS_MEMERR; return LDL_STATUS_MEMERR;
} }
apr_thread_mutex_create(&globals.flag_mutex, APR_THREAD_MUTEX_NESTED, globals.memory_pool);
globals.log_stream = stdout; globals.log_stream = stdout;
globals.debug = debug; globals.debug = debug;
globals.id = 300; globals.id = 300;
globals.logger = default_logger; globals.logger = default_logger;
ldl_set_flag(&globals, LDL_FLAG_INIT); ldl_set_flag_locked((&globals), LDL_FLAG_INIT);
return LDL_STATUS_SUCCESS; return LDL_STATUS_SUCCESS;
} }
@ -1176,6 +1307,8 @@ int8_t ldl_handle_ready(ldl_handle_t *handle)
ldl_status ldl_handle_init(ldl_handle_t **handle, ldl_status ldl_handle_init(ldl_handle_t **handle,
char *login, char *login,
char *password, char *password,
char *server,
ldl_user_flag_t flags,
char *status_msg, char *status_msg,
ldl_loop_callback_t loop_callback, ldl_loop_callback_t loop_callback,
ldl_session_callback_t session_callback, ldl_session_callback_t session_callback,
@ -1184,19 +1317,43 @@ ldl_status ldl_handle_init(ldl_handle_t **handle,
{ {
apr_pool_t *pool; apr_pool_t *pool;
assert(ldl_test_flag(&globals, LDL_FLAG_INIT)); assert(ldl_test_flag(&globals, LDL_FLAG_INIT));
*handle = NULL;
if ((apr_pool_create(&pool, globals.memory_pool)) != LDL_STATUS_SUCCESS) { if ((apr_pool_create(&pool, globals.memory_pool)) != LDL_STATUS_SUCCESS) {
return LDL_STATUS_MEMERR; return LDL_STATUS_MEMERR;
} }
if (!login) {
globals.logger(DL_LOG_ERR, "No login supplied!\n");
return LDL_STATUS_FALSE;
}
if (!password) {
globals.logger(DL_LOG_ERR, "No password supplied!\n");
return LDL_STATUS_FALSE;
}
if ((*handle = apr_palloc(pool, sizeof(ldl_handle_t)))) { if ((*handle = apr_palloc(pool, sizeof(ldl_handle_t)))) {
ldl_handle_t *new_handle = *handle; ldl_handle_t *new_handle = *handle;
memset(new_handle, 0, sizeof(ldl_handle_t)); memset(new_handle, 0, sizeof(ldl_handle_t));
new_handle->log_stream = globals.log_stream; new_handle->log_stream = globals.log_stream;
new_handle->login = apr_pstrdup(pool, login); new_handle->login = apr_pstrdup(pool, login);
new_handle->password = apr_pstrdup(pool, password); new_handle->password = apr_pstrdup(pool, password);
if (server) {
char *p;
new_handle->server = apr_pstrdup(pool, server);
if ((p = strchr(new_handle->server, ':'))) {
*p++ = '\0';
new_handle->port = atoi(p);
}
}
if (status_msg) {
new_handle->status_msg = apr_pstrdup(pool, status_msg); new_handle->status_msg = apr_pstrdup(pool, status_msg);
}
if (loop_callback) { if (loop_callback) {
new_handle->loop_callback = loop_callback; new_handle->loop_callback = loop_callback;
@ -1212,17 +1369,24 @@ ldl_status ldl_handle_init(ldl_handle_t **handle,
new_handle->private_info = private_info; new_handle->private_info = private_info;
new_handle->pool = pool; new_handle->pool = pool;
new_handle->flags |= flags;
apr_queue_create(&new_handle->queue, LDL_HANDLE_QLEN, new_handle->pool); apr_queue_create(&new_handle->queue, LDL_HANDLE_QLEN, new_handle->pool);
apr_queue_create(&new_handle->retry_queue, LDL_HANDLE_QLEN, new_handle->pool); apr_queue_create(&new_handle->retry_queue, LDL_HANDLE_QLEN, new_handle->pool);
new_handle->features |= IKS_STREAM_BIND|IKS_STREAM_SESSION |IKS_STREAM_SASL_PLAIN; new_handle->features |= IKS_STREAM_BIND|IKS_STREAM_SESSION;
if (new_handle->flags & LDL_FLAG_SASL_PLAIN) {
new_handle->features |= IKS_STREAM_SASL_PLAIN;
} else if (new_handle->flags & LDL_FLAG_SASL_MD5) {
new_handle->features |= IKS_STREAM_SASL_MD5;
}
new_handle->sessions = apr_hash_make(new_handle->pool); new_handle->sessions = apr_hash_make(new_handle->pool);
new_handle->retry_hash = apr_hash_make(new_handle->pool); new_handle->retry_hash = apr_hash_make(new_handle->pool);
new_handle->probe_hash = apr_hash_make(new_handle->pool); new_handle->probe_hash = apr_hash_make(new_handle->pool);
apr_thread_mutex_create(&new_handle->lock, APR_THREAD_MUTEX_NESTED, new_handle->pool); apr_thread_mutex_create(&new_handle->lock, APR_THREAD_MUTEX_NESTED, new_handle->pool);
apr_thread_mutex_create(&new_handle->flag_mutex, APR_THREAD_MUTEX_NESTED, new_handle->pool);
return LDL_STATUS_SUCCESS; return LDL_STATUS_SUCCESS;
} else {
*handle = NULL;
} }
return LDL_STATUS_FALSE; return LDL_STATUS_FALSE;
@ -1230,14 +1394,14 @@ ldl_status ldl_handle_init(ldl_handle_t **handle,
void ldl_handle_run(ldl_handle_t *handle) void ldl_handle_run(ldl_handle_t *handle)
{ {
ldl_set_flag(handle, LDL_FLAG_RUNNING); ldl_set_flag_locked(handle, LDL_FLAG_RUNNING);
xmpp_connect(handle, handle->login, handle->password); xmpp_connect(handle, handle->login, handle->password);
ldl_clear_flag(handle, LDL_FLAG_RUNNING); ldl_clear_flag_locked(handle, LDL_FLAG_RUNNING);
} }
void ldl_handle_stop(ldl_handle_t *handle) void ldl_handle_stop(ldl_handle_t *handle)
{ {
ldl_clear_flag(handle, LDL_FLAG_RUNNING); ldl_clear_flag_locked(handle, LDL_FLAG_RUNNING);
} }
ldl_status ldl_handle_destroy(ldl_handle_t **handle) ldl_status ldl_handle_destroy(ldl_handle_t **handle)
@ -1246,6 +1410,7 @@ ldl_status ldl_handle_destroy(ldl_handle_t **handle)
ldl_flush_queue(*handle); ldl_flush_queue(*handle);
apr_pool_destroy(pool); apr_pool_destroy(pool);
*handle = NULL; *handle = NULL;
return LDL_STATUS_SUCCESS; return LDL_STATUS_SUCCESS;

View File

@ -100,9 +100,16 @@ typedef enum {
LDL_FLAG_RUNNING = (1 << 1), LDL_FLAG_RUNNING = (1 << 1),
LDL_FLAG_AUTHORIZED = (1 << 2), LDL_FLAG_AUTHORIZED = (1 << 2),
LDL_FLAG_READY = (1 << 3), LDL_FLAG_READY = (1 << 3),
LDL_FLAG_CONNECTED = (1 << 4) LDL_FLAG_CONNECTED = (1 << 4),
LDL_FLAG_QUEUE_RUNNING = (1 << 5),
} ldl_flag_t; } ldl_flag_t;
typedef enum {
LDL_FLAG_TLS = (1 << 10),
LDL_FLAG_SASL_PLAIN = (1 << 11),
LDL_FLAG_SASL_MD5 = (1 << 12)
} ldl_user_flag_t;
typedef enum { typedef enum {
LDL_SIGNAL_NONE, LDL_SIGNAL_NONE,
LDL_SIGNAL_INITIATE, LDL_SIGNAL_INITIATE,
@ -171,6 +178,23 @@ typedef void (*ldl_logger_t)(char *file, const char *func, int line, int level,
*/ */
#define ldl_clear_flag(obj, flag) (obj)->flags &= ~(flag) #define ldl_clear_flag(obj, flag) (obj)->flags &= ~(flag)
/*!
\brief Set a flag on an arbitrary object while locked
\param obj the object to set the flags on
\param flag the or'd list of flags to set
*/
#define ldl_set_flag_locked(obj, flag) assert(obj->flag_mutex != NULL);\
apr_thread_mutex_lock(obj->flag_mutex);\
(obj)->flags |= (flag);\
apr_thread_mutex_unlock(obj->flag_mutex);
/*!
\brief Clear a flag on an arbitrary object
\param obj the object to test
\param flag the or'd list of flags to clear
*/
#define ldl_clear_flag_locked(obj, flag) apr_thread_mutex_lock(obj->flag_mutex); (obj)->flags &= ~(flag); apr_thread_mutex_unlock(obj->flag_mutex);
/*! /*!
\brief Copy flags from one arbitrary object to another \brief Copy flags from one arbitrary object to another
\param dest the object to copy the flags to \param dest the object to copy the flags to
@ -389,6 +413,8 @@ int8_t ldl_handle_ready(ldl_handle_t *handle);
ldl_status ldl_handle_init(ldl_handle_t **handle, ldl_status ldl_handle_init(ldl_handle_t **handle,
char *login, char *login,
char *password, char *password,
char *server,
ldl_user_flag_t flags,
char *status_msg, char *status_msg,
ldl_loop_callback_t loop_callback, ldl_loop_callback_t loop_callback,
ldl_session_callback_t session_callback, ldl_session_callback_t session_callback,

View File

@ -97,10 +97,12 @@ struct mdl_profile {
char *ip; char *ip;
char *extip; char *extip;
char *lanaddr; char *lanaddr;
char *server;
char *exten; char *exten;
char *context; char *context;
ldl_handle_t *handle; ldl_handle_t *handle;
unsigned int flags; uint32_t flags;
uint32_t user_flags;
}; };
struct private_object { struct private_object {
@ -1146,6 +1148,8 @@ static switch_status_t init_profile(struct mdl_profile *profile, uint8_t login)
if (ldl_handle_init(&handle, if (ldl_handle_init(&handle,
profile->login, profile->login,
profile->password, profile->password,
profile->server,
profile->user_flags,
profile->message, profile->message,
handle_loop, handle_loop,
handle_signalling, handle_signalling,
@ -1224,8 +1228,20 @@ static void set_profile_val(struct mdl_profile *profile, char *var, char *val)
profile->ip = switch_core_strdup(module_pool, val); profile->ip = switch_core_strdup(module_pool, val);
} else if (!strcasecmp(var, "ext-rtp-ip")) { } else if (!strcasecmp(var, "ext-rtp-ip")) {
profile->extip = switch_core_strdup(module_pool, val); profile->extip = switch_core_strdup(module_pool, val);
} else if (!strcasecmp(var, "server")) {
profile->server = switch_core_strdup(module_pool, val);
} else if (!strcasecmp(var, "lanaddr")) { } else if (!strcasecmp(var, "lanaddr")) {
profile->lanaddr = switch_core_strdup(module_pool, val); profile->lanaddr = switch_core_strdup(module_pool, val);
} else if (!strcasecmp(var, "tls")) {
if (switch_true(val)) {
profile->user_flags |= LDL_FLAG_TLS;
}
} else if (!strcasecmp(var, "sasl")) {
if (!strcasecmp(val, "plain")) {
profile->user_flags |= LDL_FLAG_SASL_PLAIN;
} else if (!strcasecmp(val, "md5")) {
profile->user_flags |= LDL_FLAG_SASL_MD5;
}
} else if (!strcasecmp(var, "exten")) { } else if (!strcasecmp(var, "exten")) {
profile->exten = switch_core_strdup(module_pool, val); profile->exten = switch_core_strdup(module_pool, val);
} else if (!strcasecmp(var, "context")) { } else if (!strcasecmp(var, "context")) {