繁体   English   中英

来自 TCP 套接字的 HTTP 服务器(在 Go 中)

[英]HTTP server from TCP socket (in Go)

我正在尝试在 Go 中创建一个 TCP 套接字,将其绑定到VRF 接口并在该特定接口中建立一个 HTTP 服务器。 VRF 绑定工作正常,但启动 HTTP 服务器会返回一条错误消息,指出“accept tcp 127.0.0.1:80: accept: invalid argument” 我是否正确假设插座有某种缺陷并且我创建它是错误的?

以下是重现问题的简化版本。 VRF 部分被注释掉,因为它不会影响实际问题,但我将它留在这里是因为我试图避免人们告诉我只使用 net.Listen 而不是套接字。 VRF 需要先绑定,然后才能使用,因此 net.Listen 不是一个选项。

package main

import (
    "fmt"
    "net"
    "net/http"
    "os"
    "syscall"
)

func main() {
    fd, err := syscall.Socket(syscall.AF_INET, syscall.SOCK_STREAM, syscall.IPPROTO_TCP)
    if err != nil {
        fmt.Printf("Error creating socket: %v", err)
        os.Exit(1)
    }

    // if err = syscall.SetsockoptString(fd, syscall.SOL_SOCKET, syscall.SO_BINDTODEVICE, "vrfiface"); err != nil {
    //  fmt.Printf("Error binding to vrf: %v", err)
    //  os.Exit(1)
    // }

    sockAddr := &syscall.SockaddrInet4{
        Port: 80,
        Addr: [4]byte{127, 0, 0, 1},
    }

    if err = syscall.Bind(fd, sockAddr); err != nil {
        fmt.Printf("Error binding to IP and port: %v", err)
        os.Exit(1)
    }

    file := os.NewFile(uintptr(fd), "socketfile")
    if file == nil {
        fmt.Println("Error creating file")
        os.Exit(1)
    }

    listener, err := net.FileListener(file)
    if err != nil {
        fmt.Printf("Error creating a listener: %v", err)
        os.Exit(1)
    }

    http.HandleFunc("/", TestServer)
    if err = http.Serve(listener, nil); err != nil {
        fmt.Printf("Error serving HTTP requests: %v", err)
    }
}

func TestServer(w http.ResponseWriter, r *http.Request) {
    fmt.Fprintf(w, "Test, %s!", r.URL.Path[1:])
}

任何有关解决此问题的指针将不胜感激。 谢谢!

您可以使用net.ListenConfigsyscall.Bind之前注入所需的套接字选项。 这也确保套接字设置正确完成,并且与net包预期的方式相同。

ListenConfig.Control函数为您提供了一个syscall.RawConn ,您可以在其上使用闭包调用Control ,您可以在其中访问套接字设置期间正在使用的原始文件描述符。

func main() {
    lc := net.ListenConfig{Control: controlOnConnSetup}

    ln, err := lc.Listen(context.Background(), "tcp", "127.0.0.1:80")
    if err != nil {
        log.Fatal(err)
    }
    ln.Close()
}

func controlOnConnSetup(network string, address string, c syscall.RawConn) error {
    var operr error
    fn := func(fd uintptr) {
        operr = syscall.SetsockoptString(int(fd), syscall.SOL_SOCKET, syscall.SO_BINDTODEVICE, "vrfiface")
    }
    if err := c.Control(fn); err != nil {
        return err
    }
    if operr != nil {
        return operr
    }
    return nil
}

正如C Han已经在评论中提到的:你需要倾听。 创建服务器的顺序是创建socket,绑定IP和端口,调用listen ,然后接受传入的连接。 如果你忘记听,那么你会得到你看到的错误。 因此:

if err = syscall.Bind(fd, sockAddr); err != nil {
     ...
}

if err = syscall.Listen(fd, 10); err != nil {
    fmt.Printf("Error listening %v", err)
    os.Exit(1)
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM