1
0
mirror of https://github.com/v2fly/v2ray-core.git synced 2025-01-23 09:46:55 -05:00
v2fly/transport/internet/http/dialer.go

165 lines
4.5 KiB
Go
Raw Normal View History

2018-03-01 07:16:52 -05:00
package http
import (
"context"
gotls "crypto/tls"
gonet "net"
2018-03-01 07:16:52 -05:00
"net/http"
"net/url"
"sync"
"golang.org/x/net/http2"
2021-02-16 15:31:50 -05:00
core "github.com/v2fly/v2ray-core/v5"
"github.com/v2fly/v2ray-core/v5/common"
"github.com/v2fly/v2ray-core/v5/common/buf"
"github.com/v2fly/v2ray-core/v5/common/net"
"github.com/v2fly/v2ray-core/v5/transport/internet"
"github.com/v2fly/v2ray-core/v5/transport/internet/security"
"github.com/v2fly/v2ray-core/v5/transport/pipe"
2018-03-01 07:16:52 -05:00
)
var (
2018-03-21 18:28:28 -04:00
globalDialerMap map[net.Destination]*http.Client
globalDialerAccess sync.Mutex
2018-03-01 07:16:52 -05:00
)
type dialerCanceller func()
func getHTTPClient(ctx context.Context, dest net.Destination, securityEngine *security.Engine, streamSettings *internet.MemoryStreamConfig) (*http.Client, dialerCanceller) {
globalDialerAccess.Lock()
defer globalDialerAccess.Unlock()
2018-03-01 07:16:52 -05:00
canceller := func() {
globalDialerAccess.Lock()
defer globalDialerAccess.Unlock()
delete(globalDialerMap, dest)
}
2018-03-21 18:28:28 -04:00
if globalDialerMap == nil {
globalDialerMap = make(map[net.Destination]*http.Client)
}
2018-03-01 07:16:52 -05:00
if client, found := globalDialerMap[dest]; found {
return client, canceller
2018-03-01 07:16:52 -05:00
}
transport := &http2.Transport{
2023-10-24 10:36:09 -04:00
DialTLSContext: func(_ context.Context, network, addr string, tlsConfig *gotls.Config) (gonet.Conn, error) {
2018-03-01 07:16:52 -05:00
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)
detachedContext := core.ToBackgroundDetachedContext(ctx)
pconn, err := internet.DialSystem(detachedContext, net.TCPDestination(address, port), streamSettings.SocketSettings)
2018-03-01 07:16:52 -05:00
if err != nil {
return nil, err
}
cn, err := (*securityEngine).Client(pconn,
security.OptionWithDestination{Dest: dest})
if err != nil {
return nil, err
}
protocol := ""
if connAPLNGetter, ok := cn.(security.ConnectionApplicationProtocol); ok {
connectionALPN, err := connAPLNGetter.GetConnectionApplicationProtocol()
if err != nil {
return nil, newError("failed to get connection ALPN").Base(err).AtWarning()
}
protocol = connectionALPN
}
if protocol != http2.NextProtoTLS {
return nil, newError("http2: unexpected ALPN protocol " + protocol + "; want q" + http2.NextProtoTLS).AtError()
}
return cn, nil
2018-03-01 07:16:52 -05:00
},
}
client := &http.Client{
Transport: transport,
}
globalDialerMap[dest] = client
return client, canceller
2018-03-01 07:16:52 -05:00
}
// Dial dials a new TCP connection to the given destination.
func Dial(ctx context.Context, dest net.Destination, streamSettings *internet.MemoryStreamConfig) (internet.Connection, error) {
httpSettings := streamSettings.ProtocolSettings.(*Config)
securityEngine, err := security.CreateSecurityEngineFromSettings(ctx, streamSettings)
if err != nil {
return nil, newError("unable to create security engine").Base(err)
}
if securityEngine == nil {
return nil, newError("TLS must be enabled for http transport.").AtWarning()
2018-03-01 08:22:33 -05:00
}
client, canceller := getHTTPClient(ctx, dest, &securityEngine, streamSettings)
2018-03-01 07:16:52 -05:00
2018-05-25 06:08:28 -04:00
opts := pipe.OptionsFromContext(ctx)
preader, pwriter := pipe.New(opts...)
2018-04-20 18:54:53 -04:00
breader := &buf.BufferedReader{Reader: preader}
httpMethod := "PUT"
if httpSettings.Method != "" {
httpMethod = httpSettings.Method
}
httpHeaders := make(http.Header)
for _, httpHeader := range httpSettings.Header {
for _, httpHeaderValue := range httpHeader.Value {
httpHeaders.Set(httpHeader.Name, httpHeaderValue)
}
}
2018-03-01 07:16:52 -05:00
request := &http.Request{
Method: httpMethod,
2018-03-01 08:22:33 -05:00
Host: httpSettings.getRandomHost(),
2018-04-20 18:54:53 -04:00
Body: breader,
2018-03-01 07:16:52 -05:00
URL: &url.URL{
Scheme: "https",
Host: dest.NetAddr(),
2018-03-01 08:22:33 -05:00
Path: httpSettings.getNormalizedPath(),
2018-03-01 07:16:52 -05:00
},
Proto: "HTTP/2",
ProtoMajor: 2,
ProtoMinor: 0,
Header: httpHeaders,
2018-03-01 07:16:52 -05:00
}
// Disable any compression method from server.
request.Header.Set("Accept-Encoding", "identity")
response, err := client.Do(request) // nolint: bodyclose
2018-03-01 07:16:52 -05:00
if err != nil {
canceller()
2018-03-01 07:16:52 -05:00
return nil, newError("failed to dial to ", dest).Base(err).AtWarning()
}
if response.StatusCode != 200 {
return nil, newError("unexpected status", response.StatusCode).AtWarning()
}
2018-04-16 08:57:13 -04:00
bwriter := buf.NewBufferedWriter(pwriter)
common.Must(bwriter.SetBuffered(false))
return net.NewConnection(
net.ConnectionOutput(response.Body),
net.ConnectionInput(bwriter),
2019-01-01 14:16:04 -05:00
net.ConnectionOnClose(common.ChainedClosable{breader, bwriter, response.Body}),
), nil
2018-03-01 07:16:52 -05:00
}
func init() {
2018-08-06 07:48:35 -04:00
common.Must(internet.RegisterTransportDialer(protocolName, Dial))
2018-03-01 07:16:52 -05:00
}