123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110 |
- //go:build with_quic
- package v2rayquic
- import (
- "context"
- "net"
- "sync"
- "github.com/sagernet/quic-go"
- "github.com/sagernet/quic-go/http3"
- "github.com/sagernet/sing-box/adapter"
- "github.com/sagernet/sing-box/common/tls"
- C "github.com/sagernet/sing-box/constant"
- "github.com/sagernet/sing-box/option"
- "github.com/sagernet/sing-quic"
- "github.com/sagernet/sing/common"
- "github.com/sagernet/sing/common/bufio"
- M "github.com/sagernet/sing/common/metadata"
- N "github.com/sagernet/sing/common/network"
- )
- var _ adapter.V2RayClientTransport = (*Client)(nil)
- type Client struct {
- ctx context.Context
- dialer N.Dialer
- serverAddr M.Socksaddr
- tlsConfig tls.Config
- quicConfig *quic.Config
- connAccess sync.Mutex
- conn common.TypedValue[*quic.Conn]
- rawConn net.Conn
- }
- func NewClient(ctx context.Context, dialer N.Dialer, serverAddr M.Socksaddr, options option.V2RayQUICOptions, tlsConfig tls.Config) (adapter.V2RayClientTransport, error) {
- quicConfig := &quic.Config{
- DisablePathMTUDiscovery: !C.IsLinux && !C.IsWindows,
- }
- if len(tlsConfig.NextProtos()) == 0 {
- tlsConfig.SetNextProtos([]string{http3.NextProtoH3})
- }
- return &Client{
- ctx: ctx,
- dialer: dialer,
- serverAddr: serverAddr,
- tlsConfig: tlsConfig,
- quicConfig: quicConfig,
- }, nil
- }
- func (c *Client) offer() (*quic.Conn, error) {
- conn := c.conn.Load()
- if conn != nil && !common.Done(conn.Context()) {
- return conn, nil
- }
- c.connAccess.Lock()
- defer c.connAccess.Unlock()
- conn = c.conn.Load()
- if conn != nil && !common.Done(conn.Context()) {
- return conn, nil
- }
- conn, err := c.offerNew()
- if err != nil {
- return nil, err
- }
- return conn, nil
- }
- func (c *Client) offerNew() (*quic.Conn, error) {
- udpConn, err := c.dialer.DialContext(c.ctx, "udp", c.serverAddr)
- if err != nil {
- return nil, err
- }
- packetConn := bufio.NewUnbindPacketConn(udpConn)
- quicConn, err := qtls.Dial(c.ctx, packetConn, udpConn.RemoteAddr(), c.tlsConfig, c.quicConfig)
- if err != nil {
- packetConn.Close()
- return nil, err
- }
- c.conn.Store(quicConn)
- c.rawConn = udpConn
- return quicConn, nil
- }
- func (c *Client) DialContext(ctx context.Context) (net.Conn, error) {
- conn, err := c.offer()
- if err != nil {
- return nil, err
- }
- stream, err := conn.OpenStream()
- if err != nil {
- return nil, err
- }
- return &StreamWrapper{Conn: conn, Stream: stream}, nil
- }
- func (c *Client) Close() error {
- c.connAccess.Lock()
- defer c.connAccess.Unlock()
- conn := c.conn.Swap(nil)
- if conn != nil {
- conn.CloseWithError(0, "")
- }
- if c.rawConn != nil {
- c.rawConn.Close()
- }
- c.rawConn = nil
- return nil
- }
|