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…

WPF中TextWrapping

在 WPF(Windows Presentation Foundation)中,TextWrapping 是一个与文本布局相关的属性,用于控制文本在遇到容器边界时是否自动换行。这个属性常用于文本展示控件,如 TextBlock、TextBox、Label 等,以确保文…

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

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

Django -- 模型层

模型和字段 一个模型(model)就是一个单独的、确定的数据的信息源,包含了数据的字段和操作方法。通常,每个模型映射为一张数据库中的表。 基本的原则如下: 每个模型在Django中的存在形式为一个Python类每个类都是dja…

C#的Thread.CurrentThread.IsBackground的作用

当一个线程,被设置为IsBackground true的时候,它就会放手,让主线程不用等,而主线程一退出,它就会退出。 为False时,则是要求主线程等待其执行完毕,它先退出,主线程再退出。 参考官方…

使用阿里云试用Elasticsearch学习:1.6 基础入门——排序与相关性

默认情况下,返回的结果是按照 相关性 进行排序的——最相关的文档排在最前。 在本章的后面部分,我们会解释 相关性 意味着什么以及它是如何计算的, 不过让我们首先看看 sort 参数以及如何使用它。 排序 为了按照相关性来排序,需…

Linux存储的基本管理

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

代码随想录算法训练营第四十三天|1049. 最后一块石头的重量 II 494. 目标和 474.一和零

1049. 最后一块石头的重量 II 本题就和 昨天的 416. 分割等和子集 很像了,可以尝试先自己思考做一做。 视频讲解:https://www.bilibili.com/video/BV14M411C7oV https://programmercarl.com/1049.%E6%9C%80%E5%90%8E%E4%B8%80%E5%9D%97%E7%9F%B3%E5%A4%…

Python性能优化:提升代码执行效率的秘诀

在Python编程中,性能优化是一个不可忽视的方面。无论是处理大数据集,还是构建需要快速响应的系统,高效的代码都是至关重要的。本文将讨论一些Python性能优化的关键策略,包括选择正确的数据结构、避免常见的性能陷阱以及使用并行计…

牛的学术圈(c++实现)

题目 由于对计算机科学的热爱,以及有朝一日成为 「Bessie 博士」的诱惑,奶牛 Bessie 开始攻读计算机科学博士学位。 经过一段时间的学术研究,她已经发表了 N 篇论文,并且她的第 i 篇论文得到了来自其他研究文献的 ci 次引用。 B…

Python常用算法思想--快速解决24点游戏案例【附源码】

算法的起源:欧几里德的《几何原本》中阐述的求两个数的最大公约数的过程。 算法的定义:解题方案的准确而完整的描述,是一系列解决问题的清晰指令,算法代表用系统的方法描述解决问题的策略机制。 算法的本质:算法是程序的灵魂,也是衡量一位程序员水平高低的最好参照物。…

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

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

在国企特定的环境中,如何激励低效能员工?

导读: 总额高达4万亿元的巨额投资,曾经让国企在应对百年不遇的金融危机中交出一份靓丽的成绩单,然而随着“4万亿经济刺激措施”逐步退出,国企问题又开始暴露出来。 2012年度国内上市公司财务报告显示,国企成为亏损大户…