package proxy import ( "net" "net/http" "sync" tls "github.com/refraction-networking/utls" "github.com/rs/zerolog/log" "golang.org/x/net/http2" ) type utlsRoundTripper struct { mu sync.Mutex connections map[string]*http2.ClientConn pending map[string]*sync.Cond } func newUtlsRoundTripper() *utlsRoundTripper { return &utlsRoundTripper{ connections: make(map[string]*http2.ClientConn), pending: make(map[string]*sync.Cond), } } func (t *utlsRoundTripper) getOrCreateConnection(host, addr string) (*http2.ClientConn, error) { t.mu.Lock() if h2Conn, ok := t.connections[host]; ok && h2Conn.CanTakeNewRequest() { t.mu.Unlock() return h2Conn, nil } if cond, ok := t.pending[host]; ok { cond.Wait() if h2Conn, ok := t.connections[host]; ok && h2Conn.CanTakeNewRequest() { t.mu.Unlock() return h2Conn, nil } } cond := sync.NewCond(&t.mu) t.pending[host] = cond t.mu.Unlock() h2Conn, err := t.createConnection(host, addr) t.mu.Lock() defer t.mu.Unlock() delete(t.pending, host) cond.Broadcast() if err != nil { return nil, err } t.connections[host] = h2Conn return h2Conn, nil } func (t *utlsRoundTripper) createConnection(host, addr string) (*http2.ClientConn, error) { conn, err := net.Dial("tcp", addr) if err != nil { return nil, err } tlsConfig := &tls.Config{ServerName: host} tlsConn := tls.UClient(conn, tlsConfig, tls.HelloChrome_Auto) if err := tlsConn.Handshake(); err != nil { conn.Close() return nil, err } tr := &http2.Transport{} h2Conn, err := tr.NewClientConn(tlsConn) if err != nil { tlsConn.Close() return nil, err } return h2Conn, nil } func (t *utlsRoundTripper) RoundTrip(req *http.Request) (*http.Response, error) { hostname := req.URL.Hostname() port := req.URL.Port() if port == "" { port = "443" } addr := net.JoinHostPort(hostname, port) log.Debug().Str("addr", addr).Msg("uTLS round trip") h2Conn, err := t.getOrCreateConnection(hostname, addr) if err != nil { return nil, err } resp, err := h2Conn.RoundTrip(req) if err != nil { t.mu.Lock() if cached, ok := t.connections[hostname]; ok && cached == h2Conn { delete(t.connections, hostname) } t.mu.Unlock() return nil, err } return resp, nil }