mirror of
https://github.com/apernet/hysteria.git
synced 2025-04-03 20:47:38 +03:00
feat: reconnect
This commit is contained in:
parent
62fddff137
commit
a59111faad
4 changed files with 136 additions and 5 deletions
|
@ -35,7 +35,7 @@ type HyUDPConn interface {
|
|||
}
|
||||
|
||||
func NewClient(config *Config) (Client, error) {
|
||||
if err := config.fill(); err != nil {
|
||||
if err := config.verifyAndFill(); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
c := &clientImpl{
|
||||
|
|
|
@ -24,11 +24,16 @@ type Config struct {
|
|||
QUICConfig QUICConfig
|
||||
BandwidthConfig BandwidthConfig
|
||||
FastOpen bool
|
||||
|
||||
filled bool // whether the fields have been verified and filled
|
||||
}
|
||||
|
||||
// fill fills the fields that are not set by the user with default values when possible,
|
||||
// and returns an error if the user has not set a required field.
|
||||
func (c *Config) fill() error {
|
||||
// verifyAndFill fills the fields that are not set by the user with default values when possible,
|
||||
// and returns an error if the user has not set a required field or has set an invalid value.
|
||||
func (c *Config) verifyAndFill() error {
|
||||
if c.filled {
|
||||
return nil
|
||||
}
|
||||
if c.ConnFactory == nil {
|
||||
c.ConnFactory = &udpConnFactory{}
|
||||
}
|
||||
|
@ -66,6 +71,8 @@ func (c *Config) fill() error {
|
|||
return errors.ConfigError{Field: "QUICConfig.KeepAlivePeriod", Reason: "must be between 2s and 60s"}
|
||||
}
|
||||
c.QUICConfig.DisablePathMTUDiscovery = c.QUICConfig.DisablePathMTUDiscovery || pmtud.DisablePathMTUDiscovery
|
||||
|
||||
c.filled = true
|
||||
return nil
|
||||
}
|
||||
|
||||
|
|
115
core/client/reconnect.go
Normal file
115
core/client/reconnect.go
Normal file
|
@ -0,0 +1,115 @@
|
|||
package client
|
||||
|
||||
import (
|
||||
"net"
|
||||
"sync"
|
||||
|
||||
coreErrs "github.com/apernet/hysteria/core/errors"
|
||||
)
|
||||
|
||||
// reconnectableClientImpl is a wrapper of Client, which can reconnect when the connection is closed,
|
||||
// except when the caller explicitly calls Close() to permanently close this client.
|
||||
type reconnectableClientImpl struct {
|
||||
config *Config
|
||||
client Client
|
||||
count int
|
||||
connectedFunc func(int) // called when successfully connected
|
||||
m sync.Mutex
|
||||
closed bool // permanent close
|
||||
}
|
||||
|
||||
func NewReconnectableClient(config *Config, connectedFunc func(int), lazy bool) (Client, error) {
|
||||
// Make sure we capture any error in config and return it here,
|
||||
// so that the caller doesn't have to wait until the first call
|
||||
// to TCP() or UDP() to get the error (when lazy is true).
|
||||
if err := config.verifyAndFill(); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
rc := &reconnectableClientImpl{
|
||||
config: config,
|
||||
connectedFunc: connectedFunc,
|
||||
}
|
||||
if !lazy {
|
||||
if err := rc.reconnect(); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
return rc, nil
|
||||
}
|
||||
|
||||
func (rc *reconnectableClientImpl) reconnect() error {
|
||||
if rc.client != nil {
|
||||
_ = rc.client.Close()
|
||||
}
|
||||
var err error
|
||||
rc.client, err = NewClient(rc.config)
|
||||
if err != nil {
|
||||
return err
|
||||
} else {
|
||||
rc.count++
|
||||
if rc.connectedFunc != nil {
|
||||
rc.connectedFunc(rc.count)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
}
|
||||
|
||||
func (rc *reconnectableClientImpl) TCP(addr string) (net.Conn, error) {
|
||||
rc.m.Lock()
|
||||
defer rc.m.Unlock()
|
||||
if rc.closed {
|
||||
return nil, coreErrs.ClosedError{}
|
||||
}
|
||||
if rc.client == nil {
|
||||
// First time
|
||||
if err := rc.reconnect(); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
conn, err := rc.client.TCP(addr)
|
||||
if _, ok := err.(coreErrs.ClosedError); ok {
|
||||
// Connection closed, reconnect
|
||||
if err := rc.reconnect(); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return rc.client.TCP(addr)
|
||||
} else {
|
||||
// OK or some other temporary error
|
||||
return conn, err
|
||||
}
|
||||
}
|
||||
|
||||
func (rc *reconnectableClientImpl) UDP() (HyUDPConn, error) {
|
||||
rc.m.Lock()
|
||||
defer rc.m.Unlock()
|
||||
if rc.closed {
|
||||
return nil, coreErrs.ClosedError{}
|
||||
}
|
||||
if rc.client == nil {
|
||||
// First time
|
||||
if err := rc.reconnect(); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
conn, err := rc.client.UDP()
|
||||
if _, ok := err.(coreErrs.ClosedError); ok {
|
||||
// Connection closed, reconnect
|
||||
if err := rc.reconnect(); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return rc.client.UDP()
|
||||
} else {
|
||||
// OK or some other temporary error
|
||||
return conn, err
|
||||
}
|
||||
}
|
||||
|
||||
func (rc *reconnectableClientImpl) Close() error {
|
||||
rc.m.Lock()
|
||||
defer rc.m.Unlock()
|
||||
rc.closed = true
|
||||
if rc.client != nil {
|
||||
return rc.client.Close()
|
||||
}
|
||||
return nil
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue