Python实现海康威视SDK二次开发(开源库)

此贴能起到的作用

通过这个帖子,能了解到如何用Python调用海康SDK,实现业务逻辑需要结合哪些资料,这些接口的参数是怎么样的,如何翻译成Python,如何传参,参数中的一些变量,常量可以怎样查找。

源码传送门 github仓库readme中有社区群联系方式可供大家参与讨论,目前已经有数百人

海康威视SDK下载 https://www.hikvision.com/cn/download_61.html

SDK只有对linux和windows的支持,没有对mac的支持,所以mac开发比较累

基于SDK开发

  • 从官方给到的SDK中有.chm的文件,是一个接口文档,里面详细的介绍了该SDK的所有描述。
  • SDK中给到了基于Java,C#等demo,但是没有python。(这些demo可以帮我们理解一些比较晦涩难懂的逻辑,但是demo写的也没有很严格,部分变量类型定义不是完全按照文档来的,需要自己消化)
  • linux和windows的SDK中分别是.so和.dll,对于python我们需要ctypes库来完成二次开发
  • 官方开发文档 ,这里有详细硬件功能调用链,很详细,不过demo是c++的,另外demo中的一些变量或者常量不能查看引用,所以可以与第一项中提到的文档结合,如果两者描述不符,以当前SDK中的.chm文件优先。 https://open.hikvision.com/hardware/definitions/ {接口或实体}.html
    这个是在线接口详细文档。

    目前开源库以更新迭代,以下为原始代码,如果需要用到用到Python开发,可以直接使用开源库。

  • 基础SDK调用实现
  • from ctypes import *
    import os
    import logging
    import hkws.model.login as login
    import hkws.model.preview as preview
    from hkws.callback import hikFunc
    from hkws.callback import g_real_data_call_back
    class HKAdapter:
        so_list = []
        # 加载目录下所有so文件
        def add_lib(self, path, suffix):
            files = os.listdir(path)
            for file in files:
                if not os.path.isdir(path + file):
                    if file.endswith(suffix):
                        self.so_list.append(path + file)
                else:
                    self.add_lib(path + file + "/", suffix)
        # python 调用 sdk 指定方法
        def call_cpp(self, func_name, *args):
            for so_lib in self.so_list:
                    lib = cdll.LoadLibrary(so_lib)
                        value = eval("lib.%s" % func_name)(*args)
                        logging.info("调用的库:" + so_lib)
                        logging.info("执行成功,返回值:" + str(value))
                        return value
                    except:
                        continue
                except:
                    continue
                # logging.info("库文件载入失败:" + so_lib )
            logging.error("没有找到接口!")
            return False
    
  • 初始化SDK及释放SDK
  •     # 初始化海康微视 sdk
        def init_sdk(self):
            init_res = self.call_cpp("NET_DVR_Init")  # SDK初始化
            if init_res:
                logging.info("SDK初始化成功")
                return True
            else:
                error_info = self.call_cpp("NET_DVR_GetLastError")
                logging.error("SDK初始化错误:" + str(error_info))
                return False
        # 释放sdk
        def sdk_clean(self):
            result = self.call_cpp("NET_DVR_Cleanup")
            logging.info("释放资源", result)
    
  • 用户设备登录
  • 请求所用参数,这里需要用python ctypes参照https://open.hikvision.com/hardware/definitions/NET_DVR_Login_V40.html 所给出的结构写出对应的python类,有些常量具体数值是没有的,需要结合之前所说的Java,C#的demo看。

    class NET_DVR_USER_LOGIN_INFO(Structure):
        _fields_ = [
            ("sDeviceAddress", c_byte * 129),  # 设备地址,IP或者普通域名
            ("byUseTransport", c_byte),  # 是否启用能力透传 0:不启动,默认  1:启动
            ("wPort", c_uint16),  # 设备端口号
            ("sUserName", c_byte * 64),  # 登录用户名
            ("sPassword", c_byte * 64),  # 登录密码
            # ("fLoginResultCallBack",)  #
            ("bUseAsynLogin", c_bool),  # 是否异步登录, 0:否 1:是
            ("byProxyType", c_byte),  # 代理服务器类型:0- 不使用代理,1- 使用标准代理,2- 使用EHome代理
            # 是否使用UTC时间:
            # 0 - 不进行转换,默认;
            # 1 - 输入输出UTC时间,SDK进行与设备时区的转换;
            # 2 - 输入输出平台本地时间,SDK进行与设备时区的转换
            ("byUseUTCTime", c_byte),
            # 登录模式(不同模式具体含义详见“Remarks”说明):
            # 0- SDK私有协议,
            # 1- ISAPI协议,
            # 2- 自适应(设备支持协议类型未知时使用,一般不建议)
            ("byLoginMode", c_byte),
            # ISAPI协议登录时是否启用HTTPS(byLoginMode为1时有效):
            # 0 - 不启用,
            # 1 - 启用,
            # 2 - 自适应(设备支持协议类型未知时使用,一般不建议)
            ("byHttps", c_byte),
            # 代理服务器序号,添加代理服务器信息时相对应的服务器数组下表值
            ("iProxyID", c_long),
            # 保留,置为0
            ("byRes3", c_byte * 120),
    # 设备参数结构体。
    class NET_DVR_DEVICEINFO_V30(Structure):
        _fields_ = [
            ("sSerialNumber", c_byte * 48),  # 序列号
            ("byAlarmInPortNum", c_byte),  # 模拟报警输入个数
            ("byAlarmOutPortNum", c_byte),  # 模拟报警输出个数
            ("byDiskNum", c_byte),  # 硬盘个数
            ("byDVRType", c_byte),  # 设备类型,详见下文列表
            ("byChanNum", c_byte),  # 设备模拟通道个数,数字(IP)通道最大个数为byIPChanNum + byHighDChanNum*256
            ("byStartChan", c_byte),  # 模拟通道的起始通道号,从1开始。数字通道的起始通道号见下面参数byStartDChan
            ("byAudioChanNum", c_byte),  # 设备语音对讲通道数
            ("byIPChanNum", c_byte),
            # 设备最大数字通道个数,低8位,搞8位见byHighDChanNum. 可以根据ip通道个数是否调用NET_DVR_GetDVRConfig (配置命令NET_DVR_GET_IPPARACFG_V40)获得模拟和数字通道的相关参数
            ("byZeroChanNum", c_byte),  # 零通道编码个数
            ("byMainProto", c_byte),  # 主码流传输协议类型: 0 - private, 1 - rtsp, 2- 同时支持私有协议和rtsp协议去留(默认采用私有协议取流)
            ("bySubProto", c_byte),  # 字码流传输协议类型: 0 - private , 1 - rtsp , 2 - 同时支持私有协议和rtsp协议取流 (默认采用私有协议取流)
            # 能力,位与结果为0表示不支持,1
            # 表示支持
            # bySupport & 0x1,表示是否支持智能搜索
            # bySupport & 0x2,表示是否支持备份
            # bySupport & 0x4,表示是否支持压缩参数能力获取
            # bySupport & 0x8, 表示是否支持双网卡
            # bySupport & 0x10, 表示支持远程SADP
            # bySupport & 0x20, 表示支持Raid卡功能
            # bySupport & 0x40, 表示支持IPSAN目录查找
            # bySupport & 0x80, 表示支持rtp over rtsp
            ("bySupport", c_byte),
            # 能力集扩充,位与结果为0表示不支持,1
            # 表示支持
            # bySupport1 & 0x1, 表示是否支持snmp
            # v30
            # bySupport1 & 0x2, 表示是否支持区分回放和下载
            # bySupport1 & 0x4, 表示是否支持布防优先级
            # bySupport1 & 0x8, 表示智能设备是否支持布防时间段扩展
            # bySupport1 & 0x10, 表示是否支持多磁盘数(超过33个)
            # bySupport1 & 0x20, 表示是否支持rtsp over http
            # bySupport1 & 0x80, 表示是否支持车牌新报警信息,且还表示是否支持NET_DVR_IPPARACFG_V40配置
            ("bySupport1", c_byte),
            # 能力集扩充,位与结果为0表示不支持,1
            # 表示支持
            # bySupport2 & 0x1, 表示解码器是否支持通过URL取流解码
            # bySupport2 & 0x2, 表示是否支持FTPV40
            # bySupport2 & 0x4, 表示是否支持ANR(断网录像)
            # bySupport2 & 0x20, 表示是否支持单独获取设备状态子项
            # bySupport2 & 0x40, 表示是否是码流加密设备
            ("bySupport2", c_byte),
            ("wDevType", c_uint16),  # 设备型号,详见下文列表
            # 能力集扩展,位与结果:0 - 不支持,1 - 支持
            # bySupport3 & 0x1, 表示是否支持多码流
            # bySupport3 & 0x4, 表示是否支持按组配置,具体包含通道图像参数、报警输入参数、IP报警输入 / 输出接入参数、用户参数、设备工作状态、JPEG抓图、定时和时间抓图、硬盘盘组管理等
            # bySupport3 & 0x20,表示是否支持通过DDNS域名解析取流
            ("bySupport3", c_byte),
            # 是否支持多码流,按位表示,位与结果:0 - 不支持,1 - 支持
            # byMultiStreamProto & 0x1, 表示是否支持码流3
            # byMultiStreamProto & 0x2, 表示是否支持码流4
            # byMultiStreamProto & 0x40, 表示是否支持主码流
            # byMultiStreamProto & 0x80, 表示是否支持子码流
            ("byMultiStreamProto", c_byte),
            ("byStartDChan", c_byte),  # 起始数字通道号,0表示无数字通道,比如DVR或IPC
            ("byStartDTalkChan", c_byte),  # 起始数字对讲通道号,区别于模拟对讲通道号,0表示无数字对讲通道
            ("byHighDChanNum", c_byte),  # 数字通道个数,高8位
            # 能力集扩展,按位表示,位与结果:0 - 不支持,1 - 支持
            # bySupport4 & 0x01, 表示是否所有码流类型同时支持RTSP和私有协议
            # bySupport4 & 0x10, 表示是否支持域名方式挂载网络硬盘
            ("bySupport4", c_byte),
            # 支持语种能力,按位表示,位与结果:0 - 不支持,1 - 支持
            # byLanguageType == 0,表示老设备,不支持该字段
            # byLanguageType & 0x1,表示是否支持中文
            # byLanguageType & 0x2,表示是否支持英文
            ("byLanguageType", c_byte),
            ("byVoiceInChanNum", c_byte),  # 音频输入通道数
            ("byStartVoiceInChanNo", c_byte),  # 音频输入起始通道号,0表示无效
            ("byRes3", c_byte * 2),  # 保留,置为0
            ("byMirrorChanNum", c_byte),  # 镜像通道个数,录播主机中用于表示导播通道
            ("wStartMirrorChanNo", c_uint16),  # 起始镜像通道号
            ("byRes2", c_byte * 2)]  # 保留,置为0
    class NET_DVR_DEVICEINFO_V40(Structure):
        _fields_ = [
            ("struDeviceV30", NET_DVR_DEVICEINFO_V30),  # 设备参数
            ("bySupportLock", c_byte),  # 设备是否支持锁定功能,bySuportLock 为1时,dwSurplusLockTime和byRetryLoginTime有效
            ("byRetryLoginTime", c_byte),  # 剩余可尝试登陆的次数,用户名,密码错误时,此参数有效
            # 密码安全等级: 0-无效,1-默认密码,2-有效密码,3-风险较高的密码,
            # 当管理员用户的密码为出厂默认密码(12345)或者风险较高的密码时,建议上层客户端提示用户名更改密码
            ("byPasswordLevel", c_byte),
            ("byProxyType", c_byte),  # 代理服务器类型,0-不使用代理,1-使用标准代理,2-使用EHome代理
            # 剩余时间,单位:秒,用户锁定时次参数有效。在锁定期间,用户尝试登陆,不算用户名密码输入对错
            # 设备锁定剩余时间重新恢复到30分钟
            ("dwSurplusLockTime", c_ulong),
            # 字符编码类型(SDK所有接口返回的字符串编码类型,透传接口除外):
            # 0 - 无字符编码信息(老设备)
            # 1 - GB2312
            ("byCharEncodeType", c_byte),
            # 支持v50版本的设备参数获取,设备名称和设备类型名称长度扩展为64字节
            ("bySupportDev5", c_byte),
            # 登录模式(不同的模式具体含义详见"Remarks"说明:0- SDK私有协议,1- ISAPI协议)
            ("byLoginMode", c_byte),
            # 保留,置为0
            ("byRes2", c_byte * 253),
    class NET_DVR_Login_V40(Structure):
        _fields_ = [
            ("pLoginInfo", NET_DVR_USER_LOGIN_INFO),
            ("lpDeviceInfo", NET_DVR_DEVICEINFO_V40)
    
     # 用户登录指定摄像机设备
        def login(self, address="192.168.1.1", port=8000, user="admin", pwd="admin"):
            # 设置连接时间
            set_overtime = self.call_cpp("NET_DVR_SetConnectTime", 5000, 4)  # 设置超时
            if set_overtime:
                logging.info(address + ", 设置超时时间成功")
            else:
                error_info = self.call_cpp("NET_DVR_GetLastError")
                logging.error(address + ", 设置超时错误信息:" + str(error_info))
                return False
            # 设置重连
            self.call_cpp("NET_DVR_SetReconnect", 10000, True)
            b_address = bytes(address, "ascii")
            b_user = bytes(user, "ascii")
            b_pwd = bytes(pwd, "ascii")
            struLoginInfo = login.NET_DVR_USER_LOGIN_INFO()
            struLoginInfo.bUseAsynLogin = 0  # 同步登陆
            i = 0
            for o in b_address:
                struLoginInfo.sDeviceAddress[i] = o
                i += 1
            struLoginInfo.wPort = port
            i = 0
            for o in b_user:
                struLoginInfo.sUserName[i] = o
                i += 1
            i = 0
            for o in b_pwd:
                struLoginInfo.sPassword[i] = o
                i += 1
            device_info = login.NET_DVR_DEVICEINFO_V40()
            loginInfo1 = byref(struLoginInfo)
            loginInfo2 = byref(device_info)
            user_id = self.call_cpp("NET_DVR_Login_V40", loginInfo1, loginInfo2)
            logging.info(address + ", 登录结果:" + str(user_id))
            if user_id == -1:  # -1表示失败,其他值表示返回的用户ID值。
                error_info = self.call_cpp("NET_DVR_GetLastError")
                logging.error(address + ", 登录错误信息:" + str(error_info))
            return user_id
    
  • 调用网络摄像机获得视频数据流
  • 这里会有callback的概念,这里是针对视频流的回调,得到视频流后可以自定义视频流的处理,比如直接对接openCV等

    # 定义callback
    @CFUNCTYPE(None, c_long, c_ulong, c_byte, c_ulong, c_ulong)
    def g_real_data_call_back(lRealPlayHandle: c_long,
                              dwDataType: c_ulong,
                              pBuffer: c_byte,
                              dwBufSize: c_ulong,
                              dwUser: c_ulong):
        print('callback pBufSize is ', lRealPlayHandle, dwBufSize)
    # 预览参数结构体
    class NET_DVR_PREVIEWINFO(Structure):
        _fields_ = [
            # 通道号,目前设备模拟通道号从1开始,数字通道的起始通道号通过
            # NET_DVR_GetDVRConfig(配置命令NET_DVR_GET_IPPARACFG_V40)获取(dwStartDChan)
            ('lChannel', c_long),
            # 码流类型:0-主码流,1-子码流,2-三码流,3-虚拟码流,以此类推
            ('dwStreamType', c_ulong),
            # 连接方式:0-TCP方式,1-UDP方式,2-多播方式,3-RTP方式,4-RTP/RTSP,5-RTP/HTTP,6-HRUDP(可靠传输)
            ('dwLinkMode', c_ulong),
            # 播放窗口的句柄,为NULL表示不解码显示
            ('hPlayWnd', c_void_p),
            # 0-非阻塞取流,1- 阻塞取流
            # 若设为不阻塞,表示发起与设备的连接就认为连接成功,如果发生码流接收失败、播放失败等
            # 情况以预览异常的方式通知上层。在循环播放的时候可以减短停顿的时间,与NET_DVR_RealPlay
            # 处理一致。
            # 若设为阻塞,表示直到播放操作完成才返回成功与否,网络异常时SDK内部connect失败将会有5s
            # 的超时才能够返回,不适合于轮询取流操作。
            ('bBlocked', c_bool),
            # 是否启用录像回传: 0-不启用录像回传,1-启用录像回传。ANR断网补录功能,
            # 客户端和设备之间网络异常恢复之后自动将前端数据同步过来,需要设备支持。
            ('bPassbackRecord', c_bool),
            # 延迟预览模式:0-正常预览,1-延迟预览
            ('byPreviewMode', c_byte),
            # 流ID,为字母、数字和"_"的组合,IChannel为0xffffffff时启用此参数
            ('byStreamID', c_byte * 32),
            # 应用层取流协议:0-私有协议,1-RTSP协议。
            # 主子码流支持的取流协议通过登录返回结构参数NET_DVR_DEVICEINFO_V30的byMainProto、bySubProto值得知。
            # 设备同时支持私协议和RTSP协议时,该参数才有效,默认使用私有协议,可选RTSP协议。
            ('byProtoType', c_byte),
            # 保留,置为0
            ('byRes1', c_byte),
            # 码流数据编解码类型:0-通用编码数据,1-热成像探测器产生的原始数据
            # (温度数据的加密信息,通过去加密运算,将原始数据算出真实的温度值)
            ('byVideoCodingType', c_byte),
            # 播放库播放缓冲区最大缓冲帧数,取值范围:1、6(默认,自适应播放模式)   15:置0时默认为1
            ('dwDisplayBufNum', c_ulong),
            # 保留,置为0
            ('byRes', c_byte * 216),
    

    视频流回调可以在NET_DVR_RealPlay_V40中直接得到,也可以用以下返回的lRealPlayHandle去调用callback_real_data()获得,回调所得的数据可以在回调函数里面操作

        def start_preview(self, cbFunc: hikFunc, userId=0):
            req = preview.NET_DVR_PREVIEWINFO()
            req.hPlayWnd = None
            req.lChannel = 1  # 预览通道号
            req.dwStreamType = 0  # 码流类型:0-主码流,1-子码流,2-三码流,3-虚拟码流,以此类推
            req.dwLinkMode = 0  # 连接方式:0-TCP方式,1-UDP方式,2-多播方式,3-RTP方式,4-RTP/RTSP,5-RTP/HTTP,6-HRUDP(可靠传输)
            req.bBlocked = 1  # 0-非阻塞 1-阻塞
            struPlayInfo = byref(req)
            # 这个回调函数不适合长时间占用
            # fRealDataCallBack_V30 = preview.REALDATACALLBACK
            lRealPlayHandle = self.call_cpp("NET_DVR_RealPlay_V40", userId, struPlayInfo, cbFunc, None)
            print("start_preview lrealPlayHandle is ", lRealPlayHandle)
            if lRealPlayHandle < 0:
                self.logout(userId)
                self.sdk_clean()
            return lRealPlayHandle
        def stop_preview(self, lRealPlayHandle):
            self.call_cpp("NET_DVR_StopRealPlay", lRealPlayHandle)
        def callback_real_data(self, lRealPlayHandle: c_long, cbFunc: g_real_data_call_back, dwUser: c_ulong):
            return self.call_cpp("NET_DVR_SetRealDataCallBack", lRealPlayHandle, cbFunc, dwUser)