1
0
mirror of https://github.com/v2fly/v2ray-core.git synced 2025-01-05 00:47:51 -05:00
v2fly/transport/internet/quic/dialer.go

221 lines
4.6 KiB
Go
Raw Normal View History

2018-11-21 10:47:06 -05:00
package quic
import (
"context"
"sync"
2018-11-22 05:57:17 -05:00
"time"
2018-11-21 10:47:06 -05:00
"github.com/lucas-clemente/quic-go"
2021-02-16 15:31:50 -05:00
"github.com/v2fly/v2ray-core/v5/common"
"github.com/v2fly/v2ray-core/v5/common/net"
"github.com/v2fly/v2ray-core/v5/common/task"
"github.com/v2fly/v2ray-core/v5/transport/internet"
"github.com/v2fly/v2ray-core/v5/transport/internet/tls"
2018-11-21 10:47:06 -05:00
)
type connectionContext struct {
2018-11-26 12:00:41 -05:00
rawConn *sysConn
conn quic.Connection
2018-11-25 15:56:24 -05:00
}
var errConnectionClosed = newError("connection closed")
2018-11-25 15:56:24 -05:00
func (c *connectionContext) openStream(destAddr net.Addr) (*interConn, error) {
if !isActive(c.conn) {
return nil, errConnectionClosed
2018-11-25 15:56:24 -05:00
}
stream, err := c.conn.OpenStream()
2018-11-25 15:56:24 -05:00
if err != nil {
return nil, err
}
conn := &interConn{
2018-11-26 12:00:41 -05:00
stream: stream,
local: c.conn.LocalAddr(),
2018-11-26 12:00:41 -05:00
remote: destAddr,
2018-11-25 15:56:24 -05:00
}
return conn, nil
}
type clientConnections struct {
access sync.Mutex
conns map[net.Destination][]*connectionContext
cleanup *task.Periodic
2018-11-24 16:18:17 -05:00
}
func isActive(s quic.Connection) bool {
2018-11-24 16:18:17 -05:00
select {
case <-s.Context().Done():
return false
default:
return true
}
2018-11-23 11:04:53 -05:00
}
func removeInactiveConnections(conns []*connectionContext) []*connectionContext {
activeConnections := make([]*connectionContext, 0, len(conns))
for _, s := range conns {
if isActive(s.conn) {
activeConnections = append(activeConnections, s)
2018-11-25 15:56:24 -05:00
continue
}
if err := s.conn.CloseWithError(0, ""); err != nil {
newError("failed to close connection").Base(err).WriteToLog()
2018-11-25 15:56:24 -05:00
}
if err := s.rawConn.Close(); err != nil {
2018-11-26 12:00:41 -05:00
newError("failed to close raw connection").Base(err).WriteToLog()
2018-11-23 11:04:53 -05:00
}
}
if len(activeConnections) < len(conns) {
return activeConnections
2018-11-23 11:04:53 -05:00
}
return conns
2018-11-23 11:04:53 -05:00
}
func openStream(conns []*connectionContext, destAddr net.Addr) *interConn {
for _, s := range conns {
if !isActive(s.conn) {
2018-11-24 17:10:04 -05:00
continue
}
2018-11-25 15:56:24 -05:00
conn, err := s.openStream(destAddr)
2018-11-23 11:04:53 -05:00
if err != nil {
continue
}
2018-11-25 15:56:24 -05:00
return conn
}
return nil
}
func (s *clientConnections) cleanConnections() error {
2018-11-25 15:56:24 -05:00
s.access.Lock()
defer s.access.Unlock()
if len(s.conns) == 0 {
2018-11-25 15:56:24 -05:00
return nil
}
newConnMap := make(map[net.Destination][]*connectionContext)
2018-11-25 15:56:24 -05:00
for dest, conns := range s.conns {
conns = removeInactiveConnections(conns)
if len(conns) > 0 {
newConnMap[dest] = conns
2018-11-25 15:56:24 -05:00
}
2018-11-23 11:04:53 -05:00
}
s.conns = newConnMap
2018-11-25 15:56:24 -05:00
return nil
2018-11-21 10:47:06 -05:00
}
func (s *clientConnections) openConnection(destAddr net.Addr, config *Config, tlsConfig *tls.Config, sockopt *internet.SocketConfig) (internet.Connection, error) {
2018-11-21 10:47:06 -05:00
s.access.Lock()
defer s.access.Unlock()
if s.conns == nil {
s.conns = make(map[net.Destination][]*connectionContext)
2018-11-21 10:47:06 -05:00
}
dest := net.DestinationFromAddr(destAddr)
var conns []*connectionContext
if s, found := s.conns[dest]; found {
conns = s
2018-11-23 11:04:53 -05:00
}
{
conn := openStream(conns, destAddr)
2018-11-25 15:56:24 -05:00
if conn != nil {
return conn, nil
2018-11-24 17:10:04 -05:00
}
2018-11-21 10:47:06 -05:00
}
conns = removeInactiveConnections(conns)
2022-04-06 11:48:16 -04:00
newError("dialing QUIC to ", dest).WriteToLog()
2018-11-24 17:10:04 -05:00
2018-11-21 16:02:19 -05:00
rawConn, err := internet.ListenSystemPacket(context.Background(), &net.UDPAddr{
2018-11-21 10:47:06 -05:00
IP: []byte{0, 0, 0, 0},
Port: 0,
}, sockopt)
if err != nil {
return nil, err
}
2018-11-21 16:02:19 -05:00
quicConfig := &quic.Config{
ConnectionIDLength: 12,
HandshakeIdleTimeout: time.Second * 8,
MaxIdleTimeout: time.Second * 30,
KeepAlive: true,
2018-11-21 10:47:06 -05:00
}
sysConn, err := wrapSysConn(rawConn.(*net.UDPConn), config)
2018-11-21 10:47:06 -05:00
if err != nil {
2018-11-21 16:02:19 -05:00
rawConn.Close()
return nil, err
}
conn, err := quic.DialContext(context.Background(), sysConn, destAddr, "", tlsConfig.GetTLSConfig(tls.WithDestination(dest)), quicConfig)
2018-11-21 16:02:19 -05:00
if err != nil {
sysConn.Close()
2018-11-21 10:47:06 -05:00
return nil, err
}
context := &connectionContext{
conn: conn,
rawConn: sysConn,
2018-11-23 11:04:53 -05:00
}
s.conns[dest] = append(conns, context)
2018-11-25 15:56:24 -05:00
return context.openStream(destAddr)
2018-11-21 10:47:06 -05:00
}
var client clientConnections
2018-11-21 10:47:06 -05:00
2018-11-25 15:56:24 -05:00
func init() {
client.conns = make(map[net.Destination][]*connectionContext)
2018-11-25 15:56:24 -05:00
client.cleanup = &task.Periodic{
Interval: time.Minute,
Execute: client.cleanConnections,
2018-11-25 15:56:24 -05:00
}
common.Must(client.cleanup.Start())
}
2018-11-21 10:47:06 -05:00
func Dial(ctx context.Context, dest net.Destination, streamSettings *internet.MemoryStreamConfig) (internet.Connection, error) {
tlsConfig := tls.ConfigFromStreamSettings(streamSettings)
if tlsConfig == nil {
2018-11-21 16:02:19 -05:00
tlsConfig = &tls.Config{
ServerName: internalDomain,
AllowInsecure: true,
}
2018-11-21 10:47:06 -05:00
}
2018-11-25 15:56:24 -05:00
var destAddr *net.UDPAddr
if dest.Address.Family().IsIP() {
destAddr = &net.UDPAddr{
IP: dest.Address.IP(),
Port: int(dest.Port),
}
} else {
addr, err := net.ResolveUDPAddr("udp", dest.NetAddr())
if err != nil {
return nil, err
}
destAddr = addr
2018-11-21 10:47:06 -05:00
}
2018-11-21 16:02:19 -05:00
config := streamSettings.ProtocolSettings.(*Config)
2018-11-23 11:04:53 -05:00
return client.openConnection(destAddr, config, tlsConfig, streamSettings.SocketSettings)
2018-11-21 10:47:06 -05:00
}
func init() {
common.Must(internet.RegisterTransportDialer(protocolName, Dial))
}