zoukankan      html  css  js  c++  java
  • 多个程序监听同一端口

    https://www.jianshu.com/p/ce277812eca2

    对于多个程序绑定同一个端口我们遇到最多的是(Port 80 was already in use),也就是说端口被占用,不能重复绑定,但是操作系统内核支持通过配置socket参数的方式来实现多个进程绑定同一个端口。

    简单示例

    package main
    
    import (
        "context"
        "golang.org/x/sys/windows"
        "net"
        "syscall"
    )
    
    var listenConfig = net.ListenConfig{
        Control:   MyControl,
    }
    
    func MyControl(network, address string, c syscall.RawConn) error {
        return c.Control(func(fd uintptr) {
            err := windows.SetsockoptInt(windows.Handle(fd), windows.SOL_SOCKET, windows.SO_REUSEADDR, 1)
            if err != nil {
                panic(err)
            }
        })
    }
    
    func main() {
        listener, err := listenConfig.Listen(context.Background(), "tcp", "127.0.0.1:8080")
    
        if err != nil {
            panic(err)
        }
        defer listener.Close()
        for {
            conn, err := listener.Accept()
            if err == nil {
                println("new connection ", conn.RemoteAddr().String())
            }
        }
    }
    
    

    执行该程序后发现多个进程可以绑定同一端口

    port_reuse.png

    进程12720,3632和13612同时绑定了8080端口

    原理解析

    这个例子关键代码是设置socket的SO_REUSEADDR参数
    实现多个程序绑定一个端口windows下需要设置SO_REUSEADDR参数
    linux下需要设置SO_REUSEADDR和SO_REUSEPORT参数

    对于windows和linux系统SO_REUSEADDR参数含义并不相同
    Windows系统:
    using-so-reuseaddr-and-so-exclusiveaddruse
    so-exclusiveaddruse

    Linux系统:
    The SO_REUSEPORT socket option

    socket man

    相关问题:
    how-do-so-reuseaddr-and-so-reuseport-differ
    可以看出Windows下SO_REUSEADDR可以用来端口复用,在Linux下SO_REUSEADDR为了实现TIME_WAIT阶段的快速绑定,SO_REUSEPORT用来配置端口复用

    安全问题

    由此引发了一个安全问题,如果一个正常的web程序监听80端口提供服务,其它恶意程序同样监听了80端口,那么发送到80端口的请求就会被恶意程序接收并处理,这是我们不愿看到的。

    Linux下处理方式为所有端口复用的进程必须在同一个用户下
    Windows下处理方式为添加SO_EXECLUSIVEADDRUSE参数,程序设置该参数后,其它程序就不能复用这个端口

    SO_REUSEADDR: Allows a socket to bind to an address and port already in use. The SO_EXCLUSIVEADDRUSE option can prevent this.
    

    开箱即用

    c语言程序直接调用setsockopt(fd, SOL_SOCKET, SO_REUSEPORT, &reuse_port, sizeof(reuse_port));函数即可

    golang包greuse提供了开箱即用的端口复用功能,兼容多系统 https://github.com/gogf/greuse

    其它语言程序可以通过调用bindp项目实现https://github.com/yongboy/bindp

    作者:写个代码容易么
    链接:https://www.jianshu.com/p/ce277812eca2
    来源:简书
    著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

  • 相关阅读:
    CF827D Best Edge Weight
    克鲁斯卡尔重构树总结
    模拟赛 提米树 题解 (DP+思维)
    luogu P4781 【模板】拉格朗日插值
    luogu P5826 【模板】子序列自动机
    子序列自动机
    luogu P1368 工艺 /【模板】最小表示法
    最小表示法
    SP1812 LCS2
    FZOJ 3602 T2
  • 原文地址:https://www.cnblogs.com/94cool/p/13501284.html
Copyright © 2011-2022 走看看