2016-06-14 20:54:08 +00:00
|
|
|
package tcp
|
|
|
|
|
|
|
|
import (
|
2016-09-30 14:53:40 +00:00
|
|
|
"crypto/tls"
|
2016-12-04 08:10:47 +00:00
|
|
|
"net"
|
|
|
|
"v2ray.com/core/common/errors"
|
2016-08-20 18:55:45 +00:00
|
|
|
"v2ray.com/core/common/log"
|
|
|
|
v2net "v2ray.com/core/common/net"
|
|
|
|
"v2ray.com/core/transport/internet"
|
2016-11-24 22:16:05 +00:00
|
|
|
"v2ray.com/core/transport/internet/internal"
|
2016-10-02 21:43:58 +00:00
|
|
|
v2tls "v2ray.com/core/transport/internet/tls"
|
2016-06-14 20:54:08 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2016-11-24 22:16:05 +00:00
|
|
|
globalCache = internal.NewConnectionPool()
|
2016-06-14 20:54:08 +00:00
|
|
|
)
|
|
|
|
|
2016-09-30 14:53:40 +00:00
|
|
|
func Dial(src v2net.Address, dest v2net.Destination, options internet.DialerOptions) (internet.Connection, error) {
|
2016-11-22 07:23:03 +00:00
|
|
|
log.Info("Internet|TCP: Dailing TCP to ", dest)
|
2016-06-14 20:54:08 +00:00
|
|
|
if src == nil {
|
|
|
|
src = v2net.AnyIP
|
|
|
|
}
|
2016-10-02 21:43:58 +00:00
|
|
|
networkSettings, err := options.Stream.GetEffectiveNetworkSettings()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
tcpSettings := networkSettings.(*Config)
|
|
|
|
|
2016-11-24 22:16:05 +00:00
|
|
|
id := internal.NewConnectionId(src, dest)
|
2016-06-14 20:54:08 +00:00
|
|
|
var conn net.Conn
|
2016-10-17 12:35:13 +00:00
|
|
|
if dest.Network == v2net.Network_TCP && tcpSettings.ConnectionReuse.IsEnabled() {
|
2016-06-14 20:54:08 +00:00
|
|
|
conn = globalCache.Get(id)
|
|
|
|
}
|
|
|
|
if conn == nil {
|
|
|
|
var err error
|
|
|
|
conn, err = internet.DialToDest(src, dest)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2016-10-31 23:41:46 +00:00
|
|
|
if options.Stream != nil && options.Stream.HasSecuritySettings() {
|
|
|
|
securitySettings, err := options.Stream.GetEffectiveSecuritySettings()
|
|
|
|
if err != nil {
|
|
|
|
log.Error("TCP: Failed to get security settings: ", err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
tlsConfig, ok := securitySettings.(*v2tls.Config)
|
|
|
|
if ok {
|
|
|
|
config := tlsConfig.GetTLSConfig()
|
|
|
|
if dest.Address.Family().IsDomain() {
|
|
|
|
config.ServerName = dest.Address.Domain()
|
|
|
|
}
|
|
|
|
conn = tls.Client(conn, config)
|
2016-10-16 12:22:21 +00:00
|
|
|
}
|
2016-09-30 14:53:40 +00:00
|
|
|
}
|
2016-11-02 21:26:21 +00:00
|
|
|
if tcpSettings.HeaderSettings != nil {
|
|
|
|
headerConfig, err := tcpSettings.HeaderSettings.GetInstance()
|
|
|
|
if err != nil {
|
2016-12-04 08:43:33 +00:00
|
|
|
return nil, errors.Base(err).Message("Interent|TCP: Failed to get header settings.")
|
2016-11-02 21:26:21 +00:00
|
|
|
}
|
|
|
|
auth, err := internet.CreateConnectionAuthenticator(tcpSettings.HeaderSettings.Type, headerConfig)
|
|
|
|
if err != nil {
|
2016-12-04 08:43:33 +00:00
|
|
|
return nil, errors.Base(err).Message("Internet|TCP: Failed to create header authenticator.")
|
2016-11-02 21:26:21 +00:00
|
|
|
}
|
|
|
|
conn = auth.Client(conn)
|
|
|
|
}
|
2016-09-30 14:53:40 +00:00
|
|
|
}
|
2016-10-02 21:43:58 +00:00
|
|
|
return NewConnection(id, conn, globalCache, tcpSettings), nil
|
2016-06-14 20:54:08 +00:00
|
|
|
}
|
|
|
|
|
2016-09-30 14:53:40 +00:00
|
|
|
func DialRaw(src v2net.Address, dest v2net.Destination, options internet.DialerOptions) (internet.Connection, error) {
|
2016-11-22 07:23:03 +00:00
|
|
|
log.Info("Internet|TCP: Dailing Raw TCP to ", dest)
|
2016-06-14 20:54:08 +00:00
|
|
|
conn, err := internet.DialToDest(src, dest)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2016-09-30 14:53:40 +00:00
|
|
|
// TODO: handle dialer options
|
2016-06-14 20:54:08 +00:00
|
|
|
return &RawConnection{
|
|
|
|
TCPConn: *conn.(*net.TCPConn),
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
internet.TCPDialer = Dial
|
|
|
|
internet.RawTCPDialer = DialRaw
|
|
|
|
}
|