forked from monzo/typhon
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathclient.go
133 lines (116 loc) · 4.12 KB
/
client.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
package typhon
import (
"context"
"crypto/tls"
"net"
"net/http"
"time"
"github.com/monzo/terrors"
"golang.org/x/net/http2"
)
var (
// Client is used to send all requests by default. It can be overridden globally but MUST only be done before use
// takes place; access is not synchronised.
Client Service = BareClient
// RoundTripper chooses HTTP1, or H2C based on a context flag (see WithH2C)
RoundTripper http.RoundTripper = dynamicRoundTripper{}
// HTTPRoundTripper is a HTTP1 and TLS HTTP2 client
HTTPRoundTripper http.RoundTripper = &http.Transport{
Proxy: http.ProxyFromEnvironment,
DisableKeepAlives: false,
DisableCompression: false,
IdleConnTimeout: 10 * time.Minute,
MaxIdleConnsPerHost: 10,
}
// H2cRoundTripper is a prior-knowledge H2c client. It does not support ProxyFromEnvironment.
H2cRoundTripper http.RoundTripper = &http2.Transport{
AllowHTTP: true,
// This monstrosity is needed to get the http2 Transport to dial over cleartext.
// See https://github.com/thrawn01/h2c-golang-example
DialTLSContext: func(ctx context.Context, network, addr string, _ *tls.Config) (net.Conn, error) {
var d net.Dialer
return d.DialContext(ctx, network, addr)
},
}
)
// A ResponseFuture is a container for a Response which will materialise at some point.
type ResponseFuture struct {
done <-chan struct{} // guards access to r
r Response
}
// WaitC returns a channel which can be waited upon until the response is available
func (f *ResponseFuture) WaitC() <-chan struct{} {
return f.done
}
// Response provides access to the response object, blocking until it is available
func (f *ResponseFuture) Response() Response {
<-f.WaitC()
return f.r
}
// HttpService returns a Service which sends requests via the given net/http RoundTripper.
// Only use this if you need to do something custom at the transport level.
func HttpService(rt http.RoundTripper) Service {
return func(req Request) Response {
ctx := req.unwrappedContext()
httpRsp, err := rt.RoundTrip(req.Request.WithContext(ctx))
// When the calling context is cancelled, close the response body
// This protects callers that forget to call Close(), or those which proxy responses upstream
//
// If the calling context is infinite (ie. returns nil for Done()), it can never signal cancellation
// so we bypass this as a performance optimisation
if httpRsp != nil && httpRsp.Body != nil && ctx.Done() != nil {
body := newDoneReader(httpRsp.Body, httpRsp.ContentLength)
httpRsp.Body = body
go func() {
select {
case <-body.closed:
case <-ctx.Done():
body.Close()
}
}()
}
return Response{
Request: &req,
Response: httpRsp,
Error: terrors.Wrap(err, nil)}
}
}
// BareClient is the most basic way to send a request, using the default http RoundTripper
func BareClient(req Request) Response {
return HttpService(RoundTripper)(req)
}
// SendVia round-trips the request via the passed Service. It does not block, instead returning a ResponseFuture
// representing the asynchronous operation to produce the response.
func SendVia(req Request, svc Service) *ResponseFuture {
done := make(chan struct{}, 0)
f := &ResponseFuture{
done: done}
go func() {
defer close(done) // makes the response available to waiters
f.r = svc(req)
}()
return f
}
// Send round-trips the request via the default Client. It does not block, instead returning a ResponseFuture
// representing the asynchronous operation to produce the response. It is equivalent to:
//
// SendVia(req, Client)
func Send(req Request) *ResponseFuture {
return SendVia(req, Client)
}
type withH2C struct{}
// WithH2C instructs the dynamicRoundTripper to use prior-knowledge cleartext HTTP2 instead of HTTP1.1
func WithH2C(ctx context.Context) context.Context {
return context.WithValue(ctx, withH2C{}, true)
}
func isH2C(ctx context.Context) bool {
b, _ := ctx.Value(withH2C{}).(bool)
return b
}
type dynamicRoundTripper struct{}
func (d dynamicRoundTripper) RoundTrip(r *http.Request) (*http.Response, error) {
if r.URL.Scheme == "http" && isH2C(r.Context()) {
return H2cRoundTripper.RoundTrip(r)
}
return HTTPRoundTripper.RoundTrip(r)
}