关于python:限定源端口访问目标

41次阅读

共计 3705 个字符,预计需要花费 10 分钟才能阅读完成。

限定源端口拜访指标

1.1. 起因

在浸透测试时,客户须要对咱们的测试 IP 进行加白,然而此次客户要求准确到固定端口或者小范畴端口(不能 1 -65535),依据以前的教训,默认是加白 IP 和全端口,因为代理建设连贯应用的端口是随机的,所以这次算是从头查找材料总结一下各种指定源端口的形式。

这里的端口是指与指标建设连贯时应用的源端口,而不是代理监听的端口。

1.2. 留神

最好应用服务器(VPS)发送申请。如果在局域网外向外网发动申请,即便本机指定了端口,也杯水车薪,因为理论发出请求的是最外层网络设备。

1.3. 办法一:NC

# 指定 12345 端口,毛病是一次一用,比拟麻烦
nc <IP> <Port> -p 12345

1.4. 办法二:批量占用端口

查找了很多材料,但都没能找到适合的工具。最终我想到了一种蠢笨的办法:强制占用所有可用的端口。这样当代理发动申请时,就会主动抉择未被占用的端口。python 代码举例如下:

import socket
import time

# 要占用的端口范畴
start_port = 1
end_port = 60000

# 创立套接字
socks = []
for port in range(start_port, end_port+1):
    try:
        sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
        sock.bind(('0.0.0.0', port))
        sock.listen(1)
        socks.append(sock)
    except:
        print(f"{port} 被其余利用占用")

print(f'已占用端口范畴: {start_port} - {end_port}')

# 长期占用端口
while True:
    time.sleep(60)

成果如下,可见都是 60000 后的端口连贯。

1.5. 办法三:手动实现代理转发

本人实现一个代理。监听一个端口,而后将流量发送到代理中,再由代理去固定源端口发送申请并返回后果。

Warning

因为都是走的这个端口去建设连贯,一旦出现异常状况,那这个端口可能会长期保持 TIME_WAIT 状态从而导致无奈失常应用。

python 代码如下:

import http.server                  # 导入 HTTP 服务器相干的模块
import socketserver                 # 导入 socket 服务器相干的模块
import http.client                  # 导入 HTTP 客户端相干的模块
import urllib.parse                 # 导入 URL 解析相干的模块
import socket                       # 导入 socket 相干的模块
from loguru import logger           # 导入日志库

LISTEN_PORT = 8080                  # 本地监听端口
CONNECT_PORT = 22333                # 用于建设代理服务器与指标服务器连贯的本地端口

# 继承 http.server.BaseHTTPRequestHandler 类,实现代理服务器的申请解决
class ProxyRequestHandler(http.server.BaseHTTPRequestHandler):

    # 解决 CONNECT 办法
    def do_CONNECT(self):
        # 解析 URL
        u = urllib.parse.urlparse('http://' + self.path)
        # 建设一个新的套接字
        p_sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        # 将该套接字绑定到本地端口
        p_sock.bind(('0.0.0.0', CONNECT_PORT))
        # 建设连贯
        p_sock.connect((u.hostname, u.port or 80))
        # 返回 200 连贯已建设响应
        self.wfile.write(b'HTTP/1.1 200 Connection Established\r\n\r\n')
        # 将该套接字设置为非阻塞模式
        p_sock.setblocking(False)
        self.connection.setblocking(False)
        while True:
            try:
                # 从代理服务器接收数据
                data = self.connection.recv(1024)
                if not data:
                    break
                # 将数据发送到指标服务器
                p_sock.sendall(data)
            except:
                pass
            try:
                # 从指标服务器接收数据
                data = p_sock.recv(1024)
                if not data:
                    break
                # 将数据发送到代理服务器
                self.connection.sendall(data)
            except:
                pass
        # 敞开连贯
        self.connection.close()
        p_sock.close()

    # 解决 GET 办法
    def do_GET(self):
        # 解析 URL
        u = urllib.parse.urlparse(self.path)
        # 创立一个 HTTP 连贯
        conn = http.client.HTTPConnection(u.hostname, port=u.port or 80, timeout=10, source_address=('0.0.0.0', CONNECT_PORT))
        # 发送 HTTP 申请
        conn.request(self.command, self.path, headers=self.headers)
        # 获取 HTTP 响应
        resp = conn.getresponse()
        # 发送 HTTP 响应头
        self.send_response(resp.status)
        for header, value in resp.getheaders():
            self.send_header(header, value)
        self.end_headers()
        # 发送 HTTP 响应体
        while True:
            data = resp.read(1024)
            if not data:
                break
            self.wfile.write(data)

    # 解决 POST 办法
    def do_POST(self):
        # 解析 URL
        u = urllib.parse.urlparse(self.path)
        # 获取申请体长度
        content_len = int(self.headers.get('Content-Length', 0))
        # 读取申请体
        content_len = int(self.headers.get('Content-Length', 0))  # 获取申请体长度
        body = self.rfile.read(content_len)  # 读取申请体数据
        conn = http.client.HTTPConnection(u.hostname, port=u.port or 80, timeout=10,
                                          source_address=('0.0.0.0', CONNECT_PORT))  # 创立 HTTP 连贯对象
        conn.request(self.command, self.path, body=body, headers=self.headers)  # 发送 HTTP POST 申请
        resp = conn.getresponse()  # 获取响应对象

        self.send_response(resp.status)  # 发送 HTTP 响应状态码
        for header, value in resp.getheaders():  # 遍历响应头部信息
            self.send_header(header, value)  # 发送响应头部信息
        self.end_headers()  # 完结响应头部信息的发送

        while True:
            data = resp.read(1024)  # 依照每次读取 1024 字节的形式获取响应数据
            if not data:  # 如果读取结束,跳出循环
                break
            self.wfile.write(data)  # 将响应数据写入连贯,响应给客户端

if __name__ == '__main__':
    try:
        # 创立一个 TCPServer 对象,绑定到 0.0.0.0:LISTEN_PORT 的地址,用于监听来自客户端的连贯申请,当有连贯申请时,将应用 ProxyRequestHandler 类进行解决。httpd = socketserver.TCPServer(('0.0.0.0', LISTEN_PORT), ProxyRequestHandler)
        logger.info(f"Proxy server is listening on port {LISTEN_PORT}")
        #  开始监听并承受来自客户端的连贯申请,直到呈现 KeyboardInterrupt 异样为止
        httpd.serve_forever()
    except KeyboardInterrupt:
        httpd.server_close()

成果如下,均为 22333 端口。

1.6. 尝试过的失败办法记录

  • Nginx 反向代理
  • SSH 动静代理 ssh -N -D 0.0.0.0:9999 ubuntu@hongkong
  • socat sudo socat TCP-LISTEN:<new_source_port>,fork,reuseaddr TCP:<destination_IP>:<destination_port>

正文完
 0