什么是TCP代理

TCP代理是一种网络代理技术,它允许客户端和服务器之间通过一个位于中间的第三方TCP代理服务器进行通信。TCP代理的工作方式是客户端向代理服务器发送TCP连接请求,代理服务器将此请求转发到目标服务器,然后等待目标服务器响应。当目标服务器响应时,代理服务器将响应转发回客户端。

整体功能规划

(1)显示本地设备与远程设备之间的通信过程

(2)处理代理两端发送和接收的数据

(3)创建和管理socket

打印设备之间的通信过程

  1. # 所有可打印字符(长度为3)保持不变,不可打印字符改为句点“.”表示
  2. # chr函数用于将i转换为相应的ASCII字符
  3. # repr函数返回一个对象的字符串表示形式,通常用于调试和打印目的。对于大多数字符,repr(chr(i))返回一个长度为 3 的字符串,其中第一个和最后一个字符是单引号(')或双引号("),中间的字符是该字符本身
  4. HEX_FILTER = ''.join([(len(repr(chr(i))) == 3) and chr(i) or '.' for i in range(256)])
  5. # hexdump函数能接收bytes或string类型的输入,并将其转换为十六进制格式输出到屏幕上
  6. # 它能同时以十六进制数和ASCII可打印字符的格式,输出数据包的详细内容。
  7. # 这有助于理解未知协议的格式,或是在明文协议里查找用户的身份凭证等
  8. def hexdump(src, length=16, show=True):
  9. # 如果传进来的参数是bytes类型,调用decode函数将它转换为string类型
  10. if isinstance(src, bytes):
  11. src = src.decode()
  12. results = list()
  13. for i in range(0, len(src), length):
  14. # 从src中取出一段数据放入word变量
  15. word = str(src[i : i + length])
  16. # 调用内置的translate函数把整段数据转换成可打印字符的格式,保存到printable变量里,HEX_FILTER为翻译表
  17. printable = word.translate(HEX_FILTER)
  18. # 将word中的数据转换为十六进制保存在变量hexa中
  19. hexa = ' '.join([f'{ord(c):02x}' for c in word])
  20. hexwidth = length * 3
  21. # 将word变量起始点的偏移、其十六进制表示和可打印字符表示形式打包成一行字符串,放入results数组
  22. results.append(f'{i : 04x} {hexa:<{hexwidth}} {printable}')
  23. if show:
  24. for line in results:
  25. print(line)
  26. else:
  27. return results

处理代理两端发送和接收的数据

  1. # 从代理两端接收数据
  2. def receive_from(connection):
  3. # 用来存储socket对象返回的数据
  4. buffer = b""
  5. connection.settimeout(5)
  6. try:
  7. # 创建一个循环,不断把返回的数据写进buffer,直到数据读完或者连接超时为止。最后,把buffer返回给调用方.
  8. while True:
  9. data = connection.recv(4096)
  10. if not data:
  11. break
  12. buffer += data
  13. except Exception as e:
  14. pass
  15. return buffer
  16. def request_handler(buffer):
  17. # 此处可添加修改请求包操作
  18. return buffer
  19. def response_handler(buffer):
  20. # 此处可添加修改返回包操作
  21. return buffer
  22. def proxy_handler(client_socket, remote_host, remote_port, receive_first):
  23. # 连接远程主机
  24. remote_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
  25. remote_socket.connect((remote_host, remote_port))
  26. # 进入主循环之前,先确认是否需要先从远端设备接收数据
  27. if receive_first:
  28. remote_buffer = receive_from(remote_socket)
  29. hexdump(remote_buffer)
  30. remote_buffer = response_handler(remote_buffer)
  31. if len(remote_buffer):
  32. print("[<==] sending %d bytes to localhost." % len(remote_buffer))
  33. client_socket.send(remote_buffer)
  34. while True:
  35. # 从本地客户端读取数据,打印本地数据,处理请求数据,转发给远程服务器
  36. local_buffer = receive_from(client_socket)
  37. if len(local_buffer):
  38. line = "[==>]Received %d bytes from localhost." % len(local_buffer)
  39. print(line)
  40. hexdump(local_buffer)
  41. local_buffer = request_handler(local_buffer)
  42. remote_socket.send(local_buffer)
  43. print("[==>] Sent to remote.")
  44. # 从远程服务器读取数据,打印远端数据,处理响应数据,转发给本地客户端
  45. remote_buffer = receive_from(remote_socket)
  46. if len(remote_buffer):
  47. print("[<==]Received %d bytes from remote." % len(remote_socket))
  48. hexdump(local_buffer)
  49. remote_buffer = request_handler(remote_buffer)
  50. client_socket.send(remote_buffer)
  51. print("[<==] Sent to localhost.")
  52. # 当通信两端都没有任何数据时关闭两端的socket,退出代理循环
  53. if not len(local_buffer) or not len(remote_buffer):
  54. client_socket.close()
  55. remote_socket.close()
  56. print("[*]No more data. closing connections.")
  57. break

创建和管理socket

  1. def server_loop(local_host, local_port, remote_host, remote_port, receive_first):
  2. server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
  3. # 创建一个socket并绑定到本地端口开始监听
  4. try:
  5. server.bind((local_host, local_port))
  6. except Exception as e:
  7. print(" problem on bind: %r" % e)
  8. print("[!!] Failed to listen on %s:%d" % (local_host, local_port))
  9. print("[!!] check for other listening sockets or correct permissions. ")
  10. sys.exit(e)
  11. print("[*] Listening on %s: %d" % (local_host, local_port))
  12. server.listen(5)
  13. # 每出现一个新连接就新开一个线程,将新连接交给proxy_handler函数
  14. while True:
  15. client_socket, addr = server.accept()
  16. line = "> Received incoming connection from %s:%d" % (addr[0], addr[1])
  17. print(line)
  18. proxy_thread = threading.Thread(
  19. target=proxy_handler,
  20. args=(client_socket, remote_host, remote_port, receive_first)
  21. )
  22. proxy_thread.start()

整体代码

完整代码可以参考我的Github仓库:Github

代码测试

这里使用kali虚拟机(192.168.131.135)作为本地机器,使用ubuntu虚拟机(192.168.131.132)作为TCP代理机器和FTP服务器。

首先在ubuntu上运行该代码:

接着,在本地kali机器上连接ubuntu的9000端口进行登录:



此时可以看到ubuntu机器上已经打印出了数据:

Python实现简易版TCP代理的更多相关文章

  1. Python模拟简易版淘宝客服机器人

    对于用Python制作一个简易版的淘宝客服机器人,大概思路是:首先从数据库中用sql语句获取相关数据信息并将其封装成函数,然后定义机器问答的主体函数,对于问题的识别可以利用正则表达式来进行分析,结合现 ...

  2. Black Hat Python之#2:TCP代理

    在本科做毕设的时候就接触到TCP代理这东西,当时需要使用代理来对发送和收到的数据做修改,同时使用代理也让我对HTTP协议有了更深的了解. TCP Proxy用到的一个主要的东西就是socket.pro ...

  3. 一个简单的tcp代理实现

    There are a number of reasons to have a TCP proxy in your tool belt, bothfor forwarding traffic to b ...

  4. Python写地铁的到站的原理简易版

    Python地铁的到站流程及原理(个人理解) 今天坐地铁看着站牌就莫名的想如果用Python写其工作原理 是不是很简单就小试牛刀了下大佬们勿喷纯属小弟个人理解 首先来看看地铁上显示的站牌如下: 就想这 ...

  5. python简易版微信或QQ轰炸

    ​ 在讲解代码之前我们先来回忆一下,平时我们发送消息时,先打开微信或QQ的界面,在信息栏中输入你要发送的内容在点击发送或通过快捷键发送.如果要发送表情时,先打开微信或QQ的界面,在点击表情包中你要发送 ...

  6. django框架简介及自定义简易版框架

    web应用与web框架本质 概念 什么是web应用程序呢? Web应用程序就一种可以通过互联网来访问资源的应用程序, 用户可以只需要用一个浏览器而不需要安装其他程序就可以访问自己需要的资源. 应用软件 ...

  7. 简易版的TimSort排序算法

    欢迎探讨,如有错误敬请指正 如需转载,请注明出处http://www.cnblogs.com/nullzx/ 1. 简易版本TimSort排序算法原理与实现 TimSort排序算法是Python和Ja ...

  8. Android学习之路——简易版微信为例(三)

    最近好久没有更新博文,一则是因为公司最近比较忙,另外自己在Android学习过程和简易版微信的开发过程中碰到了一些绊脚石,所以最近一直在学习充电中.下面来列举一下自己所走过的弯路: (1)本来打算前端 ...

  9. 第十八篇:简易版web服务器开发

    在上篇有实现了一个静态的web服务器,可以接收web浏览器的请求,随后对请求消息进行解析,获取客户想要文件的文件名,随后根据文件名返回响应消息:那么这篇我们对该web服务器进行改善,通过多任务.非阻塞 ...

  10. Netty核心组件介绍及手写简易版Tomcat

    Netty是什么: 异步事件驱动框架,用于快速开发高i性能服务端和客户端 封装了JDK底层BIO和NIO模型,提供高度可用的API 自带编码解码器解决拆包粘包问题,用户只用关心业务逻辑 精心设计的Re ...

随机推荐

  1. MySQL-分区表和分区介绍

    一.MySQL分区简介 1.数据库分区 MySQL是一种常用的关系型数据库管理系统,分区表是一种在MySQL数据库中处理大规模数据的最佳方案之一,其主要目的是为了在特定的SQL操作中减少数据读写的总量 ...

  2. Linux-双网卡绑定bond详解

    1.什么是bond 网卡bond是通过多张物理网卡绑定为一个逻辑网卡,实现本地网卡的冗余,带宽扩容和负载均衡,在生产场景中是一种常用的技术.Kernels 2.4.12及以后的版本均供bonding模 ...

  3. 在Windows上使用.NET部署到Docker 《让孩子们走出大坑》

    折腾Docker 有几天了,整别的都没这个糟心.目前已经顺利部署运行起来了.顺便给大家分享下处理在Windows上使用.NET部署到Docker的相关问题解决方法. 1.  Docker无法安装问题 ...

  4. JS 判断两个数组是否相等,元素以及顺序相等,顺序不同但元素相等

    壹 ❀ 引 在日常开发中,判断两个数组是否相等应该是较为常见的场景,因为常用,所以想着简单记录下.关于判断数组相等,这里我分为两种场景,第一种是数组完全相等,即数组元素相同且元素顺序一致:第二则为元素 ...

  5. Freaktab将于12月底关闭

    出过众多优秀固件的电视盒子论坛Freaktab, 将于2021年12月31日关闭 R.I.P

  6. mysqlGTID主从同步出现1236错误问题

    从主库xtrabackup备份,配置好gtid复制,从主库的从库复制.一直报错误 Last_IO_Error: Got fatal error 1236 from master when readin ...

  7. java: -source 1.5 中不支持 diamond 运算符

    1.问题说明 平常在用idea编译spring boot多模块项目时,老是无端提示: Error:(107, 55) java: -source 1.5 中不支持 diamond 运算符 (请使用 - ...

  8. celery正常启动后能接收任务但不执行(已解决)

    错误截图:celery接收到任务却不执行(多出在windows系统中) 解决方法1 添加–pool=solo参数 celery -A celery_tasks.main worker --pool=s ...

  9. 自然周算法-javascript实现

    获取自然周 js获取自然周 本文作者:bigroc 本文链接:https://www.cnblogs.com/bigroc/p/14888550.html 代码 function getWeeks() ...

  10. 【Azure Redis】Redis服务负载达到100%后的影响及有何优化方法

    问题描述 Redis服务负载达到100%后的影响及有何优化方法 问题解答 Redis的负载达到100% 意味着 Redis 服务器繁忙,无法跟上请求,导致客户端发送出来的请求超时. 常规情况下有一下几 ...