Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/posener/h2conn
HTTP2 client-server full-duplex connection
https://github.com/posener/h2conn
go golang http2
Last synced: 6 days ago
JSON representation
HTTP2 client-server full-duplex connection
- Host: GitHub
- URL: https://github.com/posener/h2conn
- Owner: posener
- License: apache-2.0
- Created: 2018-07-14T12:49:30.000Z (over 6 years ago)
- Default Branch: master
- Last Pushed: 2023-12-04T02:54:07.000Z (about 1 year ago)
- Last Synced: 2025-01-02T08:12:57.883Z (13 days ago)
- Topics: go, golang, http2
- Language: Go
- Homepage:
- Size: 60.5 KB
- Stars: 133
- Watchers: 4
- Forks: 22
- Open Issues: 3
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- awesome-hacking-lists - posener/h2conn - HTTP2 client-server full-duplex connection (Go)
README
# h2conn
`h2conn` provides HTTP2 client-server full-duplex communication connection.
[![Build Status](https://travis-ci.org/posener/h2conn.svg?branch=master)](https://travis-ci.org/posener/h2conn)
[![codecov](https://codecov.io/gh/posener/h2conn/branch/master/graph/badge.svg)](https://codecov.io/gh/posener/h2conn)
[![GoDoc](https://godoc.org/github.com/posener/h2conn?status.svg)](http://godoc.org/github.com/posener/h2conn)
[![Go Report Card](https://goreportcard.com/badge/github.com/posener/h2conn)](https://goreportcard.com/report/github.com/posener/h2conn)## Motivation
Go has a wonderful HTTP2 support that is integrated seamlessly into the HTTP1.1 implementation.
There is a nice demo on [https://http2.golang.org](https://http2.golang.org) in which you can see it in action.
The code for the demo is available [here](https://github.com/golang/net/tree/master/http2/h2demo).I became interested how HTTP2 can work with full-duplex communication,
and saw the [echo handler implementation](https://github.com/golang/net/blob/a680a1efc54dd51c040b3b5ce4939ea3cf2ea0d1/http2/h2demo/h2demo.go#L136-L164),
and a suggested client side implementation for this handler in this
[Github issue](https://github.com/golang/go/issues/13444#issuecomment-161115822).This library provides a simpler API for the same sort of "advanced usage" / "low level" / "hard core" implementation.
## Examples
Check out the [example](https://github.com/posener/h2conn/tree/master/example) directory.
### Server
On the server side, in an implementation of `http.Handler`, the `h2conn.Accept` function
can be used to get a full-duplex connection to the client.```go
type handler struct{}func (h handler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
conn, err := h2conn.Accept(w, r)
if err != nil {
log.Printf("Failed creating connection from %s: %s", r.RemoteAddr, err)
http.Error(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
return
}
defer conn.Close()
// [ Use conn ... ]
// The connection will be left open until this function will return.
// If there is a need to wait to the client to close the connection,
// we can wait on the request context: `<-r.Context().Done()`.
}
```### Client
On the client side, the `h2conn.Connect` function can be used in order to connect to an HTTP2 server
with full-duplex communication.```go
func main() {
conn, resp, err := h2conn.Connect(ctx, url, nil)
if err != nil {
log.Fatalf("Initiate conn: %s", err)
}
defer conn.Close()// Check server status code
if resp.StatusCode != http.StatusOK {
log.Fatalf("Bad status code: %d", resp.StatusCode)
}// [ Use conn ... ]
}
```### Using the Connection
The server and the client need to decide on message format.
Here are few examples that demonstrate how the client and server can communicate over the created pipe.#### 1. JSON
Sending and receiving JSON format is a very common thing to do.
```go
import "encoding/json"func main() {
// [ Create a connection ... ]
// Create an encoder and decoder from the connection
var in, out = json.NewDecoder(conn), json.NewEncoder(conn)
// Sending data into the connection using the out encoder.
// Any type can be sent - the important thing is that the other side will read with a
// variable of the same type.
// In this case, we just use a simple string.
err = out.Encode("hello")
// [ handle err ... ]
// Receiving data from the connection using the in decoder and a variable.
// Any type can be received - the important thing is that the other side will write data
// to the connection of the same type.
// In this case we assume that the other side sent us a string.
var resp string
err = in.Decode(&resp)
// [ handle err, use resp ... ]
}
```#### 2. GOB
GOB is more efficient message format, but requires both client and server to be written in Go.
The example is exactly the same as in the json encoding, just switch `json` with `gob`.```go
import "encoding/gob"func main() {
// [ Create a connection ... ]
var in, out = gob.NewDecoder(conn), gob.NewEncoder(conn)
// Sending data into the connection using the out encoder.
// Any type can be sent - the important thing is that the other side will read with a
// variable of the same type.
// In this case, we just use a simple string.
err = out.Encode("hello")
// [ handle err ... ]
// Receiving data from the connection using the in decoder and a variable.
// Any type can be received - the important thing is that the other side will write data
// to the connection of the same type.
// In this case we assume that the other side sent us a string.
var resp string
err = in.Decode(&resp)
// [ handle err, use resp ... ]
}
```#### 3. Constant Buffer Size
```go
// Create constant size buffer
const bufSize = 10func main () {
// [ Create a connection ... ]
buf := make([]byte, bufSize)// Write to the connection:
// [ Write something to buf... ]
_, err = conn.Write(buf)// Read from the connection:
_, err = conn.Read(buf)
// [ Use buf... ]
}
```