[英]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.ListenConfig
在syscall.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.