簡體   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