-
Notifications
You must be signed in to change notification settings - Fork 4.4k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
7e12068
commit 9ab8b62
Showing
39 changed files
with
1,842 additions
and
1,065 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,82 @@ | ||
/* | ||
* | ||
* Copyright 2024 gRPC authors. | ||
* | ||
* Licensed under the Apache License, Version 2.0 (the "License"); | ||
* you may not use this file except in compliance with the License. | ||
* You may obtain a copy of the License at | ||
* | ||
* http://www.apache.org/licenses/LICENSE-2.0 | ||
* | ||
* Unless required by applicable law or agreed to in writing, software | ||
* distributed under the License is distributed on an "AS IS" BASIS, | ||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
* See the License for the specific language governing permissions and | ||
* limitations under the License. | ||
* | ||
*/ | ||
|
||
package encoding | ||
|
||
import ( | ||
"strings" | ||
|
||
"google.golang.org/grpc/mem" | ||
) | ||
|
||
// CodecV2 defines the interface gRPC uses to encode and decode messages. Note | ||
// that implementations of this interface must be thread safe; a CodecV2's | ||
// methods can be called from concurrent goroutines. | ||
type CodecV2 interface { | ||
// Marshal returns the wire format of v. The buffers in the returned | ||
// [mem.BufferSlice] must have at least one reference each, which will be freed | ||
// by gRPC when they are no longer needed. | ||
Marshal(v any) (out mem.BufferSlice, err error) | ||
// Unmarshal parses the wire format into v. Note that data will be freed as soon | ||
// as this function returns. If the codec wishes to guarantee access to the data | ||
// after this function, it must take its own reference that it frees when it is | ||
// no longer needed. | ||
Unmarshal(data mem.BufferSlice, v any) error | ||
// Name returns the name of the Codec implementation. The returned string | ||
// will be used as part of content type in transmission. The result must be | ||
// static; the result cannot change between calls. | ||
Name() string | ||
} | ||
|
||
var registeredV2Codecs = make(map[string]CodecV2) | ||
|
||
// RegisterCodecV2 registers the provided CodecV2 for use with all gRPC clients and | ||
// servers. | ||
// | ||
// The CodecV2 will be stored and looked up by result of its Name() method, which | ||
// should match the content-subtype of the encoding handled by the CodecV2. This | ||
// is case-insensitive, and is stored and looked up as lowercase. If the | ||
// result of calling Name() is an empty string, RegisterCodecV2 will panic. See | ||
// Content-Type on | ||
// https://github.com/grpc/grpc/blob/master/doc/PROTOCOL-HTTP2.md#requests for | ||
// more details. | ||
// | ||
// If both a Codec and CodecV2 are registered with the same name, the CodecV2 | ||
// will be used. | ||
// | ||
// NOTE: this function must only be called during initialization time (i.e. in | ||
// an init() function), and is not thread-safe. If multiple Codecs are | ||
// registered with the same name, the one registered last will take effect. | ||
func RegisterCodecV2(codec CodecV2) { | ||
if codec == nil { | ||
panic("cannot register a nil CodecV2") | ||
} | ||
if codec.Name() == "" { | ||
panic("cannot register CodecV2 with empty string result for Name()") | ||
} | ||
contentSubtype := strings.ToLower(codec.Name()) | ||
registeredV2Codecs[contentSubtype] = codec | ||
} | ||
|
||
// GetCodecV2 gets a registered CodecV2 by content-subtype, or nil if no CodecV2 is | ||
// registered for the content-subtype. | ||
// | ||
// The content-subtype is expected to be lowercase. | ||
func GetCodecV2(contentSubtype string) CodecV2 { | ||
return registeredV2Codecs[contentSubtype] | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,81 @@ | ||
/* | ||
* | ||
* Copyright 2024 gRPC authors. | ||
* | ||
* Licensed under the Apache License, Version 2.0 (the "License"); | ||
* you may not use this file except in compliance with the License. | ||
* You may obtain a copy of the License at | ||
* | ||
* http://www.apache.org/licenses/LICENSE-2.0 | ||
* | ||
* Unless required by applicable law or agreed to in writing, software | ||
* distributed under the License is distributed on an "AS IS" BASIS, | ||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
* See the License for the specific language governing permissions and | ||
* limitations under the License. | ||
* | ||
*/ | ||
|
||
package proto | ||
|
||
import ( | ||
"fmt" | ||
|
||
"google.golang.org/grpc/encoding" | ||
"google.golang.org/grpc/mem" | ||
"google.golang.org/protobuf/proto" | ||
) | ||
|
||
func init() { | ||
encoding.RegisterCodecV2(&codecV2{}) | ||
} | ||
|
||
// codec is a CodecV2 implementation with protobuf. It is the default codec for | ||
// gRPC. | ||
type codecV2 struct{} | ||
|
||
var _ encoding.CodecV2 = (*codecV2)(nil) | ||
|
||
func (c *codecV2) Marshal(v any) (data mem.BufferSlice, err error) { | ||
vv := messageV2Of(v) | ||
if vv == nil { | ||
return nil, fmt.Errorf("proto: failed to marshal, message is %T, want proto.Message", v) | ||
} | ||
|
||
size := proto.Size(vv) | ||
if mem.IsBelowBufferPoolingThreshold(size) { | ||
buf, err := proto.Marshal(vv) | ||
if err != nil { | ||
return nil, err | ||
} | ||
data = append(data, mem.SliceBuffer(buf)) | ||
} else { | ||
pool := mem.DefaultBufferPool() | ||
buf := pool.Get(size) | ||
if _, err := (proto.MarshalOptions{}).MarshalAppend((*buf)[:0], vv); err != nil { | ||
pool.Put(buf) | ||
return nil, err | ||
} | ||
data = append(data, mem.NewBuffer(buf, pool)) | ||
} | ||
|
||
return data, nil | ||
} | ||
|
||
func (c *codecV2) Unmarshal(data mem.BufferSlice, v any) (err error) { | ||
vv := messageV2Of(v) | ||
if vv == nil { | ||
return fmt.Errorf("failed to unmarshal, message is %T, want proto.Message", v) | ||
} | ||
|
||
buf := data.MaterializeToBuffer(mem.DefaultBufferPool()) | ||
defer buf.Free() | ||
// TODO: Upgrade proto.Unmarshal to support mem.BufferSlice. Right now, it's not | ||
// really possible without a major overhaul of the proto package, but the | ||
// vtprotobuf library may be able to support this. | ||
return proto.Unmarshal(buf.ReadOnlyData(), vv) | ||
} | ||
|
||
func (c *codecV2) Name() string { | ||
return Name | ||
} |
Oops, something went wrong.