feat: support Android protect path

about: 1ac9d4956b

Signed-off-by: HystericalDragon <HystericalDragons@proton.me>
This commit is contained in:
HystericalDragon 2024-03-31 10:58:34 +08:00
parent f91efbeded
commit 03c8b5e6b9
No known key found for this signature in database
GPG key ID: E205724F8931D99B
6 changed files with 103 additions and 7 deletions

View file

@ -30,6 +30,7 @@ import (
"github.com/apernet/hysteria/core/client"
"github.com/apernet/hysteria/extras/correctnet"
"github.com/apernet/hysteria/extras/obfs"
"github.com/apernet/hysteria/extras/protect"
"github.com/apernet/hysteria/extras/transport/udphop"
)
@ -55,6 +56,7 @@ func initClientFlags() {
type clientConfig struct {
Server string `mapstructure:"server"`
ProtectPath string `mapstructure:"protectPath"`
Auth string `mapstructure:"auth"`
Transport clientConfigTransport `mapstructure:"transport"`
Obfs clientConfigObfs `mapstructure:"obfs"`
@ -202,11 +204,11 @@ func (c *clientConfig) fillConnFactory(hyConfig *client.Config) error {
if hyConfig.ServerAddr.Network() == "udphop" {
hopAddr := hyConfig.ServerAddr.(*udphop.UDPHopAddr)
newFunc = func(addr net.Addr) (net.PacketConn, error) {
return udphop.NewUDPHopPacketConn(hopAddr, c.Transport.UDP.HopInterval, nil)
return udphop.NewUDPHopPacketConn(hopAddr, c.Transport.UDP.HopInterval, protect.ListenUDP(c.ProtectPath))
}
} else {
newFunc = func(addr net.Addr) (net.PacketConn, error) {
return net.ListenUDP("udp", nil)
return protect.ListenUDP(c.ProtectPath)()
}
}
default:

View file

@ -11,6 +11,7 @@ require (
github.com/txthinking/socks5 v0.0.0-20230325130024-4230056ae301
golang.org/x/crypto v0.19.0
golang.org/x/net v0.21.0
golang.org/x/sys v0.17.0
google.golang.org/protobuf v1.33.0
)
@ -28,7 +29,6 @@ require (
go.uber.org/mock v0.4.0 // indirect
golang.org/x/exp v0.0.0-20221205204356-47842c84f3db // indirect
golang.org/x/mod v0.12.0 // indirect
golang.org/x/sys v0.17.0 // indirect
golang.org/x/text v0.14.0 // indirect
golang.org/x/tools v0.11.1 // indirect
gopkg.in/yaml.v3 v3.0.1 // indirect

View file

@ -0,0 +1,9 @@
// Package protect set VPN protect for every conns to bypass route.
package protect
import (
"net"
)
// ListenUDPFunc listen UDP with VPN protect.
type ListenUDPFunc func() (net.PacketConn, error)

View file

@ -0,0 +1,72 @@
//go:build linux
package protect
import (
"errors"
"net"
"golang.org/x/sys/unix"
)
const (
timevalSec = 3
)
func protect(connFd int, path string) error {
if path == "" {
return nil
}
socketFd, err := unix.Socket(unix.AF_UNIX, unix.SOCK_STREAM, unix.PROT_NONE)
if err != nil {
return err
}
defer unix.Close(socketFd)
_ = unix.SetsockoptTimeval(socketFd, unix.SOL_SOCKET, unix.SO_RCVTIMEO, &unix.Timeval{Sec: timevalSec})
_ = unix.SetsockoptTimeval(socketFd, unix.SOL_SOCKET, unix.SO_SNDTIMEO, &unix.Timeval{Sec: timevalSec})
err = unix.Connect(socketFd, &unix.SockaddrUnix{Name: path})
if err != nil {
return err
}
err = unix.Sendmsg(socketFd, nil, unix.UnixRights(connFd), nil, 0)
if err != nil {
return err
}
dummy := []byte{1}
n, err := unix.Read(socketFd, dummy)
if err != nil {
return err
}
if n != 1 {
return errors.New("protect failed")
}
return nil
}
func ListenUDP(protectPath string) ListenUDPFunc {
return func() (net.PacketConn, error) {
udpConn, err := net.ListenUDP("udp", nil)
if err != nil {
return nil, err
}
udpFile, err := udpConn.File()
if err != nil {
return nil, err
}
err = protect(int(udpFile.Fd()), protectPath)
if err != nil {
_ = udpConn.Close()
return nil, err
}
return udpConn, nil
}
}

View file

@ -0,0 +1,13 @@
//go:build !linux
package protect
import (
"net"
)
func ListenUDP(protectPath string) ListenUDPFunc {
return func() (net.PacketConn, error) {
return net.ListenUDP("udp", nil)
}
}

View file

@ -7,6 +7,8 @@ import (
"sync"
"syscall"
"time"
"github.com/apernet/hysteria/extras/protect"
)
const (
@ -20,7 +22,7 @@ type udpHopPacketConn struct {
Addr net.Addr
Addrs []net.Addr
HopInterval time.Duration
ListenUDPFunc ListenUDPFunc
ListenUDPFunc protect.ListenUDPFunc
connMutex sync.RWMutex
prevConn net.PacketConn
@ -44,9 +46,7 @@ type udpPacket struct {
Err error
}
type ListenUDPFunc func() (net.PacketConn, error)
func NewUDPHopPacketConn(addr *UDPHopAddr, hopInterval time.Duration, listenUDPFunc ListenUDPFunc) (net.PacketConn, error) {
func NewUDPHopPacketConn(addr *UDPHopAddr, hopInterval time.Duration, listenUDPFunc protect.ListenUDPFunc) (net.PacketConn, error) {
if hopInterval == 0 {
hopInterval = defaultHopInterval
} else if hopInterval < 5*time.Second {