zoukankan      html  css  js  c++  java
  • onvif开发之设备发现功能的实现--转

    忙了一个多月,onvif总算告一段落了。这几个星期忙着其他的项目,也没有好好整理一下onvif的东西。接下来得好好整理一下自己的项目思路和项目经验,同时将自己的一些心得写出来,希望对人有所帮助。

            相信大多数兄弟和我一样,onvif开发,最开始做的就是发现功能。这两天登录onvif的官网看才发现,onvif版本在八月份有更新,已经更新到V2.4了,于是下载最新的版本来进行。代码的生成可以详见我的前一篇文章。V2.4版本新增了一个wsdl文件,现在用于生成源码的文件一共有18个。为了保证全功能,最好一次性生成包含所有功能的源码。然后根据最新生成的源码来实现onvif客户端和服务端的发现功能。

    1. 创建onvif_test目录。以下这些源码由最新的gsoap(2.8.16)和最新的onvif的wsdl文件(2.4)生成。(截止2013.09.16)

    onvif.h

    soapClientLib.c

    soapServerLib.c

    soapC.c

    soapClient.c

    soapH.h

    soapServer.c

    soapStub.h

    2.以下文件来自gsoap_2.8.16gsoap-2.8gsoap

    stdsoap2.c

    stdsoap2.h

    3.以下文件来自gsoap_2.8.16gsoap-2.8gsoapcustom

    duration.c

    4.生成的soapClientLib.c和soapServerLib.c无实际作用,可直接删除。

    5.增加自定义文件:

    onvif_server.c   onvif服务端实现代码

    onvif_client.c   onvif客户端实现代码

    onvif_server_interface.c   onvif服务端接口实现

    onvif_function.c  onvif实现函数,公用

    onvif_function.h  onvif实现函数,公用

    onvif_server.c为服务端的实现代码,主要是定义main函数,服务端主要是监听,并处理和应答消息。

    Main函数定义如下:

    [cpp] view plain copy
     
    1. int main(int argc,char ** argv)  
    2. {  
    3.     printf("[%s][%d][%s][%s] start  ", __FILE__, __LINE__, __TIME__, __func__);  
    4.   
    5.     int count = 0;  
    6.     struct soap ServerSoap;  
    7.     struct ip_mreq mcast;  
    8.       
    9.     soap_init1(&ServerSoap, SOAP_IO_UDP | SOAP_XML_IGNORENS);  
    10.     soap_set_namespaces(&ServerSoap,  namespaces);  
    11.       
    12.     printf("[%s][%d][%s][%s] ServerSoap.version = %d  ", __FILE__, __LINE__, __TIME__, __func__, ServerSoap.version);  
    13.       
    14.     if(!soap_valid_socket(soap_bind(&ServerSoap, NULL, ONVIF_LISTEN_PORT, 10)))  
    15.     {  
    16.         soap_print_fault(&ServerSoap, stderr);  
    17.         exit(1);  
    18.     }  
    19.       
    20.     mcast.imr_multiaddr.s_addr = inet_addr("239.255.255.250");  
    21.     mcast.imr_interface.s_addr = htonl(INADDR_ANY);  
    22.   
    23.     if(setsockopt(ServerSoap.master, IPPROTO_IP, IP_ADD_MEMBERSHIP, (char*)&mcast, sizeof(mcast)) < 0)  
    24.     {  
    25.             printf("setsockopt error! error code = %d,err string = %s ",errno,strerror(errno));  
    26.         return 0;  
    27.     }  
    28.   
    29.     for(;;)  
    30.     {  
    31.         if(soap_serve(&ServerSoap))  
    32.         {  
    33.             soap_print_fault(&ServerSoap, stderr);  
    34.         }  
    35.   
    36.         soap_destroy(&ServerSoap);  
    37.         soap_end(&ServerSoap);  
    38.   
    39.         //客户端的IP地址  
    40.         printf("RECEIVE count %d, connection from IP = %lu.%lu.%lu.%lu socket = %d  ", count, ((ServerSoap.ip)>>24)&0xFF, ((ServerSoap.ip)>>16)&0xFF, ((ServerSoap.ip)>>8)&0xFF,(ServerSoap.ip)&0xFF, (ServerSoap.socket));  
    41.         count++;  
    42.     }  
    43.   
    44.     //分离运行时的环境  
    45.     soap_done(&ServerSoap);  
    46.   
    47.     return 0;  
    48. }  

    onvif_server_interface.c   此文件用来定义所有服务端需要填充的接口。这里我们填充__wsdd__Probe接口即可,其他赞不支持的接口可以用宏统一处理。

    __wsdd__Probe的填充如下:

    [cpp] view plain copy
     
    1. SOAP_FMAC5 int SOAP_FMAC6  __wsdd__Probe(struct soap* soap, struct wsdd__ProbeType *wsdd__Probe)  
    2. {  
    3.   
    4.     #define MACH_ADDR_LENGTH 6  
    5.     #define INFO_LENGTH 512  
    6.     #define LARGE_INFO_LENGTH 1024  
    7.     #define SMALL_INFO_LENGTH 512  
    8.       
    9.     printf("[%d] __wsdd__Probe start ! ", __LINE__);  
    10.       
    11.     unsigned char macaddr[6] = {0};  
    12.     char _IPAddr[INFO_LENGTH] = {0};  
    13.     char _HwId[1024] = {0};  
    14.       
    15.     wsdd__ProbeMatchesType ProbeMatches;  
    16.     ProbeMatches.ProbeMatch = (struct wsdd__ProbeMatchType *)soap_malloc(soap, sizeof(struct wsdd__ProbeMatchType));  
    17.     ProbeMatches.ProbeMatch->XAddrs = (char *)soap_malloc(soap, sizeof(char) * INFO_LENGTH);  
    18.     ProbeMatches.ProbeMatch->Types = (char *)soap_malloc(soap, sizeof(char) * INFO_LENGTH);  
    19.     ProbeMatches.ProbeMatch->Scopes = (struct wsdd__ScopesType*)soap_malloc(soap,sizeof(struct wsdd__ScopesType));  
    20.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ReferenceProperties = (struct wsa__ReferencePropertiesType*)soap_malloc(soap,sizeof(struct wsa__ReferencePropertiesType));  
    21.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ReferenceParameters = (struct wsa__ReferenceParametersType*)soap_malloc(soap,sizeof(struct wsa__ReferenceParametersType));  
    22.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ServiceName = (struct wsa__ServiceNameType*)soap_malloc(soap,sizeof(struct wsa__ServiceNameType));  
    23.     ProbeMatches.ProbeMatch->wsa__EndpointReference.PortType = (char **)soap_malloc(soap, sizeof(char *) * SMALL_INFO_LENGTH);  
    24.     ProbeMatches.ProbeMatch->wsa__EndpointReference.__any = (char **)soap_malloc(soap, sizeof(char*) * SMALL_INFO_LENGTH);  
    25.     ProbeMatches.ProbeMatch->wsa__EndpointReference.__anyAttribute = (char *)soap_malloc(soap, sizeof(char) * SMALL_INFO_LENGTH);  
    26.     ProbeMatches.ProbeMatch->wsa__EndpointReference.Address = (char *)soap_malloc(soap, sizeof(char) * INFO_LENGTH);  
    27.   
    28.     netGetMac("eth4", macaddr); //eth0  根据实际情况填充  
    29.     macaddr[0]=0x01;macaddr[1]=0x01;macaddr[2]=0x01;macaddr[3]=0x01;macaddr[4]=0x01;macaddr[5]=0x01;  
    30.     sprintf(_HwId,"urn:uuid:2419d68a-2dd2-21b2-a205-%02X%02X%02X%02X%02X%02X",macaddr[0], macaddr[1], macaddr[2], macaddr[3], macaddr[4], macaddr[5]);  
    31.   
    32.   
    33.     unsigned int localIp = 0;  
    34.     netGetIp("eth4", &localIp); //eth0 根据实际情况填充  
    35.     sprintf(_IPAddr, "http://%s/onvif/device_service", inet_ntoa(*((struct in_addr *)&localIp)));  
    36.     printf("[%d] _IPAddr ==== %s ", __LINE__, _IPAddr);  
    37.       
    38.     ProbeMatches.__sizeProbeMatch = 1;  
    39.     ProbeMatches.ProbeMatch->Scopes->__item =(char *)soap_malloc(soap, 1024);  
    40.     memset(ProbeMatches.ProbeMatch->Scopes->__item,0,sizeof(ProbeMatches.ProbeMatch->Scopes->__item));    
    41.   
    42.     //Scopes MUST BE  
    43.     strcat(ProbeMatches.ProbeMatch->Scopes->__item, "onvif://www.onvif.org/type/NetworkVideoTransmitter");  
    44.   
    45.     ProbeMatches.ProbeMatch->Scopes->MatchBy = NULL;  
    46.     strcpy(ProbeMatches.ProbeMatch->XAddrs, _IPAddr);  
    47.     strcpy(ProbeMatches.ProbeMatch->Types, wsdd__Probe->Types);  
    48.     printf("wsdd__Probe->Types=%s ",wsdd__Probe->Types);  
    49.     ProbeMatches.ProbeMatch->MetadataVersion = 1;  
    50.       
    51.     //ws-discovery规定 为可选项  
    52.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ReferenceProperties->__size = 0;  
    53.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ReferenceProperties->__any = NULL;  
    54.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ReferenceParameters->__size = 0;  
    55.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ReferenceParameters->__any = NULL;  
    56.       
    57.     ProbeMatches.ProbeMatch->wsa__EndpointReference.PortType[0] = (char *)soap_malloc(soap, sizeof(char) * SMALL_INFO_LENGTH);  
    58.     //ws-discovery规定 为可选项  
    59.     strcpy(ProbeMatches.ProbeMatch->wsa__EndpointReference.PortType[0], "ttl");  
    60.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ServiceName->__item = NULL;  
    61.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ServiceName->PortName = NULL;  
    62.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ServiceName->__anyAttribute = NULL;  
    63.     ProbeMatches.ProbeMatch->wsa__EndpointReference.__any[0] = (char *)soap_malloc(soap, sizeof(char) * SMALL_INFO_LENGTH);  
    64.     strcpy(ProbeMatches.ProbeMatch->wsa__EndpointReference.__any[0], "Any");  
    65.     strcpy(ProbeMatches.ProbeMatch->wsa__EndpointReference.__anyAttribute, "Attribute");  
    66.     ProbeMatches.ProbeMatch->wsa__EndpointReference.__size = 0;  
    67.     strcpy(ProbeMatches.ProbeMatch->wsa__EndpointReference.Address, _HwId);  
    68.       
    69.     soap->header->wsa__To = "http://schemas.xmlsoap.org/ws/2004/08/addressing/role/anonymous";  
    70.     soap->header->wsa__Action = "http://schemas.xmlsoap.org/ws/2005/04/discovery/ProbeMatches";  
    71.     soap->header->wsa__RelatesTo = (struct wsa__Relationship*)soap_malloc(soap, sizeof(struct wsa__Relationship));  
    72.     soap->header->wsa__RelatesTo->__item = soap->header->wsa__MessageID;  
    73.     soap->header->wsa__RelatesTo->RelationshipType = NULL;  
    74.     soap->header->wsa__RelatesTo->__anyAttribute = NULL;  
    75.   
    76.     soap->header->wsa__MessageID =(char *)soap_malloc(soap, sizeof(char) * INFO_LENGTH);  
    77.     strcpy(soap->header->wsa__MessageID,_HwId+4);  
    78.   
    79.         if (SOAP_OK == soap_send___wsdd__ProbeMatches(soap, "http://", NULL, &ProbeMatches))  
    80.         {  
    81.         printf("send ProbeMatches success ! ");  
    82.         return SOAP_OK;  
    83.     }  
    84.   
    85.     printf("[%d] soap error: %d, %s, %s ", __LINE__, soap->error, *soap_faultcode(soap), *soap_faultstring(soap));    
    86.       
    87.     return soap->error;;  
    88.   
    89. }  

    onvif_client.c   onvif客户端实现代码,主要是定义客户端的main函数:

    Main定义如下:

    [cpp] view plain copy
     
    1. int main()    
    2. {    
    3.     printf("[%s][%d][%s][%s] start  ", __FILE__, __LINE__, __TIME__, __func__);  
    4.   
    5.     int result = 0;    
    6.     wsdd__ProbeType req;  
    7.     struct __wsdd__ProbeMatches resp;  
    8.     wsdd__ScopesType sScope;  
    9.     struct SOAP_ENV__Header header;    
    10.       
    11.     struct soap *soap;    
    12.     soap = soap_new();    
    13.     if(NULL == soap )    
    14.     {    
    15.         printf("sopa new error ");    
    16.         return -1;    
    17.     }    
    18.   
    19.     soap->recv_timeout = 10;    
    20.     soap_set_namespaces(soap, namespaces);    
    21.     soap_default_SOAP_ENV__Header(soap, &header);    
    22.   
    23.     uuid_t uuid;  
    24.     char guid_string[100];  
    25.     uuid_generate(uuid);  
    26.     uuid_unparse(uuid, guid_string);  
    27.   
    28.     header.wsa__MessageID = guid_string;   
    29.     header.wsa__To = "urn:schemas-xmlsoap-org:ws:2005:04:discovery";    
    30.     header.wsa__Action = "http://schemas.xmlsoap.org/ws/2005/04/discovery/Probe";    
    31.     soap->header = &header;    
    32.       
    33.     soap_default_wsdd__ScopesType(soap, &sScope);    
    34.     sScope.__item = "";    
    35.     soap_default_wsdd__ProbeType(soap, &req);    
    36.     req.Scopes = &sScope;    
    37.     req.Types = ""; //"dn:NetworkVideoTransmitter";    
    38.   
    39.     int i = 0;        
    40.        result = soap_send___wsdd__Probe(soap, MULTICAST_ADDRESS, NULL, &req);    
    41.        while(result == SOAP_OK)    
    42.        {    
    43.         result = soap_recv___wsdd__ProbeMatches(soap, &resp);    
    44.         if(result == SOAP_OK)    
    45.         {    
    46.             if(soap->error)    
    47.             {    
    48.                 printf("soap error 1: %d, %s, %s ", soap->error, *soap_faultcode(soap), *soap_faultstring(soap));    
    49.                 result = soap->error;    
    50.             }    
    51.             else    
    52.             {    
    53.                 printf("guog ********************************************* ");    
    54.                 if(soap->header->wsa__MessageID)    
    55.                 {    
    56.                     printf("MessageID   : %s ", soap->header->wsa__MessageID);    
    57.                 }    
    58.                 if(soap->header->wsa__RelatesTo && soap->header->wsa__RelatesTo->__item)    
    59.                 {    
    60.                     printf("RelatesTo   : %s ", soap->header->wsa__RelatesTo->__item);    
    61.                 }    
    62.                 if(soap->header->wsa__To)    
    63.                 {    
    64.                     printf("To          : %s ", soap->header->wsa__To);    
    65.                 }    
    66.                 if(soap->header->wsa__Action)    
    67.                 {    
    68.                     printf("Action      : %s ", soap->header->wsa__Action);    
    69.                 }    
    70.   
    71.                 for(i = 0; i < resp.wsdd__ProbeMatches->__sizeProbeMatch; i++)    
    72.                 {    
    73.                     printf("__sizeProbeMatch        : %d ", resp.wsdd__ProbeMatches->__sizeProbeMatch);    
    74.                     printf("wsa__EndpointReference       : %p ", resp.wsdd__ProbeMatches->ProbeMatch->wsa__EndpointReference);    
    75.                     printf("Target EP Address       : %s ", resp.wsdd__ProbeMatches->ProbeMatch->wsa__EndpointReference.Address);    
    76.                     printf("Target Type             : %s ", resp.wsdd__ProbeMatches->ProbeMatch->Types);    
    77.                     printf("Target Service Address  : %s ", resp.wsdd__ProbeMatches->ProbeMatch->XAddrs);    
    78.                     printf("Target Metadata Version : %d ", resp.wsdd__ProbeMatches->ProbeMatch->MetadataVersion);    
    79.                     if(resp.wsdd__ProbeMatches->ProbeMatch->Scopes)    
    80.                     {    
    81.                         printf("Target Scopes Address   : %s ", resp.wsdd__ProbeMatches->ProbeMatch->Scopes->__item);    
    82.                     }    
    83.                 }  
    84.             }    
    85.         }    
    86.         else if (soap->error)    
    87.         {    
    88.             printf("[%d] soap error 2: %d, %s, %s ", __LINE__, soap->error, *soap_faultcode(soap), *soap_faultstring(soap));    
    89.             result = soap->error;    
    90.         }    
    91.        }    
    92.   
    93.     soap_destroy(soap);   
    94.     soap_end(soap);   
    95.     soap_free(soap);  
    96.   
    97.     printf("[%d] guog discover over ! ", __LINE__);  
    98.       
    99.     return result;    
    100. }    

    6.将wsdd.nsmap改为nsmap.h,并删除其余的*.nsmap(都一样)

    7.编写makefile文件。注意在makefile中打开开关DEBUG,以便跟踪日志。

    8.tcpdump为gcc环境的抓包工具,调试的时候用。

    make编译通过,运行,客户端发现功能ok;

        但是服务端的发现功能却不行;别急,这是由于SOAP的版本问题;soap的版本是根据命名空间来自动确定的;在soap结构体的version字段表示soap版本;

    以下命名空间表示SOAP1.1版本:

    {"SOAP-ENV", "http://schemas.xmlsoap.org/soap/envelope/", "http://www.w3.org/*/soap-envelope", NULL},

    {"SOAP-ENC", "http://schemas.xmlsoap.org/soap/encoding/", "http://www.w3.org/*/soap-encoding", NULL}, //1.1

    以下命名空间表示SOAP1.2版本:

    {"SOAP-ENV", "http://www.w3.org/2003/05/soap-envelope", "http://schemas.xmlsoap.org/soap/envelope/", NULL},

    {"SOAP-ENC", "http://www.w3.org/2003/05/soap-encoding", "http://schemas.xmlsoap.org/soap/encoding/", NULL},  //1.2

        注意确定自己当前的命名空间。我们用SOAP1.2版本才能被测试工具发现(ONVIF Device Test Tool version 13.06)。

    不清楚可以查看soap_set_namespaces接口和soap_set_local_namespaces接口;

    于是将nsmap.h中的命名空间前两行改为:

    {"SOAP-ENV", "http://www.w3.org/2003/05/soap-envelope", "http://schemas.xmlsoap.org/soap/envelope/", NULL},

    {"SOAP-ENC", "http://www.w3.org/2003/05/soap-encoding", "http://schemas.xmlsoap.org/soap/encoding/", NULL},  //1.2

    再编译运行,发现功能ok。

     

    最后,将所用的工具和源码提供给大家,大家根据需要下载。


    onvif测试工具V13.06:http://download.csdn.net/detail/u011597695/6288593


    gsoap 2.8.16版本: http://download.csdn.net/detail/u011597695/6288615


    onvif v2.4版本的wsdl文件:http://download.csdn.net/detail/u011597695/6288627


    onvif v2.4版本的wsdl文件(适于离线生成源码):http://download.csdn.net/detail/u011597695/6288647


    onvif源代码V2.4:http://download.csdn.net/detail/u011597695/6288663

    from:http://blog.csdn.net/love_xjhu/article/details/11821037

  • 相关阅读:
    抓包的原理
    在ASP.NET MVC中使用JQ插件datatable
    如何禁用Visual Studio 2013的Browser Link功能
    SVN中tag branch trunk用法详解
    ASP.NET MVC和jQuery DataTable整合
    随便看的
    SQL查询今天、昨天、7天内、30天
    在DataTable数据类型最后增加一列,列名为“Column”,内容都为“AAA”
    validform表单验证插件最终版
    context.Session[“xxx”]详解
  • 原文地址:https://www.cnblogs.com/lidabo/p/6552798.html
Copyright © 2011-2022 走看看