mirror of
https://github.com/SagerNet/sing-quic.git
synced 2025-04-06 05:17:39 +03:00
Compare commits
1 commit
main
...
v0.4.0-alp
Author | SHA1 | Date | |
---|---|---|---|
|
255ab5a070 |
8 changed files with 34 additions and 383 deletions
14
go.mod
14
go.mod
|
@ -4,9 +4,9 @@ go 1.20
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/gofrs/uuid/v5 v5.3.0
|
github.com/gofrs/uuid/v5 v5.3.0
|
||||||
github.com/sagernet/quic-go v0.49.0-beta.1
|
github.com/sagernet/quic-go v0.48.1-beta.1
|
||||||
github.com/sagernet/sing v0.6.0-beta.9
|
github.com/sagernet/sing v0.6.0-alpha.14
|
||||||
golang.org/x/crypto v0.32.0
|
golang.org/x/crypto v0.28.0
|
||||||
golang.org/x/exp v0.0.0-20240506185415-9bf2ced13842
|
golang.org/x/exp v0.0.0-20240506185415-9bf2ced13842
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -16,10 +16,8 @@ require (
|
||||||
github.com/onsi/ginkgo/v2 v2.9.7 // indirect
|
github.com/onsi/ginkgo/v2 v2.9.7 // indirect
|
||||||
github.com/quic-go/qpack v0.4.0 // indirect
|
github.com/quic-go/qpack v0.4.0 // indirect
|
||||||
github.com/quic-go/qtls-go1-20 v0.4.1 // indirect
|
github.com/quic-go/qtls-go1-20 v0.4.1 // indirect
|
||||||
golang.org/x/mod v0.20.0 // indirect
|
golang.org/x/net v0.30.0 // indirect
|
||||||
golang.org/x/net v0.34.0 // indirect
|
golang.org/x/sys v0.26.0 // indirect
|
||||||
golang.org/x/sync v0.10.0 // indirect
|
golang.org/x/text v0.19.0 // indirect
|
||||||
golang.org/x/sys v0.29.0 // indirect
|
|
||||||
golang.org/x/text v0.21.0 // indirect
|
|
||||||
golang.org/x/tools v0.24.0 // indirect
|
golang.org/x/tools v0.24.0 // indirect
|
||||||
)
|
)
|
||||||
|
|
28
go.sum
28
go.sum
|
@ -19,27 +19,23 @@ github.com/quic-go/qpack v0.4.0 h1:Cr9BXA1sQS2SmDUWjSofMPNKmvF6IiIfDRmgU0w1ZCo=
|
||||||
github.com/quic-go/qpack v0.4.0/go.mod h1:UZVnYIfi5GRk+zI9UMaCPsmZ2xKJP7XBUvVyT1Knj9A=
|
github.com/quic-go/qpack v0.4.0/go.mod h1:UZVnYIfi5GRk+zI9UMaCPsmZ2xKJP7XBUvVyT1Knj9A=
|
||||||
github.com/quic-go/qtls-go1-20 v0.4.1 h1:D33340mCNDAIKBqXuAvexTNMUByrYmFYVfKfDN5nfFs=
|
github.com/quic-go/qtls-go1-20 v0.4.1 h1:D33340mCNDAIKBqXuAvexTNMUByrYmFYVfKfDN5nfFs=
|
||||||
github.com/quic-go/qtls-go1-20 v0.4.1/go.mod h1:X9Nh97ZL80Z+bX/gUXMbipO6OxdiDi58b/fMC9mAL+k=
|
github.com/quic-go/qtls-go1-20 v0.4.1/go.mod h1:X9Nh97ZL80Z+bX/gUXMbipO6OxdiDi58b/fMC9mAL+k=
|
||||||
github.com/sagernet/quic-go v0.49.0-beta.1 h1:3LdoCzVVfYRibZns1tYWSIoB65fpTmrwy+yfK8DQ8Jk=
|
github.com/sagernet/quic-go v0.48.1-beta.1 h1:ElPaV5yzlXIKZpqFMAcUGax6vddi3zt4AEpT94Z0vwo=
|
||||||
github.com/sagernet/quic-go v0.49.0-beta.1/go.mod h1:uesWD1Ihrldq1M3XtjuEvIUqi8WHNsRs71b3Lt1+p/U=
|
github.com/sagernet/quic-go v0.48.1-beta.1/go.mod h1:1WgdDIVD1Gybp40JTWketeSfKA/+or9YMLaG5VeTk4k=
|
||||||
github.com/sagernet/sing v0.6.0-beta.9 h1:P8lKa5hN53fRNAVCIKy5cWd6/kLO5c4slhdsfehSmHs=
|
github.com/sagernet/sing v0.6.0-alpha.14 h1:ORh6yQwLL+/nv+rklrO2W4k+zgf3ZzaOl/83vQbJUl4=
|
||||||
github.com/sagernet/sing v0.6.0-beta.9/go.mod h1:ARkL0gM13/Iv5VCZmci/NuoOlePoIsW0m7BWfln/Hak=
|
github.com/sagernet/sing v0.6.0-alpha.14/go.mod h1:ARkL0gM13/Iv5VCZmci/NuoOlePoIsW0m7BWfln/Hak=
|
||||||
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
||||||
github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
||||||
github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg=
|
github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg=
|
||||||
golang.org/x/crypto v0.32.0 h1:euUpcYgM8WcP71gNpTqQCn6rC2t6ULUPiOzfWaXVVfc=
|
golang.org/x/crypto v0.28.0 h1:GBDwsMXVQi34v5CCYUm2jkJvu4cbtru2U4TN2PSyQnw=
|
||||||
golang.org/x/crypto v0.32.0/go.mod h1:ZnnJkOaASj8g0AjIduWNlq2NRxL0PlBrbKVyZ6V/Ugc=
|
golang.org/x/crypto v0.28.0/go.mod h1:rmgy+3RHxRZMyY0jjAJShp2zgEdOqj2AO7U0pYmeQ7U=
|
||||||
golang.org/x/exp v0.0.0-20240506185415-9bf2ced13842 h1:vr/HnozRka3pE4EsMEg1lgkXJkTFJCVUX+S/ZT6wYzM=
|
golang.org/x/exp v0.0.0-20240506185415-9bf2ced13842 h1:vr/HnozRka3pE4EsMEg1lgkXJkTFJCVUX+S/ZT6wYzM=
|
||||||
golang.org/x/exp v0.0.0-20240506185415-9bf2ced13842/go.mod h1:XtvwrStGgqGPLc4cjQfWqZHG1YFdYs6swckp8vpsjnc=
|
golang.org/x/exp v0.0.0-20240506185415-9bf2ced13842/go.mod h1:XtvwrStGgqGPLc4cjQfWqZHG1YFdYs6swckp8vpsjnc=
|
||||||
golang.org/x/mod v0.20.0 h1:utOm6MM3R3dnawAiJgn0y+xvuYRsm1RKM/4giyfDgV0=
|
golang.org/x/net v0.30.0 h1:AcW1SDZMkb8IpzCdQUaIq2sP4sZ4zw+55h6ynffypl4=
|
||||||
golang.org/x/mod v0.20.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c=
|
golang.org/x/net v0.30.0/go.mod h1:2wGyMJ5iFasEhkwi13ChkO/t1ECNC4X4eBKkVFyYFlU=
|
||||||
golang.org/x/net v0.34.0 h1:Mb7Mrk043xzHgnRM88suvJFwzVrRfHEHJEl5/71CKw0=
|
golang.org/x/sys v0.26.0 h1:KHjCJyddX0LoSTb3J+vWpupP9p0oznkqVk/IfjymZbo=
|
||||||
golang.org/x/net v0.34.0/go.mod h1:di0qlW3YNM5oh6GqDGQr92MyTozJPmybPK4Ev/Gm31k=
|
golang.org/x/sys v0.26.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
|
||||||
golang.org/x/sync v0.10.0 h1:3NQrjDixjgGwUOCaF8w2+VYHv0Ve/vGYSbdkTa98gmQ=
|
golang.org/x/text v0.19.0 h1:kTxAhCbGbxhK0IwgSKiMO5awPoDQ0RpfiVYBfK860YM=
|
||||||
golang.org/x/sync v0.10.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
|
golang.org/x/text v0.19.0/go.mod h1:BuEKDfySbSR4drPmRPG/7iBdf8hvFMuRexcpahXilzY=
|
||||||
golang.org/x/sys v0.29.0 h1:TPYlXGxvx1MGTn2GiZDhnjPA9wZzZeGKHHmKhHYvgaU=
|
|
||||||
golang.org/x/sys v0.29.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
|
|
||||||
golang.org/x/text v0.21.0 h1:zyQAAkrwaneQ066sspRyJaG9VNi/YJ1NfzcGB3hZ/qo=
|
|
||||||
golang.org/x/text v0.21.0/go.mod h1:4IBbMaMmOPCJ8SecivzSH54+73PCFmPWxNTLm+vZkEQ=
|
|
||||||
golang.org/x/tools v0.24.0 h1:J1shsA93PJUEVaUSaay7UXAyE8aimq3GW0pjlolpa24=
|
golang.org/x/tools v0.24.0 h1:J1shsA93PJUEVaUSaay7UXAyE8aimq3GW0pjlolpa24=
|
||||||
golang.org/x/tools v0.24.0/go.mod h1:YhNqVBIfWHdzvTLs0d8LCuMhkKUgSUKldakyV7W/WDQ=
|
golang.org/x/tools v0.24.0/go.mod h1:YhNqVBIfWHdzvTLs0d8LCuMhkKUgSUKldakyV7W/WDQ=
|
||||||
google.golang.org/protobuf v1.28.0 h1:w43yiav+6bVFTBQFZX0r7ipe9JQ1QsbMgHwbBziscLw=
|
google.golang.org/protobuf v1.28.0 h1:w43yiav+6bVFTBQFZX0r7ipe9JQ1QsbMgHwbBziscLw=
|
||||||
|
|
|
@ -159,7 +159,7 @@ func (s *Service[U]) loopConnections(listener qtls.Listener) {
|
||||||
Service: s,
|
Service: s,
|
||||||
ctx: s.ctx,
|
ctx: s.ctx,
|
||||||
quicConn: connection,
|
quicConn: connection,
|
||||||
source: M.SocksaddrFromNet(connection.RemoteAddr()).Unwrap(),
|
source: M.SocksaddrFromNet(connection.RemoteAddr()),
|
||||||
connDone: make(chan struct{}),
|
connDone: make(chan struct{}),
|
||||||
udpConnMap: make(map[uint32]*udpPacketConn),
|
udpConnMap: make(map[uint32]*udpPacketConn),
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,14 +3,11 @@ package hysteria2
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"io"
|
"io"
|
||||||
"math"
|
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
"os"
|
"os"
|
||||||
"runtime"
|
"runtime"
|
||||||
"strconv"
|
|
||||||
"strings"
|
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
@ -39,8 +36,6 @@ type ClientOptions struct {
|
||||||
Logger logger.Logger
|
Logger logger.Logger
|
||||||
BrutalDebug bool
|
BrutalDebug bool
|
||||||
ServerAddress M.Socksaddr
|
ServerAddress M.Socksaddr
|
||||||
ServerPorts []string
|
|
||||||
HopInterval time.Duration
|
|
||||||
SendBPS uint64
|
SendBPS uint64
|
||||||
ReceiveBPS uint64
|
ReceiveBPS uint64
|
||||||
SalamanderPassword string
|
SalamanderPassword string
|
||||||
|
@ -55,8 +50,6 @@ type Client struct {
|
||||||
logger logger.Logger
|
logger logger.Logger
|
||||||
brutalDebug bool
|
brutalDebug bool
|
||||||
serverAddr M.Socksaddr
|
serverAddr M.Socksaddr
|
||||||
serverPorts []uint16
|
|
||||||
hopInterval time.Duration
|
|
||||||
sendBPS uint64
|
sendBPS uint64
|
||||||
receiveBPS uint64
|
receiveBPS uint64
|
||||||
salamanderPassword string
|
salamanderPassword string
|
||||||
|
@ -83,22 +76,12 @@ func NewClient(options ClientOptions) (*Client, error) {
|
||||||
if len(options.TLSConfig.NextProtos()) == 0 {
|
if len(options.TLSConfig.NextProtos()) == 0 {
|
||||||
options.TLSConfig.SetNextProtos([]string{http3.NextProtoH3})
|
options.TLSConfig.SetNextProtos([]string{http3.NextProtoH3})
|
||||||
}
|
}
|
||||||
var serverPorts []uint16
|
|
||||||
if len(options.ServerPorts) > 0 {
|
|
||||||
var err error
|
|
||||||
serverPorts, err = parsePorts(options.ServerPorts)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return &Client{
|
return &Client{
|
||||||
ctx: options.Context,
|
ctx: options.Context,
|
||||||
dialer: options.Dialer,
|
dialer: options.Dialer,
|
||||||
logger: options.Logger,
|
logger: options.Logger,
|
||||||
brutalDebug: options.BrutalDebug,
|
brutalDebug: options.BrutalDebug,
|
||||||
serverAddr: options.ServerAddress,
|
serverAddr: options.ServerAddress,
|
||||||
serverPorts: serverPorts,
|
|
||||||
hopInterval: options.HopInterval,
|
|
||||||
sendBPS: options.SendBPS,
|
sendBPS: options.SendBPS,
|
||||||
receiveBPS: options.ReceiveBPS,
|
receiveBPS: options.ReceiveBPS,
|
||||||
salamanderPassword: options.SalamanderPassword,
|
salamanderPassword: options.SalamanderPassword,
|
||||||
|
@ -109,38 +92,6 @@ func NewClient(options ClientOptions) (*Client, error) {
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func parsePorts(serverPorts []string) ([]uint16, error) {
|
|
||||||
var portList []uint16
|
|
||||||
for _, portRange := range serverPorts {
|
|
||||||
if !strings.Contains(portRange, ":") {
|
|
||||||
return nil, E.New("bad port range: ", portRange)
|
|
||||||
}
|
|
||||||
subIndex := strings.Index(portRange, ":")
|
|
||||||
var (
|
|
||||||
start, end uint64
|
|
||||||
err error
|
|
||||||
)
|
|
||||||
if subIndex > 0 {
|
|
||||||
start, err = strconv.ParseUint(portRange[:subIndex], 10, 16)
|
|
||||||
if err != nil {
|
|
||||||
return nil, E.Cause(err, E.Cause(err, "bad port range: ", portRange))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if subIndex == len(portRange)-1 {
|
|
||||||
end = math.MaxUint16
|
|
||||||
} else {
|
|
||||||
end, err = strconv.ParseUint(portRange[subIndex+1:], 10, 16)
|
|
||||||
if err != nil {
|
|
||||||
return nil, E.Cause(err, E.Cause(err, "bad port range: ", portRange))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for i := start; i <= end; i++ {
|
|
||||||
portList = append(portList, uint16(i))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return portList, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Client) offer(ctx context.Context) (*clientQUICConnection, error) {
|
func (c *Client) offer(ctx context.Context) (*clientQUICConnection, error) {
|
||||||
conn := c.conn
|
conn := c.conn
|
||||||
if conn != nil && conn.active() {
|
if conn != nil && conn.active() {
|
||||||
|
@ -160,34 +111,19 @@ func (c *Client) offer(ctx context.Context) (*clientQUICConnection, error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Client) offerNew(ctx context.Context) (*clientQUICConnection, error) {
|
func (c *Client) offerNew(ctx context.Context) (*clientQUICConnection, error) {
|
||||||
dialFunc := func(serverAddr M.Socksaddr) (net.PacketConn, error) {
|
udpConn, err := c.dialer.DialContext(c.ctx, "udp", c.serverAddr)
|
||||||
udpConn, err := c.dialer.DialContext(c.ctx, "udp", serverAddr)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
var packetConn net.PacketConn
|
|
||||||
packetConn = bufio.NewUnbindPacketConn(udpConn)
|
|
||||||
if c.salamanderPassword != "" {
|
|
||||||
packetConn = NewSalamanderConn(packetConn, []byte(c.salamanderPassword))
|
|
||||||
}
|
|
||||||
return packetConn, nil
|
|
||||||
}
|
|
||||||
var (
|
|
||||||
packetConn net.PacketConn
|
|
||||||
err error
|
|
||||||
)
|
|
||||||
if len(c.serverPorts) == 0 {
|
|
||||||
packetConn, err = dialFunc(c.serverAddr)
|
|
||||||
} else {
|
|
||||||
packetConn, err = NewHopPacketConn(dialFunc, c.serverAddr, c.serverPorts, c.hopInterval)
|
|
||||||
}
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
var packetConn net.PacketConn
|
||||||
|
packetConn = bufio.NewUnbindPacketConn(udpConn)
|
||||||
|
if c.salamanderPassword != "" {
|
||||||
|
packetConn = NewSalamanderConn(packetConn, []byte(c.salamanderPassword))
|
||||||
|
}
|
||||||
var quicConn quic.EarlyConnection
|
var quicConn quic.EarlyConnection
|
||||||
http3Transport, err := qtls.CreateTransport(packetConn, &quicConn, c.serverAddr, c.tlsConfig, c.quicConfig)
|
http3Transport, err := qtls.CreateTransport(packetConn, &quicConn, c.serverAddr, c.tlsConfig, c.quicConfig)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
packetConn.Close()
|
udpConn.Close()
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
request := &http.Request{
|
request := &http.Request{
|
||||||
|
@ -205,14 +141,14 @@ func (c *Client) offerNew(ctx context.Context) (*clientQUICConnection, error) {
|
||||||
if quicConn != nil {
|
if quicConn != nil {
|
||||||
quicConn.CloseWithError(0, "")
|
quicConn.CloseWithError(0, "")
|
||||||
}
|
}
|
||||||
packetConn.Close()
|
udpConn.Close()
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if response.StatusCode != protocol.StatusAuthOK {
|
if response.StatusCode != protocol.StatusAuthOK {
|
||||||
if quicConn != nil {
|
if quicConn != nil {
|
||||||
quicConn.CloseWithError(0, "")
|
quicConn.CloseWithError(0, "")
|
||||||
}
|
}
|
||||||
packetConn.Close()
|
udpConn.Close()
|
||||||
return nil, E.New("authentication failed, status code: ", response.StatusCode)
|
return nil, E.New("authentication failed, status code: ", response.StatusCode)
|
||||||
}
|
}
|
||||||
response.Body.Close()
|
response.Body.Close()
|
||||||
|
@ -236,7 +172,7 @@ func (c *Client) offerNew(ctx context.Context) (*clientQUICConnection, error) {
|
||||||
}
|
}
|
||||||
conn := &clientQUICConnection{
|
conn := &clientQUICConnection{
|
||||||
quicConn: quicConn,
|
quicConn: quicConn,
|
||||||
rawConn: packetConn,
|
rawConn: udpConn,
|
||||||
connDone: make(chan struct{}),
|
connDone: make(chan struct{}),
|
||||||
udpDisabled: !authResponse.UDPEnabled,
|
udpDisabled: !authResponse.UDPEnabled,
|
||||||
udpConnMap: make(map[uint32]*udpPacketConn),
|
udpConnMap: make(map[uint32]*udpPacketConn),
|
||||||
|
|
269
hysteria2/hop.go
269
hysteria2/hop.go
|
@ -1,269 +0,0 @@
|
||||||
package hysteria2
|
|
||||||
|
|
||||||
import (
|
|
||||||
"errors"
|
|
||||||
"math/rand"
|
|
||||||
"net"
|
|
||||||
"os"
|
|
||||||
"sync"
|
|
||||||
"syscall"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/sagernet/sing/common"
|
|
||||||
"github.com/sagernet/sing/common/buf"
|
|
||||||
M "github.com/sagernet/sing/common/metadata"
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
|
||||||
packetQueueSize = 1024
|
|
||||||
udpBufferSize = 2048
|
|
||||||
defaultHopInterval = 30 * time.Second
|
|
||||||
)
|
|
||||||
|
|
||||||
type HopPacketConn struct {
|
|
||||||
dialFunc func(M.Socksaddr) (net.PacketConn, error)
|
|
||||||
destination M.Socksaddr
|
|
||||||
ports []uint16
|
|
||||||
interval time.Duration
|
|
||||||
access sync.Mutex
|
|
||||||
prevConn net.PacketConn
|
|
||||||
currentConn net.PacketConn
|
|
||||||
portIndex int
|
|
||||||
readBufferSize int
|
|
||||||
writeBufferSize int
|
|
||||||
packetChan chan *buf.Buffer
|
|
||||||
errChan chan error
|
|
||||||
doneChan chan struct{}
|
|
||||||
done bool
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewHopPacketConn(
|
|
||||||
dialFunc func(M.Socksaddr) (net.PacketConn, error),
|
|
||||||
destination M.Socksaddr,
|
|
||||||
ports []uint16,
|
|
||||||
interval time.Duration,
|
|
||||||
) (*HopPacketConn, error) {
|
|
||||||
if interval == 0 {
|
|
||||||
interval = defaultHopInterval
|
|
||||||
}
|
|
||||||
hopConn := &HopPacketConn{
|
|
||||||
dialFunc: dialFunc,
|
|
||||||
destination: destination,
|
|
||||||
ports: ports,
|
|
||||||
interval: interval,
|
|
||||||
packetChan: make(chan *buf.Buffer, packetQueueSize),
|
|
||||||
errChan: make(chan error),
|
|
||||||
doneChan: make(chan struct{}),
|
|
||||||
}
|
|
||||||
currentConn, err := dialFunc(hopConn.nextAddr())
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
hopConn.currentConn = currentConn
|
|
||||||
go hopConn.recvLoop(currentConn)
|
|
||||||
go hopConn.hopLoop()
|
|
||||||
return hopConn, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *HopPacketConn) nextAddr() M.Socksaddr {
|
|
||||||
c.portIndex = rand.Intn(len(c.ports))
|
|
||||||
return M.Socksaddr{
|
|
||||||
Addr: c.destination.Addr,
|
|
||||||
Fqdn: c.destination.Fqdn,
|
|
||||||
Port: c.ports[c.portIndex],
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *HopPacketConn) recvLoop(conn net.PacketConn) {
|
|
||||||
for {
|
|
||||||
buffer := buf.NewSize(udpBufferSize)
|
|
||||||
n, _, err := conn.ReadFrom(buffer.FreeBytes())
|
|
||||||
if err != nil {
|
|
||||||
buffer.Release()
|
|
||||||
var netErr net.Error
|
|
||||||
if errors.As(err, &netErr) && netErr.Timeout() {
|
|
||||||
// Only pass through timeout errors here, not permanent errors
|
|
||||||
// like connection closed. Connection close is normal as we close
|
|
||||||
// the old connection to exit this loop every time we hop.
|
|
||||||
c.errChan <- netErr
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
buffer.Truncate(n)
|
|
||||||
select {
|
|
||||||
case c.packetChan <- buffer:
|
|
||||||
default:
|
|
||||||
buffer.Release()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *HopPacketConn) hopLoop() {
|
|
||||||
ticker := time.NewTicker(c.interval)
|
|
||||||
defer ticker.Stop()
|
|
||||||
for {
|
|
||||||
select {
|
|
||||||
case <-ticker.C:
|
|
||||||
c.hop()
|
|
||||||
case <-c.doneChan:
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *HopPacketConn) hop() {
|
|
||||||
c.access.Lock()
|
|
||||||
defer c.access.Unlock()
|
|
||||||
if c.done {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
nextAddr := c.nextAddr()
|
|
||||||
newConn, err := c.dialFunc(nextAddr)
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if c.prevConn != nil {
|
|
||||||
c.prevConn.Close()
|
|
||||||
}
|
|
||||||
c.prevConn = c.currentConn
|
|
||||||
c.currentConn = newConn
|
|
||||||
if c.readBufferSize > 0 {
|
|
||||||
_ = trySetReadBuffer(newConn, c.readBufferSize)
|
|
||||||
}
|
|
||||||
if c.writeBufferSize > 0 {
|
|
||||||
_ = trySetWriteBuffer(newConn, c.writeBufferSize)
|
|
||||||
}
|
|
||||||
go c.recvLoop(newConn)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *HopPacketConn) ReadFrom(b []byte) (n int, addr net.Addr, err error) {
|
|
||||||
for {
|
|
||||||
select {
|
|
||||||
case packet := <-c.packetChan:
|
|
||||||
n = copy(b, packet.Bytes())
|
|
||||||
packet.Release()
|
|
||||||
return n, (*hopFakeAddr)(nil), nil
|
|
||||||
case err = <-c.errChan:
|
|
||||||
return 0, nil, err
|
|
||||||
case <-c.doneChan:
|
|
||||||
return 0, nil, net.ErrClosed
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *HopPacketConn) WriteTo(b []byte, _ net.Addr) (n int, err error) {
|
|
||||||
c.access.Lock()
|
|
||||||
defer c.access.Unlock()
|
|
||||||
if c.done {
|
|
||||||
return 0, net.ErrClosed
|
|
||||||
}
|
|
||||||
return c.currentConn.WriteTo(b, (*hopFakeAddr)(nil))
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *HopPacketConn) Close() error {
|
|
||||||
c.access.Lock()
|
|
||||||
defer c.access.Unlock()
|
|
||||||
if c.done {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
if c.prevConn != nil {
|
|
||||||
_ = c.prevConn.Close()
|
|
||||||
}
|
|
||||||
err := c.currentConn.Close()
|
|
||||||
close(c.doneChan)
|
|
||||||
c.done = true
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *HopPacketConn) LocalAddr() net.Addr {
|
|
||||||
c.access.Lock()
|
|
||||||
defer c.access.Unlock()
|
|
||||||
return c.currentConn.LocalAddr()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *HopPacketConn) SetDeadline(t time.Time) error {
|
|
||||||
c.access.Lock()
|
|
||||||
defer c.access.Unlock()
|
|
||||||
if c.prevConn != nil {
|
|
||||||
_ = c.prevConn.SetDeadline(t)
|
|
||||||
}
|
|
||||||
return c.currentConn.SetDeadline(t)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *HopPacketConn) SetReadDeadline(t time.Time) error {
|
|
||||||
c.access.Lock()
|
|
||||||
defer c.access.Unlock()
|
|
||||||
if c.prevConn != nil {
|
|
||||||
_ = c.prevConn.SetReadDeadline(t)
|
|
||||||
}
|
|
||||||
return c.currentConn.SetReadDeadline(t)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *HopPacketConn) SetWriteDeadline(t time.Time) error {
|
|
||||||
c.access.Lock()
|
|
||||||
defer c.access.Unlock()
|
|
||||||
if c.prevConn != nil {
|
|
||||||
_ = c.prevConn.SetWriteDeadline(t)
|
|
||||||
}
|
|
||||||
return c.currentConn.SetWriteDeadline(t)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *HopPacketConn) SetReadBuffer(bytes int) error {
|
|
||||||
c.access.Lock()
|
|
||||||
defer c.access.Unlock()
|
|
||||||
c.readBufferSize = bytes
|
|
||||||
if c.prevConn != nil {
|
|
||||||
_ = trySetReadBuffer(c.prevConn, bytes)
|
|
||||||
}
|
|
||||||
return trySetReadBuffer(c.currentConn, bytes)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *HopPacketConn) SetWriteBuffer(bytes int) error {
|
|
||||||
c.access.Lock()
|
|
||||||
defer c.access.Unlock()
|
|
||||||
c.writeBufferSize = bytes
|
|
||||||
if c.prevConn != nil {
|
|
||||||
_ = trySetWriteBuffer(c.prevConn, bytes)
|
|
||||||
}
|
|
||||||
return trySetWriteBuffer(c.currentConn, bytes)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *HopPacketConn) SyscallConn() (syscall.RawConn, error) {
|
|
||||||
c.access.Lock()
|
|
||||||
defer c.access.Unlock()
|
|
||||||
rawConn, isRawConn := common.Cast[syscall.Conn](c.currentConn)
|
|
||||||
if !isRawConn {
|
|
||||||
return nil, os.ErrInvalid
|
|
||||||
}
|
|
||||||
return rawConn.SyscallConn()
|
|
||||||
}
|
|
||||||
|
|
||||||
func trySetReadBuffer(pc any, bytes int) error {
|
|
||||||
udpConn, isUDPConn := common.Cast[interface {
|
|
||||||
SetReadBuffer(bytes int) error
|
|
||||||
}](pc)
|
|
||||||
if !isUDPConn {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return udpConn.SetReadBuffer(bytes)
|
|
||||||
}
|
|
||||||
|
|
||||||
func trySetWriteBuffer(pc any, bytes int) error {
|
|
||||||
udpConn, isUDPConn := common.Cast[interface {
|
|
||||||
SetWriteBuffer(bytes int) error
|
|
||||||
}](pc)
|
|
||||||
if !isUDPConn {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return udpConn.SetWriteBuffer(bytes)
|
|
||||||
}
|
|
||||||
|
|
||||||
type hopFakeAddr struct{}
|
|
||||||
|
|
||||||
func (a *hopFakeAddr) Network() string {
|
|
||||||
return "udphop"
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a *hopFakeAddr) String() string {
|
|
||||||
return "<udphop>"
|
|
||||||
}
|
|
|
@ -69,10 +69,6 @@ func (s *SalamanderPacketConn) WriteTo(p []byte, addr net.Addr) (n int, err erro
|
||||||
return len(p), nil
|
return len(p), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *SalamanderPacketConn) Upstream() any {
|
|
||||||
return s.PacketConn
|
|
||||||
}
|
|
||||||
|
|
||||||
type VectorisedSalamanderPacketConn struct {
|
type VectorisedSalamanderPacketConn struct {
|
||||||
SalamanderPacketConn
|
SalamanderPacketConn
|
||||||
writer N.VectorisedPacketWriter
|
writer N.VectorisedPacketWriter
|
||||||
|
|
|
@ -156,7 +156,7 @@ func (s *Service[U]) handleConnection(connection quic.Connection) {
|
||||||
Service: s,
|
Service: s,
|
||||||
ctx: s.ctx,
|
ctx: s.ctx,
|
||||||
quicConn: connection,
|
quicConn: connection,
|
||||||
source: M.SocksaddrFromNet(connection.RemoteAddr()).Unwrap(),
|
source: M.SocksaddrFromNet(connection.RemoteAddr()),
|
||||||
connDone: make(chan struct{}),
|
connDone: make(chan struct{}),
|
||||||
udpConnMap: make(map[uint32]*udpPacketConn),
|
udpConnMap: make(map[uint32]*udpPacketConn),
|
||||||
}
|
}
|
||||||
|
@ -188,7 +188,7 @@ func (s *serverSession[U]) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
||||||
protocol.AuthResponseToHeader(w.Header(), protocol.AuthResponse{
|
protocol.AuthResponseToHeader(w.Header(), protocol.AuthResponse{
|
||||||
UDPEnabled: !s.udpDisabled,
|
UDPEnabled: !s.udpDisabled,
|
||||||
Rx: s.receiveBPS,
|
Rx: s.receiveBPS,
|
||||||
RxAuto: s.receiveBPS == 0 && s.ignoreClientBandwidth,
|
RxAuto: s.ignoreClientBandwidth,
|
||||||
})
|
})
|
||||||
w.WriteHeader(protocol.StatusAuthOK)
|
w.WriteHeader(protocol.StatusAuthOK)
|
||||||
return
|
return
|
||||||
|
@ -201,12 +201,7 @@ func (s *serverSession[U]) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
||||||
}
|
}
|
||||||
s.authUser = user
|
s.authUser = user
|
||||||
s.authenticated = true
|
s.authenticated = true
|
||||||
var rxAuto bool
|
if !s.ignoreClientBandwidth && request.Rx > 0 {
|
||||||
if s.receiveBPS > 0 && s.ignoreClientBandwidth && request.Rx == 0 {
|
|
||||||
s.logger.Debug("process connection from ", r.RemoteAddr, ": BBR disabled by server")
|
|
||||||
s.masqueradeHandler.ServeHTTP(w, r)
|
|
||||||
return
|
|
||||||
} else if !(s.receiveBPS == 0 && s.ignoreClientBandwidth) && request.Rx > 0 {
|
|
||||||
rx := request.Rx
|
rx := request.Rx
|
||||||
if s.sendBPS > 0 && rx > s.sendBPS {
|
if s.sendBPS > 0 && rx > s.sendBPS {
|
||||||
rx = s.sendBPS
|
rx = s.sendBPS
|
||||||
|
@ -222,12 +217,11 @@ func (s *serverSession[U]) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
||||||
congestion.ByteCount(s.quicConn.Config().InitialPacketSize),
|
congestion.ByteCount(s.quicConn.Config().InitialPacketSize),
|
||||||
congestion.ByteCount(congestion_meta1.InitialCongestionWindow),
|
congestion.ByteCount(congestion_meta1.InitialCongestionWindow),
|
||||||
))
|
))
|
||||||
rxAuto = true
|
|
||||||
}
|
}
|
||||||
protocol.AuthResponseToHeader(w.Header(), protocol.AuthResponse{
|
protocol.AuthResponseToHeader(w.Header(), protocol.AuthResponse{
|
||||||
UDPEnabled: !s.udpDisabled,
|
UDPEnabled: !s.udpDisabled,
|
||||||
Rx: s.receiveBPS,
|
Rx: s.receiveBPS,
|
||||||
RxAuto: rxAuto,
|
RxAuto: s.ignoreClientBandwidth,
|
||||||
})
|
})
|
||||||
w.WriteHeader(protocol.StatusAuthOK)
|
w.WriteHeader(protocol.StatusAuthOK)
|
||||||
if s.ctx.Done() != nil {
|
if s.ctx.Done() != nil {
|
||||||
|
|
|
@ -163,7 +163,7 @@ func (s *Service[U]) handleConnection(connection quic.Connection) {
|
||||||
Service: s,
|
Service: s,
|
||||||
ctx: s.ctx,
|
ctx: s.ctx,
|
||||||
quicConn: connection,
|
quicConn: connection,
|
||||||
source: M.SocksaddrFromNet(connection.RemoteAddr()).Unwrap(),
|
source: M.SocksaddrFromNet(connection.RemoteAddr()),
|
||||||
connDone: make(chan struct{}),
|
connDone: make(chan struct{}),
|
||||||
authDone: make(chan struct{}),
|
authDone: make(chan struct{}),
|
||||||
udpConnMap: make(map[uint16]*udpPacketConn),
|
udpConnMap: make(map[uint16]*udpPacketConn),
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue