github.com/yaling888/clash@v1.53.0/transport/snell/pool.go (about)

     1  package snell
     2  
     3  import (
     4  	"context"
     5  	"net"
     6  	"time"
     7  
     8  	"github.com/yaling888/clash/component/pool"
     9  	"github.com/yaling888/clash/transport/shadowsocks/shadowaead"
    10  )
    11  
    12  type Pool struct {
    13  	pool *pool.Pool[*Snell]
    14  }
    15  
    16  func (p *Pool) Get() (net.Conn, error) {
    17  	return p.GetContext(context.Background())
    18  }
    19  
    20  func (p *Pool) GetContext(ctx context.Context) (net.Conn, error) {
    21  	elm, err := p.pool.GetContext(ctx)
    22  	if err != nil {
    23  		return nil, err
    24  	}
    25  
    26  	return &PoolConn{elm, p}, nil
    27  }
    28  
    29  func (p *Pool) Put(conn *Snell) {
    30  	if err := HalfClose(conn); err != nil {
    31  		_ = conn.Close()
    32  		return
    33  	}
    34  
    35  	p.pool.Put(conn)
    36  }
    37  
    38  type PoolConn struct {
    39  	*Snell
    40  	pool *Pool
    41  }
    42  
    43  func (pc *PoolConn) Read(b []byte) (int, error) {
    44  	// save old status of reply (it mutable by Read)
    45  	reply := pc.Snell.reply
    46  
    47  	n, err := pc.Snell.Read(b)
    48  	if err == shadowaead.ErrZeroChunk {
    49  		// if reply is false, it should be client halfclose.
    50  		// ignore error and read data again.
    51  		if !reply {
    52  			pc.Snell.reply = false
    53  			return pc.Snell.Read(b)
    54  		}
    55  	}
    56  	return n, err
    57  }
    58  
    59  func (pc *PoolConn) Write(b []byte) (int, error) {
    60  	return pc.Snell.Write(b)
    61  }
    62  
    63  func (pc *PoolConn) Close() error {
    64  	// clash use SetReadDeadline to break bidirectional copy between client and server.
    65  	// reset it before reuse connection to avoid io timeout error.
    66  	_ = pc.Snell.Conn.SetReadDeadline(time.Time{})
    67  	pc.pool.Put(pc.Snell)
    68  	return nil
    69  }
    70  
    71  func NewPool(factory func(context.Context) (*Snell, error)) *Pool {
    72  	p := pool.New[*Snell](
    73  		func(ctx context.Context) (*Snell, error) {
    74  			return factory(ctx)
    75  		},
    76  		pool.WithAge[*Snell](15000),
    77  		pool.WithSize[*Snell](10),
    78  		pool.WithEvict[*Snell](func(item *Snell) {
    79  			_ = item.Close()
    80  		}),
    81  	)
    82  
    83  	return &Pool{pool: p}
    84  }