mirror of
https://github.com/SagerNet/sing-box.git
synced 2026-04-11 17:47:20 +10:00
206 lines
3.7 KiB
Go
206 lines
3.7 KiB
Go
package transport
|
|
|
|
import (
|
|
"context"
|
|
"net"
|
|
"sync"
|
|
)
|
|
|
|
type ConnectorCallbacks[T any] struct {
|
|
IsClosed func(connection T) bool
|
|
Close func(connection T)
|
|
Reset func(connection T)
|
|
}
|
|
|
|
type Connector[T any] struct {
|
|
dial func(ctx context.Context) (T, error)
|
|
callbacks ConnectorCallbacks[T]
|
|
|
|
access sync.Mutex
|
|
connection T
|
|
hasConnection bool
|
|
connecting chan struct{}
|
|
|
|
closeCtx context.Context
|
|
closed bool
|
|
}
|
|
|
|
func NewConnector[T any](closeCtx context.Context, dial func(context.Context) (T, error), callbacks ConnectorCallbacks[T]) *Connector[T] {
|
|
return &Connector[T]{
|
|
dial: dial,
|
|
callbacks: callbacks,
|
|
closeCtx: closeCtx,
|
|
}
|
|
}
|
|
|
|
func NewSingleflightConnector(closeCtx context.Context, dial func(context.Context) (*Connection, error)) *Connector[*Connection] {
|
|
return NewConnector(closeCtx, dial, ConnectorCallbacks[*Connection]{
|
|
IsClosed: func(connection *Connection) bool {
|
|
return connection.IsClosed()
|
|
},
|
|
Close: func(connection *Connection) {
|
|
connection.CloseWithError(ErrTransportClosed)
|
|
},
|
|
Reset: func(connection *Connection) {
|
|
connection.CloseWithError(ErrConnectionReset)
|
|
},
|
|
})
|
|
}
|
|
|
|
func (c *Connector[T]) Get(ctx context.Context) (T, error) {
|
|
var zero T
|
|
for {
|
|
c.access.Lock()
|
|
|
|
if c.closed {
|
|
c.access.Unlock()
|
|
return zero, ErrTransportClosed
|
|
}
|
|
|
|
if c.hasConnection && !c.callbacks.IsClosed(c.connection) {
|
|
connection := c.connection
|
|
c.access.Unlock()
|
|
return connection, nil
|
|
}
|
|
|
|
c.hasConnection = false
|
|
|
|
if c.connecting != nil {
|
|
connecting := c.connecting
|
|
c.access.Unlock()
|
|
|
|
select {
|
|
case <-connecting:
|
|
continue
|
|
case <-ctx.Done():
|
|
return zero, ctx.Err()
|
|
case <-c.closeCtx.Done():
|
|
return zero, ErrTransportClosed
|
|
}
|
|
}
|
|
|
|
c.connecting = make(chan struct{})
|
|
c.access.Unlock()
|
|
|
|
connection, err := c.dialWithCancellation(ctx)
|
|
|
|
c.access.Lock()
|
|
close(c.connecting)
|
|
c.connecting = nil
|
|
|
|
if err != nil {
|
|
c.access.Unlock()
|
|
return zero, err
|
|
}
|
|
|
|
if c.closed {
|
|
c.callbacks.Close(connection)
|
|
c.access.Unlock()
|
|
return zero, ErrTransportClosed
|
|
}
|
|
|
|
c.connection = connection
|
|
c.hasConnection = true
|
|
result := c.connection
|
|
c.access.Unlock()
|
|
|
|
return result, nil
|
|
}
|
|
}
|
|
|
|
func (c *Connector[T]) dialWithCancellation(ctx context.Context) (T, error) {
|
|
dialCtx, cancel := context.WithCancel(ctx)
|
|
defer cancel()
|
|
|
|
go func() {
|
|
select {
|
|
case <-c.closeCtx.Done():
|
|
cancel()
|
|
case <-dialCtx.Done():
|
|
}
|
|
}()
|
|
|
|
return c.dial(dialCtx)
|
|
}
|
|
|
|
func (c *Connector[T]) Close() error {
|
|
c.access.Lock()
|
|
defer c.access.Unlock()
|
|
|
|
if c.closed {
|
|
return nil
|
|
}
|
|
c.closed = true
|
|
|
|
if c.hasConnection {
|
|
c.callbacks.Close(c.connection)
|
|
c.hasConnection = false
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (c *Connector[T]) Reset() {
|
|
c.access.Lock()
|
|
defer c.access.Unlock()
|
|
|
|
if c.hasConnection {
|
|
c.callbacks.Reset(c.connection)
|
|
c.hasConnection = false
|
|
}
|
|
}
|
|
|
|
type Connection struct {
|
|
net.Conn
|
|
|
|
closeOnce sync.Once
|
|
done chan struct{}
|
|
closeError error
|
|
}
|
|
|
|
func WrapConnection(conn net.Conn) *Connection {
|
|
return &Connection{
|
|
Conn: conn,
|
|
done: make(chan struct{}),
|
|
}
|
|
}
|
|
|
|
func (c *Connection) Done() <-chan struct{} {
|
|
return c.done
|
|
}
|
|
|
|
func (c *Connection) IsClosed() bool {
|
|
select {
|
|
case <-c.done:
|
|
return true
|
|
default:
|
|
return false
|
|
}
|
|
}
|
|
|
|
func (c *Connection) CloseError() error {
|
|
select {
|
|
case <-c.done:
|
|
if c.closeError != nil {
|
|
return c.closeError
|
|
}
|
|
return ErrTransportClosed
|
|
default:
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func (c *Connection) Close() error {
|
|
return c.CloseWithError(ErrTransportClosed)
|
|
}
|
|
|
|
func (c *Connection) CloseWithError(err error) error {
|
|
var returnError error
|
|
c.closeOnce.Do(func() {
|
|
c.closeError = err
|
|
returnError = c.Conn.Close()
|
|
close(c.done)
|
|
})
|
|
return returnError
|
|
}
|