2019-06-03 20:44:43 +03:00
|
|
|
package grpc
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
"github.com/golang/protobuf/proto"
|
|
|
|
"github.com/micro/go-micro/codec"
|
2019-06-17 22:05:58 +03:00
|
|
|
"github.com/micro/go-micro/codec/bytes"
|
2019-06-03 20:44:43 +03:00
|
|
|
"github.com/micro/go-micro/codec/jsonrpc"
|
|
|
|
"github.com/micro/go-micro/codec/protorpc"
|
|
|
|
"google.golang.org/grpc/encoding"
|
|
|
|
)
|
|
|
|
|
|
|
|
type jsonCodec struct{}
|
|
|
|
type bytesCodec struct{}
|
|
|
|
type protoCodec struct{}
|
2019-06-17 22:05:58 +03:00
|
|
|
type wrapCodec struct { encoding.Codec }
|
2019-06-03 20:44:43 +03:00
|
|
|
|
|
|
|
var (
|
|
|
|
defaultGRPCCodecs = map[string]encoding.Codec{
|
|
|
|
"application/json": jsonCodec{},
|
|
|
|
"application/proto": protoCodec{},
|
|
|
|
"application/protobuf": protoCodec{},
|
|
|
|
"application/octet-stream": protoCodec{},
|
|
|
|
"application/grpc": protoCodec{},
|
|
|
|
"application/grpc+json": jsonCodec{},
|
|
|
|
"application/grpc+proto": protoCodec{},
|
|
|
|
"application/grpc+bytes": bytesCodec{},
|
|
|
|
}
|
|
|
|
|
|
|
|
defaultRPCCodecs = map[string]codec.NewCodec{
|
|
|
|
"application/json": jsonrpc.NewCodec,
|
|
|
|
"application/json-rpc": jsonrpc.NewCodec,
|
|
|
|
"application/protobuf": protorpc.NewCodec,
|
|
|
|
"application/proto-rpc": protorpc.NewCodec,
|
|
|
|
"application/octet-stream": protorpc.NewCodec,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2019-06-17 22:05:58 +03:00
|
|
|
func (w wrapCodec) String() string {
|
|
|
|
return w.Codec.Name()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w wrapCodec) Marshal(v interface{}) ([]byte, error) {
|
|
|
|
b, ok := v.(*bytes.Frame)
|
|
|
|
if ok {
|
|
|
|
return b.Data, nil
|
|
|
|
}
|
|
|
|
return w.Codec.Marshal(v)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w wrapCodec) Unmarshal(data []byte, v interface{}) error {
|
|
|
|
b, ok := v.(*bytes.Frame)
|
|
|
|
if ok {
|
|
|
|
b.Data = data
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return w.Codec.Unmarshal(data, v)
|
|
|
|
}
|
|
|
|
|
2019-06-03 20:44:43 +03:00
|
|
|
func (protoCodec) Marshal(v interface{}) ([]byte, error) {
|
|
|
|
return proto.Marshal(v.(proto.Message))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (protoCodec) Unmarshal(data []byte, v interface{}) error {
|
|
|
|
return proto.Unmarshal(data, v.(proto.Message))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (protoCodec) Name() string {
|
|
|
|
return "proto"
|
|
|
|
}
|
|
|
|
|
|
|
|
func (jsonCodec) Marshal(v interface{}) ([]byte, error) {
|
|
|
|
return json.Marshal(v)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (jsonCodec) Unmarshal(data []byte, v interface{}) error {
|
|
|
|
return json.Unmarshal(data, v)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (jsonCodec) Name() string {
|
|
|
|
return "json"
|
|
|
|
}
|
|
|
|
|
|
|
|
func (bytesCodec) Marshal(v interface{}) ([]byte, error) {
|
|
|
|
b, ok := v.(*[]byte)
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("failed to marshal: %v is not type of *[]byte", v)
|
|
|
|
}
|
|
|
|
return *b, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (bytesCodec) Unmarshal(data []byte, v interface{}) error {
|
|
|
|
b, ok := v.(*[]byte)
|
|
|
|
if !ok {
|
|
|
|
return fmt.Errorf("failed to unmarshal: %v is not type of *[]byte", v)
|
|
|
|
}
|
|
|
|
*b = data
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (bytesCodec) Name() string {
|
|
|
|
return "bytes"
|
|
|
|
}
|