Python 3.7 - 检测Linux系统上指定端口是否被占用

检测端口占用

import socket
import sys

# 定义待检测的端口
reserved_ports = [8080, 9090, "30000-32767"]

# 将声明的待检测端口 展开并转换为整数
def convert_ports(ports):
    list_int_port = []
    for port in ports:
        if type(port) == int:
            list_int_port.append(port)
        elif type(port) == str:
            if '-' in port:
                start, end = map(int, port.split('-'))
                for p in range(start, end+1):
                    list_int_port.append(p)
            else:
                list_int_port.append(int(port))
    return list_int_port


# function 1: 检测端口处于 LISTEN 状态
def is_port_listening(port):
    s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    try:
        s.bind(('0.0.0.0', port))
    except socket.error as e:
        if e.errno == 98:
            # 端口已被占用
            return True
        else:
            return False
    s.close()
    return False

# function 2: 检测端口是否可以建联
def is_port_connect(port):
    with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
        try: 
            s.settimeout(0.2)
            s.connect(('127.0.0.1', port))
        except Exception:
            # 建联失败,端口未被占用
            return False
        s.close()
        return True

# 检测所有待检测端口的状态
def check_ports(ports):
    results = {}
    for port in ports:
        results[port] = is_port_listening(port)
    return results

if __name__ == '__main__':
    ports = convert_ports(reserved_ports)
    results = check_ports(ports)

    opend_ports = []
    closed_ports = []
    for port, status in results.items():
        if status:
            opend_ports.append(port)
        else:
            closed_ports.append(port)
    if len(opend_ports) > 0:
        print(f'Some reserved ports have already been opened({len(opend_ports)}): {opend_ports}')

Q&A

Q: connect的方式在k8s NodePort场景下存在建联不响应的情况(k8s网络插件自动创建iptables规则-ipvs模式下,drop 127.0.0.1 + NodePort的访问请求)
A: 由connect改为bind,端口绑定成功则说明未被占用,否则端口已被占用

Q: 使用0.0.0.0 bind端口,如果端口被127.0.0.1 connect(非listen)使用,bind依然失败
A: 由bind改为connect,并添加超时失败s.settimeout(0.2)

AI-辅助写代码

检测端口连通性

端口监听

import socket
import select

# List of ports to listen on
ports = [8000, 10000, 12000]

def handle_client(connection, address):
    try:
        print(f"New connection from {address[0]}:{address[1]}")
        while True:
            data = connection.recv(1024)
            if not data:
                break
            print(f"Received data from {address[0]}:{address[1]}: {data.decode()}")

    except socket.error as e:
        print(f"Error: {e}")

    finally:
        connection.close()
        print(f"Connection closed with {address[0]}:{address[1]}")


if __name__ == '__main__':
    # Create server sockets for each port
    server_sockets = []
    for port in ports:
        try:
            server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
            server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
            server_socket.bind(('0.0.0.0', port))
            server_socket.listen(5)
            server_sockets.append(server_socket)
            print(f"Listening on port {port}...")
        except socket.error as e:
            print(f"Failed to listen on port {port}...")
            print(f"Error: {e}")

    while True:
        # Use select to monitor sockets for incoming connections
        readable, _, _ = select.select(server_sockets, [], [])

        for sock in readable:
            connection, address = sock.accept()
            handle_client(connection, address)

端口建联

import socket
import sys


# List of ports to listen on
ports = [8000, 10000, 12000]

# 检测端口是否处于 LISTEN 状态
def is_port_listening(ip, port):
    with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
        try: 
            s.settimeout(0.2)
            s.connect((ip, port))
        except Exception:
            # 建联失败,端口无法建联
            return False
        s.close()
        return True

# 检测到所有待检测端口的建联状态
def check_ports(ip, ports):
    results = {}
    for port in ports:
        results[port] = is_port_listening(ip, port)
    return results

#  get the Host IP address
def get_host_ip():
    try:
        # Get the hostname
        hostname = socket.gethostname()
        # Get the IP address
        ip = socket.gethostbyname(hostname)
        return ip
    except socket.error as e:
        print(f"Error: {e}")
        return None

if __name__ == '__main__':
    local_ip = get_host_ip()
    try:
        remote_ip  = sys.argv[1]
    except IndexError:
        sys.stdout.write(f"No arg: remote host ip")
        sys.exit(1)

    results = check_ports(remote_ip, ports)
    connected_ports = []
    refused_ports = []
    for port, status in results.items():
        if status:
            connected_ports.append(port)
        else:
            refused_ports.append(port)
    if len(refused_ports) > 0:
        sys.stdout.write(f'Cannot connect to some ports({refused_ports}). From {local_ip} To {remote_ip}')
        sys.exit(1)
    sys.stdout.write(f"Pass the ports connection check! ports({connected_ports}) From {local_ip} To {remote_ip}\n")

FAQ

Failed to listen on port 443, Permission denied

1024以下的端口为特权端口,需使用root账号启动监听进程

AI - 辅助写代码

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,445评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,889评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,047评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,760评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,745评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,638评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,011评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,669评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,923评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,655评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,740评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,406评论 4 320
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,995评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,961评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,023评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,483评论 2 342

推荐阅读更多精彩内容