freeswitch/libs/libdingaling/src/libdingaling.h

751 lines
20 KiB
C

/*
* libDingaLing XMPP Jingle Library
* Copyright (C) 2005-2014, Anthony Minessale II <anthm@freeswitch.org>
*
* Version: MPL 1.1
*
* The contents of this file are subject to the Mozilla Public License Version
* 1.1 (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
* http://www.mozilla.org/MPL/
*
* Software distributed under the License is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
* for the specific language governing rights and limitations under the
* License.
*
* The Original Code is libDingaLing XMPP Jingle Library
*
* The Initial Developer of the Original Code is
* Anthony Minessale II <anthm@freeswitch.org>
* Portions created by the Initial Developer are Copyright (C)
* the Initial Developer. All Rights Reserved.
*
* Contributor(s):
*
* Anthony Minessale II <anthm@freeswitch.org>
*
* libdingaling.h -- Main Header File
*
*/
/*! \file libdingaling.h
\brief Main Header File
*/
/*!
\defgroup core1 libDingaLing Library
\ingroup LIBDINGALING
\{
*/
/* OMG */
#ifdef _MSC_VER
#define __LDL_FUNC__ __FUNCTION__
#define inline __inline
#else
#define __LDL_FUNC__ (const char *)__func__
#endif
#ifndef LIBDINGALING_H
#define LIBDINGALING_H
#ifdef __cplusplus
extern "C" {
#endif
#ifdef __STUPIDFORMATBUG__
}
#endif
#if defined (__SVR4) && defined (__sun)
#define __EXTENSIONS__ 1
#include <strings.h>
#endif
#define LDL_HANDLE_QLEN 2000
#define LDL_MAX_CANDIDATES 25
#define LDL_MAX_PAYLOADS 50
#define LDL_RETRY 3
#define IKS_NS_COMPONENT "jabber:component:accept"
/* period between keep alive signals in 1sec units*/
#define LDL_KEEPALIVE_TIMEOUT 300
typedef struct ldl_crypto_data_s {
char *tag;
char *suite;
char *key;
} ldl_crypto_data_t;
/*! \brief A structure to store a jingle candidate */
struct ldl_candidate {
/*! the transport id of the candidate */
char *tid;
/*! the name of the candidate */
char *name;
/*! the type of the candidate */
char *type;
/*! the protocol of the candidate */
char *protocol;
/*! the STUN username of the candidate */
char *username;
/*! the STUN password of the candidate */
char *password;
/*! the ip address of the candidate */
char *address;
/*! the stun/rtp port of the candidate */
uint16_t port;
/*! the preference level of the candidate */
double pref;
};
typedef struct ldl_candidate ldl_candidate_t;
typedef enum {
LDL_PAYLOAD_AUDIO,
LDL_PAYLOAD_VIDEO
} ldl_payload_type_t;
/*! \brief A structure to store a jingle audio payload */
struct ldl_payload {
/*! the type of the payload */
ldl_payload_type_t type;
/*! the iana name of the payload type */
char *name;
/*! the iana id of the payload type */
unsigned int id;
/* Audio */
/*! the transfer rate of the payload type */
unsigned int rate;
/*! the bits per second of the payload type */
unsigned int bps;
/* Video */
/*! the width of the video payload type */
unsigned int width;
/*! the width of the video payload type */
unsigned int height;
/*! the framerate of the video payload type */
unsigned int framerate;
unsigned int ptime;
};
typedef struct ldl_payload ldl_payload_t;
enum ldl_transport_type {
LDL_TPORT_RTP,
LDL_TPORT_VIDEO_RTP,
LDL_TPORT_RTCP,
LDL_TPORT_VIDEO_RTCP,
/* Nothing below that line */
LDL_TPORT_MAX
};
typedef enum ldl_transport_type ldl_transport_type_t;
static inline const char *ldl_transport_type_str(ldl_transport_type_t type)
{
static const char *name[] = { "rtp", "video_rtp", "rtcp", "video_rtcp" };
return type >= LDL_TPORT_MAX ? NULL : name[type];
}
static inline ldl_transport_type_t ldl_transport_type_parse(const char *type) {
if (!strcasecmp(type, "rtp")) {
return LDL_TPORT_RTP;
} else if (!strcasecmp(type, "rtcp")) {
return LDL_TPORT_RTCP;
} else if (!strcasecmp(type, "video_rtp")) {
return LDL_TPORT_VIDEO_RTP;
} else if (!strcasecmp(type, "video_rtcp")) {
return LDL_TPORT_VIDEO_RTCP;
} else {
return LDL_TPORT_MAX;
}
}
#if 0
/*! \brief A structure to store a jingle video payload */
struct ldl_vpayload {
/*! the iana name of the video payload type */
char *name;
/*! the iana id of the video payload type */
unsigned int id;
/*! the width of the video payload type */
unsigned int width;
/*! the width of the video payload type */
unsigned int height;
/*! the framerate of the video payload type */
unsigned int framerate;
};
typedef struct ldl_vpayload ldl_vpayload_t;
#endif
struct ldl_handle;
typedef struct ldl_handle ldl_handle_t;
struct ldl_session;
typedef struct ldl_session ldl_session_t;
typedef enum {
LDL_STATUS_SUCCESS,
LDL_STATUS_FALSE,
LDL_STATUS_MEMERR,
} ldl_status;
typedef enum {
LDL_FLAG_INIT = (1 << 0),
LDL_FLAG_RUNNING = (1 << 1),
LDL_FLAG_AUTHORIZED = (1 << 2),
LDL_FLAG_READY = (1 << 3),
LDL_FLAG_CONNECTED = (1 << 4),
LDL_FLAG_STOPPED = (1 << 5),
LDL_FLAG_BREAK = (1 << 6)
} ldl_flag_t;
typedef enum {
LDL_FLAG_NONE = 0,
LDL_FLAG_TLS = (1 << 10),
LDL_FLAG_SASL_PLAIN = (1 << 11),
LDL_FLAG_SASL_MD5 = (1 << 12),
LDL_FLAG_COMPONENT = (1 << 13),
LDL_FLAG_OUTBOUND = (1 << 14),
LDL_FLAG_GATEWAY = (1 << 15),
LDL_FLAG_JINGLE = (1 << 16)
} ldl_user_flag_t;
typedef enum {
LDL_SIGNAL_NONE,
LDL_SIGNAL_INITIATE,
LDL_SIGNAL_CANDIDATES,
LDL_SIGNAL_MSG,
LDL_SIGNAL_PRESENCE_IN,
LDL_SIGNAL_PRESENCE_OUT,
LDL_SIGNAL_PRESENCE_PROBE,
LDL_SIGNAL_ROSTER,
LDL_SIGNAL_SUBSCRIBE,
LDL_SIGNAL_UNSUBSCRIBE,
LDL_SIGNAL_VCARD,
LDL_SIGNAL_TERMINATE,
LDL_SIGNAL_ERROR,
LDL_SIGNAL_LOGIN_SUCCESS,
LDL_SIGNAL_LOGIN_FAILURE,
LDL_SIGNAL_CONNECTED,
LDL_SIGNAL_TRANSPORT_ACCEPT,
LDL_SIGNAL_REJECT,
LDL_SIGNAL_REDIRECT
} ldl_signal_t;
typedef enum {
LDL_REPLY_ACK,
LDL_REPLY_NACK,
} ldl_reply_t;
typedef enum {
LDL_STATE_NEW,
LDL_STATE_ANSWERED,
LDL_STATE_DESTROYED,
} ldl_state_t;
typedef enum {
LDL_DESCRIPTION_INITIATE,
LDL_DESCRIPTION_ACCEPT
} ldl_description_t;
typedef enum {
LDL_QUEUE_NONE,
LDL_QUEUE_SENT
} ldl_queue_t;
#define DL_PRE __FILE__, __LDL_FUNC__, __LINE__
#define DL_LOG_DEBUG DL_PRE, 7
#define DL_LOG_INFO DL_PRE, 6
#define DL_LOG_NOTICE DL_PRE, 5
#define DL_LOG_WARNING DL_PRE, 4
#define DL_LOG_ERR DL_PRE, 3
#define DL_LOG_CRIT DL_PRE, 2
#define DL_LOG_ALERT DL_PRE, 1
#define DL_LOG_EMERG DL_PRE, 0
typedef ldl_status (*ldl_loop_callback_t)(ldl_handle_t *);
typedef ldl_status (*ldl_session_callback_t)(ldl_handle_t *, ldl_session_t *, ldl_signal_t, char *, char *, char *, char *);
typedef ldl_status (*ldl_response_callback_t)(ldl_handle_t *, char *);
typedef void (*ldl_logger_t)(char *file, const char *func, int line, int level, char *fmt, ...);
#define ldl_yield(ms) apr_sleep(ms * 10); apr_thread_yield();
/*!
\brief Test for a common domain in 2 jid
\param id_a the first id
\param id_b the second id
\return 1 if the domains match 0 if they dont or -1 if either id is invalid
\note the id may or may not contain a user and/or resource
*/
static inline int ldl_jid_domcmp(char *id_a, char *id_b)
{
char *id_a_host, *id_b_host, *id_a_r, *id_b_r;
id_a_host = strchr(id_a, '@');
if (id_a_host) {
id_a_host++;
} else {
id_a_host = id_a;
}
id_b_host = strchr(id_b, '@');
if (id_b_host) {
id_b_host++;
} else {
id_b_host = id_b;
}
if (id_a_host && id_b_host) {
size_t id_a_len = 0, id_b_len = 0, len = 0;
id_a_r = strchr(id_a_host, '/');
if (id_a_r) {
id_a_len = id_a_r - id_a_host;
} else {
id_a_len = strlen(id_a_host);
}
id_b_r = strchr(id_b_host, '/');
if (id_b_r) {
id_b_len = id_b_r - id_b_host;
} else {
id_b_len = strlen(id_b_host);
}
if (id_a_len > id_b_len) {
len = id_b_len;
} else {
len = id_a_len;
}
return strncasecmp(id_a_host, id_b_host, len) ? 0 : 1;
}
return -1;
}
/*!
\brief Test for the existance of a flag on an arbitary object
\param obj the object to test
\param flag the or'd list of flags to test
\return true value if the object has the flags defined
*/
#define ldl_test_flag(obj, flag) ((obj)->flags & flag)
/*!
\brief Set a flag on an arbitrary object
\param obj the object to set the flags on
\param flag the or'd list of flags to set
*/
#define ldl_set_flag(obj, flag) (obj)->flags |= (flag)
/*!
\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(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
\param dest the object to copy the flags to
\param src the object to copy the flags from
\param flags the flags to copy
*/
#define ldl_copy_flags(dest, src, flags) (dest)->flags &= ~(flags); (dest)->flags |= ((src)->flags & (flags))
/*!
\brief Test for NULL or zero length string
\param s the string to test
\return true value if the string is NULL or zero length
*/
#define ldl_strlen_zero(s) (s && *s != '\0') ? 0 : 1
/*!
\brief Destroy a Jingle Session
\param session_p the session to destroy
\return SUCCESS OR FAILURE
*/
ldl_status ldl_session_destroy(ldl_session_t **session_p);
/*!
\brief Get a value from a session
\param session the session
\param key the key to look up
\return the value
*/
char *ldl_session_get_value(ldl_session_t *session, char *key);
/*!
\brief Set a value on a session
\param session the session
\param key the key to set
\param val the value of the key
*/
void ldl_session_set_value(ldl_session_t *session, const char *key, const char *val);
/*!
\brief Create a Jingle Session
\param session_p pointer to reference the session
\param handle handle to associate the session with
\param id the id to use for the session
\param them the id of the other end of the call
\param me the id of our end of the call
\param flags user flags
\return SUCCESS OR FAILURE
*/
ldl_status ldl_session_create(ldl_session_t **session_p, ldl_handle_t *handle, char *id, char *them, char *me, ldl_user_flag_t flags);
/*!
\brief get the id of a session
\param session the session to get the id of
\return the requested id
*/
char *ldl_session_get_id(ldl_session_t *session);
/*!
\brief Get the caller name of a session
\param session the session to get the caller from
\return the caller name
*/
char *ldl_session_get_caller(ldl_session_t *session);
/*!
\brief Get the callee name of a session
\param session the session to get the callee from
\return the callee name
*/
char *ldl_session_get_callee(ldl_session_t *session);
/*!
\brief Set the ip of a session
\param session the session to set the ip on
\param ip the ip
*/
void ldl_session_set_ip(ldl_session_t *session, char *ip);
/*!
\brief Get the ip of a session
\param session the session to get the ip from
\return the ip
*/
char *ldl_session_get_ip(ldl_session_t *session);
/*!
\brief Set a private pointer to associate with the session
\param session the session to set the data pointer to
\param private_data the data to associate
*/
void ldl_session_set_private(ldl_session_t *session, void *private_data);
/*!
\brief Get a private pointer from a session
\param session the session to get the data from
\return the data
*/
void *ldl_session_get_private(ldl_session_t *session);
/*!
\brief Accept a candidate
\param session the session to accept on
\param candidate the candidate to accept
*/
void ldl_session_accept_candidate(ldl_session_t *session, ldl_candidate_t *candidate);
/*!
\brief turn logging on/off
\param on (TRUE or FALSE)
\return current state
*/
int ldl_global_debug(int on);
/*!
\brief Set a custom logger
\param logger the logger function
*/
void ldl_global_set_logger(ldl_logger_t logger);
/*!
\brief Perform a probe on a given id to resolve the proper Jingle Resource
\param handle the connection handle to use.
\param id the id to probe
\param from the from string
\param buf a string to store the result
\param len the size in bytes of the string
\return a pointer to buf if a successful lookup was made otherwise NULL
*/
char *ldl_handle_probe(ldl_handle_t *handle, char *id, char *from, char *buf, unsigned int len);
/*!
\brief Perform a discovery on a given id to resolve the proper Jingle Resource
\param handle the connection handle to use.
\param id the id to probe
\param from the from string
\param buf a string to store the result
\param len the size in bytes of the string
\return a pointer to buf if a successful lookup was made otherwise NULL
*/
char *ldl_handle_disco(ldl_handle_t *handle, char *id, char *from, char *buf, unsigned int len);
/*!
\brief Signal a termination request on a given session
\param session the session to terminate
\return TRUE if the signal was sent.
*/
unsigned int ldl_session_terminate(ldl_session_t *session);
/*!
\brief Get the private data of a connection handle
\param handle the conection handle
\return the requested data
*/
void *ldl_handle_get_private(ldl_handle_t *handle);
/*!
\brief Get the full login of a connection handle
\param handle the conection handle
\return the requested data
*/
char *ldl_handle_get_login(ldl_handle_t *handle);
/*!
\brief Send a message to a session
\param session the session handle
\param subject optional subject
\param body body of the message
*/
void ldl_session_send_msg(ldl_session_t *session, char *subject, char *body);
/*!
\brief Send a presence notification to a target
\param handle the handle to send with
\param from the from address
\param to the to address
\param type the type of presence
\param rpid data for the icon
\param message a status message
\param avatar the path to an avatar image
*/
void ldl_handle_send_presence(ldl_handle_t *handle, char *from, char *to, char *type, char *rpid, char *message, char *avatar);
/*!
\brief Send a vcard
\param handle the handle to send with
\param from the from address
\param to the to address
\param id the request id
\param vcard the text xml of the vcard
*/
void ldl_handle_send_vcard(ldl_handle_t *handle, char *from, char *to, char *id, char *vcard);
/*!
\brief Send a message
\param handle the conection handle
\param from the message sender
\param to the message recipiant
\param subject optional subject
\param body body of the message
*/
void ldl_handle_send_msg(ldl_handle_t *handle, char *from, char *to, const char *subject, const char *body);
/*!
\brief Offer candidates to a potential session
\param session the session to send candidates on
\param candidates an array of candidate description objects
\param clen the number of elements in the candidates array
\return the message_id of the generated xmpp request
*/
unsigned int ldl_session_candidates(ldl_session_t *session,
ldl_candidate_t *candidates,
unsigned int clen);
unsigned int ldl_session_transport(ldl_session_t *session,
ldl_candidate_t *candidates,
unsigned int clen);
/*!
\brief Initiate or Accept a new session and provide transport options
\param session the session to initiate or accept
\param payloads an array of payload description objects
\param plen the number of elements in the payloads array
\param description the type of description LDL_DESCRIPTION_INITIATE or LDL_DESCRIPTION_ACCEPT
\return the message_id of the generated xmpp request
*/
unsigned int ldl_session_describe(ldl_session_t *session,
ldl_payload_t *payloads,
unsigned int plen,
ldl_description_t description, unsigned int *audio_ssrc, unsigned int *video_ssrc,
ldl_crypto_data_t *audio_crypto_data, ldl_crypto_data_t *video_crypto_data);
/*!
\brief get a session's state
\param session a session to get the state from
\return the state
*/
ldl_state_t ldl_session_get_state(ldl_session_t *session);
/*!
\brief get the candidates
\param session the session
\param tport type of transport (rtp,rtcp,video_rtp,video_rtcp,etc.)
\param candidates pointer to point at array of the candidates
\param len the resulting len of the array pointer
\return success or failure
*/
ldl_status ldl_session_get_candidates(ldl_session_t *session, ldl_transport_type_t tport, ldl_candidate_t **candidates, unsigned int *len);
/*!
\brief get the payloads
\param session the session
\param payloads pointer to point at array of the payloads
\param len the resulting len of the array pointer
\return success or failure
*/
ldl_status ldl_session_get_payloads(ldl_session_t *session, ldl_payload_t **payloads, unsigned int *len);
/*!
\brief Initilize libDingaLing
\param debug debug level
\return success or failure
*/
ldl_status ldl_global_init(int debug);
/*!
\brief Destroy libDingaLing
\return success or failure
*/
ldl_status ldl_global_destroy(void);
/*!
\brief Set the log stream
\param log_stream the new log stream
*/
void ldl_global_set_log_stream(FILE *log_stream);
int8_t ldl_handle_ready(ldl_handle_t *handle);
/*!
\brief Initilize a new libDingaLing handle
\param handle the Dingaling handle to initialize
\param login the xmpp login
\param password the password
\param server the server address
\param flags user flags
\param status_msg status message to advertise
\param loop_callback optional loop callback
\param session_callback function to call on session signalling
\param response_callback function to call on responses
\param private_info optional pointer to private data
\return success or failure
*/
ldl_status ldl_handle_init(ldl_handle_t **handle,
char *login,
char *password,
char *server,
ldl_user_flag_t flags,
char *status_msg,
ldl_loop_callback_t loop_callback,
ldl_session_callback_t session_callback,
ldl_response_callback_t response_callback,
void *private_info);
/*!
\brief Run a libDingaLing handle
\param handle the Dingaling handle to run
*/
void ldl_handle_run(ldl_handle_t *handle);
int ldl_session_gateway(ldl_session_t *handle);
void ldl_session_set_gateway(ldl_session_t *session);
/*!
\brief Stop a libDingaLing handle
\param handle the Dingaling handle to stop
*/
void ldl_handle_stop(ldl_handle_t *handle);
int ldl_handle_running(ldl_handle_t *handle);
int ldl_handle_connected(ldl_handle_t *handle);
int ldl_handle_authorized(ldl_handle_t *handle);
/*!
\brief Destroy a libDingaLing handle
\param handle the Dingaling handle to destroy
\return success or failure
*/
ldl_status ldl_handle_destroy(ldl_handle_t **handle);
/*!
\brief Set the log stream on a handle
\param handle the Dingaling handle
\param log_stream the new log stream
*/
void ldl_handle_set_log_stream(ldl_handle_t *handle, FILE *log_stream);
ldl_status ldl_global_terminate(void);
///\}
#ifdef __cplusplus
}
#endif
/** \mainpage libDingaling
* libDingaling - Cross Platform Jingle (Google Talk) voip signaling library
* \section intro Introduction
*
* \section supports Supported Platforms
* libDingaling has been built on the following platforms:
*
* - Linux (x86, x86_64)
* - Windows (MSVC 2005)
* - Mac OS X (intel & ppc )
*
* \section depends Dependencies
* libDingaling makes use of the following external libraries.
*
* - APR (http://apr.apache.org)
* - iksemel (http://iksemel.jabberstudio.org/)
*
* \section license Licensing
*
* libDingaling is licensed under the terms of the MPL 1.1
*
*/
#endif
/* For Emacs:
* Local Variables:
* mode:c
* indent-tabs-mode:t
* tab-width:4
* c-basic-offset:4
* End:
* For VIM:
* vim:set softtabstop=4 shiftwidth=4 tabstop=4 noet:
*/