我正在尝试设置一个接受和解码HTTP / 2数据的TCP服务器。可以在本文中找到用于实际解析框架的代码:
Decoding http2 frame header/data in Go
我在设置服务器时遇到问题。连接被接受,但挂在framer.ReadFrame()
上。这是一个代码示例:
// generate with: openssl req -x509 -newkey rsa:4096 -keyout server.key -out server.pem -days 365 -nodes
cert, err := tls.LoadX509KeyPair("server.pem", "server.key")
if err != nil {
log.Fatal(err)
}
tlsCfg := &tls.Config{
Certificates: []tls.Certificate{cert},
NextProtos: []string{"h2"},
}
l, err := tls.Listen("tcp", ":8787", tlsCfg)
if err != nil {
log.Fatal(err)
}
defer l.Close()
conn, err := l.Accept()
if err != nil {
log.Fatalln(err)
}
defer conn.Close()
framer := http2.NewFramer(conn, conn)
frame, _ := framer.ReadFrame() // Here it hangs
我使用curl请求服务器
curl -v https://127.0.0.1:8787/ -k --http2
最佳答案
对话doesn't start with a frame。首先客户端发送字符串
"PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n"
您可以通过检查客户端发送的字节来判断:
framer := http2.NewFramer(conn, io.TeeReader(conn, hex.Dumper(os.Stdout)))
// 00000000 50 52 49 20 2a 20 48 54 54 50 2f 32 2e 30 0d 0a |PRI * HTTP/2.0..|
// 00000010 0d 0a 53 4d 0d 0a 0d 0a 00 00 12 04 00 00 00 00 |..SM............|
// 00000020 00 00 03 00 00 00 64 00 04 40 00 00 00 00 02 00 |......d..@......|
// 00000030 00 00 00 00 00 04 08 00 00 00 00 00 3f ff 00 01 |............?...|
// 00000040 00 00 1e 01 05 00 00 00 01 82 84 87 41 8a a0 e4 |............A...|
// 00000050 1d 13 9d 09 b8 f3 af 3b 7a 88 25 b6 50 c3 ab b6 |.......;z.%.P...|
// 00000060 fa e0 53 03 2a 2f 2a ^C
如果您修改代码以首先读取该字符串,则它将按预期工作:
conn, err := l.Accept()
if err != nil {
log.Fatalln(err)
}
defer conn.Close()
const preface = "PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n"
b := make([]byte, len(preface))
if _, err := io.ReadFull(conn, b); err != nil {
log.Fatalln(err)
}
if string(b) != preface {
log.Fatalln("invalid preface")
}
framer := http2.NewFramer(conn, conn)
frame, err := framer.ReadFrame()
fmt.Println(frame, err)
// Output:
// [FrameHeader SETTINGS len=18] <nil>
如果发现这些文章对自己使用HTTP2很有用: