| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125 | 
							- package http
 
- import (
 
- 	"context"
 
- 	gotls "crypto/tls"
 
- 	"net/http"
 
- 	"net/url"
 
- 	"sync"
 
- 	"golang.org/x/net/http2"
 
- 	"v2ray.com/core/common"
 
- 	"v2ray.com/core/common/buf"
 
- 	"v2ray.com/core/common/net"
 
- 	"v2ray.com/core/transport/internet"
 
- 	"v2ray.com/core/transport/internet/tls"
 
- 	"v2ray.com/core/transport/pipe"
 
- )
 
- var (
 
- 	globalDialerMap    map[net.Destination]*http.Client
 
- 	globalDailerAccess sync.Mutex
 
- )
 
- func getHTTPClient(ctx context.Context, dest net.Destination) (*http.Client, error) {
 
- 	globalDailerAccess.Lock()
 
- 	defer globalDailerAccess.Unlock()
 
- 	if globalDialerMap == nil {
 
- 		globalDialerMap = make(map[net.Destination]*http.Client)
 
- 	}
 
- 	if client, found := globalDialerMap[dest]; found {
 
- 		return client, nil
 
- 	}
 
- 	config := tls.ConfigFromContext(ctx)
 
- 	if config == nil {
 
- 		return nil, newError("TLS must be enabled for http transport.").AtWarning()
 
- 	}
 
- 	transport := &http2.Transport{
 
- 		DialTLS: func(network string, addr string, tlsConfig *gotls.Config) (net.Conn, error) {
 
- 			rawHost, rawPort, err := net.SplitHostPort(addr)
 
- 			if err != nil {
 
- 				return nil, err
 
- 			}
 
- 			if len(rawPort) == 0 {
 
- 				rawPort = "443"
 
- 			}
 
- 			port, err := net.PortFromString(rawPort)
 
- 			if err != nil {
 
- 				return nil, err
 
- 			}
 
- 			address := net.ParseAddress(rawHost)
 
- 			pconn, err := internet.DialSystem(context.Background(), nil, net.TCPDestination(address, port))
 
- 			if err != nil {
 
- 				return nil, err
 
- 			}
 
- 			return gotls.Client(pconn, tlsConfig), nil
 
- 		},
 
- 		TLSClientConfig: config.GetTLSConfig(tls.WithDestination(dest), tls.WithNextProto("h2")),
 
- 	}
 
- 	client := &http.Client{
 
- 		Transport: transport,
 
- 	}
 
- 	globalDialerMap[dest] = client
 
- 	return client, nil
 
- }
 
- // Dial dials a new TCP connection to the given destination.
 
- func Dial(ctx context.Context, dest net.Destination) (internet.Connection, error) {
 
- 	rawSettings := internet.TransportSettingsFromContext(ctx)
 
- 	httpSettings, ok := rawSettings.(*Config)
 
- 	if !ok {
 
- 		return nil, newError("HTTP config is not set.").AtError()
 
- 	}
 
- 	client, err := getHTTPClient(ctx, dest)
 
- 	if err != nil {
 
- 		return nil, err
 
- 	}
 
- 	opts := pipe.OptionsFromContext(ctx)
 
- 	preader, pwriter := pipe.New(opts...)
 
- 	breader := &buf.BufferedReader{Reader: preader}
 
- 	request := &http.Request{
 
- 		Method: "PUT",
 
- 		Host:   httpSettings.getRandomHost(),
 
- 		Body:   breader,
 
- 		URL: &url.URL{
 
- 			Scheme: "https",
 
- 			Host:   dest.NetAddr(),
 
- 			Path:   httpSettings.getNormalizedPath(),
 
- 		},
 
- 		Proto:      "HTTP/2",
 
- 		ProtoMajor: 2,
 
- 		ProtoMinor: 0,
 
- 		Header:     make(http.Header),
 
- 	}
 
- 	// Disable any compression method from server.
 
- 	request.Header.Set("Accept-Encoding", "identity")
 
- 	response, err := client.Do(request)
 
- 	if err != nil {
 
- 		return nil, newError("failed to dial to ", dest).Base(err).AtWarning()
 
- 	}
 
- 	if response.StatusCode != 200 {
 
- 		return nil, newError("unexpected status", response.StatusCode).AtWarning()
 
- 	}
 
- 	bwriter := buf.NewBufferedWriter(pwriter)
 
- 	common.Must(bwriter.SetBuffered(false))
 
- 	return net.NewConnection(
 
- 		net.ConnectionOutput(response.Body),
 
- 		net.ConnectionInput(bwriter),
 
- 		net.ConnectionOnClose(common.NewChainedClosable(breader, bwriter, response.Body)),
 
- 	), nil
 
- }
 
- func init() {
 
- 	common.Must(internet.RegisterTransportDialer(internet.TransportProtocol_HTTP, Dial))
 
- }
 
 
  |