Skip to content

Commit

Permalink
Add ignore-unknown-fields flag
Browse files Browse the repository at this point in the history
  • Loading branch information
codesoap committed Oct 10, 2024
1 parent 6f2963f commit 2076368
Show file tree
Hide file tree
Showing 11 changed files with 74 additions and 34 deletions.
15 changes: 12 additions & 3 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,8 @@ The following features can be generated:

- `unmarshal`: generates a `func (p *YourProto) UnmarshalVT(data []byte)` that behaves similarly to calling `proto.Unmarshal(data, p)` on the message, except the unmarshalling is performed by unrolled codegen without using reflection and allocating as little memory as possible. If the receiver `p` is **not** fully zeroed-out, the unmarshal call will actually behave like `proto.Merge(data, p)`. This is because the `proto.Unmarshal` in the ProtoBuf API is implemented by resetting the destination message and then calling `proto.Merge` on it. To ensure proper `Unmarshal` semantics, ensure you've called `proto.Reset` on your message before calling `UnmarshalVT`, or that your message has been newly allocated.

- The `ignoreUnknownFields` option can be used to ignore unknown fields in protobuf messages and further reduce memory allocations.

- `unmarshal_unsafe` generates a `func (p *YourProto) UnmarshalVTUnsafe(data []byte)` that behaves like `UnmarshalVT`, except it unsafely casts slices of data to `bytes` and `string` fields instead of copying them to newly allocated arrays, so that it performs less allocations. **Data received from the wire has to be left untouched for the lifetime of the message.** Otherwise, the message's `bytes` and `string` fields can be corrupted.

- `pool`: generates the following helper methods
Expand Down Expand Up @@ -122,7 +124,14 @@ message Label {
--go-vtproto_opt=pool=vitess.io/vitess/go/vt/proto/query.Row \
--go-vtproto_opt=pool=vitess.io/vitess/go/vt/proto/binlogdata.VStreamRowsResponse \
```
6. (Optional) if you want to selectively compile the generate `vtprotobuf` files, the `--vtproto_opt=buildTag=<tag>` can be used.
6. (Optional) If you are handling messages containing unknown fields and don't intend to forward these messages to a tool that might expect these fields, you can ignore them using the `ignoreUnknownFields` option.
- You can tag messages explicitly in the `.proto` files with `option (vtproto.ignore_unknown_fields)`. Take a look at the example using `option (vtproto.mempool)` above.
- Alternatively, you can enumerate the objects with `--go-vtproto_opt=ignoreUnknownFields=<import>.<message>` flags passed via the CLI. Take a look at the example using `--go-vtproto_opt=pool=...` above.
7. (Optional) if you want to selectively compile the generate `vtprotobuf` files, the `--vtproto_opt=buildTag=<tag>` can be used.
When using this option, the generated code will only be compiled in if a build tag is provided.
Expand All @@ -133,9 +142,9 @@ message Label {
This can be done with type assertions before using `vtprotobuf` generated methods.
The `grpc.Codec{}` object (discussed below) shows an example.
7. Compile the `.proto` files in your project. You should see `_vtproto.pb.go` files next to the `.pb.go` and `_grpc.pb.go` files that were already being generated.
8. Compile the `.proto` files in your project. You should see `_vtproto.pb.go` files next to the `.pb.go` and `_grpc.pb.go` files that were already being generated.
8. (Optional) Switch your RPC framework to use the optimized helpers (see following sections)
9. (Optional) Switch your RPC framework to use the optimized helpers (see following sections)
## `vtprotobuf` package and well-known types
Expand Down
2 changes: 2 additions & 0 deletions cmd/protoc-gen-go-vtproto/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -24,8 +24,10 @@ func main() {
f.BoolVar(&cfg.AllowEmpty, "allow-empty", false, "allow generation of empty files")
cfg.Poolable = generator.NewObjectSet()
cfg.PoolableExclude = generator.NewObjectSet()
cfg.IgnoreUnknownFields = generator.NewObjectSet()
f.Var(&cfg.Poolable, "pool", "use memory pooling for this object")
f.Var(&cfg.PoolableExclude, "pool-exclude", "do not use memory pooling for this object")
f.Var(&cfg.IgnoreUnknownFields, "ignoreUnknownFields", "ignore unknown fields instead of saving them")
f.BoolVar(&cfg.Wrap, "wrap", false, "generate wrapper types")
f.StringVar(&features, "features", "all", "list of features to generate (separated by '+')")
f.StringVar(&cfg.BuildTag, "buildTag", "", "the go:build tag to set on generated files")
Expand Down
6 changes: 3 additions & 3 deletions features/clone/clone.go
Original file line number Diff line number Diff line change
Expand Up @@ -154,7 +154,7 @@ func (p *clone) cloneField(lhsBase, rhsBase string, allFieldsNullable bool, fiel
func (p *clone) generateCloneMethodsForMessage(proto3 bool, message *protogen.Message) {
ccTypeName := message.GoIdent.GoName
p.P(`func (m *`, ccTypeName, `) `, cloneName, `() *`, ccTypeName, ` {`)
p.body(!proto3, ccTypeName, message, true)
p.body(!proto3, ccTypeName, message)
p.P(`}`)
p.P()

Expand All @@ -169,7 +169,7 @@ func (p *clone) generateCloneMethodsForMessage(proto3 bool, message *protogen.Me
// body generates the code for the actual cloning logic of a structure containing the given fields.
// In practice, those can be the fields of a message.
// The object to be cloned is assumed to be called "m".
func (p *clone) body(allFieldsNullable bool, ccTypeName string, message *protogen.Message, cloneUnknownFields bool) {
func (p *clone) body(allFieldsNullable bool, ccTypeName string, message *protogen.Message) {
// The method body for a message or a oneof wrapper always starts with a nil check.
p.P(`if m == nil {`)
// We use an explicitly typed nil to avoid returning the nil interface in the oneof wrapper
Expand Down Expand Up @@ -220,7 +220,7 @@ func (p *clone) body(allFieldsNullable bool, ccTypeName string, message *protoge
p.cloneField("r", "m", allFieldsNullable, field)
}

if cloneUnknownFields && !p.Wrapper() {
if !p.Wrapper() && !p.ShouldIgnoreUnknownFields(message) {
// Clone unknown fields, if any
p.P(`if len(m.unknownFields) > 0 {`)
p.P(`r.unknownFields = make([]byte, len(m.unknownFields))`)
Expand Down
2 changes: 1 addition & 1 deletion features/equal/equal.go
Original file line number Diff line number Diff line change
Expand Up @@ -113,7 +113,7 @@ func (p *equal) message(proto3 bool, message *protogen.Message) {
}
}

if p.Wrapper() {
if p.Wrapper() || p.ShouldIgnoreUnknownFields(message) {
p.P(`return true`)
} else {
p.P(`return string(this.unknownFields) == string(that.unknownFields)`)
Expand Down
2 changes: 1 addition & 1 deletion features/marshal/marshalto.go
Original file line number Diff line number Diff line change
Expand Up @@ -599,7 +599,7 @@ func (p *marshal) message(message *protogen.Message) {
p.P(`var l int`)
p.P(`_ = l`)

if !p.Wrapper() {
if !p.Wrapper() && !p.ShouldIgnoreUnknownFields(message) {
p.P(`if m.unknownFields != nil {`)
p.P(`i -= len(m.unknownFields)`)
p.P(`copy(dAtA[i:], m.unknownFields)`)
Expand Down
2 changes: 1 addition & 1 deletion features/size/size.go
Original file line number Diff line number Diff line change
Expand Up @@ -320,7 +320,7 @@ func (p *size) message(message *protogen.Message) {
}
}
}
if !p.Wrapper() {
if !p.Wrapper() && !p.ShouldIgnoreUnknownFields(message) {
p.P(`n+=len(m.unknownFields)`)
}
p.P(`return n`)
Expand Down
2 changes: 1 addition & 1 deletion features/unmarshal/unmarshal.go
Original file line number Diff line number Diff line change
Expand Up @@ -856,7 +856,7 @@ func (p *unmarshal) message(proto3 bool, message *protogen.Message) {
p.P(`iNdEx += skippy`)
p.P(`} else {`)
}
if !p.Wrapper() {
if !p.Wrapper() && !p.ShouldIgnoreUnknownFields(message) {
p.P(`m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...)`)
}
p.P(`iNdEx += skippy`)
Expand Down
10 changes: 10 additions & 0 deletions generator/generatedfile.go
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,16 @@ func (b *GeneratedFile) ShouldPool(message *protogen.Message) bool {
return false
}

func (b *GeneratedFile) ShouldIgnoreUnknownFields(message *protogen.Message) bool {
if b.Config.IgnoreUnknownFields.Contains(message.GoIdent) {
return true
}

ext := proto.GetExtension(message.Desc.Options(), vtproto.E_IgnoreUnknownFields)
ignoreUnknownFields, ok := ext.(bool)
return ok && ignoreUnknownFields
}

func (b *GeneratedFile) Alloc(vname string, message *protogen.Message, isQualifiedIdent bool) {
ident := message.GoIdent.GoName
if isQualifiedIdent {
Expand Down
10 changes: 6 additions & 4 deletions generator/generator.go
Original file line number Diff line number Diff line change
Expand Up @@ -56,10 +56,12 @@ type Config struct {
Poolable ObjectSet
// PoolableExclude rules determines if pool feature disabled for particular message
PoolableExclude ObjectSet
Wrap bool
WellKnownTypes bool
AllowEmpty bool
BuildTag string
// IgnoreUnknownFields contains messages for which unknown fields shall be ignored
IgnoreUnknownFields ObjectSet
Wrap bool
WellKnownTypes bool
AllowEmpty bool
BuildTag string
}

type Generator struct {
Expand Down
3 changes: 2 additions & 1 deletion include/github.com/planetscale/vtprotobuf/vtproto/ext.proto
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ option go_package = "github.com/planetscale/vtprotobuf/vtproto";

extend google.protobuf.MessageOptions {
optional bool mempool = 64101;
optional bool ignore_unknown_fields = 64102;
}

extend google.protobuf.FieldOptions {
Expand All @@ -19,4 +20,4 @@ extend google.protobuf.FieldOptions {
// applying them to some of the fields in protobuf
message Opts {
optional bool unique = 1;
}
}
54 changes: 35 additions & 19 deletions vtproto/ext.pb.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

0 comments on commit 2076368

Please sign in to comment.