RtspServer之LibRtsp解决闪退 新增鉴权(用户名密码登录)

news2024/11/24 22:50:32

        编程语言:C

        目标平台:arm(hi3519)注:因为代码是纯C语言按道理可以移植到各种平台   

        基础库:librtsp(存在功能补全,有一定的bug)    

        因为项目需求需要在海思平台实现IPC的全部功能,实现RTSPServer的方案很多,冲浪了github平台发现有很多开源的方案,但是也有一些假开源比如EasyRtspServer(一个平台授权几万米^_^),有一些RtspServer是用C++写的,目前海思的交叉编译工具对C++高版本(大于C++11)支持并不好虽然可以编译过去但是运行的时候会有很多意向不到的问题,这也是本文为什么选用librtsp的原因。

        因为以前接触过librtsp,并在海思平台实现了RtspServer(相对来讲还是比较稳定的),但项目需求需要新增鉴权的功能(使用用户名密码登录),翻遍github发现能支持鉴权的开源项目并不多而且多数移植存在兼容性的问题比如前面提到的问题,因此下定决心在librtsp的基础上新增用户名密码鉴权的功能,并修复以前就存在闪退的bug。


1、首先通读代码

        主要的代码集中在了rtsp_demo.c和rtsp_msg.c里,demo里主要解决客户端连接和相关消息发送的功能,msg主要封装了协议解析的部分,因此本文的功能新增也都是在这两个文件中处理的。

2、熟悉鉴权协议和信息交互流程

        新增鉴权的功能,首先需要了解RTSP协议鉴权的步骤《参见》。

3、抓包分析

        3.1、找来海康的摄像头,使用Wireshark抓包工具对鉴权的过程进行分析,如下图

         3.2、未通过验证客户端(VLC)的发送和服务器应答如下(海康摄像头应答)

        VLC发送

        服务器应答如下(海康摄像头) 

         其中红框内指明了密码加密的方式,目前有两种一种是Digest,MD5加解密,一种是Basic,base64加解密,因为base64相对好实现一些因此本文先采用Basic的验证方式(VLC播放器会自动匹配),当然服务器(相机)可以做两种鉴权方式都支持(接下来再去完善MD5加解密的功能)。

        3.3、客户端回复如下(VLC播放器)

         3.4、通过以上的抓包分析可知,只需要在librtsp内完善DESCRIBE请求中的验证和解密即可(因为本文先用BASIC即base64加解密)


4、修改代码

        4.1、首先在rtsp client结构体内新增passed标志,表明该连接是否通过验证,未通过验证则不发送数据包,如下图

         4.2、在rtsp_do_event函数修改发送的地方新增passed判断(数据发送地方很多自行查找,涉及到音频和视频的发送)

         4.3、处理DESCRIBE请求

        在rtsp_handle_DESCRIBE函数内,新增加解密的功能,如下

         4.4、处理完DESCRIBE请求后还有socket的接收和发送的处理,完善authorization的相关处理函数

        4.5、base64代码如下 《参考》

static int base64Decode(uint8_t *input)
{
	unsigned int inlen = strlen(input);
	uint8_t output[1024] = {0};
	//解码需要一张反着的表
	const char *base64_tbl = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";

	int off = 0;
	unsigned int i;
	uint8_t reverse_tbl[256] = {0};

	if (NULL == input) {
		return -1;
	}
	if (inlen == 0) {
		return -2;
	}

	uint8_t *p = input;
	while (isprint(*p) && *p != ' ') {
		p++;
	}
	p++;
	inlen = inlen - (p - input);
	
	if (inlen % 4 != 0) {
		/*error not a base64*/
		dbprt("decode auth message is not a base64 message.");
		return -1;
	}

	for (i = 0; i < 64; i++) {
		reverse_tbl[base64_tbl[i]] = i;
	}

	for (i = 0; i < inlen - 4; i += 4) {
		output[off++] = (((reverse_tbl[p[i]] << 2) | (reverse_tbl[p[i + 1]] >> 4)) & 0xFF);
		output[off++] = (((reverse_tbl[p[i + 1]] << 4) | (reverse_tbl[p[i + 2]] >> 2)) & 0xFF);
		output[off++] = (((reverse_tbl[p[i + 2]] << 6) | (reverse_tbl[p[i + 3]])) & 0xFF);
	}

	if (p[i + 2] == '=') {
		output[off++] = (((reverse_tbl[p[i]] << 2) | (reverse_tbl[p[i + 1]] >> 4)) & 0xFF);
	}
	else if (p[i + 3] == '=') {
		output[off++] = (((reverse_tbl[p[i]] << 2) | (reverse_tbl[p[i + 1]] >> 4)) & 0xFF);
		output[off++] = (((reverse_tbl[p[i + 1]] << 4) | (reverse_tbl[p[i + 2]] >> 2)) & 0xFF);
	}
	else {
		output[off++] = (((reverse_tbl[p[i]] << 2) | (reverse_tbl[p[i + 1]] >> 4)) & 0xFF);
		output[off++] = (((reverse_tbl[p[i + 1]] << 4) | (reverse_tbl[p[i + 2]] >> 2)) & 0xFF);
		output[off++] = (((reverse_tbl[p[i + 2]] << 6) | (reverse_tbl[p[i + 3]])) & 0xFF);
	}

	dbprt("decode auth message is : %s", output);

	if (strstr(output, "admin:123456") == NULL) {
		return -1;
	}

	return 0;
}

5、相机程序异常退出的问题解决,因为linux平台在处理socket数据发送的时候如果socket客户端已关闭,会出现程序异常闪退的问题,参见下面博客解决《参考》,如下图send的地方都需要修改

 6、本工程暂不开源,后续完善MD5加解密后再github见,如需交流可文章后留言联系。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/47209.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

C++ Reference: Standard C++ Library reference: Containers: list: list: crend

C官网参考链接&#xff1a;https://cplusplus.com/reference/list/list/crend/ 公有成员函数 <list> std::list::crend const_reverse_iterator crend() const noexcept; 返回反向结束的常量反向迭代器 返回一个const_reverse_iterator&#xff0c;指向容器中第一个元素…

领悟《信号与系统》之 采样定理

采样定理采样定理一、采样定理结论二、奈奎斯特间隔和频率计算公式及例题&#xff1a;三、信号自然采样&#xff08;脉冲采样&#xff09;四、信号理想采样&#xff08;冲激采样&#xff09;采样定理 连续时间信号也叫模拟信号。在一定条件之下&#xff0c;模拟信号可以用该信…

LDAP客户端操作方法

本文章简单介绍LDAP Admin客户端的使用方法&#xff0c;客户端工具地址为&#xff1a; https://download.csdn.net/download/u013896064/87209312 1、前提 我使用的是Ubuntu的虚拟机安装了一个LDAP服务&#xff0c;初始化完成并配置好密码&#xff0c;本次例子里面我使用的U…

nodejs+vue音乐网站与分享平台

目 录 摘 要 I 1 绪论 1 1.1研究背景 1 1.2研究现状 1 1.3研究内容 2 2 系统关键技术 3 2.1 Spring Boot框架 3 2.2 JAVA技术 3 2.3 MYSQL数据库 4 2.4 B/S结构 4 3 系统分析 5 3.1 可行性分析 5 3.1.1 技术可行性 5 3.1.2经济可行性…

Windows OpenGL 图像色彩替换

目录 一.OpenGL 图像色彩替换 1.原始图片2.效果演示 二.OpenGL 图像色彩替换源码下载三.猜你喜欢 零基础 OpenGL ES 学习路线推荐 : OpenGL ES 学习目录 >> OpenGL ES 基础 零基础 OpenGL ES 学习路线推荐 : OpenGL ES 学习目录 >> OpenGL ES 特效 零基础 OpenGL…

[附源码]计算机毕业设计JAVA网上花店系统

[附源码]计算机毕业设计JAVA网上花店系统 项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis M…

SpringBoot基于若依项目工时统计成本核算管理源码带文字教程

该系统是前后端分离的架构&#xff0c;前端使用Vue2&#xff0c;后端使用SpringBoot2。 技术架构 技术框架&#xff1a;SpringBoot2.0.0 Mybatis1.3.2 Shiro swagger-ui jpa lombok Vue2 Mysql5.7 运行环境&#xff1a;jdk8 IntelliJ IDEA maven 宝塔面板 本地搭建文…

MySQL数据库行级锁之间隙锁、临键锁

间隙锁 默认情况下&#xff0c;InnoDB在 REPEATABLE READ事务隔离级别运行&#xff0c;InnoDB使用 next-key 锁进行搜索和索引扫描&#xff0c;以防止幻读。 索引上的等值查询(唯一索引)&#xff0c;给不存在的记录加锁时, 优化为间隙锁 。索引上的等值查询(非唯一普通索引)&…

数据字典功能

我们在项目中会有很多的下拉框,这些下拉框都有一个特点,就是键值对的存在 实现方案: 直接硬编码写在html代码中, 缺点:新增修改需要修改代码为每个下拉框都设计一个表&#xff0c;然后提供CRUD功能 , 缺点 : 很多表,很多重复的CRUD代码使用数据字典的方式 ,使用字典类型表和字…

CTFHUB-web-SQL注入

整数型注入 字符型注入 得到数据库名 sqli http://challenge-f7a63a00793e62c6.sandbox.ctfhub.com:10800/?id-1 union select 1,database() 爆sqli数据库的数据表 爆flag表的所有列&#xff1a; http://challenge-f7a63a00793e62c6.sandbox.ctfhub.com:10800/?id-1 union s…

[Azkaban] No active executors found分析

启动AzkabanWebServer报如下错误: 点击进入 ActiveExecutors.setupExecutors发现调用的是 loadExecutors()继续深入 private ImmutableSet<Executor> loadExecutors() throws ExecutorManagerException {logger.info("Initializing executors from database.&q…

window 下 达梦数据库的备份和还原

DM 提供的各种工具进行备份还原与恢复的操作&#xff0c;包括 DIsql工具、DMRMAN 工具、图形化客户端管理工具 MANAGER 和 CONSOLE。 DIsql 工具用于执 行联机的数据备份与数据还原&#xff0c;包括数、归档备份据库备份、表空间备份与还原、表备份与 还原&#xff1b; DMRMA…

docker-compose安装nacos

文章目录一、安装nacos1.docker-compose.yml2.nacos数据库表结构3.docker-compose 构建运行4.检查nacos日志6.测试访问&#xff1a;TODO:二、nacos为注册中心&#xff1a;项目测试1.新建maven项目&#xff1a;2.配置pom.xml3.配置application.properties4.激活服务发现客户端5.…

117.Django-缓存redis

1. 概述 ​ 动态网站的基本权衡是&#xff0c;它们是动态的。每次用户请求页面时&#xff0c;Web服务器都会进行各种计算 - 从数据库查询到模板呈现再到业务逻辑 - 以创建站点访问者看到的页面。从处理开销的角度来看&#xff0c;这比标准的文件读取文件系统服务器要耗时多了。…

数据库、计算机网络,操作系统刷题笔记4

数据库、计算机网络&#xff0c;操作系统刷题笔记4 2022找工作是学历、能力和运气的超强结合体&#xff0c;遇到寒冬&#xff0c;大厂不招人&#xff0c;可能很多算法学生都得去找开发&#xff0c;测开 测开的话&#xff0c;你就得学数据库&#xff0c;sql&#xff0c;oracle&…

【POJ No. 1195】 矩形区域查询 Mobile phones

【POJ No. 1195】 矩形区域查询 Mobile phones 北大 OJ 题目地址 【题意】 移动电话的基站区域分为多个正方形单元&#xff0c;形成S S 矩阵&#xff0c;行和列的编号为0&#xff5e;S -1&#xff0c;每个单元都包含一个基站。一个单元内活动手机的数量可能发生变化&#xff…

[附源码]Python计算机毕业设计Django大学生考勤管理系统论文

项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等等。 环境需要 1.运行环境&#xff1a;最好是python3.7.7&#xff0c;…

区块链存储优化——从MPT树到KV存储

MPT树存储的优缺点 区块链如果采用MPT树存储&#xff0c;大概会有以下优点&#xff1a; 可用全局数据的根哈希做共识&#xff0c;数据篡改会被立即发现&#xff1b;可以查询任意历史区块对应时刻的所有数据&#xff1b;方便从指定区块开始同步数据&#xff0c;因为正如上面所…

Codeforces Round #790 (Div. 4) G. White-Black Balanced Subtrees 感觉很好的树形dp的板子题

翻译&#xff1a; 您得到一个有根的树&#xff0c;其中包含从1到&#x1d45b;编号为&#x1d45b;的顶点。根结点是顶点1。还有一个字符串&#x1d460;表示每个顶点的颜色:如果&#x1d460;&#x1d456;&#x1d671;&#xff0c;那么顶点&#x1d456;是黑色的&#xff0…

MongoDB实战:应用场景以及Spring和mongodb的整合

前言 mongodb是非关系型数据库&#xff0c;他的存储数据可以超过上亿条&#xff08;老版本的mongodb有丢数据的情况&#xff0c;新版本不会有&#xff0c;网上说的&#xff09;&#xff0c;mongodb适合存储 一些量大表关系较简单的数据&#xff0c;例如用户信息&#xff0c;用户…