hysteria/sdk/client.go

261 lines
7.4 KiB
Go

// Package sdk provides an official API for integrating Hysteria client into other projects.
// It aims to be as stable & simple as possible, so that it can be easily maintained and
// widely adopted.
package sdk
import (
"crypto/tls"
"crypto/x509"
"errors"
"net"
"time"
"github.com/apernet/hysteria/pkg/core"
"github.com/lucas-clemente/quic-go"
)
const (
defaultALPN = "hysteria"
defaultStreamReceiveWindow = 16777216 // 16 MB
defaultConnectionReceiveWindow = defaultStreamReceiveWindow * 5 / 2 // 40 MB
defaultClientIdleTimeoutSec = 20
defaultClientHopIntervalSec = 10
)
type (
Protocol string
ResolveFunc func(network string, address string) (net.Addr, error)
ListenUDPFunc func(network string, laddr *net.UDPAddr) (*net.UDPConn, error)
)
const (
ProtocolUDP Protocol = "udp"
ProtocolWeChat Protocol = "wechat"
ProtocolFakeTCP Protocol = "faketcp"
)
// Client is a Hysteria client.
type Client interface {
// DialTCP dials a TCP connection to the specified address.
// The remote address must be in "host:port" format.
DialTCP(addr string) (net.Conn, error)
// DialUDP dials a UDP connection.
// It is bound to a fixed port on the server side.
// Can be used to send and receive UDP packets to/from any address.
DialUDP() (HyUDPConn, error)
// Close closes the client.
Close() error
}
// HyUDPConn is a Hysteria-proxied UDP connection.
type HyUDPConn interface {
// ReadFrom reads a packet from the connection.
// It returns the data, the source address (in "host:port" format) and any error encountered.
ReadFrom() ([]byte, string, error)
// WriteTo writes a packet to the connection.
// The remote address must be in "host:port" format.
WriteTo([]byte, string) error
// Close closes the connection.
Close() error
}
// ClientConfig is the configuration for a Hysteria client.
type ClientConfig struct {
// ServerAddress is the address of the Hysteria server.
// It must be in "host:port" format.
ServerAddress string
// ResolveFunc is the function used to resolve the server address.
// If not set, the default resolver will be used.
ResolveFunc ResolveFunc
// ListenUDPFunc is the function used to listen on a UDP port.
// If not set, the default listener will be used.
// Please note that ProtocolFakeTCP does NOT use this function,
// as it is not a UDP-based protocol and has its own stack.
ListenUDPFunc ListenUDPFunc
// Protocol is the protocol to use.
// It must be one of the following:
// - ProtocolUDP
// - ProtocolWeChat
// - ProtocolFakeTCP
Protocol Protocol
// Obfs is the obfuscation password.
// Empty = no obfuscation.
Obfs string
// HopInterval is the port hopping interval.
// 0 = default 10s.
HopInterval time.Duration
// Auth is the authentication payload to be sent to the server.
// It can be empty or nil if no authentication is required.
Auth []byte
// SendBPS is the maximum sending speed in bytes per second.
// Required and cannot be 0.
SendBPS uint64
// RecvBPS is the maximum receiving speed in bytes per second.
// Required and cannot be 0.
RecvBPS uint64
// ALPN is the ALPN protocol to be used.
// Empty = default "hysteria".
ALPN string
// ServerName is the SNI to be used.
// Empty = get from ServerAddress.
ServerName string
// Insecure is whether to skip certificate verification.
// It is not recommended to set this to true.
Insecure bool
// RootCAs is the root CA certificates to be used.
// Empty = use system default.
RootCAs *x509.CertPool
// ReceiveWindowConn is the flow control receive window size for each connection.
// 0 = default 16MB.
ReceiveWindowConn uint64
// ReceiveWindow is the flow control receive window size for the whole client.
// 0 = default 40MB.
ReceiveWindow uint64
// HandshakeTimeout is the timeout for the initial handshake.
// 0 = default 5s.
HandshakeTimeout time.Duration
// IdleTimeout is the timeout for idle connections.
// The client will send a heartbeat packet every 2/5 of this value.
// If the server does not respond within IdleTimeout, the connection will be closed.
// 0 = default 20s.
IdleTimeout time.Duration
// DisableMTUDiscovery is whether to disable MTU discovery.
// Only disable this if you are having MTU issues.
DisableMTUDiscovery bool
// TLSConfig, if not nil, will override all TLS-related fields above!!!
// Only set this if you know what you are doing.
TLSConfig *tls.Config
// QUICConfig, if not nil, will override all QUIC-related fields above!!!
// Only set this if you know what you are doing.
QUICConfig *quic.Config
}
// fill in the default values (if not set) for the configuration.
func (c *ClientConfig) fill() {
if c.ResolveFunc == nil {
c.ResolveFunc = func(network string, address string) (net.Addr, error) {
switch network {
case "tcp", "tcp4", "tcp6":
return net.ResolveTCPAddr(network, address)
case "udp", "udp4", "udp6":
return net.ResolveUDPAddr(network, address)
case "ip", "ip4", "ip6":
return net.ResolveIPAddr(network, address)
default:
return nil, errors.New("unsupported network type")
}
}
}
if c.ListenUDPFunc == nil {
c.ListenUDPFunc = net.ListenUDP
}
if c.Protocol == "" {
c.Protocol = ProtocolUDP
}
if c.HopInterval == 0 {
c.HopInterval = defaultClientHopIntervalSec * time.Second
}
if c.ALPN == "" {
c.ALPN = defaultALPN
}
if c.ReceiveWindowConn == 0 {
c.ReceiveWindowConn = defaultStreamReceiveWindow
}
if c.ReceiveWindow == 0 {
c.ReceiveWindow = defaultConnectionReceiveWindow
}
if c.IdleTimeout == 0 {
c.IdleTimeout = defaultClientIdleTimeoutSec * time.Second
}
}
// NewClient creates a new Hysteria client.
func NewClient(config ClientConfig) (Client, error) {
// Fill in default values
config.fill()
// TLS config
var tlsConfig *tls.Config
if config.TLSConfig != nil {
tlsConfig = config.TLSConfig
} else {
tlsConfig = &tls.Config{
NextProtos: []string{config.ALPN},
ServerName: config.ServerName,
InsecureSkipVerify: config.Insecure,
RootCAs: config.RootCAs,
MinVersion: tls.VersionTLS13,
}
}
// QUIC config
var quicConfig *quic.Config
if config.QUICConfig != nil {
quicConfig = config.QUICConfig
} else {
quicConfig = &quic.Config{
InitialStreamReceiveWindow: config.ReceiveWindowConn,
MaxStreamReceiveWindow: config.ReceiveWindowConn,
InitialConnectionReceiveWindow: config.ReceiveWindow,
MaxConnectionReceiveWindow: config.ReceiveWindow,
HandshakeIdleTimeout: config.HandshakeTimeout,
MaxIdleTimeout: config.IdleTimeout,
KeepAlivePeriod: config.IdleTimeout * 2 / 5,
DisablePathMTUDiscovery: config.DisableMTUDiscovery,
EnableDatagrams: true,
}
}
// Packet conn func
pff := clientPacketConnFuncFactoryMap[config.Protocol]
if pff == nil {
return nil, errors.New("unsupported protocol")
}
pf := pff(config.Obfs, config.HopInterval, config.ResolveFunc, config.ListenUDPFunc)
c, err := core.NewClient(config.ServerAddress, config.Auth, tlsConfig, quicConfig, pf,
config.SendBPS, config.RecvBPS, nil)
if err != nil {
return nil, err
}
return &clientImpl{c}, nil
}
type clientImpl struct {
*core.Client
}
func (c *clientImpl) DialTCP(addr string) (net.Conn, error) {
return c.Client.DialTCP(addr)
}
func (c *clientImpl) DialUDP() (HyUDPConn, error) {
conn, err := c.Client.DialUDP()
return HyUDPConn(conn), err
}
func (c *clientImpl) Close() error {
return c.Client.Close()
}