Python实现简易版TCP代理
什么是TCP代理
TCP代理是一种网络代理技术,它允许客户端和服务器之间通过一个位于中间的第三方TCP代理服务器进行通信。TCP代理的工作方式是客户端向代理服务器发送TCP连接请求,代理服务器将此请求转发到目标服务器,然后等待目标服务器响应。当目标服务器响应时,代理服务器将响应转发回客户端。
整体功能规划
(1)显示本地设备与远程设备之间的通信过程
(2)处理代理两端发送和接收的数据
(3)创建和管理socket
打印设备之间的通信过程
# 所有可打印字符(长度为3)保持不变,不可打印字符改为句点“.”表示
# chr函数用于将i转换为相应的ASCII字符
# repr函数返回一个对象的字符串表示形式,通常用于调试和打印目的。对于大多数字符,repr(chr(i))返回一个长度为 3 的字符串,其中第一个和最后一个字符是单引号(')或双引号("),中间的字符是该字符本身
HEX_FILTER = ''.join([(len(repr(chr(i))) == 3) and chr(i) or '.' for i in range(256)])
# hexdump函数能接收bytes或string类型的输入,并将其转换为十六进制格式输出到屏幕上
# 它能同时以十六进制数和ASCII可打印字符的格式,输出数据包的详细内容。
# 这有助于理解未知协议的格式,或是在明文协议里查找用户的身份凭证等
def hexdump(src, length=16, show=True):
# 如果传进来的参数是bytes类型,调用decode函数将它转换为string类型
if isinstance(src, bytes):
src = src.decode()
results = list()
for i in range(0, len(src), length):
# 从src中取出一段数据放入word变量
word = str(src[i : i + length])
# 调用内置的translate函数把整段数据转换成可打印字符的格式,保存到printable变量里,HEX_FILTER为翻译表
printable = word.translate(HEX_FILTER)
# 将word中的数据转换为十六进制保存在变量hexa中
hexa = ' '.join([f'{ord(c):02x}' for c in word])
hexwidth = length * 3
# 将word变量起始点的偏移、其十六进制表示和可打印字符表示形式打包成一行字符串,放入results数组
results.append(f'{i : 04x} {hexa:<{hexwidth}} {printable}')
if show:
for line in results:
print(line)
else:
return results
处理代理两端发送和接收的数据
# 从代理两端接收数据
def receive_from(connection):
# 用来存储socket对象返回的数据
buffer = b""
connection.settimeout(5)
try:
# 创建一个循环,不断把返回的数据写进buffer,直到数据读完或者连接超时为止。最后,把buffer返回给调用方.
while True:
data = connection.recv(4096)
if not data:
break
buffer += data
except Exception as e:
pass
return buffer
def request_handler(buffer):
# 此处可添加修改请求包操作
return buffer
def response_handler(buffer):
# 此处可添加修改返回包操作
return buffer
def proxy_handler(client_socket, remote_host, remote_port, receive_first):
# 连接远程主机
remote_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
remote_socket.connect((remote_host, remote_port))
# 进入主循环之前,先确认是否需要先从远端设备接收数据
if receive_first:
remote_buffer = receive_from(remote_socket)
hexdump(remote_buffer)
remote_buffer = response_handler(remote_buffer)
if len(remote_buffer):
print("[<==] sending %d bytes to localhost." % len(remote_buffer))
client_socket.send(remote_buffer)
while True:
# 从本地客户端读取数据,打印本地数据,处理请求数据,转发给远程服务器
local_buffer = receive_from(client_socket)
if len(local_buffer):
line = "[==>]Received %d bytes from localhost." % len(local_buffer)
print(line)
hexdump(local_buffer)
local_buffer = request_handler(local_buffer)
remote_socket.send(local_buffer)
print("[==>] Sent to remote.")
# 从远程服务器读取数据,打印远端数据,处理响应数据,转发给本地客户端
remote_buffer = receive_from(remote_socket)
if len(remote_buffer):
print("[<==]Received %d bytes from remote." % len(remote_socket))
hexdump(local_buffer)
remote_buffer = request_handler(remote_buffer)
client_socket.send(remote_buffer)
print("[<==] Sent to localhost.")
# 当通信两端都没有任何数据时关闭两端的socket,退出代理循环
if not len(local_buffer) or not len(remote_buffer):
client_socket.close()
remote_socket.close()
print("[*]No more data. closing connections.")
break
创建和管理socket
def server_loop(local_host, local_port, remote_host, remote_port, receive_first):
server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 创建一个socket并绑定到本地端口开始监听
try:
server.bind((local_host, local_port))
except Exception as e:
print(" problem on bind: %r" % e)
print("[!!] Failed to listen on %s:%d" % (local_host, local_port))
print("[!!] check for other listening sockets or correct permissions. ")
sys.exit(e)
print("[*] Listening on %s: %d" % (local_host, local_port))
server.listen(5)
# 每出现一个新连接就新开一个线程,将新连接交给proxy_handler函数
while True:
client_socket, addr = server.accept()
line = "> Received incoming connection from %s:%d" % (addr[0], addr[1])
print(line)
proxy_thread = threading.Thread(
target=proxy_handler,
args=(client_socket, remote_host, remote_port, receive_first)
)
proxy_thread.start()
整体代码
完整代码可以参考我的Github仓库:Github
代码测试
这里使用kali虚拟机(192.168.131.135)作为本地机器,使用ubuntu虚拟机(192.168.131.132)作为TCP代理机器和FTP服务器。
首先在ubuntu上运行该代码:

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

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

Python实现简易版TCP代理的更多相关文章
- Python模拟简易版淘宝客服机器人
对于用Python制作一个简易版的淘宝客服机器人,大概思路是:首先从数据库中用sql语句获取相关数据信息并将其封装成函数,然后定义机器问答的主体函数,对于问题的识别可以利用正则表达式来进行分析,结合现 ...
- Black Hat Python之#2:TCP代理
在本科做毕设的时候就接触到TCP代理这东西,当时需要使用代理来对发送和收到的数据做修改,同时使用代理也让我对HTTP协议有了更深的了解. TCP Proxy用到的一个主要的东西就是socket.pro ...
- 一个简单的tcp代理实现
There are a number of reasons to have a TCP proxy in your tool belt, bothfor forwarding traffic to b ...
- Python写地铁的到站的原理简易版
Python地铁的到站流程及原理(个人理解) 今天坐地铁看着站牌就莫名的想如果用Python写其工作原理 是不是很简单就小试牛刀了下大佬们勿喷纯属小弟个人理解 首先来看看地铁上显示的站牌如下: 就想这 ...
- python简易版微信或QQ轰炸
在讲解代码之前我们先来回忆一下,平时我们发送消息时,先打开微信或QQ的界面,在信息栏中输入你要发送的内容在点击发送或通过快捷键发送.如果要发送表情时,先打开微信或QQ的界面,在点击表情包中你要发送 ...
- django框架简介及自定义简易版框架
web应用与web框架本质 概念 什么是web应用程序呢? Web应用程序就一种可以通过互联网来访问资源的应用程序, 用户可以只需要用一个浏览器而不需要安装其他程序就可以访问自己需要的资源. 应用软件 ...
- 简易版的TimSort排序算法
欢迎探讨,如有错误敬请指正 如需转载,请注明出处http://www.cnblogs.com/nullzx/ 1. 简易版本TimSort排序算法原理与实现 TimSort排序算法是Python和Ja ...
- Android学习之路——简易版微信为例(三)
最近好久没有更新博文,一则是因为公司最近比较忙,另外自己在Android学习过程和简易版微信的开发过程中碰到了一些绊脚石,所以最近一直在学习充电中.下面来列举一下自己所走过的弯路: (1)本来打算前端 ...
- 第十八篇:简易版web服务器开发
在上篇有实现了一个静态的web服务器,可以接收web浏览器的请求,随后对请求消息进行解析,获取客户想要文件的文件名,随后根据文件名返回响应消息:那么这篇我们对该web服务器进行改善,通过多任务.非阻塞 ...
- Netty核心组件介绍及手写简易版Tomcat
Netty是什么: 异步事件驱动框架,用于快速开发高i性能服务端和客户端 封装了JDK底层BIO和NIO模型,提供高度可用的API 自带编码解码器解决拆包粘包问题,用户只用关心业务逻辑 精心设计的Re ...
随机推荐
- 【译】使用.NET将WebAssembly扩展到云(二)
原文 | Richard Lander 翻译 | 郑子铭 轻量级功能 嗯--但是如果我们使用 Wasm 更像是一个典型的功能而不是一个应用程序,我们可能不会计算一百万个单词,而是做一些更轻量级的事情. ...
- NC16527 [NOIP2013]货车运输
题目链接 题目 题目描述 A 国有 n 座城市,编号从 1 到 n ,城市之间有 m 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在有 q 辆货车在运输货物, 司机们想知道每辆车在不超过车辆 ...
- NC23501 小A的回文串
题目链接 题目 题目描述 小A非常喜欢回文串,当然我们都知道回文串这种情况是非常特殊的.所以小A只想知道给定的一个字符串的最大回文子串是多少,但是小A对这个结果并不是非常满意.现在小A可以对这个字符串 ...
- Docker 容器逃逸漏洞 (CVE-2020-15257)
漏洞详情 Docker发布一个容器逃逸漏洞,攻击者利用该漏洞可以实现容器逃逸,提升特权并破坏主机. containerd使用的抽象套接字仅使用UID做验证,即任意UID为0的进程均可访问此API. 当 ...
- java 从零开始手写 redis(十)缓存淘汰算法 LFU 最少使用频次
前言 java从零手写实现redis(一)如何实现固定大小的缓存? java从零手写实现redis(三)redis expire 过期原理 java从零手写实现redis(三)内存数据如何重启不丢失? ...
- HTMLElement对象
HTMLElement对象 任何HTML元素都继承于HTMLElement对象,一些元素直接实现这个接口,而另一些元素通过多层继承来实现它. 属性 从其父元素Element继承属性,并从Documen ...
- haproxy ssl证书配置
通常情况下,web应用程序的ssl证书放置于nginx的服务器,但很多时候前面会加一次负载均衡,使用HAProxy可以实现https的证书安全,从客户浏览器到HAProxy代理服务器之间为ssl加密传 ...
- spring boot+bootstrap实现动态轮播图实战
1.bootstrap轮播图 最近开发了个网站需要用到轮播图,正好前端用的是Bootstrap,这里就实战一下. 水平一般能力有限,仅供参考. 前提条件: bootstrap4.5 jquery 3张 ...
- Python之正则表达式匹配电话号码和邮箱
代码 #! python3 # phoneAndEmail.py - Finds phone numbers and email addresses on Clipboard import pyper ...
- FileBeat简单使用
简介 首先要了解ELK架构 这种结构因为需要在各个服务器上部署 Logstash,而它比较消耗 CPU 和内存资源,所以比较适合计算资源丰富的服务器,否则容易造成服务器性能下降,甚至可能导致无法正常工 ...