大家好,又见面了,我是你们的朋友全栈君。
最近接到任务,要写一个SIP协议转换的服务器。但个人对SIP了解非常少,幸好学长写有一个类似的东西,于是向学长请教。但是由于个人悟性问题(微笑),我除了知道了用了osip和exosip这个库之外,其他一无所知。于是回实验室开始看源码+网上查资料,发现了一个问题:目前学长的程序还有网上的资料千篇一律全是只添加了一个body(即SDP部分),但是根据实际,我需要添加两个body(SDP和XML),但网上的资料和手里的程序没有任何地方提到了添加第二个body的方法,并且第二个body还是xml格式的更没人提到了。
查了两三个小时的资料后,我在网上找到了osip_message_set_content_type(osip_message_t * sip, const char *hvalue)函数的原型,需要用这个函数来设置SIP信令中的content_type头域,但是源码中却说 if (sip->content_type != NULL),就要return,不做任何操作。但我需要两个body啊,word哥。。于是到了昨晚凌晨0点,我甚至认为osip和exosip这个库无法添加两个body,打算看有没有其它好用的库,心灰意冷的回宿舍洗洗睡了。
但我是个只要有点心事就睡不着的人,凌晨三点醒了,掏出了电脑,想着毕竟osip和exosip项目的那么多人都是国外大神,肯定考虑了这个问题,就换了个思路——硬着头皮开始看osip和exosip的源码。一直到今天中午12点多总算有点头绪了。。
一、构建SIP信令的body
我看了我需要构建的SIP信令格式,发现了我之前一直忽略了的一个东西,如图:
这是第一个content_type的内容,填的是multipart/mixed。在exosip的offans.c中,有一个sdp_message_t *eXosip_get_sdp_info(osip_message_t * message)函数,其中有一段代码如下描述:
if (osip_strcasecmp(ctt->type, “multipart”) == 0) {
/* probably within the multipart attachement */
} else if (osip_strcasecmp(ctt->type, “application”) != 0 ||osip_strcasecmp(ctt->subtype, “sdp”) != 0)
return NULL;
可见,肯定一上来就是要先通过 osip_message_set_content_type函数来将整个message的content_type设置为multipart,这样才能添加多个body。
那么问题来了,怎样添加第一个body?首先看看osip中osip_message.h关于osip_message_t这个数据结构的定义(图中为其中的一部分):
其中,osip_list_t bodies这个元素非常重要,个人感觉有点像个链表一样,存储着所有body的相关内容。再看osip_body.c中的osip_message_set_body(osip_message_t * sip, const char *buf, size_t length)函数:
所以,可以通过 osip_message_set_body向osip_message_t这个数据结构的bodies元素添加字符串,字符串可以是SDP,也可以是其它格式的东西。通过图中所示的osip_list_add,就可以向bodies中添加(存储的数据结构与链表有点像,就不多说了)。
那么,SDP和XML这两个body总算是塞进去了,那么怎样添加这两个body的content_type呢?网上又没资料(微笑),我在osip.c中,找到了osip_body_set_contenttype(osip_body_t * body, const char *hvalue),函数体如下图所示:
其中,osip_body_t的数据结构如下图所示:
那么问题又来了,如何从sip->bodids中,获得osip_body_t类型的body呢?我在忘了哪个.c文件里看到了一种写法:body = (osip_body_t *) osip_list_get(&sip->bodies, pos); (其中,如果只有一个body,pos取0;如果取第二个body,pos取1)。在osip_list.c中, osip_list_get的函数体如下图所示:
既然得到了body,那就set_content_type吧,可以为啊application/sdp,也可以为application/**xml等。至此,SIP两个body的构建应该就能行了,至于长度之类的,如果原来的库中没计算,用sizeof计算一下,填上去就行了吧。(但是我个人觉得应该是计算了的,没计算的话再议)。
二、解析SIP信令的body
学长给的程序中,只有一个eXosip_get_sdp_info的使用,因为只有一个body,那就是sdp。exosip库提供了API用来解析sdp,非常方便。 eXosip_get_sdp_info的函数体如下图所示:
可见,如果有两个body,该函数仍然能够解析出SDP部分的内容,并且返回一个sdp_message_t的结构体。如何通过API得到第二个body(即XML)的内容呢?在osip_body.c中,有一个osip_message_get_body的函数,pos填1就行,函数体如下图所示:
得到了osip_body_t后,通过dest->body得到字符串,再通过libxml相关库函数即可解析出xml中的数据。
三、写在后面
以上内容,全是准备工作,没有经过实际验证,但是我在想应该是没有啥问题,毕竟是一个成熟的库,希望不要坑我(破涕为笑)。如果有问题,待我完成服务器后,再写吧。
其实我写这个主要是为了我在短时间之内不会忘,趁热打铁,毕竟学渣一枚,忘了就又要重来, 毕竟网上这方面的资料不一样的好少。。。。。。。
不管了,我要回宿舍补觉了,下周和下下周的考试裸考就裸考吧,暂时还不想复习(憨笑)。
发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/147998.html原文链接:https://javaforall.cn
【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛
【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...