libeXosip2(1-1) -- How-To initialize libeXosip2.
How-To initialize libeXosip2.
Initialize eXosip and prepare transport layer
When using eXosip, your first task is to initialize both eXosip context and libosip library (parser and state machines). This must be done prior to any use of libeXosip2.
Also, you need to prepare the transport layer and choose either UDP, TCP, TLS or DTLS.
Here is the basic mandatory setup:
- Initialize the osip trace (compile this code with -DENABLE_TRACE)
#include <eXosip2/eXosip.h>
eXosip_t *ctx;
int i;
int port=5060;
TRACE_INITIALIZE (6, NULL);
- Initialize eXosip (and osip) stack
ctx = eXosip_malloc();
if (ctx==NULL)
return -1;
i=eXosip_init(ctx);
if (i!=0)
return -1;
- Open a UDP socket for signalling
i = eXosip_listen_addr (ctx, IPPROTO_UDP, NULL, port, AF_INET, 0);
if (i!=0)
{
eXosip_quit(ctx);
fprintf (stderr, "could not initialize transport layer\n");
return -1;
}
Choosing UDP, TCP, TLS or DTLS
If you wish to use other transport protocol, you can select:
- UDP:
i = eXosip_listen_addr (ctx, IPPROTO_UDP, NULL, 5060, AF_INET, 0);
- TCP:
i = eXosip_listen_addr (ctx, IPPROTO_TCP, NULL, 5060, AF_INET, 0);
- TLS:
i = eXosip_listen_addr (ctx, IPPROTO_TCP, NULL, 5061, AF_INET, 1);
- DTLS:
i = eXosip_listen_addr (ctx, IPPROTO_UDP, NULL, 5061, AF_INET, 1);
Specific setup for TLS
TLS may requires specific setup. TLS introduce in fact two interesting features:
- Using certiticates and keys, it helps to trust/verify the remote server.
- It also encrypts data so that no man-in-the-middle could read the SIP traffic.
If you don't need server verification, TLS is very easy to setup. You don't need to configure any certificate, key or root certificate...
Here is the code to disable certificate verification:
int val=0;
i = eXosip_set_option (ctx, EXOSIP_OPT_SET_TLS_VERIFY_CERTIFICATE, (void*)&val);
If you require validation, a few work still needs to be done. What you need depends on your platform/os.
- Certificate on windows:
On windows, exosip has built-in support for "Windows Certificate Store". Thus, you only need to add your certifcate and keys in the official "Windows Certificate Store".
- Certificate on macosx:
On macosx, exosip has built-in support for the certificate store.
- Certificate on other platforms:
eXosip_tls_ctx_t tls_info;
memset(&tls_info, 0, sizeof(eXosip_tls_ctx_t));
snprintf(tls_info.client.cert, sizeof(tls_info.client.cert), "user-cert.crt");
snprintf(tls_info.client.priv_key, sizeof(tls_info.client.priv_key), "user-privkey.crt");
snprintf(tls_info.client.priv_key_pw, sizeof(tls_info.client.priv_key_pw), "password");
snprintf(tls_info.root_ca_cert, sizeof(tls_info.root_ca_cert), "cacert.crt"); i = eXosip_set_option (ctx, EXOSIP_OPT_SET_TLS_CERTIFICATES_INFO, (void*)&tls_info);
Additionnal setup
A few options can be modified in eXosip2. However, most default are good values and if your sip service is configured correctly, no much settings beyond default would be required.
- EXOSIP_OPT_UDP_KEEP_ALIVE 1
- EXOSIP_OPT_UDP_LEARN_PORT 2
- EXOSIP_OPT_USE_RPORT 7
- EXOSIP_OPT_SET_IPV4_FOR_GATEWAY 8
- EXOSIP_OPT_ADD_DNS_CACHE 9
- EXOSIP_OPT_DELETE_DNS_CACHE 10
- EXOSIP_OPT_SET_IPV6_FOR_GATEWAY 12
- EXOSIP_OPT_ADD_ACCOUNT_INFO 13
- EXOSIP_OPT_DNS_CAPABILITIES 14
- EXOSIP_OPT_SET_DSCP 15
- EXOSIP_OPT_REGISTER_WITH_DATE 16
- EXOSIP_OPT_SET_HEADER_USER_AGENT 17
- EXOSIP_OPT_SET_TLS_VERIFY_CERTIFICATE 500
- EXOSIP_OPT_SET_TLS_CERTIFICATES_INFO 501
- EXOSIP_OPT_SET_TLS_CLIENT_CERTIFICATE_NAME 502
- EXOSIP_OPT_SET_TLS_SERVER_CERTIFICATE_NAME 503
Here is a basic setup that might be appropriate for usual configuration:
int val;
eXosip_set_user_agent (ctx, "exosipdemo/0.0.0");
val=17000;
eXosip_set_option (ctx, EXOSIP_OPT_UDP_KEEP_ALIVE, (void*)&val);
val=2;
eXosip_set_option (ctx, EXOSIP_OPT_DNS_CAPABILITIES, (void*)&val);
val=1;
eXosip_set_option (ctx, EXOSIP_OPT_USE_RPORT, (void*)&val);
val=26;
eXosip_set_option (ctx, EXOSIP_OPT_SET_DSCP, (void*)&dscp_value);
eXosip_set_option (ctx, EXOSIP_OPT_SET_IPV4_FOR_GATEWAY, "sip.antisip.com");
NAT and Contact header
There would be much to say about this. Most being unrelated to the eXosip2 stack itself...
The most important feature with SIP is to be able to receive SIP requests. You wouldn't be glad if your phone remains silent. However, it's in theory not possible to guess what should contain the Contact headers we are creating.
Most proxy will repair our broken contact, no matter why eXosip2 or any SIP application has provided a wrong value. The SIP specification is not very clear on the various client and server behaviors for Contact verifications.
However:
- No matter, what eXosip2, most proxy will repair correctly.
- No matter what you think is right, some people think another way is right.
- No matter sip, proxy and people: NETWORK RULES ALWAYS APPLIES FIRST! ;)
Anyway, to avoid problems:
- You should always do your best to put correct information in Contact.
- Most of the time, you can't, but it should work anyway.
Conclusion:
- Without any configuration (NAT, STUN, etc), your proxy should be able to find out how to reach you (over the existing connection).
- If it can't (whatever the reason), you can try workarounds and options.
The options you have:
- This option helps exosip2 to detect the localip when you have several ones: (VPN and eth0 for example). This will helps to detect localip for Via and Contact. The usual parameter is the proxy. (cautious: method may block because of DNS operation)
eXosip_set_option (ctx, EXOSIP_OPT_SET_IPV4_FOR_GATEWAY, "sip.antisip.com");
masquerading: When sending your first SIP request (REGISTER? OPTIONS?), the top Via of the answers will contain a "received" and "rport" parameters: those IP/port are the exact ones required for your Contact headers with the proxy. STUN will detect similar IP/port but for another destination (the stun server). Thus, STUN is not the correct way.
Thus, send a request to your proxy, check the via (received and rport) parameter and use masquerading:
eXosip_masquerade_contact (ctx, "91.121.81.212", 10456);
- EXOSIP_OPT_UDP_LEARN_PORT option: If you wish to re-use Via "received" and "rport" automatically with UDP. With the following code, the second REGISTER (after authentication?) or second outgoing REQUEST will contains the masqueraded Contact header. It should also be used if you masquerade using STUN values.
eXosip_masquerade_contact (ctx, "192.168.2.1", 5080);
val=1;
eXosip_set_option (ctx, EXOSIP_OPT_UDP_LEARN_PORT, &val);
- EXOSIP_OPT_USE_RPORT option: only use with BROKEN nat. This option remove the "rport" parameter in outgoing REQUEST. This should be never used.
val=0;
eXosip_set_option (ctx, EXOSIP_OPT_USE_RPORT, &val);
If you still have NAT issue, think about using TLS: broken NAT sometimes block SIP packets, but with encryption, broken NAT can't do anything!
About DNS
eXosip2 should be compiled with c-ares. This is very important as c-ares provides non blocking, portable and full support for all required SIP operation.
By default, SIP requires usage of specific DNS features to discover the IP address of a sip service.
- NAPTR to discover the SRV records
- SRV records are then used to receive the list of hosts.
- DNS resolution provide IPs of the host.
For complete information, check rfc3263.txt: Locating SIP servers.
- To use NAPTR:
val=2;
eXosip_set_option (ctx, EXOSIP_OPT_DNS_CAPABILITIES, &val);
If NAPTR is not set for the service you used which happens in many case, SRV record or normal DNS will be used as a fallback. It should not slow too much the procedure. However, it is still usefull in some case to disable NAPTR because there still exist a few DNS server that remains silent when sending NAPTR request. In that very specific use-case, this may lead to very slow fallback to normal DNS...
- To not use NAPTR:
val=0;
eXosip_set_option (ctx, EXOSIP_OPT_DNS_CAPABILITIES, &val);
Handle eXosip2 events (eXosip_event_t)
The eXosip_event contains all information you need:
- rid: identifier for registrations.
- tid: identifier for transactions.
- cid, did identifiers for calls.
- sid, did: identifier for outgoing subscriptions.
- nid, did: identifier for incoming subscriptions.
- request: outgoing or incoming request for the event
- answer: outgoing or incoming answer for the event
- ack: outgoing or incoming ACK for the event
Those identifiers are re-used in related eXosip2 API to make it simpler to control context. The request, answer and ack are fully duplicated so that you can access them without locking the eXosip2 context.
Now you have to handle eXosip events. Here is some code to get eXosip_event from the eXosip2 stack.
Note: For advanced users, or more real time app, you may also use lower level API so that you can get woken up on a select call when an event occurs.
eXosip_event_t *evt;
for (;;)
{
evt = eXosip_event_wait (ctx, 0, 50);
eXosip_lock(ctx);
eXosip_automatic_action (ctx);
eXosip_unlock(ctx);
if (evt == NULL)
continue;
if (evt->type == EXOSIP_CALL_NEW)
{
....
....
}
else if (evt->type == EXOSIP_CALL_ACK)
{
....
....
}
else if (evt->type == EXOSIP_CALL_ANSWERED)
{
....
....
}
else .....
....
....
eXosip_event_free(evt);
}
You will receive one event for each SIP message sent. Each event contains the original request of the affected transaction and the last response that triggers the event when available.
You can access all headers from those messages and store them in your own context for other actions or graphic displays.
For example, when you receive a REFER request for a call transfer, you'll typically retreive the "refer-To" header:
osip_header_t *referto_head = NULL;
i = osip_message_header_get_byname (evt->sip, "refer-to", 0, &referto_head);
if (referto_head == NULL || referto_head->hvalue == NULL)
Here are a few examples:
- Answer 180 Ringing to an incoming INVITE:
if (evt->type == EXOSIP_CALL_NEW)
{
eXosip_lock (ctx);
eXosip_call_send_answer (ctx, evt->tid, 180, NULL);
eXosip_unlock (ctx);
}
- Answer 200 Ok to an incoming MESSAGE: (also check the attachment in evt->request!)
if (evt->type == EXOSIP_MESSAGE_NEW && osip_strcasecmp (minfo.method, "MESSAGE") == 0) {
{
osip_message_t *answer=NULL;
int i;
eXosip_lock (ctx);
i = eXosip_message_build_answer (ctx, evt->tid, 200, &answer);
i = eXosip_message_send_answer (ctx, evt->tid, 200, answer);
eXosip_unlock (ctx);
}
- Handle incoming REFER within dialog (call transfer): fg
if (evt->type == EXOSIP_CALL_MESSAGE_NEW && osip_strcasecmp (minfo.method, "REFER") == 0) {
osip_header_t *refer_to = NULL;
eXosip_lock (ctx);
i = eXosip_call_build_answer (ctx, evt->tid, 202, &answer);
i = eXosip_call_send_answer (ctx, evt->tid, 202, answer);
i = osip_message_header_get_byname (evt->request, "refer-to", 0, &refer_to);
if (i >= 0) {
printf ("you must start call to: %s\n", refer_to->hvalue);
...
}
else {
}
eXosip_call_terminate (ctx, evt->cid, evt->did, 486);
eXosip_unlock (ctx);
}
libeXosip2(1-1) -- How-To initialize libeXosip2.的更多相关文章
- libeXosip2(1) -- Modules
Modules Here is a list of all modules: [detail level 12] The eXtented eXosip stack LibeXosip2 Versio ...
- SIP UserAgent (B2BUA client)——libosip2 libeXosip2
1. libosip2 libeXosip2 http://www.antisip.com/download/exosip2/ Features:--------eXosip2 has support ...
- libeXosip2(3-1) -- eXosip2 INVITE and Call Management
eXosip2 INVITE and Call Management SIP messages and call control API Functions int eXosip_call_set_ ...
- libeXosip2(3) -- SIP messages and call control API
SIP messages and call control API The SIP messages and call control API. More... Modules eXosip2 INV ...
- libeXosip2(2-3) -- eXosip2 event API
eXosip2 event API General purpose API. Data Structures struct eXosip_event Enumerations enum eXo ...
- libeXosip2(2-2) -- eXosip2 network API
eXosip2 network API General purpose API. Functions int eXosip_transport_set (osip_message_t *msg, c ...
- libeXosip2(2-1) -- eXosip2 configuration API
eXosip2 configuration API General purpose API. Data Structures struct eXosip_dns_cache struct eX ...
- libeXosip2(2) -- General purpose API.
General purpose API. general purpose API in libeXosip2-4.0.0. More... Modules eXosip2 configuration ...
- libeXosip2(1-3) -- How-To send or update registrations.
How-To send or update registrations. The eXtented eXosip stack Initiate a registration To start a re ...
随机推荐
- CSS的基本认识
1.定义: 级联样式表(Cascading Style Sheet)简称“CSS”,通常又称为“风格样式表(Style Sheet)”,它是用来进行网页风格设计的. 2.对CSS的基本认识: CSS是 ...
- 原生sql语句执行
public function Text() { $nation = D("Nation"); $sqla = "select * from nation"; ...
- Annotation(一)——注解开发介绍
<p>在编程中,一直强调的一点就是注释的编写,注释的规范等等.尤其是越是核心,程序越复杂,逻辑越多的清空下,注释的编写对我们以后的阅读代码,维护软件起着至关重要的作用.一款软件有着好的注释 ...
- 56个PHP开发常用代码
2016/02/14 6203 4 在编写代码的时候有个神奇的工具总是好的!下面这里收集了 50+ PHP 代码片段,可以帮助你开发 PHP 项目. 这些 PHP 片段对于 PHP 初学者也非常 ...
- (转)iOS7界面设计规范(13) - UI基础 - 与iOS的系统整合
突然就到了周日傍晚.你永远不会知道自己的生活在接下来的一周当中能够发生多少变化:各种不可预知性所带来的更多是快感还是焦虑与不安,冷暖自知.相比之下,白天工作当中那些需求列表与排期文档就显得那么可爱了, ...
- 使用PLSql连接Oracle时报错ORA-12541: TNS: 无监听程序
非常多时候为了优化我们的启动项把oracle的服务禁止了.但是重新启动启动之后使用PLSQL登陆oracle时会出现无监听程序,这说明我们有一些服务没有启动.我们先查看一下oracle的服务是否启动, ...
- [Cycle.js] Customizing effects from the main function
How can we show one string on the DOM, and a completely different string on Console log? This lesson ...
- [Regular Expressions] Find Plain Text Patterns
The simplest use of Regular Expressions is to find a plain text pattern. In this lesson we'll look a ...
- c#利用VM_COPYDATA实现进程间通信
c#进程间的通信方式很多种,只会这种,感觉比较简单.不懂原理,能用就行. 假设有两个程序:server(主进程),client(子进程) 1.server端: /*定义一个结构体,用来接收从子进程传过 ...
- VS2015 新Web项目(C#6)出现CS1617异常的解决
VS2015 新Web项目(C#6)出现CS1617错误的解决 VS2015新增了对C#6的支持. 在新的Web项目模板中通过引入nuget包Microsoft.CodeDom.Providers.D ...