Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/autom8ter/queuerpc
a protoc plugin to generate type safe RPC client and server code that use a message queue for transport/service discovery.
https://github.com/autom8ter/queuerpc
golang message-queue protobuf protocol-buffer pubsub
Last synced: 30 days ago
JSON representation
a protoc plugin to generate type safe RPC client and server code that use a message queue for transport/service discovery.
- Host: GitHub
- URL: https://github.com/autom8ter/queuerpc
- Owner: autom8ter
- Created: 2023-02-25T20:36:41.000Z (almost 2 years ago)
- Default Branch: main
- Last Pushed: 2023-03-01T15:55:56.000Z (almost 2 years ago)
- Last Synced: 2024-11-07T15:09:11.949Z (3 months ago)
- Topics: golang, message-queue, protobuf, protocol-buffer, pubsub
- Language: Go
- Homepage:
- Size: 69.3 KB
- Stars: 1
- Watchers: 1
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
Awesome Lists containing this project
README
# queuerpc
a protoc plugin to generate type safe RPC client and server code that use a message queue for transport/service discovery.
## Installation
go install github.com/autom8ter/queuerpc/protoc-gen-queuerpc
## GoalsA protoc generator that feels like gRPC but communication between client and server happens over a message queue instead of http2.
## Usage
Given the Following protobuf service definition:
```protobuf
syntax = "proto3";package echo.v1;
option go_package = "github.com/autom8ter/example/echo/v1";message EchoRequest {
string message = 1;
}message EchoResponse {
string message = 1;
}service EchoService {
// Echo returns the same message it receives.
rpc Echo(EchoRequest) returns (EchoResponse) {}
// EchoServerStream streams the same message it initially receives.
rpc EchoServerStream(EchoRequest) returns (stream EchoResponse) {}
}
```a type safe client and server will be generated when using protoc-gen-queuerpc:
```go
package v1import "context"
import "github.com/autom8ter/queuerpc"
import "github.com/golang/protobuf/proto"// EchoServiceServer is a type safe RabbitMQ rpc server
type EchoServiceServer interface {
// Echo returns the same message it receives.
Echo(ctx context.Context, in *EchoRequest) (*EchoResponse, error)
// EchoServerStream streams the same message it initially receives.
EchoServerStream(ctx context.Context, in *EchoRequest) (<-chan *EchoResponse, error)
}// Serve starts the server and blocks until the context is canceled or the deadline is exceeded
func Serve(srv queuerpc.IServer, handler EchoServiceServer) error {
return srv.Serve(queuerpc.Handlers{
UnaryHandler: func(ctx context.Context, msg *queuerpc.Message) *queuerpc.Message {
meta := msg.Metadata
switch msg.Method {
case "Echo":
var in EchoRequest
if err := proto.Unmarshal(msg.Body, &in); err != nil {
return &queuerpc.Message{
Id: msg.Id,
Method: msg.Method,
Metadata: meta,
Error: queuerpc.ErrUnmarshal,
}
}
out, err := handler.Echo(queuerpc.NewContextWithMetadata(ctx, meta), &in)
if err != nil {
return &queuerpc.Message{
Id: msg.Id,
Method: msg.Method,
Metadata: meta,
Error: queuerpc.ErrorFrom(err),
}
}
body, err := proto.Marshal(out)
if err != nil {
return &queuerpc.Message{
Id: msg.Id,
Method: msg.Method,
Metadata: meta,
Error: queuerpc.ErrMarshal,
}
}
return &queuerpc.Message{
Id: msg.Id,
Method: msg.Method,
Metadata: meta,
Body: body,
}
}
return &queuerpc.Message{
Id: msg.Id,
Method: msg.Method,
Metadata: meta,
Error: queuerpc.ErrUnsupportedMethod,
}
},
ServerStreamHandler: func(ctx context.Context, msg *queuerpc.Message) (<-chan *queuerpc.Message, error) {
meta := msg.Metadata
ch := make(chan *queuerpc.Message)
switch msg.Method {
default:
return nil, queuerpc.ErrUnsupportedMethod
case "EchoServerStream":
var in EchoRequest
if err := proto.Unmarshal(msg.Body, &in); err != nil {
return nil, queuerpc.ErrUnmarshal
}
out, err := handler.EchoServerStream(queuerpc.NewContextWithMetadata(ctx, meta), &in)
if err != nil {
return nil, err
}
go func() {
defer close(ch)
for {
select {
case <-ctx.Done():
return
case out, ok := <-out:
if !ok {
return
}
body, _ := proto.Marshal(out)
ch <- &queuerpc.Message{
Id: msg.Id,
Method: msg.Method,
Metadata: meta,
Body: body,
}
}
}
}()
return ch, nil
}
},
})
}// EchoServiceClient is a type safe RabbitMQ rpc client
type EchoServiceClient struct {
client queuerpc.IClient
}// NewEchoServiceClient returns a new EchoServiceClientwith the given rpc client
func NewEchoServiceClient(client queuerpc.IClient) *EchoServiceClient {
return &EchoServiceClient{client: client}
}// Echo returns the same message it receives.
func (c *EchoServiceClient) Echo(ctx context.Context, in *EchoRequest) (*EchoResponse, error) {
meta := queuerpc.MetadataFromContext(ctx)
var out EchoResponse
body, err := proto.Marshal(in)
if err != nil {
return nil, err
}
msg, err := c.client.Request(ctx, &queuerpc.Message{Method: "Echo", Body: body, Metadata: meta})
if err != nil {
return nil, err
}
if msg.Error != nil {
return nil, msg.Error
}
if err := proto.Unmarshal(msg.Body, &out); err != nil {
return nil, err
}
return &out, nil
}// EchoServerStream streams the same message it initially receives.
func (c *EchoServiceClient) EchoServerStream(ctx context.Context, in *EchoRequest, handler func(*EchoResponse)) error {
meta := queuerpc.MetadataFromContext(ctx)
body, err := proto.Marshal(in)
if err != nil {
return err
}
return c.client.ServerStream(ctx, &queuerpc.Message{Method: "EchoServerStream", Body: body, Metadata: meta}, func(msg *queuerpc.Message) {
var out EchoResponse
if err := proto.Unmarshal(msg.Body, &out); err != nil {
return
}
handler(&out)
})
}```
see [example](example) for a full example of a client and server
example code generated using [Makefile](Makefile)## Limitations
- only supports proto3
- only supports unary and server streaming rpcs
- only supports rabbitmq and nats as a transport (for now)
- only supports generating clients and servers for go (for now)## TODO
- [ ] kafka message queue provider
- [x] nats message queue provider
- [ ] google pubsub message queue provider
- [ ] aws sqs message queue provider
- [ ] redis message queue provider
- [ ] client -> server streaming support
- [ ] bidirectional streaming support
- [ ] support for generating an openapi proxy
- [ ] typescript client/server code gen
- [ ] rust client/server code gen
- [ ] python client/server code gen