Golang TCP 服务器 : how to write HTTP2 data

标签 go tcp http2

我是 HTTP/2.0 的新手,我正在尝试设置一个用 Golang 编写的 TCP 服务器,它接收和写入 HTTP/2.0 帧。我在将任何数据写回客户端时遇到问题。

以下代码片段显示了如何处理请求。

conn, err := l.Accept()
if err != nil {
    log.Fatal("could not accept connection:", err)
}
defer conn.Close()

// Every connection starts with a connection preface send first, which has to be read prior
// to reading any frames (RFC 7540, section 3.5)
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.Fatal("could not read from connection:", err)
}
if string(b) != preface {
    log.Fatal("invalid preface")
}

framer := http2.NewFramer(conn, conn)

// Read client request (SETTINGS and HEADERS)
readFrames(framer)

// Send empty SETTINGS frame to the client
framer.WriteRawFrame(http2.FrameSettings, 0, 0, []byte{})

// Read clients response (contains empty SETTINGS with END_STREAM flag)
readFrames(framer)

// Prepare HEADERS
hbuf := bytes.NewBuffer([]byte{})
encoder := hpack.NewEncoder(hbuf)
encoder.WriteField(hpack.HeaderField{Name: ":status:", Value: "200"})
encoder.WriteField(hpack.HeaderField{Name: "date", Value: time.Now().UTC().Format(http.TimeFormat)})
encoder.WriteField(hpack.HeaderField{Name: "content-length", Value: strconv.Itoa(len("ok"))})
encoder.WriteField(hpack.HeaderField{Name: "content-type", Value: "text/html"})

// Write HEADERS frame 
err = framer.WriteHeaders(http2.HeadersFrameParam{StreamID: 2, BlockFragment: hbuf.Bytes(), EndHeaders: true})
if err != nil {
    log.Fatal("could not write headers: ", err)
}

// Clients response contains GOAWAY
readFrames(framer)

framer.WriteData(2, true, []byte("ok"))
conn.Close()

可以在这里找到完整的服务器:https://play.golang.org/p/ONA_OoyAMg-

通过 curl 调用它:

curl -kv https://127.0.0.1:8080 --http2

据我所知,在 SETTINGS 帧之后,连接应该可以进行通信了。应该通过发送一个 HEADERS 帧来打开一个流,之后可以在打开的流上发送一个 DATA 帧。但是,在发送 HEADERS 帧后,我收到以下错误消息:

curl: (16) Error in the HTTP2 framing layer

客户以 GOAWAY 回应。

最佳答案

我自己发现了这个问题,如果事实上有两个。我在这篇文章中发现了这两个问题:http://undertow.io/blog/2015/04/27/An-in-depth-overview-of-HTTP2.html

首先,包含状态代码的必需响应 header 应该是:status 而不是:status:

最后,我没有响应发出请求的同一个流 ID,而是尝试打开一个新流。

所以我应该遍历请求帧,查找流 ID,并使用它来写出响应 header 。

// Read client request (SETTINGS and HEADERS)
readFrames(framer)
var streamID uint32
for _, frame := range frames {
    if headersframe, ok := frame.(*http2.HeadersFrame); ok {
        streamID = headersframe.StreamID
    }
}

//... Prepare headers

// Write HEADERS frame 
err = framer.WriteHeaders(http2.HeadersFrameParam{StreamID: streamID, BlockFragment: hbuf.Bytes(), EndHeaders: true})
if err != nil {
    log.Fatal("could not write headers: ", err)
}

framer.WriteData(streamID, true, []byte("ok"))
conn.Close()

关于Golang TCP 服务器 : how to write HTTP2 data,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52002623/

相关文章:

tcp - Julia ZMQ - 连接到其他 WebSockets 产生 StateError

tcp - erlang: gen_tcp:接受限制

java - 使用 Netty 创建一对多的 TCP 代理

ssl - HTTP2、NginX 但 TTFB 仍然很高

go - 获取 Go 项目的所有依赖项

go - 同时运行多个 go routines

go - 将 []int 转换为字符串的单行器

mongodb - 结构/结果的倒序时间顺序

linux - 如何使 Wildfly 10.1.0 在 Linux Ubuntu 16.04 中使用 h2 (HTTP/2) 协议(protocol)在端口 80 和 443 (SSL) 上工作

curl - curl在Mac上不支持http2