动环监控系统中B接口的实现

动环监控系统简述

1.术语介绍

1.1 省集中监控中心-Province Supervision Center(PSC)

面向多FSU管理的高级监控层次,即省集中监控中心,通过开放的数据协议,连接监控范围内的FSU。

1.2 现场监控单元-Field supervision unit(FSU)

监控系统的最小管理子系统,由若干监控模块和其它辅助设备组成,面向直接的设备数据采集、处理的监控层次,可以包含采样、数据处理、数据中继等功能,

监控范围一般为一个独立的通信局(站)或大型局(站)内相对独立的电源、空调设备及环境。

1.3 监控对象 Supervision Object(SO)

被监控的各种电源、空调设备及机房环境。

1.4 B接口

为省集中监控中心(PSC)与现场监控单元(FSU)之间的接口。(即FSU的北向接口)

2. 接口网络结构

FSU与PSC之间通过WebService和FTP方式互联,二者同时形成完整的B接口协议标准。


 

B接口在嵌入式arm监控主机上的实现

环境

宿主机平台:Ubuntu 16.04.6

目标机平台:iMX6UL

交叉编译:gcc-linaro-4.9-2014.11     arm-linux-gnueabihf-gcc        https://www.linaro.org/downloads/

SOAP/XML 关于C/C++ 语言的实现    gsoap_2.8.83      https://sourceforge.net/projects/gsoap2/files/

XML数据的生成和解析       tinyxml2

XML数据的生成和解析   libxml2   ftp://xmlsoft.org/libxml2/

结合主要开发为C环境且尽量占用较少资源,推荐使用minixml

XML数据的生成和解析      minixml   https://www.msweet.org/mxml/


 

1. gsaop 生成 B接口报文协议 C代码框架

gSOAP编译工具提供了一个SOAP/XML 关于C/C++ 语言的实现,从而让C/C++语言开发web服务或客户端程序的工作变得轻松了很多。

绝大多数的C++web服务工具包提供一组API函数类库来处理特定的SOAP数据结构,这样就使得用户必须改变程序结构来适应相关的类库。

与之相反,gSOAP利用编译器技术提供了一组透明化的SOAP API,并将与开发无关的SOAP实现细节相关的内容对用户隐藏起来。

1.1 gsaop在linux下的安装

gsoap安装编译依赖

sudo apt-get install build-essential libgtk2.0-dev libglib2.0-dev checkinstall m4 flex bison automake autoconf openssl libssl-dev

为了成功编译gSOAP,您需要安装GTK+的开发文件和GLib库(libraries)。

安装Checkinstall以便管理您系统中直接由源代码编译安装的软件。

安装YACC,YACC是Unix/Linux上一个用来生成编译器的编译器(编译器代码生成器),sudo apt-get install flex bison

安装OpenSSL,web通信的加密(https)及鉴权  sudo apt-get install openssl libssl-dev

 

安装编译步骤: 

解压

unzip gsoap_2.8.83.zip
mkdir gsoap_install
cd gsoap-2.8/

配置安装路径、编译、安装

./configure --prefix=/home/wangh/workspace/wh_tools/gsoap_install
sudo make
sudo make install

 

使用gsoap生成 fsu 代码框架

新建gsoap_fsu文件夹,从gsoap_install文件夹中拷贝以下文件

bin/soapcpp2        bin/wsdl2h

gsoap2.8/gsoap/typemap.dat

custom和import文件夹

1.使用wsdl2h工具,根据WSDL产生头文件,执行以下命令

./wsdl2h -P -x -c -s -t ./typemap.dat -o fsu.h FSUService.wsdl

其中-c为产生纯c代码,默认生成 c++代码;

-x    不产生xml文件(可用可不用,xml有一定帮助,但是太多);

-s为不使用STL库,-t为typemap.dat的标识。

详情可通过wsdl2h.exe -help查看帮助。

这里的WSDL文件,可以在wsdl2h命令中在线下载,也可以先下载到本地,然后引用本地WSDL文件,我这里是采用本地文件方式。

 

 2. 使用soapcpp2工具,根据头文件 fsu.h 产生框架代码,执行以下命令

./soapcpp2 -2 -L -c -x -I import:custom fsu.h

-2    生成 SOAP 1.2

-L    不生成客户端、服务器库文件

-c    为产生纯c代码,默认生成 c++代码

-I path  引用文件路径   

 

3. 提取有效核心代码用于应用编程,应用工程使用文件如下:

 

 

2. B接口报文协议分析与实现


 

2.1 SOAP函数接口分析

根据 FSUService.wsdl 生成 soap API 如下,在 soapStub.h 的最后定义

我们不用去关心 SOAP 接口的实现细节,直接调用对应 API 即可实现客户端与服务器的 xml 数据收发。

客户端接口过程:

soap_call_ns1__invoke() 在客户端调用,实现发送和接收(它调用send和recv);

服务器接口过程:

soap_serve() 服务器调用,部署 SOAP 服务器,它会调用soap_serve_request(),当收到 ns1:invoke 的请求时,

服务器调用 soap_serve_ns1__invoke(),在调用应用层用户接口 ns1__invoke(),来获取接收数据和填充应答数据。

2.2 SOAP通信简单示例

 我的工程目录如下图所示:

客户端程序:

/**
**********************************************************************************
* @file        main.c
* @brief       soap客户端测试程序
* @details  基于东环监控 B 接口的soap客户端程序
* @author      wanghuan  any question please send mail to 371463817@qq.com
* @date        2019-06-17
* @version     V1.0
* @copyright    Copyright (c) 2019-2022  江苏亨通光网科技有限公司
**********************************************************************************
*/
#include "soapH.h"
#include "stdsoap2.h"
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include "FSUServiceSoapBinding.nsmap"

int main(int argc, char **argv)
{
//    struct soap fsuSoap;
//    soap_init(&fsuSoap);

    struct soap *fsuSoap = soap_new();
    fsuSoap->send_timeout = fsuSoap->recv_timeout = 5;    //soap发送、接收超时
    fsuSoap->transfer_timeout = 30;        //soap消息传输超时
    soap_set_namespaces(fsuSoap, namespaces);
//    soap_set_mode(fsuSoap, SOAP_C_NOIOB);

    struct ns1__invokeResponse soap_tmp_ns1__invokeResponse;
    char * soap_tmp_SOAP_ENC__string;
    soap_default_ns1__invokeResponse(fsuSoap, &soap_tmp_ns1__invokeResponse);
    soap_tmp_SOAP_ENC__string = NULL;
    soap_tmp_ns1__invokeResponse._invokeReturn = &soap_tmp_SOAP_ENC__string;

    // 手动组成LOGIN的xml数据字符串
    char xmlData[500];
    memset(xmlData, 0x00, sizeof(xmlData));
    strcat(xmlData, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n");
    strcat(xmlData, "<Request>\n");
    strcat(xmlData, "<PK_Type>\n");
    strcat(xmlData, "<Name>LOGIN</Name>\n");
    strcat(xmlData, "</PK_Type>\n");
    strcat(xmlData, "<Info>\n");
    strcat(xmlData, "<UserName>123</UserName>\n");
    strcat(xmlData, "<PassWord>123</PassWord>\n");
    strcat(xmlData, "<FsuId>10004</FsuId>\n");
    strcat(xmlData, "<FsuIP>192.168.1.124</FsuIP>\n");
    strcat(xmlData, "<FsuPort>10000</FsuPort>\n");
    strcat(xmlData, "</Info>\n");
    strcat(xmlData, "</Request>\n");

    printf("soap_tmp_ns1__invoke._xmlData:\n");
    printf("%s", xmlData);
    
    // saop 客户端程序
//    char * server_addr = "http://10.10.62.83:8080/HT_SC/services/SCService";
    char * server_addr = "http://192.168.1.123:8080";

    int iRet = soap_call_ns1__invoke(fsuSoap, server_addr, NULL, xmlData, soap_tmp_ns1__invokeResponse._invokeReturn);
    if ( iRet == SOAP_ERR)
    {
        printf("Error while calling the soap_call_ns1__invoke");
    }
    else
    {
        printf("Calling the soap_call_ns1__invoke success。\n");
        printf("%s\n", *soap_tmp_ns1__invokeResponse._invokeReturn);
    }

    return 0;
}

 

服务端程序:

/**
**********************************************************************************
* @file        main.c
* @brief       soap服务端测试程序
* @details  基于东环监控 B 接口的soap服务器程序
* @author      wanghuan  any question please send mail to 371463817@qq.com
* @date        2019-06-17
* @version     V1.0
* @copyright    Copyright (c) 2019-2022  江苏亨通光网科技有限公司
**********************************************************************************
*/
#include "soapH.h"
#include "stdsoap2.h"
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include "FSUServiceSoapBinding.nsmap"

/**
* 启动soap服务器
*/
int main(int argc, char **argv)
{
    SOAP_SOCKET iSocket_master, iSocket_slave;
    struct soap *fsuSoap = soap_new();    //新建soap

    fsuSoap->send_timeout = fsuSoap->recv_timeout = 5;    //soap发送、接收超时
    fsuSoap->transfer_timeout = 30;        //soap消息传输超时
//    soap_set_mode(fsuSoap, SOAP_C_NOIOB);

    // saop 服务器程序 8080为端口号,最后一个参数不重要。
    iSocket_master = soap_bind(fsuSoap, NULL, 8080, 100);     //绑定到相应的IP地址和端口()NULL指本机,然后监听
    if (iSocket_master< 0)                                   //绑定出错
    {
        soap_print_fault(fsuSoap, stderr);
        exit(-1);
    }
    printf("SoapBind success,the master socket number is:%d\n",iSocket_master); //绑定成功返回监听套接字

    while(1)
    {
        iSocket_slave = soap_accept(fsuSoap);    //收到套接字连接
        if(iSocket_slave < 0)
        {
            soap_print_fault(fsuSoap, stderr);
            exit(-1);
        }
        //客户端的IP地址
        fprintf(stderr,"Accepted connection fromIP= %d.%d.%d.%d socket = %d \n",    \
                ((fsuSoap->ip)>>24)&&0xFF,((fsuSoap->ip)>>16)&0xFF,((fsuSoap->ip)>>8)&0xFF,(fsuSoap->ip)&0xFF,(fsuSoap->socket));
        printf("Socket connect success,the slave socket number is:%d\n",iSocket_slave);
        soap_serve(fsuSoap);
        printf("soap_serve end...");
        soap_end(fsuSoap);    //服务器出错才到这一步
    }
    soap_done(fsuSoap);
    free(fsuSoap);

    return 0;
}

/**
* soap服务器用户数据处理函数
* @param[in] *_xmlData            接收客户端的xml格式字符串
* @param[out] **_invokeReturn    服务器要回复大数据
* @return     SOAP_OK or error code
*/
SOAP_FMAC5 int SOAP_FMAC6 ns1__invoke(struct soap* soap, char *_xmlData, char **_invokeReturn)
{
    printf("ns1__invoke _xmlData\n");
    printf("%s\n", _xmlData);

    char xmlData[500];
    memset(xmlData, 0, sizeof(xmlData));
    strcat(xmlData, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n");
    strcat(xmlData, "<Response>\n");
    strcat(xmlData, "<PK_Type>\n");
    strcat(xmlData, "<Name>LOGIN_ACK</Name>\n");
    strcat(xmlData, "</PK_Type>\n");
    strcat(xmlData, "<Info>\n");
    strcat(xmlData, "<RightLevel/>\n");
    strcat(xmlData, "</Info>\n");
    strcat(xmlData, "</Response>");

    *_invokeReturn = xmlData;
//    *_invokeReturn = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<Response>\n<PK_Type>\n<Name>LOGIN_ACK</Name>\n</PK_Type>\n<Info>\n<RightLevel/>\n</Info>\n</Response>";

    printf("_invokeReturn _xmlData\n");
    printf("%s\n", *_invokeReturn);
    return SOAP_OK;
}

 

程序makefile:

顶层Makefile与Makefile.inc

DIRS := 
DIRS += sc_client
DIRS += sc_server
#DIRS += xml_test


# Dummy targets for building and clobbering everything in all subdirectories

all:     
    @ for dir in ${DIRS}; do (cd $${dir}; ${MAKE}) ; done

clean: 
    @ for dir in ${DIRS}; do (cd $${dir}; ${MAKE} clean) ; done

Makefile.inc

SHELL = /bin/bash

#CC           := gcc -m32
#CPP          := g++
#LD           := ld
#AR           := ar
#STRIP        := strip

CC           := arm-linux-gnueabihf-gcc
CPP          := arm-linux-gnueabihf-g++
LD           := arm-linux-gnueabihf-ld
AR           := arm-linux-gnueabihf-ar
STRIP        := arm-linux-gnueabihf-strip

# -DWITH_DOM -DWITH_OPENSSL两个宏是链接openssl时要用的;
# -DDEBUG宏用于开启SOAP协议收发日志,日志存于文件RECV.log、SENT.log、TEST.log之中
CFLAGS += -c -g -Wall
#CFLAGS += -DWITH_DOM -DWITH_OPENSSL
CFLAGS += -std=gnu99
CFLAGS += -DWITH_NO_C_LOCALE
#CFLAGS += -WITH_NONAMESPACES
CFLAGS += $(INCLUDE)

# openssl目录名
OPENSSL_DIR :=
#OPENSSL_DIR += ../openssl


# 源文件
SOURCES_FSU :=
SOURCES_FSU += ../soap_sc/soapC.c
SOURCES_FSU += ../soap_sc/soapClient.c
#SOURCES_FSU += ../soap_sc/soapServer.c
SOURCES_FSU += ../soap_sc/stdsoap2.c

SOURCES_COMM :=

# 目标文件
OBJECTS_FSU := $(patsubst %.c,$(TEMPDIR)%.o,$(filter %.c, $(SOURCES_FSU)))
OBJECTS_COMM := $(patsubst %.c,$(TEMPDIR)%.o,$(filter %.c, $(SOURCES_COMM)))

# 头文件路径
INCLUDE += -I../soap_sc/                            \
           -I../minixml/                             \
#           -I../libxml2/include                         \
#           -I../tinyxml2/                             \
#           -I../comm/                                \
#           -I$(OPENSSL_DIR)/include                  \

# 共享库链接OpenSSL
#LDLIBS += -L$(OPENSSL_DIR)/lib                      \
#          -lssl                                     \
#          -lcrypto                                  \

# 静态库链接OpenSSL
#LDLIBS += $(OPENSSL_DIR)/lib/libssl.a                \
#          $(OPENSSL_DIR)/lib/libcrypto.a             \
#          -ldl                                       \

# 静态库链接libxml2
#LDLIBS += ../libxml2/libxml2.a                          \
#          -lm

# 静态库链接minixml
LDLIBS += ../minixml/libmxml.a
#LDLIBS += -lmxml                  

# 链接库(其他)
LDLIBS += -lpthread

%.o: %.cpp
    @echo "  CPP     " $@;
    @$(CPP) $(CFLAGS) -c -o $@ $<

%.o: %.c
    @echo "  CC      " $@;
    @$(CC) $(CFLAGS) -c -o $@ $<

.PHONY: all clean

客户端client的Makefile

include ../Makefile.inc

PROGRAM = fsuc

SOURCES += main.c

OBJECTS := $(patsubst %.c,$(TEMPDIR)%.o,$(filter %.c, $(SOURCES)))

all: $(OBJECTS_FSU) $(OBJECTS_COMM) $(OBJECTS)
    $(CC) -o $(PROGRAM) $(OBJECTS_FSU) $(OBJECTS_COMM) $(OBJECTS) $(LDLIBS)

clean:
    rm -f $(OBJECTS_FSU)
    rm -f $(OBJECTS_COMM)
    rm -f $(OBJECTS)
    rm -f $(PROGRAM)

 

编译之后 先执行服务端程序,在执行客户端程序

服务端收到request

客户端发送request,并收到response

 接下来开始尝试使用 libxml2 库生成和解析 xml

 

2.3 libxml2安装移植(改用minixml)

新建安装文件夹,这样方便我们提取库文件使用

mkdir /home/wangh/workspace/wh_tools/libxml-install

编写安装及配置脚本 wh_configure.sh 如下:

#!/bin/sh

#sh文件需设置可执行权限 chmod +x wh_configure.sh
#--prefix=PATH,指定make install时目标文件存放路径
PREFIX=/home/wangh/workspace/wh_tools/libxml-install
#--host=target-platform  指定目标平台
HOST=arm-linux
#交叉编译绝对路径
CC_DIR=/home/wangh/Tools/gcc-linaro-4.9-2014.11-x86_64_arm-linux-gnueabihf/bin/arm-linux-gnueabihf-gcc
#编译配置
#sudo ./configure --prefix=$PREFIX --host=$HOST CC=$CC_DIR --with-python=/home/wangh/workspace/wh_tools/libxml2-2.9.9/python
#编译时由于提示没有python相关的头文件出错,又因为项目中不使用python相关的内容,所以没有讲python进库中,读者应该按照自己的需要要配置该选项。对于zlib同样的道理,--without-zlib不添加会编译出错
sudo ./configure --prefix=$PREFIX --host=$HOST --target=arm CC=$CC_DIR --without-zlib --without-python
sudo make
sudo make install

执行安装脚本

2.4 minixml安装移植

是一个小型的开源的XML解析器,采用 C 语言开发。该解析器最大的特点就是小型、无须依赖其他类库,在嵌入式系统中,minixml解析器很小巧(200k多点),很常用。

 

待续。。。。。

 

posted @ 2019-06-13 19:52  silencehuan  阅读(8482)  评论(1编辑  收藏  举报