123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121 |
- package tor
- import (
- std_bufio "bufio"
- "context"
- "crypto/rand"
- "encoding/hex"
- "net"
- "github.com/sagernet/sing-box/adapter"
- "github.com/sagernet/sing-box/log"
- "github.com/sagernet/sing/common"
- "github.com/sagernet/sing/common/auth"
- E "github.com/sagernet/sing/common/exceptions"
- M "github.com/sagernet/sing/common/metadata"
- N "github.com/sagernet/sing/common/network"
- "github.com/sagernet/sing/protocol/socks"
- "github.com/sagernet/sing/service"
- )
- type ProxyListener struct {
- ctx context.Context
- logger log.ContextLogger
- dialer N.Dialer
- connection adapter.ConnectionManager
- tcpListener *net.TCPListener
- username string
- password string
- authenticator *auth.Authenticator
- }
- func NewProxyListener(ctx context.Context, logger log.ContextLogger, dialer N.Dialer) *ProxyListener {
- var usernameB [64]byte
- var passwordB [64]byte
- rand.Read(usernameB[:])
- rand.Read(passwordB[:])
- username := hex.EncodeToString(usernameB[:])
- password := hex.EncodeToString(passwordB[:])
- return &ProxyListener{
- ctx: ctx,
- logger: logger,
- dialer: dialer,
- connection: service.FromContext[adapter.ConnectionManager](ctx),
- authenticator: auth.NewAuthenticator([]auth.User{{Username: username, Password: password}}),
- username: username,
- password: password,
- }
- }
- func (l *ProxyListener) Start() error {
- tcpListener, err := net.ListenTCP("tcp", &net.TCPAddr{
- IP: net.IPv4(127, 0, 0, 1),
- })
- if err != nil {
- return err
- }
- l.tcpListener = tcpListener
- go l.acceptLoop()
- return nil
- }
- func (l *ProxyListener) Port() uint16 {
- if l.tcpListener == nil {
- panic("start listener first")
- }
- return M.SocksaddrFromNet(l.tcpListener.Addr()).Port
- }
- func (l *ProxyListener) Username() string {
- return l.username
- }
- func (l *ProxyListener) Password() string {
- return l.password
- }
- func (l *ProxyListener) Close() error {
- return common.Close(l.tcpListener)
- }
- func (l *ProxyListener) acceptLoop() {
- for {
- tcpConn, err := l.tcpListener.AcceptTCP()
- if err != nil {
- return
- }
- ctx := log.ContextWithNewID(l.ctx)
- go func() {
- hErr := l.accept(ctx, tcpConn)
- if hErr != nil {
- if E.IsClosedOrCanceled(hErr) {
- l.logger.DebugContext(ctx, E.Cause(hErr, "proxy connection closed"))
- return
- }
- l.logger.ErrorContext(ctx, E.Cause(hErr, "proxy"))
- }
- }()
- }
- }
- func (l *ProxyListener) accept(ctx context.Context, conn *net.TCPConn) error {
- return socks.HandleConnectionEx(ctx, conn, std_bufio.NewReader(conn), l.authenticator, nil, l, M.SocksaddrFromNet(conn.RemoteAddr()), M.Socksaddr{}, nil)
- }
- func (l *ProxyListener) NewConnectionEx(ctx context.Context, conn net.Conn, source M.Socksaddr, destination M.Socksaddr, onClose N.CloseHandlerFunc) {
- var metadata adapter.InboundContext
- metadata.Source = source
- metadata.Destination = destination
- metadata.Network = N.NetworkTCP
- l.logger.InfoContext(ctx, "proxy connection to ", metadata.Destination)
- l.connection.NewConnection(ctx, l.dialer, conn, metadata, onClose)
- }
- func (l *ProxyListener) NewPacketConnectionEx(ctx context.Context, conn N.PacketConn, source M.Socksaddr, destination M.Socksaddr, onClose N.CloseHandlerFunc) {
- var metadata adapter.InboundContext
- metadata.Source = source
- metadata.Destination = destination
- metadata.Network = N.NetworkUDP
- l.logger.InfoContext(ctx, "proxy packet connection to ", metadata.Destination)
- l.connection.NewPacketConnection(ctx, l.dialer, conn, metadata, onClose)
- }
|