174 lines
4.7 KiB
Go
174 lines
4.7 KiB
Go
/*
|
|
Copyright NetFoundry Inc.
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
https://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
package channel
|
|
|
|
import (
|
|
"fmt"
|
|
"github.com/michaelquigley/pfxlog"
|
|
"github.com/openziti/identity"
|
|
"github.com/openziti/transport/v2"
|
|
"github.com/pkg/errors"
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
type ReconnectingDialerConfig struct {
|
|
Identity *identity.TokenId
|
|
Endpoint transport.Address
|
|
LocalBinding string
|
|
Headers map[int32][]byte
|
|
TransportConfig transport.Configuration
|
|
ReconnectHandler func()
|
|
}
|
|
|
|
type reconnectingDialer struct {
|
|
identity *identity.TokenId
|
|
endpoint transport.Address
|
|
localBinding string
|
|
headers map[int32][]byte
|
|
tcfg transport.Configuration
|
|
reconnectLock sync.Mutex
|
|
reconnectHandler func()
|
|
}
|
|
|
|
func NewReconnectingDialer(config ReconnectingDialerConfig) UnderlayFactory {
|
|
return &reconnectingDialer{
|
|
identity: config.Identity,
|
|
endpoint: config.Endpoint,
|
|
headers: config.Headers,
|
|
reconnectHandler: config.ReconnectHandler,
|
|
tcfg: config.TransportConfig,
|
|
localBinding: config.LocalBinding,
|
|
}
|
|
}
|
|
|
|
func (dialer *reconnectingDialer) Create(timeout time.Duration) (Underlay, error) {
|
|
log := pfxlog.ContextLogger(dialer.endpoint.String())
|
|
log.Debug("started")
|
|
defer log.Debug("exited")
|
|
|
|
version := uint32(2)
|
|
|
|
peer, err := dialer.dial(timeout)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
impl := newReconnectingImpl(peer, dialer, timeout)
|
|
impl.setProtocolVersion(version)
|
|
|
|
if err := dialer.sendHello(impl); err != nil {
|
|
_ = peer.Close()
|
|
// If we bump channel protocol and need to handle multiple versions,
|
|
// we'll need to reintroduce version handling code here
|
|
// version, _ = GetRetryVersion(err)
|
|
return nil, err
|
|
}
|
|
|
|
return impl, nil
|
|
}
|
|
|
|
func (dialer *reconnectingDialer) dial(timeout time.Duration) (transport.Conn, error) {
|
|
return dialer.endpoint.DialWithLocalBinding("reconnecting", dialer.localBinding, dialer.identity, timeout, dialer.tcfg)
|
|
}
|
|
|
|
func (dialer *reconnectingDialer) Reconnect(impl *reconnectingImpl) error {
|
|
log := pfxlog.ContextLogger(impl.Label() + " @" + dialer.endpoint.String())
|
|
log.Debug("starting")
|
|
defer log.Debug("exiting")
|
|
|
|
dialer.reconnectLock.Lock()
|
|
defer dialer.reconnectLock.Unlock()
|
|
|
|
if err := impl.pingInstance(); err == nil {
|
|
return nil
|
|
} else {
|
|
log.Errorf("unable to ping (%s)", err)
|
|
}
|
|
|
|
impl.reconnecting.Store(true)
|
|
defer func() {
|
|
impl.reconnecting.Store(false)
|
|
if dialer.reconnectHandler != nil {
|
|
dialer.reconnectHandler()
|
|
}
|
|
}()
|
|
|
|
attempt := 0
|
|
for {
|
|
attempt++
|
|
peer, err := dialer.dial(impl.timeout)
|
|
if err == nil {
|
|
impl.peer = peer
|
|
if err := dialer.sendHello(impl); err == nil {
|
|
return nil
|
|
} else {
|
|
if version, ok := GetRetryVersion(err); ok {
|
|
impl.setProtocolVersion(version)
|
|
}
|
|
log.Errorf("hello attempt [#%d] failed (%s)", attempt, err)
|
|
time.Sleep(5 * time.Second)
|
|
}
|
|
|
|
} else {
|
|
log.Errorf("reconnection attempt [#%d] failed (%s)", attempt, err)
|
|
time.Sleep(5 * time.Second)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (dialer *reconnectingDialer) sendHello(impl *reconnectingImpl) error {
|
|
log := pfxlog.ContextLogger(impl.Label())
|
|
defer log.Debug("exited")
|
|
log.Debug("started")
|
|
|
|
request := NewHello(dialer.identity.Token, dialer.headers)
|
|
request.sequence = HelloSequence
|
|
if impl.connectionId != "" {
|
|
request.Headers[ConnectionIdHeader] = []byte(impl.connectionId)
|
|
log.Debugf("adding connectionId header [%s]", impl.connectionId)
|
|
}
|
|
if err := impl.tx(request); err != nil {
|
|
_ = impl.peer.Close()
|
|
return err
|
|
}
|
|
|
|
response, err := impl.rx()
|
|
if err != nil {
|
|
if errors.Is(err, BadMagicNumberError) {
|
|
return errors.Errorf("could not negotiate connection with %v, invalid header", impl.peer.RemoteAddr().String())
|
|
}
|
|
return err
|
|
}
|
|
if !response.IsReplyingTo(request.sequence) || response.ContentType != ContentTypeResultType {
|
|
return fmt.Errorf("channel synchronization error, expected %v, got %v", request.sequence, response.ReplyFor())
|
|
}
|
|
result := UnmarshalResult(response)
|
|
if !result.Success {
|
|
return errors.New(result.Message)
|
|
}
|
|
impl.connectionId = string(response.Headers[ConnectionIdHeader])
|
|
|
|
if id, ok := response.GetStringHeader(IdHeader); ok {
|
|
impl.id = &identity.TokenId{Token: id}
|
|
}
|
|
|
|
impl.headers.Store(response.Headers)
|
|
|
|
return nil
|
|
}
|