五分钟技术趣谈 | MQTT设备接入时获取真实IP地址的解决方案

网络
本文通过实践演示,介绍如何在MQTT设备接入时获取真实IP地址。

Part 01

需求背景 

在OneNET平台某私有化项目中,项目方的需求是要获取设备真实IP地址,然后根据设备的IP来统计处于各个省内区域的设备数量展示到大屏上。

Part 02

查找解决方案 

以MQTT设备接入为例,由于项目方使用的外层负载是Nginx软负载,并且MQTT协议是基于TCP,只能走4层方式转发报文,Nginx转发报文的时候会将源TCP连接的IP地址改写为自己的内网IP地址,不能像F5这种硬负载可以直接将设备的源地址转发到后端服务上,因此就不能直接通过配置Nginx的方式来让MQTT接入服务获取到设备源IP地址,也就不能实现项目方的需求。

经过网上查询相关解决方案,发现一个Internet协议叫做proxy protocl(参考资料:https://www.jianshu.com/p/cc8d592582c9),该协议可以通过为TCP包添加一个很小的头信息,来传递客户端信息(协议栈、源IP、目的IP、源端口、目的端口等),在网络情况复杂又需要获取用户真实IP时非常有用。其本质是在三次握手结束后由代理在连接中插入了一个携带了原始连接四元组信息的数据包。

图片图片

proxy protocol协议流程

查阅到proxy protocol报文的格式如下图中所示,里面包含了客户端的源地址和端口等信息,能够满足我们的需求。

图片

proxy protocol报文格式

后端服务要获取这个特殊报文也需要在Nginx上配置开启proxy_protocol协议,如下图中所示。

图片

Nginx上配置开启proxy_protocol协议

通过wireshark工具抓包我们也发现包结构和查阅到的资料是一致的,接下来我们要做的就是在MQTT协议解析的时候把这个特殊的包也要解析处理并保存这个客户端的真实IP。

图片

Wireshark抓proxy_protocol协议包

Part 03

实践操作  

说干就干,我们修改MQTT接入服务的源代码是基于Netty框架实现的,于是我们在编解码的时候增加了真实IP解码器,如下图所示。

图片

真实IP解码器

随后我们在解码器的decode方法中,将原始报文解析出来,判断是否有proxy protocol报文,然后解析报文并提取里面的设备真实源IP地址和端口,并将之保持在Netty中的ChannelAttribute上下文中,方便后续获取。

注意,这里的proxy protocol报文和MQTT协议中的报文是粘包在一起的,所以我们需要提取源地址后将剩余的MQTT协议包分离处理交给后续的MQTT协议解码器进行处理,这就是整体处理流程。

图片

图片

真实IP解码器源码

Part 04

测试认证 

重新打包部署服务后,我们根据日志看到通过Nginx负载方式能够正常获取到测试设备的源IP信息,满足需求。

图片

对于4层的UDP协议获取设备源IP也可以参考本方案解决。

责任编辑:庞桂玉 来源: 移动Labs
相关推荐

2023-07-23 18:47:59

Docker开源

2023-07-16 18:46:30

2023-07-16 18:49:42

HTTP网络

2023-04-15 20:25:23

微前端

2023-07-02 16:09:57

人工智能人脸识别

2023-07-02 16:34:06

GPU虚拟化深度学习

2023-07-12 15:50:29

机器学习人工智能

2023-08-29 06:50:01

Javamaven

2023-07-30 10:09:36

MMD数据库

2023-09-12 07:10:13

Nacos架构

2023-08-15 14:46:03

2023-07-23 08:35:13

Web网络

2023-07-31 08:55:15

AI技术网络暴力

2024-12-18 14:10:33

2023-07-12 16:03:37

Android开发架构

2023-08-06 07:00:59

Openstack网络

2023-09-17 17:51:43

Android 14

2023-07-16 18:42:18

2023-08-06 06:55:29

数字可视化物联网

2023-09-02 20:22:42

自动化测试软件开发
点赞
收藏

51CTO技术栈公众号