Updated grpc servers to support new Manager API

This commit is contained in:
2025-09-05 07:07:33 +00:00
parent 582759bb3b
commit 40fda456eb
4 changed files with 70 additions and 29 deletions

View File

@@ -53,17 +53,19 @@ func main() {
}() }()
// Socket Streaming Server // Socket Streaming Server
socketStreamingServer := server.NewSocketStreamingServer(m) /*
go func() { socketStreamingServer := server.NewSocketStreamingServer(m)
socketLis, err := net.Listen("tcp", ":6000") go func() {
if err != nil { socketLis, err := net.Listen("tcp", ":6000")
log.Fatalf("Failed to listen for socket: %v", err) if err != nil {
} log.Fatalf("Failed to listen for socket: %v", err)
log.Println("Socket server listening on :6000") }
if err := socketStreamingServer.Serve(socketLis); err != nil { log.Println("Socket server listening on :6000")
log.Fatalf("Socket server error: %v", err) if err := socketStreamingServer.Serve(socketLis); err != nil {
} log.Fatalf("Socket server error: %v", err)
}() }
}()
*/
// Block main forever // Block main forever
select {} select {}

View File

@@ -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
} }

View File

@@ -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()},