mirror of
https://github.com/SagerNet/sing-shadowtls.git
synced 2025-04-03 04:07:35 +03:00
Add support for wildcard-sni
This commit is contained in:
parent
2d5886b68f
commit
6f9e732e50
3 changed files with 64 additions and 26 deletions
2
go.mod
2
go.mod
|
@ -3,7 +3,7 @@ module github.com/sagernet/sing-shadowtls
|
||||||
go 1.20
|
go 1.20
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/sagernet/sing v0.6.0
|
github.com/sagernet/sing v0.6.3
|
||||||
golang.org/x/crypto v0.32.0
|
golang.org/x/crypto v0.32.0
|
||||||
golang.org/x/exp v0.0.0-20240719175910-8a7402abbf56
|
golang.org/x/exp v0.0.0-20240719175910-8a7402abbf56
|
||||||
)
|
)
|
||||||
|
|
4
go.sum
4
go.sum
|
@ -1,7 +1,7 @@
|
||||||
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
||||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||||
github.com/sagernet/sing v0.6.0 h1:jT55zAXrG7H3x+s/FlrC15xQy3LcmuZ2GGA9+8IJdt0=
|
github.com/sagernet/sing v0.6.3 h1:J1spMc6LMlqUvRjWjvNMAcbvACDneqxB9zxfLuS0UTE=
|
||||||
github.com/sagernet/sing v0.6.0/go.mod h1:ARkL0gM13/Iv5VCZmci/NuoOlePoIsW0m7BWfln/Hak=
|
github.com/sagernet/sing v0.6.3/go.mod h1:ARkL0gM13/Iv5VCZmci/NuoOlePoIsW0m7BWfln/Hak=
|
||||||
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.32.0 h1:euUpcYgM8WcP71gNpTqQCn6rC2t6ULUPiOzfWaXVVfc=
|
||||||
golang.org/x/crypto v0.32.0/go.mod h1:ZnnJkOaASj8g0AjIduWNlq2NRxL0PlBrbKVyZ6V/Ugc=
|
golang.org/x/crypto v0.32.0/go.mod h1:ZnnJkOaASj8g0AjIduWNlq2NRxL0PlBrbKVyZ6V/Ugc=
|
||||||
|
|
84
service.go
84
service.go
|
@ -27,10 +27,19 @@ type Service struct {
|
||||||
handshake HandshakeConfig
|
handshake HandshakeConfig
|
||||||
handshakeForServerName map[string]HandshakeConfig
|
handshakeForServerName map[string]HandshakeConfig
|
||||||
strictMode bool
|
strictMode bool
|
||||||
|
wildcardSNI WildcardSNI
|
||||||
handler N.TCPConnectionHandlerEx
|
handler N.TCPConnectionHandlerEx
|
||||||
logger logger.ContextLogger
|
logger logger.ContextLogger
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type WildcardSNI int
|
||||||
|
|
||||||
|
const (
|
||||||
|
WildcardSNIOff WildcardSNI = iota
|
||||||
|
WildcardSNIAuthed
|
||||||
|
WildcardSNIAll
|
||||||
|
)
|
||||||
|
|
||||||
type ServiceConfig struct {
|
type ServiceConfig struct {
|
||||||
Version int
|
Version int
|
||||||
Password string // for protocol version 2
|
Password string // for protocol version 2
|
||||||
|
@ -38,6 +47,7 @@ type ServiceConfig struct {
|
||||||
Handshake HandshakeConfig
|
Handshake HandshakeConfig
|
||||||
HandshakeForServerName map[string]HandshakeConfig // for protocol version 2/3
|
HandshakeForServerName map[string]HandshakeConfig // for protocol version 2/3
|
||||||
StrictMode bool // for protocol version 3
|
StrictMode bool // for protocol version 3
|
||||||
|
WildcardSNI WildcardSNI // for protocol version 3
|
||||||
Handler N.TCPConnectionHandlerEx
|
Handler N.TCPConnectionHandlerEx
|
||||||
Logger logger.ContextLogger
|
Logger logger.ContextLogger
|
||||||
}
|
}
|
||||||
|
@ -60,11 +70,12 @@ func NewService(config ServiceConfig) (*Service, error) {
|
||||||
handshake: config.Handshake,
|
handshake: config.Handshake,
|
||||||
handshakeForServerName: config.HandshakeForServerName,
|
handshakeForServerName: config.HandshakeForServerName,
|
||||||
strictMode: config.StrictMode,
|
strictMode: config.StrictMode,
|
||||||
|
wildcardSNI: config.WildcardSNI,
|
||||||
handler: config.Handler,
|
handler: config.Handler,
|
||||||
logger: config.Logger,
|
logger: config.Logger,
|
||||||
}
|
}
|
||||||
|
|
||||||
if !service.handshake.Server.IsValid() {
|
if !service.handshake.Server.IsValid() && service.wildcardSNI == WildcardSNIOff {
|
||||||
return nil, E.New("missing default handshake information")
|
return nil, E.New("missing default handshake information")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -84,16 +95,6 @@ func NewService(config ServiceConfig) (*Service, error) {
|
||||||
return service, nil
|
return service, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *Service) selectHandshake(clientHelloFrame *buf.Buffer) HandshakeConfig {
|
|
||||||
serverName, err := extractServerName(clientHelloFrame.Bytes())
|
|
||||||
if err == nil {
|
|
||||||
if customHandshake, found := s.handshakeForServerName[serverName]; found {
|
|
||||||
return customHandshake
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return s.handshake
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *Service) NewConnection(ctx context.Context, conn net.Conn, source M.Socksaddr, destination M.Socksaddr, onClose N.CloseHandlerFunc) error {
|
func (s *Service) NewConnection(ctx context.Context, conn net.Conn, source M.Socksaddr, destination M.Socksaddr, onClose N.CloseHandlerFunc) error {
|
||||||
switch s.version {
|
switch s.version {
|
||||||
default:
|
default:
|
||||||
|
@ -127,8 +128,17 @@ func (s *Service) NewConnection(ctx context.Context, conn net.Conn, source M.Soc
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return E.Cause(err, "read client handshake")
|
return E.Cause(err, "read client handshake")
|
||||||
}
|
}
|
||||||
|
serverName, err := extractServerName(clientHelloFrame.Bytes())
|
||||||
handshakeConfig := s.selectHandshake(clientHelloFrame)
|
var handshakeConfig HandshakeConfig
|
||||||
|
if err == nil {
|
||||||
|
if customHandshake, found := s.handshakeForServerName[serverName]; found {
|
||||||
|
handshakeConfig = customHandshake
|
||||||
|
} else {
|
||||||
|
handshakeConfig = s.handshake
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
handshakeConfig = s.handshake
|
||||||
|
}
|
||||||
handshakeConn, err := handshakeConfig.Dialer.DialContext(ctx, N.NetworkTCP, handshakeConfig.Server)
|
handshakeConn, err := handshakeConfig.Dialer.DialContext(ctx, N.NetworkTCP, handshakeConfig.Server)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return E.Cause(err, "server handshake")
|
return E.Cause(err, "server handshake")
|
||||||
|
@ -154,28 +164,56 @@ func (s *Service) NewConnection(ctx context.Context, conn net.Conn, source M.Soc
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return E.Cause(err, "read client handshake")
|
return E.Cause(err, "read client handshake")
|
||||||
}
|
}
|
||||||
|
defer clientHelloFrame.Release()
|
||||||
handshakeConfig := s.selectHandshake(clientHelloFrame)
|
serverName, err := extractServerName(clientHelloFrame.Bytes())
|
||||||
handshakeConn, err := handshakeConfig.Dialer.DialContext(ctx, N.NetworkTCP, handshakeConfig.Server)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return E.Cause(err, "server handshake")
|
return E.Cause(err, "extract server name")
|
||||||
}
|
}
|
||||||
|
var (
|
||||||
_, err = handshakeConn.Write(clientHelloFrame.Bytes())
|
handshakeConfig HandshakeConfig
|
||||||
if err != nil {
|
isCustom bool
|
||||||
clientHelloFrame.Release()
|
)
|
||||||
return E.Cause(err, "write client handshake")
|
if customHandshake, found := s.handshakeForServerName[serverName]; found {
|
||||||
|
handshakeConfig = customHandshake
|
||||||
|
isCustom = true
|
||||||
|
} else {
|
||||||
|
handshakeConfig = s.handshake
|
||||||
|
if s.wildcardSNI != WildcardSNIOff {
|
||||||
|
handshakeConfig.Server = M.Socksaddr{
|
||||||
|
Fqdn: serverName,
|
||||||
|
Port: 443,
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
var handshakeConn net.Conn
|
||||||
user, err := verifyClientHello(clientHelloFrame.Bytes(), s.users)
|
user, err := verifyClientHello(clientHelloFrame.Bytes(), s.users)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
s.logger.WarnContext(ctx, E.Cause(err, "client hello verify failed"))
|
s.logger.WarnContext(ctx, E.Cause(err, "client hello verify failed"))
|
||||||
return bufio.CopyConn(ctx, conn, handshakeConn)
|
if s.wildcardSNI == WildcardSNIAll || isCustom {
|
||||||
|
handshakeConn, err = handshakeConfig.Dialer.DialContext(ctx, N.NetworkTCP, handshakeConfig.Server)
|
||||||
|
} else {
|
||||||
|
handshakeConn, err = s.handshake.Dialer.DialContext(ctx, N.NetworkTCP, s.handshake.Server)
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return E.Cause(err, "server handshake")
|
||||||
|
}
|
||||||
|
return bufio.CopyConn(ctx, bufio.NewCachedConn(conn, clientHelloFrame), handshakeConn)
|
||||||
}
|
}
|
||||||
if user.Name != "" {
|
if user.Name != "" {
|
||||||
ctx = auth.ContextWithUser(ctx, user.Name)
|
ctx = auth.ContextWithUser(ctx, user.Name)
|
||||||
}
|
}
|
||||||
s.logger.TraceContext(ctx, "client hello verify success")
|
s.logger.TraceContext(ctx, "client hello verify success")
|
||||||
|
|
||||||
|
handshakeConn, err = handshakeConfig.Dialer.DialContext(ctx, N.NetworkTCP, handshakeConfig.Server)
|
||||||
|
if err != nil {
|
||||||
|
return E.Cause(err, "server handshake")
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = handshakeConn.Write(clientHelloFrame.Bytes())
|
||||||
clientHelloFrame.Release()
|
clientHelloFrame.Release()
|
||||||
|
if err != nil {
|
||||||
|
return E.Cause(err, "write client handshake")
|
||||||
|
}
|
||||||
|
|
||||||
var serverHelloFrame *buf.Buffer
|
var serverHelloFrame *buf.Buffer
|
||||||
serverHelloFrame, err = extractFrame(handshakeConn)
|
serverHelloFrame, err = extractFrame(handshakeConn)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue