mirror of
https://github.com/cedar2025/hysteria.git
synced 2025-06-09 05:59:54 +00:00
182 lines
4.2 KiB
Go
182 lines
4.2 KiB
Go
package transport
|
|
|
|
import (
|
|
"crypto/tls"
|
|
"fmt"
|
|
"net"
|
|
"strconv"
|
|
"time"
|
|
|
|
"github.com/HyNetwork/hysteria/pkg/conns/faketcp"
|
|
"github.com/HyNetwork/hysteria/pkg/conns/udp"
|
|
"github.com/HyNetwork/hysteria/pkg/conns/wechat"
|
|
obfsPkg "github.com/HyNetwork/hysteria/pkg/obfs"
|
|
"github.com/HyNetwork/hysteria/pkg/sockopt"
|
|
"github.com/HyNetwork/hysteria/pkg/utils"
|
|
"github.com/lucas-clemente/quic-go"
|
|
)
|
|
|
|
type ServerTransport struct {
|
|
Dialer *net.Dialer
|
|
SOCKS5Client *SOCKS5Client
|
|
ResolvePreference ResolvePreference
|
|
LocalUDPAddr *net.UDPAddr
|
|
LocalUDPIntf *net.Interface
|
|
}
|
|
|
|
// AddrEx is like net.TCPAddr or net.UDPAddr, but with additional domain information for SOCKS5.
|
|
// At least one of Domain and IPAddr must be non-empty.
|
|
type AddrEx struct {
|
|
Domain string
|
|
IPAddr *net.IPAddr
|
|
Port int
|
|
}
|
|
|
|
func (a *AddrEx) String() string {
|
|
if a == nil {
|
|
return "<nil>"
|
|
}
|
|
var ip string
|
|
if a.IPAddr != nil {
|
|
ip = a.IPAddr.String()
|
|
}
|
|
return net.JoinHostPort(ip, strconv.Itoa(a.Port))
|
|
}
|
|
|
|
type PUDPConn interface {
|
|
ReadFromUDP([]byte) (int, *net.UDPAddr, error)
|
|
WriteToUDP([]byte, *AddrEx) (int, error)
|
|
Close() error
|
|
}
|
|
|
|
type udpConnPUDPConn struct {
|
|
Conn *net.UDPConn
|
|
}
|
|
|
|
func (c *udpConnPUDPConn) ReadFromUDP(bytes []byte) (int, *net.UDPAddr, error) {
|
|
return c.Conn.ReadFromUDP(bytes)
|
|
}
|
|
|
|
func (c *udpConnPUDPConn) WriteToUDP(bytes []byte, ex *AddrEx) (int, error) {
|
|
return c.Conn.WriteToUDP(bytes, &net.UDPAddr{
|
|
IP: ex.IPAddr.IP,
|
|
Port: ex.Port,
|
|
Zone: ex.IPAddr.Zone,
|
|
})
|
|
}
|
|
|
|
func (c *udpConnPUDPConn) Close() error {
|
|
return c.Conn.Close()
|
|
}
|
|
|
|
var DefaultServerTransport = &ServerTransport{
|
|
Dialer: &net.Dialer{
|
|
Timeout: 8 * time.Second,
|
|
},
|
|
ResolvePreference: ResolvePreferenceDefault,
|
|
}
|
|
|
|
func (st *ServerTransport) quicPacketConn(proto string, laddr string, obfs obfsPkg.Obfuscator) (net.PacketConn, error) {
|
|
if len(proto) == 0 || proto == "udp" {
|
|
laddrU, err := net.ResolveUDPAddr("udp", laddr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
conn, err := net.ListenUDP("udp", laddrU)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if obfs != nil {
|
|
oc := udp.NewObfsUDPConn(conn, obfs)
|
|
return oc, nil
|
|
} else {
|
|
return conn, nil
|
|
}
|
|
} else if proto == "wechat-video" {
|
|
laddrU, err := net.ResolveUDPAddr("udp", laddr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
conn, err := net.ListenUDP("udp", laddrU)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if obfs == nil {
|
|
obfs = obfsPkg.NewDummyObfuscator()
|
|
}
|
|
return wechat.NewObfsWeChatUDPConn(conn, obfs), nil
|
|
} else if proto == "faketcp" {
|
|
conn, err := faketcp.Listen("tcp", laddr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if obfs != nil {
|
|
oc := faketcp.NewObfsFakeTCPConn(conn, obfs)
|
|
return oc, nil
|
|
} else {
|
|
return conn, nil
|
|
}
|
|
} else {
|
|
return nil, fmt.Errorf("unsupported protocol: %s", proto)
|
|
}
|
|
}
|
|
|
|
func (st *ServerTransport) QUICListen(proto string, listen string, tlsConfig *tls.Config, quicConfig *quic.Config, obfs obfsPkg.Obfuscator) (quic.Listener, error) {
|
|
pktConn, err := st.quicPacketConn(proto, listen, obfs)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
l, err := quic.Listen(pktConn, tlsConfig, quicConfig)
|
|
if err != nil {
|
|
_ = pktConn.Close()
|
|
return nil, err
|
|
}
|
|
return l, nil
|
|
}
|
|
|
|
func (st *ServerTransport) ResolveIPAddr(address string) (*net.IPAddr, bool, error) {
|
|
ip, zone := utils.ParseIPZone(address)
|
|
if ip != nil {
|
|
return &net.IPAddr{IP: ip, Zone: zone}, false, nil
|
|
}
|
|
ipAddr, err := resolveIPAddrWithPreference(address, st.ResolvePreference)
|
|
return ipAddr, true, err
|
|
}
|
|
|
|
func (st *ServerTransport) DialTCP(raddr *AddrEx) (*net.TCPConn, error) {
|
|
if st.SOCKS5Client != nil {
|
|
return st.SOCKS5Client.DialTCP(raddr)
|
|
} else {
|
|
conn, err := st.Dialer.Dial("tcp", raddr.String())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return conn.(*net.TCPConn), nil
|
|
}
|
|
}
|
|
|
|
func (st *ServerTransport) ListenUDP() (PUDPConn, error) {
|
|
if st.SOCKS5Client != nil {
|
|
return st.SOCKS5Client.ListenUDP()
|
|
} else {
|
|
conn, err := net.ListenUDP("udp", st.LocalUDPAddr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if st.LocalUDPIntf != nil {
|
|
err = sockopt.BindUDPConn("udp", conn, st.LocalUDPIntf)
|
|
if err != nil {
|
|
conn.Close()
|
|
return nil, err
|
|
}
|
|
}
|
|
return &udpConnPUDPConn{
|
|
Conn: conn,
|
|
}, nil
|
|
}
|
|
}
|
|
|
|
func (st *ServerTransport) SOCKS5Enabled() bool {
|
|
return st.SOCKS5Client != nil
|
|
}
|