2024-05-17 17:43:28 +02:00
|
|
|
package client
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
|
2024-05-21 15:51:37 +02:00
|
|
|
"github.com/netbirdio/netbird/relay/client/dialer/udp"
|
2024-05-17 17:43:28 +02:00
|
|
|
"github.com/netbirdio/netbird/relay/messages"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2024-05-21 15:51:37 +02:00
|
|
|
bufferSize = 1500 // optimise the buffer size
|
|
|
|
serverResponseTimeout = 8 * time.Second
|
2024-05-17 17:43:28 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type connContainer struct {
|
|
|
|
conn *Conn
|
|
|
|
messages chan []byte
|
|
|
|
}
|
|
|
|
|
|
|
|
// Client Todo:
|
|
|
|
// - handle automatic reconnection
|
|
|
|
type Client struct {
|
|
|
|
serverAddress string
|
|
|
|
peerID string
|
|
|
|
|
|
|
|
channelsPending map[string]chan net.Conn // todo: protect map with mutex
|
|
|
|
channels map[uint16]*connContainer
|
|
|
|
|
|
|
|
relayConn net.Conn
|
|
|
|
relayConnState bool
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewClient(serverAddress, peerID string) *Client {
|
|
|
|
return &Client{
|
|
|
|
serverAddress: serverAddress,
|
|
|
|
peerID: peerID,
|
|
|
|
channelsPending: make(map[string]chan net.Conn),
|
|
|
|
channels: make(map[uint16]*connContainer),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) Connect() error {
|
2024-05-21 15:51:37 +02:00
|
|
|
conn, err := udp.Dial(c.serverAddress)
|
2024-05-17 17:43:28 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
c.relayConn = conn
|
|
|
|
|
|
|
|
err = c.handShake()
|
|
|
|
if err != nil {
|
|
|
|
cErr := conn.Close()
|
|
|
|
if cErr != nil {
|
|
|
|
log.Errorf("failed to close connection: %s", cErr)
|
|
|
|
}
|
|
|
|
c.relayConn = nil
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
c.relayConnState = true
|
|
|
|
go c.readLoop()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) BindChannel(remotePeerID string) (net.Conn, error) {
|
|
|
|
if c.relayConn == nil {
|
|
|
|
return nil, fmt.Errorf("client not connected")
|
|
|
|
}
|
|
|
|
|
|
|
|
bindSuccessChan := make(chan net.Conn, 1)
|
|
|
|
c.channelsPending[remotePeerID] = bindSuccessChan
|
|
|
|
msg := messages.MarshalBindNewChannelMsg(remotePeerID)
|
|
|
|
_, err := c.relayConn.Write(msg)
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("failed to write out bind message: %s", err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2024-05-21 15:51:37 +02:00
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), serverResponseTimeout)
|
2024-05-17 17:43:28 +02:00
|
|
|
defer cancel()
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
return nil, fmt.Errorf("bind timeout")
|
|
|
|
case c := <-bindSuccessChan:
|
|
|
|
return c, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) Close() error {
|
2024-05-21 15:51:37 +02:00
|
|
|
if !c.relayConnState {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-05-17 17:43:28 +02:00
|
|
|
for _, conn := range c.channels {
|
|
|
|
close(conn.messages)
|
|
|
|
}
|
|
|
|
c.channels = make(map[uint16]*connContainer)
|
|
|
|
c.relayConnState = false
|
|
|
|
err := c.relayConn.Close()
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) handShake() error {
|
|
|
|
msg, err := messages.MarshalHelloMsg(c.peerID)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
_, err = c.relayConn.Write(msg)
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("failed to send hello message: %s", err)
|
|
|
|
return err
|
|
|
|
}
|
2024-05-21 15:51:37 +02:00
|
|
|
|
|
|
|
err = c.relayConn.SetReadDeadline(time.Now().Add(serverResponseTimeout))
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("failed to set read deadline: %s", err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
buf := make([]byte, 1500) // todo: optimise buffer size
|
|
|
|
n, err := c.relayConn.Read(buf)
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("failed to read hello response: %s", err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
msgType, err := messages.DetermineServerMsgType(buf[:n])
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("failed to determine message type: %s", err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if msgType != messages.MsgTypeHelloResponse {
|
|
|
|
log.Errorf("unexpected message type: %s", msgType)
|
|
|
|
return fmt.Errorf("unexpected message type")
|
|
|
|
}
|
2024-05-17 17:43:28 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) readLoop() {
|
|
|
|
log := log.WithField("client_id", c.peerID)
|
|
|
|
var errExit error
|
|
|
|
var n int
|
|
|
|
for {
|
2024-05-20 11:38:23 +02:00
|
|
|
buf := make([]byte, bufferSize) // todo optimise buffer size, use pool
|
2024-05-17 17:43:28 +02:00
|
|
|
n, errExit = c.relayConn.Read(buf)
|
|
|
|
if errExit != nil {
|
2024-05-19 12:41:06 +02:00
|
|
|
log.Debugf("failed to read message from relay server: %s", errExit)
|
2024-05-17 17:43:28 +02:00
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
msgType, err := messages.DetermineServerMsgType(buf[:n])
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("failed to determine message type: %s", err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
switch msgType {
|
|
|
|
case messages.MsgTypeBindResponse:
|
|
|
|
channelId, peerId, err := messages.UnmarshalBindResponseMsg(buf[:n])
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("failed to parse bind response message: %v", err)
|
|
|
|
} else {
|
|
|
|
c.handleBindResponse(channelId, peerId)
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
case messages.MsgTypeTransport:
|
2024-05-17 23:29:47 +02:00
|
|
|
channelId, err := messages.UnmarshalTransportID(buf[:n])
|
2024-05-17 17:43:28 +02:00
|
|
|
if err != nil {
|
|
|
|
log.Errorf("failed to parse transport message: %v", err)
|
|
|
|
continue
|
|
|
|
}
|
2024-05-19 12:41:06 +02:00
|
|
|
go c.handleTransport(channelId, buf[:n])
|
|
|
|
|
2024-05-17 17:43:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.relayConnState {
|
|
|
|
log.Errorf("failed to read message from relay server: %s", errExit)
|
|
|
|
_ = c.relayConn.Close()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) handleBindResponse(channelId uint16, peerId string) {
|
|
|
|
bindSuccessChan, ok := c.channelsPending[peerId]
|
|
|
|
if !ok {
|
|
|
|
log.Errorf("unexpected bind response from: %s", peerId)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
delete(c.channelsPending, peerId)
|
|
|
|
|
|
|
|
messageBuffer := make(chan []byte, 10)
|
|
|
|
conn := NewConn(c, channelId, c.generateConnReaderFN(messageBuffer))
|
|
|
|
|
|
|
|
c.channels[channelId] = &connContainer{
|
|
|
|
conn,
|
|
|
|
messageBuffer,
|
|
|
|
}
|
|
|
|
log.Debugf("bind success for '%s': %d", peerId, channelId)
|
|
|
|
|
|
|
|
bindSuccessChan <- conn
|
|
|
|
}
|
|
|
|
|
2024-05-17 23:29:47 +02:00
|
|
|
func (c *Client) handleTransport(channelId uint16, msg []byte) {
|
2024-05-17 17:43:28 +02:00
|
|
|
container, ok := c.channels[channelId]
|
|
|
|
if !ok {
|
|
|
|
log.Errorf("unexpected transport message for channel: %d", channelId)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
select {
|
2024-05-17 23:29:47 +02:00
|
|
|
case container.messages <- msg:
|
2024-05-17 17:43:28 +02:00
|
|
|
default:
|
|
|
|
log.Errorf("dropping message for channel: %d", channelId)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) writeTo(channelID uint16, payload []byte) (int, error) {
|
|
|
|
msg := messages.MarshalTransportMsg(channelID, payload)
|
|
|
|
n, err := c.relayConn.Write(msg)
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("failed to write transport message: %s", err)
|
|
|
|
}
|
|
|
|
return n, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) generateConnReaderFN(messageBufferChan chan []byte) func(b []byte) (n int, err error) {
|
|
|
|
return func(b []byte) (n int, err error) {
|
|
|
|
select {
|
|
|
|
case msg, ok := <-messageBufferChan:
|
|
|
|
if !ok {
|
|
|
|
return 0, io.EOF
|
|
|
|
}
|
2024-05-17 23:29:47 +02:00
|
|
|
|
|
|
|
payload, err := messages.UnmarshalTransportPayload(msg)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
n = copy(b, payload)
|
2024-05-17 17:43:28 +02:00
|
|
|
}
|
|
|
|
return n, nil
|
|
|
|
}
|
|
|
|
}
|