nodejs实现TCP端口转发并截包的小工具

近期我正致力于开发一个基于 Go-CQHTTP 的 QQ 机器人应用程序项目,该项目现已成功实现了 Go-CQHTTP 的容器化部署,利用 Docker 技术确保其运行环境的一致性与便捷性。随着项目推进,接下来的工作重心转向部署配套的签名服务器(qsign),同样采用 Docker 进行应用封装与管理。

为了深入理解并有效监控 Go-CQHTTP 机器人与 qsign 服务间的通信交互,我计划运用 Node.js 编写一个专门的 TCP 端口转发与数据包截取工具。此工具不仅将助力调试当前项目中的通讯过程,确保二者间消息传递的准确无误,还因其通用性而有望成为日后调试其他基于 TCP 端口通信应用的理想辅助手段。

1 测试项目基本程序构成及端口设置

项目由3个js文件组成,都采用 net 模块来实现socker 通信。文件分别为:

  • server.js
    创建和启动一个 TCP 服务器,监听端口为:8888
  • tcpMapper.js
    创建一个 TCP 映射应用,包含:一个 TCP 服务端,监听端口为7777,用来接收客户端发送的数据;一个 TCP 客户端,远程连接server.js 监听的端口 8888
  • client.js
    创建一个 TCP 客户端,远程连接tcpMapper.js 监听的端口 7777

2 测试流程及实现功能:

  • 为简化设计,此例不考虑客户端自动连接功能,所以需要注意 3 个 js 的运行顺序:先启动 server.js,再启动 tcpMapper.js,最后启动 client.js(每次调试都需要重新启动一遍,确保连接成功)
  • 连接成功后,在 client.js 应用中输入任意字符串,通过 tcpMapper.js 转发到 server.js,server.js 将收到的数据转成大写后,再转发到 tcpMapper.js,tcpMapper.js 再转发到 client.js,最终在 client.js 中接收转换后的大写字符串。

3 程序代码

3.1 server.js

// 引入 net 模块
const net = require('net');// 创建本地监听服务器
const server = net.createServer((serverSocket) => {// 获取本地服务端口接收到的数据serverSocket.on('data', (data) => {data=data.toString().trim();console.log(`C -> S: 【 ${data}`);// 将data转换为大写data = data.toString().toUpperCase();// 显示要发送给客户端的数据console.log(`S -> C: 【 ${data} 】\n`);// 通过 serverSocket 将数据发送给客户端serverSocket.write(`${data}`);});
})// 启动本地监听服务器
server.listen(8888, () => {console.log(`Listening on localhost:${server.address().port}`);
});

3.2 tcpMapper.js

// 引入 net 模块
const net = require('net');// 目标服务器信息
const targetHost = '127.0.0.1';
const targetPort = 8888;// 创建本地监听服务器
const server = net.createServer((serverSocket) => {// 获取本地服务端口接收到的数据serverSocket.on('data', (data) => {console.log(`C -> M: 【 ${data}`);// 向远程服务器发送数据clientSocket.write(data);});// 创建一个客户端 Socket —— clientSocket,连接转发的服务器const clientSocket = new net.Socket();// 连接转发的目标服务器clientSocket.connect(targetPort, targetHost, () => {// 接收转发的目标服务器返回的数据clientSocket.on('data', (data) => {console.log(`M -> C: 【 ${data} 】\n`);// 向接入的客户端转发数据serverSocket.write(data);});// 错误处理['error', 'end'].forEach((event) => {serverSocket.on(event, () => clientSocket.end());clientSocket.on(event, () => serverSocket.end());});});// 远程连接错误处理clientSocket.on('error', (err) => {console.error(`Error connecting to ${targetHost}:${targetPort}:`, err);serverSocket.destroy(err);});
});// 启动本地监听服务器
server.listen(7777, () => {console.log(`Listening on localhost:${server.address().port}, forwarding to ${targetHost}:${targetPort}`);
});

3.3 client.js

// 引入 net 模块
const net = require('net');// 目标服务器信息
const targetHost = '127.0.0.1';
const targetPort = 7777;// 创建一个客户端 Socket —— clientSocket,连接目标服务器
const clientSocket = new net.Socket();
clientSocket.connect(targetPort, targetHost, () => {console.log(`连接成功! ${targetHost}:${targetPort}`)
})// 监听目标服务器返回的数据
clientSocket.on('data', (data) => {console.log(`S -> C: 【 ${data.toString()} 】\n`);
})// 键盘输入发送数据时
process.stdin.on('data', (data) => {data=data.toString().trim();console.log(`\nC -> S: 【 ${data}`);clientSocket.write(data);
})

4 调试方式

在 vscode 中,调试方式如下:
图1  tcpMapper.js 的调试方式

图1 tcpMapper.js 的调试方式

同时打开 3 个 vscode 终端,从右到左的顺序执行:server.js、tcpMapper.js、client.js,依次执行完之后,在运行 client.js 终端中输入任意字符串,回车后即可看到输出结果。

  • C -> S 表示客户端发送给服务器的数据,客户端并不知道 tcpMapper 的存在
  • S -> C 表示服务器转发给客户端的数据,服务器端也不知道 tcpMapper 的存在
  • C -> M 表示客户端转发给 tcpMapper 的数据,M -> C 表示 tcpMapper 转发给客户端的数据。tcpMapper 知道客户端和服务器的存在

5 总结

tcpMapper.js 作为一个 JavaScript 库,已经成功实现了 TCP 端口转发与数据包截取两大核心功能。然而,当前版本的 tcpMapper.js 在设计与实现上尚未充分关注系统的健壮性建设,故其定位更倾向于作为一种简易实用的工具,用于快速解决特定场景下的端口转发与数据包处理需求,而非面向复杂环境或高可用性要求的应用。

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

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

相关文章

软考111-上午题-【计算机网络】-URL和DNS

一、URL解析 org:各类组织结构(非盈利团队) 1-1、顶级域 顶级域名是域名的最后一个部分,即是域名最后一点之后的字母,例如:www.baidu.com这个域名中,顶级域是.com(或.COM&#xff…

数据结构系列-队列的结构和队列的实现

🌈个人主页:羽晨同学 💫个人格言:“成为自己未来的主人~” 队列 队列的概念及结构 队列:只允许在一端进行插入数据操作,在另一端进行删除删除数据操作的特殊线性表,队列具有先进先出FIFO,…

全面解析找不到msvcr110.dll,无法继续执行代码的解决方法

MSVCR110.dll的丢失可能导致某些应用程序无法启动。当用户试图打开依赖于该特定版本DLL文件的软件时,可能会遭遇“找不到指定模块”的错误提示,使得程序启动进程戛然而止。这种突如其来的故障不仅打断了用户的正常工作流程,也可能导致重要数据…

基于SpringBoot+微信小程序的农产品销售平台

一、项目背景介绍: 随着人们收入的不断增加、生活水平的普遍提高,对生活质量的要求也日益凸显。而作为关乎每个人的生命、健康安全的食品卫生、质量无疑更被人们所重视。所以,… 2. 其他国家的绿色有机食品所占其国家食品市场比重比较大,如德国在99年便已达到40%,美…

Mac反编译APK

文章目录 第一种方式: brew installapktool 使用说明dex2jar 使用说明 第二种方式: 下载安装包apktool 使用说明 (根据官方介绍没有操作成功,后续成功再更新这里)dex2jar 使用说明 安装 JD-GUI 查看jar包中的class文件JD-GUI 使用说明 第一种方式: brew install 安装过程可能很…

使用 mitmproxy 抓包 grpc

昨天在本地执行 grpc 的 quick start(python版本的),我了解 grpc 内部使用的是 HTTP2,所以我就想着抓包来试试,下面就来记录一下这个过程中的探索。 注意:我的电脑上面安装了 Fiddler Classic,…

微信小程序生命周期管理:从数据初始化到事件绑定

作为一个独立的应用开发平台,微信小程序提供了自己的生命周期机制,与我们熟悉的Vue.js框架有一些差异。掌握小程序生命周期的特点和使用技巧,对于开发高质量的小程序应用至关重要。深入理解和掌握小程序生命周期的使用技巧,将有助于我们构建出更加健壮和可维护的小程序应用。 小…

可视化大屏 - 项目1

文章目录 技术栈echarts 可视化需求分析代码实现 技术栈 flexible.js rem 实现不同终端下的响应式布局,根据不同屏幕宽度,自适配布局; html中引入index.js,可以改名为flexible.js;默认划分10份,可以自己修…

Linux存储的基本管理

实验环境: 系统里添加两块硬盘 ##1.设备识别## 设备接入系统后都是以文件的形式存在 设备文件名称: SATA/SAS/USB /dev/sda,/dev/sdb ##s SATA, dDISK a第几块 IDE /dev/hd0,/dev/hd1 ##h hard VIRTIO-BLOCK /de…

SQLAlchemy 建立数据库模型之间的关系

常见关系: 一对多关系多对一关系多对多关系一对一关系 一对多关系(一个作者,多篇文章) ## 一对多关系,单作者-多文章,外键不可少 ## 外键(ForeignKey)总在多的那边定义,关系(relationship)总在单的那边定…

gitee代码上传同步指南(git操作)

1.下载安装git 最开始安装git,下载链接:CNPM Binaries Mirror 选择对应版本下载git即可 2.生成ssh公钥 电脑桌面右键,选择Open Git Bash here 需要注意,在git中复制粘贴功能和windows不同,它有两种方式&#xff…

解码零跑汽车2023年报:营收增速大幅滑坡,净亏42亿,如何讲故事

在2023年的新势力车企中,有这么一家低调崛起的品牌,并没有像蔚小理那样高调,但去年全年销量却反超小鹏汽车,晋升成为新势力车企中的销量第三名,它就是第四家登陆港交所上市的新势力品牌—零跑汽车。 不过,…

蓝桥集训之斐波那契前n项和

蓝桥集训之斐波那契前n项和 核心思想&#xff1a;矩阵乘法 左边求和 右边求和 得到Sn fn2 – 1 因此只要求出fn2 即可 #include <iostream>#include <cstring>#include <algorithm>using namespace std;typedef long long LL;int n,m;int A[2][2] { …

【数据库】MySQL InnoDB存储引擎详解 - 读书笔记

MySQL InnoDB存储引擎详解 - 读书笔记 InnoDB 存储引擎概述InnoDB 存储引擎的版本InnoDB 体系架构内存缓冲池LRU List、Free List 和 Flush List重做日志缓冲&#xff08;redo log buffer&#xff09;额外的内存池 存储结构表空间系统表空间独立表空间通用表空间undo表空间临时…

学习记录:bazel和cmake运行终端指令

Bazel和CMake都是用于构建软件项目的工具&#xff0c;但它们之间有一些重要的区别和特点&#xff1a; Bazel&#xff1a; Bazel是由Google开发的构建和测试工具&#xff0c;用于构建大规模的软件项目。它采用一种称为“基于规则”的构建系统&#xff0c;它利用构建规则和依赖关…

6、【单例模式】确保了一个类在程序运行期间只有一个实例

你好&#xff0c;我是程序员雪球 在软件设计中&#xff0c;单例模式是一种常见的设计模式。它确保了一个类在程序运行期间只有一个实例&#xff0c;并提供了全局访问该实例的方式。单例模式在许多场景中都有广泛的应用&#xff0c;例如共享资源管理、数据库连接、日志记录器等…

ES6学习(五)-- Module 语法

文章目录 Module 语法1.1 痛点介绍(1) 异步加载(2) 私密(3) 重名(4) 依赖 1.2 解决方法(1) 解决异步加载问题(2) 解决私密问题(3) 重名解决方法(4) 解决依赖问题 1.3 模块化使用案例 Module 语法 之前js 出现的某些痛点&#xff1a; 在script 中引入的变量名不可以重复&#…

国内20家公司大模型岗位面试经验汇总

面试情况&#xff1a; 投过的公司&#xff1a;淘天&#xff0c;字节&#xff0c;蚂蚁&#xff0c;商汤&#xff0c;美团&#xff0c;夸克&#xff0c;腾讯&#xff0c;minimax&#xff0c;零一万物&#xff0c;阿里控股&#xff0c;潞晨科技&#xff0c;阿里巴巴国际&#xff…

【GEE实践应用】GEE下载遥感数据以及下载后在ArcGIS中的常见显示问题处理(以下载哨兵2号数据为例)

本期内容我们使用GEE进行遥感数据的下载&#xff0c;使用的相关代码如下所示&#xff0c;其中table是我们提前导入的下载遥感数据的研究区域的矢量边界数据。 var district table;var dsize district.size(); print(dsize);var district_geometry district.geometry();Map.…