Updated grpc servers to support new Manager API
This commit is contained in:
@@ -53,6 +53,7 @@ func main() {
|
|||||||
}()
|
}()
|
||||||
|
|
||||||
// Socket Streaming Server
|
// Socket Streaming Server
|
||||||
|
/*
|
||||||
socketStreamingServer := server.NewSocketStreamingServer(m)
|
socketStreamingServer := server.NewSocketStreamingServer(m)
|
||||||
go func() {
|
go func() {
|
||||||
socketLis, err := net.Listen("tcp", ":6000")
|
socketLis, err := net.Listen("tcp", ":6000")
|
||||||
@@ -64,6 +65,7 @@ func main() {
|
|||||||
log.Fatalf("Socket server error: %v", err)
|
log.Fatalf("Socket server error: %v", err)
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
*/
|
||||||
|
|
||||||
// Block main forever
|
// Block main forever
|
||||||
select {}
|
select {}
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ package server
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
"time"
|
||||||
|
|
||||||
"github.com/google/uuid"
|
"github.com/google/uuid"
|
||||||
pb "gitlab.michelsen.id/phillmichelsen/tessera/pkg/pb/data_service"
|
pb "gitlab.michelsen.id/phillmichelsen/tessera/pkg/pb/data_service"
|
||||||
@@ -21,21 +21,28 @@ func NewGRPCControlServer(m *manager.Manager) *GRPCControlServer {
|
|||||||
return &GRPCControlServer{manager: m}
|
return &GRPCControlServer{manager: m}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *GRPCControlServer) StartStream(_ context.Context, _ *pb.StartStreamRequest) (*pb.StartStreamResponse, error) {
|
// StartStream creates a new session. It does NOT attach client channels.
|
||||||
streamID, err := s.manager.StartClientStream()
|
// Your streaming RPC should later call GetChannels(sessionID, opts).
|
||||||
|
func (s *GRPCControlServer) StartStream(_ context.Context, req *pb.StartStreamRequest) (*pb.StartStreamResponse, error) {
|
||||||
|
sessionID, err := s.manager.NewSession(time.Duration(1) * time.Minute) // timeout set to 1 minute
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("failed to start stream: %w", err)
|
return nil, status.Errorf(codes.Internal, "new session: %v", err)
|
||||||
}
|
}
|
||||||
return &pb.StartStreamResponse{StreamUuid: streamID.String()}, nil
|
return &pb.StartStreamResponse{StreamUuid: sessionID.String()}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ConfigureStream sets the session's subscriptions in one shot.
|
||||||
|
// It does NOT require channels to be attached.
|
||||||
func (s *GRPCControlServer) ConfigureStream(_ context.Context, req *pb.ConfigureStreamRequest) (*pb.ConfigureStreamResponse, error) {
|
func (s *GRPCControlServer) ConfigureStream(_ context.Context, req *pb.ConfigureStreamRequest) (*pb.ConfigureStreamResponse, error) {
|
||||||
|
if req == nil {
|
||||||
|
return nil, status.Error(codes.InvalidArgument, "nil request")
|
||||||
|
}
|
||||||
streamID, err := uuid.Parse(req.StreamUuid)
|
streamID, err := uuid.Parse(req.StreamUuid)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, status.Errorf(codes.InvalidArgument, "invalid stream_uuid %q: %v", req.StreamUuid, err)
|
return nil, status.Errorf(codes.InvalidArgument, "invalid stream_uuid %q: %v", req.StreamUuid, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
var ids []domain.Identifier
|
ids := make([]domain.Identifier, 0, len(req.Identifiers))
|
||||||
for _, in := range req.Identifiers {
|
for _, in := range req.Identifiers {
|
||||||
id, e := domain.ParseIdentifier(in.Key)
|
id, e := domain.ParseIdentifier(in.Key)
|
||||||
if e != nil {
|
if e != nil {
|
||||||
@@ -44,19 +51,37 @@ func (s *GRPCControlServer) ConfigureStream(_ context.Context, req *pb.Configure
|
|||||||
ids = append(ids, id)
|
ids = append(ids, id)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := s.manager.ConfigureClientStream(streamID, ids); err != nil {
|
if err := s.manager.SetSubscriptions(streamID, ids); err != nil {
|
||||||
return nil, status.Errorf(codes.InvalidArgument, "configure failed: %v", err)
|
// Map common manager errors to gRPC codes.
|
||||||
|
switch err {
|
||||||
|
case manager.ErrSessionNotFound:
|
||||||
|
return nil, status.Errorf(codes.NotFound, "session not found: %v", err)
|
||||||
|
case manager.ErrSessionClosed:
|
||||||
|
return nil, status.Errorf(codes.FailedPrecondition, "session closed: %v", err)
|
||||||
|
default:
|
||||||
|
return nil, status.Errorf(codes.Internal, "set subscriptions: %v", err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return &pb.ConfigureStreamResponse{}, nil
|
return &pb.ConfigureStreamResponse{}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// StopStream closes the session and tears down routes and streams.
|
||||||
func (s *GRPCControlServer) StopStream(_ context.Context, req *pb.StopStreamRequest) (*pb.StopStreamResponse, error) {
|
func (s *GRPCControlServer) StopStream(_ context.Context, req *pb.StopStreamRequest) (*pb.StopStreamResponse, error) {
|
||||||
|
if req == nil {
|
||||||
|
return nil, status.Error(codes.InvalidArgument, "nil request")
|
||||||
|
}
|
||||||
streamID, err := uuid.Parse(req.StreamUuid)
|
streamID, err := uuid.Parse(req.StreamUuid)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, status.Errorf(codes.InvalidArgument, "invalid stream_uuid %q: %v", req.StreamUuid, err)
|
return nil, status.Errorf(codes.InvalidArgument, "invalid stream_uuid %q: %v", req.StreamUuid, err)
|
||||||
}
|
}
|
||||||
if err := s.manager.StopClientStream(streamID); err != nil {
|
|
||||||
return nil, status.Errorf(codes.Internal, "failed to stop stream: %v", err)
|
if err := s.manager.CloseSession(streamID); err != nil {
|
||||||
|
switch err {
|
||||||
|
case manager.ErrSessionNotFound:
|
||||||
|
return nil, status.Errorf(codes.NotFound, "session not found: %v", err)
|
||||||
|
default:
|
||||||
|
return nil, status.Errorf(codes.Internal, "close session: %v", err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return &pb.StopStreamResponse{}, nil
|
return &pb.StopStreamResponse{}, nil
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -17,25 +17,39 @@ func NewGRPCStreamingServer(m *manager.Manager) *GRPCStreamingServer {
|
|||||||
return &GRPCStreamingServer{manager: m}
|
return &GRPCStreamingServer{manager: m}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ConnectStream attaches a client to an existing session and streams outbound messages.
|
||||||
|
// This is server-streaming only; inbound use is optional and ignored here.
|
||||||
func (s *GRPCStreamingServer) ConnectStream(req *pb.ConnectStreamRequest, stream pb.DataServiceStreaming_ConnectStreamServer) error {
|
func (s *GRPCStreamingServer) ConnectStream(req *pb.ConnectStreamRequest, stream pb.DataServiceStreaming_ConnectStreamServer) error {
|
||||||
streamUUID, err := uuid.Parse(req.StreamUuid)
|
if req == nil {
|
||||||
|
return fmt.Errorf("nil request")
|
||||||
|
}
|
||||||
|
sessionID, err := uuid.Parse(req.StreamUuid)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("invalid UUID: %w", err)
|
return fmt.Errorf("invalid UUID: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
ch, err := s.manager.ConnectClientStream(streamUUID)
|
// Defaults; tune or map from req if your proto carries options.
|
||||||
if err != nil {
|
opts := manager.ChannelOpts{
|
||||||
return fmt.Errorf("failed to connect: %w", err)
|
InBufSize: 256,
|
||||||
|
OutBufSize: 1024,
|
||||||
|
DropOutbound: true, // do not let slow clients stall producers
|
||||||
|
DropInbound: true, // irrelevant here (we don't send inbound), safe default
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_, out, err := s.manager.GetChannels(sessionID, opts)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("attach channels: %w", err)
|
||||||
|
}
|
||||||
|
defer func() { _ = s.manager.DetachClient(sessionID) }()
|
||||||
|
|
||||||
|
ctx := stream.Context()
|
||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
case <-stream.Context().Done():
|
case <-ctx.Done():
|
||||||
s.manager.DisconnectClientStream(streamUUID)
|
|
||||||
return nil
|
return nil
|
||||||
case msg, ok := <-ch:
|
case msg, ok := <-out:
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil
|
return nil // session closed
|
||||||
}
|
}
|
||||||
if err := stream.Send(&pb.Message{
|
if err := stream.Send(&pb.Message{
|
||||||
Identifier: &pb.Identifier{Key: msg.Identifier.Key()},
|
Identifier: &pb.Identifier{Key: msg.Identifier.Key()},
|
||||||
|
|||||||
Reference in New Issue
Block a user