[英]Golang TCP server: how to write HTTP2 data
我是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 -kv https://127.0.0.1:8080 --http2
據我所知,在“設置”幀之后,連接應已准備就緒,可以進行通信了。 應該通過發送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
首先,所需的包含狀態碼的響應標頭應為:status
而不是:status:
最后,我沒有對發出請求的同一流ID進行響應,而是嘗試打開一個新的流。
因此,我應該遍歷請求幀,查找流ID,並將其用於寫出響應頭。
// 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()
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.