netbird/relay/server/relay.go

134 lines
2.8 KiB
Go
Raw Normal View History

package server
import (
"context"
"fmt"
"net"
"sync"
log "github.com/sirupsen/logrus"
2024-07-24 16:26:26 +02:00
"go.opentelemetry.io/otel/metric"
2024-07-05 16:12:30 +02:00
"github.com/netbirdio/netbird/relay/auth"
"github.com/netbirdio/netbird/relay/messages"
2024-07-24 16:26:26 +02:00
"github.com/netbirdio/netbird/relay/metrics"
)
type Relay struct {
2024-07-24 16:26:26 +02:00
metrics *metrics.Metrics
2024-07-05 16:12:30 +02:00
validator auth.Validator
2024-07-24 16:34:47 +02:00
store *Store
instanceURL string
closed bool
closeMu sync.RWMutex
}
2024-07-24 16:26:26 +02:00
func NewRelay(meter metric.Meter, exposedAddress string, tlsSupport bool, validator auth.Validator) (*Relay, error) {
m, err := metrics.NewMetrics(meter)
if err != nil {
return nil, fmt.Errorf("creating app metrics: %v", err)
}
2024-07-02 11:57:17 +02:00
r := &Relay{
2024-07-24 16:26:26 +02:00
metrics: m,
2024-07-05 16:12:30 +02:00
validator: validator,
store: NewStore(),
}
2024-07-02 11:57:17 +02:00
if tlsSupport {
2024-07-24 16:34:47 +02:00
r.instanceURL = fmt.Sprintf("rels://%s", exposedAddress)
2024-07-02 11:57:17 +02:00
} else {
2024-07-24 16:34:47 +02:00
r.instanceURL = fmt.Sprintf("rel://%s", exposedAddress)
2024-07-02 11:57:17 +02:00
}
2024-07-05 16:12:30 +02:00
2024-07-24 16:26:26 +02:00
return r, nil
}
func (r *Relay) Accept(conn net.Conn) {
r.closeMu.RLock()
defer r.closeMu.RUnlock()
if r.closed {
return
}
2024-07-02 11:57:17 +02:00
peerID, err := r.handShake(conn)
if err != nil {
log.Errorf("failed to handshake with %s: %s", conn.RemoteAddr(), err)
cErr := conn.Close()
if cErr != nil {
log.Errorf("failed to close connection, %s: %s", conn.RemoteAddr(), cErr)
}
return
}
peer := NewPeer(peerID, conn, r.store)
peer.log.Infof("peer connected from: %s", conn.RemoteAddr())
r.store.AddPeer(peer)
2024-07-24 16:26:26 +02:00
r.metrics.Peers.Add(context.Background(), 1)
go func() {
peer.Work()
r.store.DeletePeer(peer)
peer.log.Debugf("relay connection closed")
2024-07-24 16:26:26 +02:00
r.metrics.Peers.Add(context.Background(), -1)
}()
}
func (r *Relay) Close(ctx context.Context) {
2024-07-08 21:53:20 +02:00
log.Infof("close connection with all peers")
r.closeMu.Lock()
wg := sync.WaitGroup{}
peers := r.store.Peers()
for _, peer := range peers {
wg.Add(1)
go func(p *Peer) {
p.CloseGracefully(ctx)
wg.Done()
}(peer)
}
wg.Wait()
r.closeMu.Unlock()
}
2024-07-02 11:57:17 +02:00
func (r *Relay) handShake(conn net.Conn) ([]byte, error) {
buf := make([]byte, messages.MaxHandshakeSize)
n, err := conn.Read(buf)
if err != nil {
log.Errorf("failed to read message: %s", err)
return nil, err
}
msgType, err := messages.DetermineClientMsgType(buf[:n])
if err != nil {
return nil, err
}
2024-07-02 11:57:17 +02:00
if msgType != messages.MsgTypeHello {
tErr := fmt.Errorf("invalid message type")
log.Errorf("failed to handshake: %s", tErr)
return nil, tErr
}
2024-07-02 11:57:17 +02:00
2024-07-05 16:12:30 +02:00
peerID, authPayload, err := messages.UnmarshalHelloMsg(buf[:n])
if err != nil {
log.Errorf("failed to handshake: %s", err)
return nil, err
}
2024-07-05 16:12:30 +02:00
if err := r.validator.Validate(authPayload); err != nil {
log.Errorf("failed to authenticate connection: %s", err)
return nil, err
2024-07-05 16:12:30 +02:00
}
2024-07-24 16:34:47 +02:00
msg, _ := messages.MarshalHelloResponse(r.instanceURL)
_, err = conn.Write(msg)
if err != nil {
return nil, err
}
return peerID, nil
}
func (r *Relay) InstanceURL() string {
2024-07-24 16:34:47 +02:00
return r.instanceURL
}