netbird/signal/server/signal.go

113 lines
3.3 KiB
Go
Raw Normal View History

package server
2021-05-01 12:45:37 +02:00
import (
"context"
"fmt"
"github.com/netbirdio/netbird/signal/peer"
"github.com/netbirdio/netbird/signal/proto"
2021-05-01 12:45:37 +02:00
log "github.com/sirupsen/logrus"
"google.golang.org/grpc/codes"
"google.golang.org/grpc/metadata"
"google.golang.org/grpc/status"
"io"
)
// Server an instance of a Signal server
type Server struct {
2021-05-01 12:45:37 +02:00
registry *peer.Registry
proto.UnimplementedSignalExchangeServer
2021-05-01 12:45:37 +02:00
}
// NewServer creates a new Signal server
func NewServer() *Server {
return &Server{
2021-05-01 12:45:37 +02:00
registry: peer.NewRegistry(),
}
}
// Send forwards a message to the signal peer
func (s *Server) Send(ctx context.Context, msg *proto.EncryptedMessage) (*proto.EncryptedMessage, error) {
2021-05-01 12:45:37 +02:00
if !s.registry.IsPeerRegistered(msg.Key) {
return nil, fmt.Errorf("peer %s is not registered", msg.Key)
2021-05-01 12:45:37 +02:00
}
if dstPeer, found := s.registry.Get(msg.RemoteKey); found {
2021-05-01 12:45:37 +02:00
//forward the message to the target peer
err := dstPeer.Stream.Send(msg)
if err != nil {
log.Errorf("error while forwarding message from peer [%s] to peer [%s] %v", msg.Key, msg.RemoteKey, err)
2021-05-01 12:45:37 +02:00
//todo respond to the sender?
}
} else {
log.Debugf("message from peer [%s] can't be forwarded to peer [%s] because destination peer is not connected", msg.Key, msg.RemoteKey)
2021-05-01 12:45:37 +02:00
//todo respond to the sender?
}
return &proto.EncryptedMessage{}, nil
2021-05-01 12:45:37 +02:00
}
// ConnectStream connects to the exchange stream
func (s *Server) ConnectStream(stream proto.SignalExchange_ConnectStreamServer) error {
2021-05-01 12:45:37 +02:00
p, err := s.connectPeer(stream)
if err != nil {
return err
}
defer func() {
log.Infof("peer disconnected [%s] ", p.Id)
s.registry.Deregister(p)
}()
//needed to confirm that the peer has been registered so that the client can proceed
header := metadata.Pairs(proto.HeaderRegistered, "1")
err = stream.SendHeader(header)
if err != nil {
return err
}
log.Infof("peer connected [%s]", p.Id)
2021-05-01 12:45:37 +02:00
for {
//read incoming messages
2021-05-01 12:45:37 +02:00
msg, err := stream.Recv()
if err == io.EOF {
break
} else if err != nil {
return err
}
log.Debugf("received a new message from peer [%s] to peer [%s]", p.Id, msg.RemoteKey)
// lookup the target peer where the message is going to
if dstPeer, found := s.registry.Get(msg.RemoteKey); found {
2021-05-01 12:45:37 +02:00
//forward the message to the target peer
err := dstPeer.Stream.Send(msg)
if err != nil {
log.Errorf("error while forwarding message from peer [%s] to peer [%s] %v", p.Id, msg.RemoteKey, err)
2021-05-01 12:45:37 +02:00
//todo respond to the sender?
}
} else {
log.Debugf("message from peer [%s] can't be forwarded to peer [%s] because destination peer is not connected", p.Id, msg.RemoteKey)
2021-05-01 12:45:37 +02:00
//todo respond to the sender?
}
}
<-stream.Context().Done()
return stream.Context().Err()
}
// Handles initial Peer connection.
2021-08-20 22:33:43 +02:00
// Each connection must provide an Id header.
2021-05-01 12:45:37 +02:00
// At this moment the connecting Peer will be registered in the peer.Registry
func (s Server) connectPeer(stream proto.SignalExchange_ConnectStreamServer) (*peer.Peer, error) {
2021-05-01 12:45:37 +02:00
if meta, hasMeta := metadata.FromIncomingContext(stream.Context()); hasMeta {
if id, found := meta[proto.HeaderId]; found {
p := peer.NewPeer(id[0], stream)
s.registry.Register(p)
return p, nil
} else {
return nil, status.Errorf(codes.FailedPrecondition, "missing connection header: "+proto.HeaderId)
}
} else {
return nil, status.Errorf(codes.FailedPrecondition, "missing connection stream meta")
}
}