从网上搞了一个基于osip 库的 SIP 协议的简单的 UAC 代理客户端和 UAS 代理服务器端,并进行了编译连接,代码整理后如下:

----------- UAC 代理客户端的代码整理 ---------------

/**
 * 一个使用了 osip 和 eXosip 库的 UAC 代理客户端的演示程序
 *
 * - 只是简单的演示了使用了 osip 和 eXosip2 库的 UAC 代理客户端的如下几个功能:
 * * i 发起呼叫 INVITE 请求
 * * h 挂断会话
 * * s 执行方法 INFO
 * * m 执行方法 MESSAGE
 *
 * 编 译:g++ -I/usr/local/include -L/usr/local/lib ua_client.cpp -o ua_client -leXosip2 -losip2 -losipparser2 -lpthread
 *
 */

#include <osip2/osip_mt.h>
#include <eXosip2/eXosip.h>

#include <netinet/in.h>
#include <sys/socket.h>
#include <sys/types.h>

#include <iostream>
#include <string>

using namespace std;

int main(int argc, char* argv[])
{
    eXosip_event_t *je;
    osip_message_t *reg = NULL;
    osip_message_t *invite = NULL;
    osip_message_t *ack = NULL;
    osip_message_t *info = NULL;
    osip_message_t *message = NULL;
    int call_id, dialog_id;
    int i,flag;
    int flag1 = 1;
    int id;
    
    string strIdentity = "sip:136@133.37.55.136";
    string strRegisterer = "sip:133.37.55.136:5060"; // server ip

string strSrcCall = "sip:136@133.37.55.136";
    string strDestCall = "sip:136@133.37.55.136:5060"; // server ip

char command;
    char tmp[4096];
    char localip[128];

string strHelp = string("\n\t--> 命令字符 功能描述 <--\n\n")
                        + "\t\tr 向服务器注册\n"
                        + "\t\tc 取消注册\n"
                        + "\t\ti 发起呼叫请求\n"
                        + "\t\th 挂断\n"
                        + "\t\tq 退出程序\n"
                        + "\t\ts 执行方法 INFO\n"
                        + "\t\tm 执行方法 MESSAGE\n"
                        + "\t\te 帮助\n\n";
    cout << strHelp;

string strMsg;

i = eXosip_init ();
    if (i != 0)
    {
        cout << "\t--> Couldn't initialize eXosip! <--\n";
        return -1;
    }
    else
    {
        cout << "\t--> eXosip_init successfully! <-- \n\n";
    }

i = eXosip_listen_addr (IPPROTO_UDP, NULL, 5061, AF_INET, 0);
    if (i != 0)
    {
        eXosip_quit ();
        cerr << "\n\t--> Couldn't initialize transport layer! <-- \n\n";
        return -1;
    }
    flag = 1;
    while (flag)
    {
        cout << "请输入一个命令字符:\t";
        cin >> command;
      
        switch (command)
        {
        case 'r':
            cout << "\n\t--> This modal isn't commpleted! \n" << endl;
            break;

case 'i': // 初始化的 INVITE 请求
            i = eXosip_call_build_initial_invite (&invite,
                                                  strDestCall.c_str(),
                                                  strSrcCall.c_str(),
                                                  NULL,
                                                  "This is a call for a conversation");
            if (i != 0)
            {
                cout << "\n --> Intial INVITE failed! <-- \n";
                break;
            }

// 符合 SDP 格式, 其中属性 a 是自定义格式,也就是说可以存放自己的信息, 
            // 但是只能是两列,比如帐户信息
            // 但是经测试,格式: v o t必不可少,原因未知,估计是协议栈在传输时需要检查的

strMsg = string("v=0\r\n")
                   + "o=anonymous 0 0 IN IP4 0.0.0.0\r\n"
                   + "t=1 10\r\n"
                   + "a=username:bluesea\r\n"
                   + "a=password:123456\r\n";

osip_message_set_body (invite, strMsg.c_str(), strMsg.length());
            osip_message_set_content_type (invite, "application/sdp");
      
            // 这里使用了锁机制以保证同步
            eXosip_lock ();
            i = eXosip_call_send_initial_invite (invite);
            eXosip_unlock ();
            flag1 = 1;
            while (flag1)
            {
                je = eXosip_event_wait (0, 200);
                if (je == NULL)
                {
                    cout << "\n\t--> No response or the time is over! <--\n" << endl;
                    break;
                }
          
                switch (je->type)
                {
                case EXOSIP_CALL_INVITE:
                    cout << "\n\t--> a new invite reveived! <--\n" << endl;
                    break;

// announce processing by a remote app
                case EXOSIP_CALL_PROCEEDING:
                    cout << "\n\t--> proceeding! <--\n" << endl;
                    break;

// announce ringback
                case EXOSIP_CALL_RINGING:
                    cout << "\n\t--> ringing! <--\n"
                         << "\n\tcall_id is " << je->cid
                         << ", dialog_id is " << je->did << endl;
                    break;

// 收到请求,表示连接成功,下面发送回复确认
                case EXOSIP_CALL_ANSWERED:
                    cout << "\n\t--> ok! connected! <--\n" << endl;
                    call_id = je->cid;
                    dialog_id = je->did;
                    cout << "\n\tcall_id is " << je->cid
                         << ", dialog_id is " << je->did << endl;
                    eXosip_call_build_ack (je->did, &ack);
                    eXosip_call_send_ack (je->did, ack);
                    flag1 = 0;
                    break;

case EXOSIP_CALL_CLOSED:
                    cout << "\n\t--> the other sid closed! <--\n" << endl;
                    break;

case EXOSIP_CALL_ACK:
                    cout << "\n\t--> ACK received! <--\n" << endl;
                    break;

default:
                    cout << "\n\t--> other response!\n" <<endl;
                    break;
                }
          
                eXosip_event_free (je);
            }

break;

case 'h':
            cout << "\n\t--> Holded ! \n" << endl;
      
            eXosip_lock ();
            eXosip_call_terminate (call_id, dialog_id);
            eXosip_unlock ();
            break;

case 'c':
            cout << "\n\t--> This modal isn't commpleted! \n" << endl;
            break;

case 's':
            // 传输 INFO 方法
            eXosip_call_build_info (dialog_id, &info);
            
            snprintf (tmp , 4096, "hello,bluesea");
            osip_message_set_body (info, tmp, strlen(tmp));

// 格式可以任意设定, text/plain 代表文本信息
            osip_message_set_content_type (info, "text/plain");
            eXosip_call_send_request (dialog_id, info);
            break;

case 'm':
            // 传输 MESSAGE方法,也就是即时消息,
            // 和 INFO 方法相比,主要区别,是 MESSAGE 不用建立连接,直接传输信息,
            // 而 INFO 必须在建立 INVITE 的基础上传输。
            cout << "\n\t--> the mothed :MESSAGE \n" << endl;
            eXosip_message_build_request (&message,
                                          "MESSAGE",
                                          strDestCall.c_str(),
                                          strSrcCall.c_str(),
                                          NULL);
            strMsg = "message: hello bluesea!";
            osip_message_set_body (message, strMsg.c_str(), strMsg.length());
      
            // 假设格式是xml
            osip_message_set_content_type (message, "text/xml");
            eXosip_message_send_request (message);
            break;

case 'q':
            eXosip_quit ();
            cout << "\n\t--> Exit the setup! \n" << endl;;
            flag = 0;
            break;

case 'e':
            cout << strHelp << endl;
            break;

default:
            cout << "\n\t--> 不支持的命令 <--\n" << endl;
            break;
        }
    }

return 0;
}

----------- UAS 代理服务器端的代码整理 ---------------

/**
 * 一个使用了 osip 和 eXosip 库的 UAS 代理服务端的演示程序
 *
 * - 只是简单的演示了使用了 osip 和 eXosip2 库的 UAS 代理服务端的如下几个功能:
 *
 * 编 译:g++ -I/usr/local/include -L/usr/local/lib ua_server.cpp -o ua_server -leXosip2 -losip2 -losipparser2 -lpthread
 *
 */

#include <eXosip2/eXosip.h>

#include <netinet/in.h>
#include <sys/socket.h>
#include <sys/types.h>

#include <iostream>
#include <fstream>
#include <string>

using namespace std;

int main()
{
    eXosip_event_t *je = NULL;
    osip_message_t *ack = NULL;
    osip_message_t *invite = NULL;
    osip_message_t *answer = NULL;
    sdp_message_t *remote_sdp = NULL;
    int call_id, dialog_id;
    int i,j;
    int id;

char *sour_call = "sip:136@133.37.55.136";
    char *dest_call = "sip:136@133.37.55.136:5061"; //client ip/port

char command;
    char tmp[4096];
    char localip[128];
    int pos = 0;

// 初始化 sip
    i = eXosip_init ();
    if (i != 0)
    {
        cerr << "\n\t--> Can't initialize eXosip!\n";
        return -1;
    }
    else
    {
        cout << "\n\t--> eXosip_init successfully!\n";
    }
    
    i = eXosip_listen_addr (IPPROTO_UDP, NULL, 5060, AF_INET, 0);
    if (i != 0)
    {
        eXosip_quit ();
        cerr << "\n\t--> eXosip_listen_addr error! Couldn't initialize transport layer!\n";
    }

for(;;)
    {
        // 侦听是否有消息到来
        je = eXosip_event_wait (0, 50);

// 协议栈带有此语句,具体作用未知
        eXosip_lock ();
        eXosip_default_action (je);
        eXosip_automatic_refresh ();
        eXosip_unlock ();

if (je == NULL) // 没有接收到消息,继续
        {
            continue;
        }

switch (je->type)
        {
        case EXOSIP_MESSAGE_NEW: // 新的消息到来
            cout << "\n\t*** EXOSIP_MESSAGE_NEW!\n" << endl;

if (MSG_IS_MESSAGE (je->request)) // 如果接收到的消息类型是 MESSAGE
            {
                {
                    osip_body_t *body;
                    osip_message_get_body (je->request, 0, &body);
                    cout << "I get the msg is: " << body->body << endl;
                }

// 按照规则,需要回复 OK 信息
                eXosip_message_build_answer (je->tid, 200, &answer);
                eXosip_message_send_answer (je->tid, 200, answer);
            }
            break;

case EXOSIP_CALL_INVITE: // INVITE 请求消息
            // 得到接收到消息的具体信息
            cout << "\n\tReceived a INVITE msg from " << je->request->req_uri->host
                 << " : " << je->request->req_uri->port
                 << ", username is " << je->request->req_uri->username << endl;

// 得到消息体,认为该消息就是 SDP 格式.
            remote_sdp = eXosip_get_remote_sdp (je->did);
            call_id = je->cid;
            dialog_id = je->did;
        
            eXosip_lock ();

eXosip_call_send_answer (je->tid, 180, NULL);
            i = eXosip_call_build_answer (je->tid, 200, &answer);
            if (i != 0)
            {
                cout << "\n\t--> This request msg is invalid! Cann't response!\n" << endl;
                eXosip_call_send_answer (je->tid, 400, NULL);
            }
            else
            {
                snprintf (tmp, 4096,
                    "v=0\r\n"
                    "o=anonymous 0 0 IN IP4 0.0.0.0\r\n"
                    "t=1 10\r\n"
                    "a=username:rainfish\r\n"
                    "a=password:123\r\n");
            
                // 设置回复的SDP消息体,下一步计划分析消息体
                // 没有分析消息体,直接回复原来的消息,这一块做的不好。
                osip_message_set_body (answer, tmp, strlen(tmp));
                osip_message_set_content_type (answer, "application/sdp");
            
                eXosip_call_send_answer (je->tid, 200, answer);
                cout << "\n\t--> send 200 over!" << endl;
            }

eXosip_unlock ();
        
            // 显示出在 sdp 消息体中的 attribute 的内容,里面计划存放我们的信息
            cout << "\n\t--> The INFO is :\n" ;
            while (!osip_list_eol ( &(remote_sdp->a_attributes), pos))
            {
                sdp_attribute_t *at;
            
                //这里解释了为什么在SDP消息体中属性a里面存放必须是两列
                at = (sdp_attribute_t *) osip_list_get ( &(remote_sdp->a_attributes), pos);
                cout << "\n\t" << at->a_att_field
                     << " : " << at->a_att_value << endl;
            
                pos ++;
            }
            break;

case EXOSIP_CALL_ACK:
            cout << "\n\t--> ACK recieved!\n" << endl;
            // printf ("the cid is %s, did is %s\n", je->did, je->cid); 
            break;

case EXOSIP_CALL_CLOSED:
            cout << "\n\t--> the remote hold the session!\n" << endl;
            // eXosip_call_build_ack(dialog_id, &ack);
            // eXosip_call_send_ack(dialog_id, ack); 
            i = eXosip_call_build_answer (je->tid, 200, &answer);
            if (i != 0)
            {
                printf ("This request msg is invalid!Cann't response!\n");
                eXosip_call_send_answer (je->tid, 400, NULL);
            }
            else
            {
                eXosip_call_send_answer (je->tid, 200, answer);
                cout << "\n\t--> bye send 200 over!\n";
            }
            break;

case EXOSIP_CALL_MESSAGE_NEW:

cout << "\n\t*** EXOSIP_CALL_MESSAGE_NEW\n" << endl;
            if (MSG_IS_INFO(je->request) ) // 如果传输的是 INFO 方法
            {
                eXosip_lock ();
                i = eXosip_call_build_answer (je->tid, 200, &answer);
                if (i == 0)
                {
                    eXosip_call_send_answer (je->tid, 200, answer);
                }

eXosip_unlock ();

{
                    osip_body_t *body;
                    osip_message_get_body (je->request, 0, &body);
                    cout << "the body is " << body->body << endl;
                }
            }
            break;

default:
            cout << "\n\t--> Could not parse the msg!\n" << endl;
        }
    }

return 0;
}

一个基于 osip 库的 UAC 和 UAS 的代码整理(转)相关推荐

  1. 开发一个基于ZXing库以及安卓Studio的二维码扫描小程序(二)

    开发一个基于ZXing库以及安卓Studio的二维码扫描小程序(二) 下面我们做一个ZXing扫描二维码的例子,是通过安卓库的方式引用ZXing应用代码. 开发步骤 建立一个安卓工程(Project) ...

  2. osip和mysql_Windows下编译eXosip、osip,以及UAC和UAS的例子

    今天开始了SIP开源库的学习,我选择了osip和eXosip,但是这两个库的编译使用有些麻烦,源码下来之后编译会出现很多问题,网上也没有找到完整的编译介绍,只能一步一步的找办法解决,最后终于编译成功! ...

  3. Windows下编译eXosip、osip,以及UAC和UAS的例子

    今天开始了SIP开源库的学习,我选择了osip和eXosip,但是这两个库的编译使用有些麻烦,源码下来之后编译会出现很多问题,网上也没有找到完整的编译介绍,只能一步一步的找办法解决,最后终于编译成功! ...

  4. 基于libusb库、uac协议,获取Audio声音数据

    android_usbaudio 基于libusb,实现无驱动获取USBAudio 期望实现的功能: 通过libusb获取USBAudio数据,无需SELinux声卡权限 部分摄像头无法获取音频问题解 ...

  5. html页面发送微信朋友圈,【源码分享】分享一个基于官方share.html简化的分享js代码(微信,朋友圈,QQ)...

    基于官方demo中的share.hml中的分享内容或者链接到微博,微信,QQ,自己简化了一个通用的分享方法,进一步减少分享功能的开发量. 代码附件中,由于不能上传html文件,所以把文件扩展名改成了d ...

  6. 【Python学习系列十八】基于scikit-learn库逻辑回归训练模型(delta比赛代码3)

    为了得到一致假设而使假设变得过度严格称为过拟合.避免过拟合是分类器设计中的一个核心任务.通常采用增大数据量和测试样本集的方法对分类器性能进行评价.由于比赛中样本量是一致,目标测试集也是既定,所以我的思 ...

  7. 【Python学习系列十七】基于scikit-learn库逻辑回归训练模型(delta比赛代码2)

    机器学习任务流程:学习任务定义->数学建模->训练样本采样->特征分析和抽取->算法设计和代码->模型训练和优化(性能评估和度量)->泛化能力评估(重采样和重建模) ...

  8. 【Python学习系列十六】基于scikit-learn库逻辑回归训练模型(delta比赛代码)

    delta比赛的场景:给定数据样本,设计模型训练预测二分类结果,并通过f1-score评估结果.比赛中对特征抽取.样本扰动.过拟合.强相关特征.归一化等概念有实际的理解和应用. 这里给出的代码是基于逻 ...

  9. 磁盘使用分析器linux,ncdu:基于ncurses库的磁盘使用分析器

    Ncdu (全名为NCurses Disk Usage) 是一个基于Ncurses库的du命令的界面.它通过大家熟知的du命令,为用户提供一个快速且容易被使用的界面.它可以显示磁盘使用的百分比,且允许 ...

最新文章

  1. 移动端调取摄像头上面如何给出框_飞桨实战笔记:自编写模型如何在服务器和移动端部署...
  2. NFV节省企业宽带成本—Vecloud微云
  3. 密码技术--证书及go语言生成自签证书
  4. Chrome调试模拟iPhone6时body显示980*1742
  5. JS动画 | 用TweenMax实现收集水滴效果
  6. [设计模式] ------ 工厂方法模式
  7. plsql 快捷键设置
  8. excel操作练习_你见过最好的Excel教程有哪些?
  9. RxJS的另外四种实现方式(一)——代码最小的库
  10. 数据挖掘:模型选择——监督学习(回归)
  11. 10.25模拟 列车调度
  12. 扫雷代码(包含扩展函数)
  13. C#实战008:Excel操作-创建新的Excel工作表
  14. 慕尼黑大学计算机硕士专业,德国可以申请的英语授课的计算机硕士有哪些学校?...
  15. 个税计算--Java
  16. 拓嘉启远电商:拼多多新店多久才有流量
  17. HTB打靶(Active Directory 101 Reel)
  18. 解决 redis 存入中文,取出来是乱码wenti
  19. jquery+bootstrap 创建日历表格
  20. 【前端系列教程之JavaScript】01_JavaScript概述和引入方式

热门文章

  1. 大数据 ---(4)大数据驱动的金融业务创新(用户画像-数据架构-标签建模)
  2. Android性能优化(1) 内存泄露 amp; 解决方案
  3. python 太灵活_Python中的灵活参数
  4. Windows 10安装TensorFlow-gpu1.4 及CUDA8.0,cuDNN6.0,搞定了,包含安装方法和下载路径
  5. zynq中纯Programmable Loigc编程
  6. Ubuntu版本进化史 4.10-7.04
  7. html u方法,success 方法
  8. wgs84坐标系转换工具_ArcGIS中不同坐标系之间的转换
  9. 苹果将削减iPhone SE及AirPods产量 iPhone 13也要求减产
  10. 荣耀Magic4核心配置曝光:最强驯龙高手 性能远超iPhone 13 Pro