micro-client-grpc/codec.go

138 lines
2.6 KiB
Go
Raw Normal View History

2019-06-03 20:44:43 +03:00
package grpc
import (
"io"
"github.com/unistack-org/micro/v3/codec"
2019-06-17 22:05:58 +03:00
"google.golang.org/grpc"
2019-06-18 20:51:52 +03:00
"google.golang.org/grpc/encoding"
2019-06-03 20:44:43 +03:00
)
var (
_ encoding.Codec = &wrapMicroCodec{}
_ codec.Codec = &wrapGrpcCodec{}
)
type wrapStream struct{ grpc.ClientStream }
2019-06-03 20:44:43 +03:00
func (w *wrapStream) Write(d []byte) (int, error) {
n := len(d)
err := w.ClientStream.SendMsg(&codec.Frame{Data: d})
return n, err
2019-06-17 22:05:58 +03:00
}
func (w *wrapStream) Read(d []byte) (int, error) {
m := &codec.Frame{}
err := w.ClientStream.RecvMsg(m)
copy(d, m.Data)
return len(d), err
2019-06-03 20:44:43 +03:00
}
type wrapMicroCodec struct{ codec.Codec }
2019-06-03 20:44:43 +03:00
func (w *wrapMicroCodec) Name() string {
return w.Codec.String()
2019-06-03 20:44:43 +03:00
}
func (w *wrapMicroCodec) Marshal(v interface{}) ([]byte, error) {
if m, ok := v.(*codec.Frame); ok {
return m.Data, nil
}
return w.Codec.Marshal(v)
}
func (w *wrapMicroCodec) Unmarshal(d []byte, v interface{}) error {
if d == nil || v == nil {
return nil
}
if m, ok := v.(*codec.Frame); ok {
m.Data = d
return nil
}
return w.Codec.Unmarshal(d, v)
}
type wrapGrpcCodec struct{ encoding.Codec }
2019-06-03 20:44:43 +03:00
func (w *wrapGrpcCodec) String() string {
return w.Codec.Name()
2019-06-03 20:44:43 +03:00
}
func (w *wrapGrpcCodec) Marshal(v interface{}, opts ...codec.Option) ([]byte, error) {
if m, ok := v.(*codec.Frame); ok {
return m.Data, nil
}
return w.Codec.Marshal(v)
2019-06-03 20:44:43 +03:00
}
func (w *wrapGrpcCodec) Unmarshal(d []byte, v interface{}, opts ...codec.Option) error {
if d == nil || v == nil {
return nil
}
if m, ok := v.(*codec.Frame); ok {
m.Data = d
return nil
}
return w.Codec.Unmarshal(d, v)
2019-06-03 20:44:43 +03:00
}
2019-06-17 22:05:58 +03:00
/*
2019-06-17 22:05:58 +03:00
type grpcCodec struct {
grpc.ServerStream
2019-06-17 22:05:58 +03:00
// headers
2019-06-18 20:51:52 +03:00
id string
target string
method string
2019-06-17 22:05:58 +03:00
endpoint string
c encoding.Codec
}
*/
func (w *wrapGrpcCodec) ReadHeader(conn io.Reader, m *codec.Message, mt codec.MessageType) error {
/*
if m == nil {
m = codec.NewMessage(codec.Request)
}
if md, ok := metadata.FromIncomingContext(g.ServerStream.Context()); ok {
if m.Header == nil {
m.Header = meta.New(len(md))
}
for k, v := range md {
m.Header[k] = strings.Join(v, ",")
}
}
m.Id = g.id
m.Target = g.target
m.Method = g.method
m.Endpoint = g.endpoint
*/
2019-06-17 22:05:58 +03:00
return nil
}
func (w *wrapGrpcCodec) ReadBody(conn io.Reader, v interface{}) error {
// caller has requested a frame
if m, ok := v.(*codec.Frame); ok {
_, err := conn.Read(m.Data)
return err
2019-06-17 22:05:58 +03:00
}
return codec.ErrInvalidMessage
2019-06-17 22:05:58 +03:00
}
func (w *wrapGrpcCodec) Write(conn io.Writer, m *codec.Message, v interface{}) error {
2019-06-17 22:05:58 +03:00
// if we don't have a body
2019-06-18 20:51:52 +03:00
if v != nil {
b, err := w.Marshal(v)
if err != nil {
return err
}
m.Body = b
2019-06-17 22:05:58 +03:00
}
// write the body using the framing codec
_, err := conn.Write(m.Body)
return err
2019-06-17 22:05:58 +03:00
}