package grpc import ( "io" "go.unistack.org/micro/v3/codec" "google.golang.org/grpc" "google.golang.org/grpc/encoding" ) var ( _ encoding.Codec = &wrapMicroCodec{} _ codec.Codec = &wrapGrpcCodec{} ) type wrapStream struct{ grpc.ClientStream } func (w *wrapStream) Write(d []byte) (int, error) { n := len(d) err := w.ClientStream.SendMsg(&codec.Frame{Data: d}) return n, err } func (w *wrapStream) Read(d []byte) (int, error) { m := &codec.Frame{} err := w.ClientStream.RecvMsg(m) copy(d, m.Data) return len(d), err } type wrapMicroCodec struct{ codec.Codec } func (w *wrapMicroCodec) Name() string { return w.Codec.String() } func (w *wrapMicroCodec) Marshal(v interface{}) ([]byte, error) { return w.Codec.Marshal(v) } func (w *wrapMicroCodec) Unmarshal(d []byte, v interface{}) error { return w.Codec.Unmarshal(d, v) } type wrapGrpcCodec struct{ encoding.Codec } func (w *wrapGrpcCodec) String() string { return w.Codec.Name() } 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) } 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) } /* type grpcCodec struct { grpc.ServerStream // headers id string target string method string 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 */ 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 } return codec.ErrInvalidMessage } func (w *wrapGrpcCodec) Write(conn io.Writer, m *codec.Message, v interface{}) error { // if we don't have a body if v != nil { b, err := w.Marshal(v) if err != nil { return err } m.Body = b } // write the body using the framing codec _, err := conn.Write(m.Body) return err }