package internal import ( "net" "sync" "time" "v2ray.com/core/common/signal" ) // ConnectionRecyler is the interface for recycling connections. type ConnectionRecyler interface { // Put returns a connection back to a connection pool. Put(ConnectionID, net.Conn) } type NoOpConnectionRecyler struct{} func (NoOpConnectionRecyler) Put(ConnectionID, net.Conn) {} // ExpiringConnection is a connection that will expire in certain time. type ExpiringConnection struct { conn net.Conn expire time.Time } // Expired returns true if the connection has expired. func (ec *ExpiringConnection) Expired() bool { return ec.expire.Before(time.Now()) } // Pool is a connection pool. type Pool struct { sync.RWMutex connsByDest map[ConnectionID][]*ExpiringConnection cleanupToken *signal.Semaphore } // NewConnectionPool creates a new Pool. func NewConnectionPool() *Pool { p := &Pool{ connsByDest: make(map[ConnectionID][]*ExpiringConnection), cleanupToken: signal.NewSemaphore(1), } return p } // Get returns a connection with matching connection ID. Nil if not found. func (p *Pool) Get(id ConnectionID) net.Conn { p.Lock() defer p.Unlock() list, found := p.connsByDest[id] if !found { return nil } connIdx := -1 for idx, conn := range list { if !conn.Expired() { connIdx = idx break } } if connIdx == -1 { return nil } listLen := len(list) conn := list[connIdx] if connIdx != listLen-1 { list[connIdx] = list[listLen-1] } list = list[:listLen-1] p.connsByDest[id] = list return conn.conn } func (p *Pool) isEmpty() bool { p.RLock() defer p.RUnlock() return len(p.connsByDest) == 0 } func (p *Pool) cleanup() { defer p.cleanupToken.Signal() for !p.isEmpty() { time.Sleep(time.Second * 5) expiredConns := make([]net.Conn, 0, 16) p.Lock() for dest, list := range p.connsByDest { validConns := make([]*ExpiringConnection, 0, len(list)) for _, conn := range list { if conn.Expired() { expiredConns = append(expiredConns, conn.conn) } else { validConns = append(validConns, conn) } } if len(validConns) != len(list) { p.connsByDest[dest] = validConns } } p.Unlock() for _, conn := range expiredConns { conn.Close() } } } // Put implements ConnectionRecyler.Put(). func (p *Pool) Put(id ConnectionID, conn net.Conn) { expiringConn := &ExpiringConnection{ conn: conn, expire: time.Now().Add(time.Second * 4), } p.Lock() defer p.Unlock() list, found := p.connsByDest[id] if !found { list = []*ExpiringConnection{expiringConn} } else { list = append(list, expiringConn) } p.connsByDest[id] = list select { case <-p.cleanupToken.Wait(): go p.cleanup() default: } }