github.com/whiteCcinn/protobuf-go@v1.0.9/README.md (about) 1 # Go support for Protocol Buffers 2 3 [](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go) 4 [](https://travis-ci.org/protocolbuffers/protobuf-go) 5 6 This project hosts the Go implementation for 7 [protocol buffers](https://developers.google.com/protocol-buffers), which is a 8 language-neutral, platform-neutral, extensible mechanism for serializing 9 structured data. The protocol buffer language is a language for specifying the 10 schema for structured data. This schema is compiled into language specific 11 bindings. This project provides both a tool to generate Go code for the 12 protocol buffer language, and also the runtime implementation to handle 13 serialization of messages in Go. See the 14 [protocol buffer developer guide](https://developers.google.com/protocol-buffers/docs/overview) 15 for more information about protocol buffers themselves. 16 17 This project is comprised of two components: 18 19 * Code generator: The 20 [`protoc-gen-go`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/cmd/protoc-gen-go) 21 tool is a compiler plugin to `protoc`, the protocol buffer compiler. It 22 augments the `protoc` compiler so that it knows how to 23 [generate Go specific code for a given `.proto` file](https://developers.google.com/protocol-buffers/docs/reference/go-generated). 24 25 * Runtime library: The 26 [`protobuf`](https://pkg.go.dev/mod/github.com/whiteCcinn/protobuf-go) module 27 contains a set of Go packages that form the runtime implementation of 28 protobufs in Go. This provides the set of interfaces that 29 [define what a message is](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/reflect/protoreflect) 30 and functionality to serialize message in various formats (e.g., 31 [wire](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/proto), 32 [JSON](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/encoding/protojson), 33 and 34 [text](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/encoding/prototext)). 35 36 See the 37 [developer guide for protocol buffers in Go](https://developers.google.com/protocol-buffers/docs/gotutorial) 38 for a general guide for how to get started using protobufs in Go. 39 40 This project is the second major revision of the Go protocol buffer API 41 implemented by the 42 [`github.com/whiteCcinn/protobuf-go`](https://pkg.go.dev/mod/github.com/whiteCcinn/protobuf-go) 43 module. The first major version is implemented by the 44 [`github.com/golang/protobuf`](https://pkg.go.dev/mod/github.com/golang/protobuf) 45 module. 46 47 ## Package index 48 49 Summary of the packages provided by this module: 50 51 * [`proto`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/proto): Package 52 `proto` provides functions operating on protobuf messages such as cloning, 53 merging, and checking equality, as well as binary serialization. 54 * [`encoding/protojson`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/encoding/protojson): 55 Package `protojson` serializes protobuf messages as JSON. 56 * [`encoding/prototext`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/encoding/prototext): 57 Package `prototext` serializes protobuf messages as the text format. 58 * [`encoding/protowire`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/encoding/protowire): 59 Package `protowire` parses and formats the low-level raw wire encoding. Most 60 users should use package `proto` to serialize messages in the wire format. 61 * [`reflect/protoreflect`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/reflect/protoreflect): 62 Package `protoreflect` provides interfaces to dynamically manipulate 63 protobuf messages. 64 * [`reflect/protoregistry`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/reflect/protoregistry): 65 Package `protoregistry` provides data structures to register and lookup 66 protobuf descriptor types. 67 * [`reflect/protodesc`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/reflect/protodesc): 68 Package `protodesc` provides functionality for converting 69 `descriptorpb.FileDescriptorProto` messages to/from the reflective 70 `protoreflect.FileDescriptor`. 71 * [`reflect/protopath`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/reflect/protopath): 72 Package `protopath` provides a representation of a sequence of 73 protobuf reflection operations on a message. 74 * [`reflect/protorange`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/reflect/protorange): 75 Package `protorange` provides functionality to traverse a protobuf message. 76 * [`testing/protocmp`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/testing/protocmp): 77 Package `protocmp` provides protobuf specific options for the `cmp` package. 78 * [`testing/protopack`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/testing/protopack): 79 Package `protopack` aids manual encoding and decoding of the wire format. 80 * [`testing/prototest`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/testing/prototest): 81 Package `prototest` exercises the protobuf reflection implementation for 82 concrete message types. 83 * [`types/dynamicpb`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/types/dynamicpb): 84 Package `dynamicpb` creates protobuf messages at runtime from protobuf 85 descriptors. 86 * [`types/known/anypb`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/types/known/anypb): 87 Package `anypb` is the generated package for `google/protobuf/any.proto`. 88 * [`types/known/timestamppb`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/types/known/timestamppb): 89 Package `timestamppb` is the generated package for 90 `google/protobuf/timestamp.proto`. 91 * [`types/known/durationpb`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/types/known/durationpb): 92 Package `durationpb` is the generated package for 93 `google/protobuf/duration.proto`. 94 * [`types/known/wrapperspb`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/types/known/wrapperspb): 95 Package `wrapperspb` is the generated package for 96 `google/protobuf/wrappers.proto`. 97 * [`types/known/structpb`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/types/known/structpb): 98 Package `structpb` is the generated package for 99 `google/protobuf/struct.proto`. 100 * [`types/known/fieldmaskpb`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/types/known/fieldmaskpb): 101 Package `fieldmaskpb` is the generated package for 102 `google/protobuf/field_mask.proto`. 103 * [`types/known/apipb`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/types/known/apipb): 104 Package `apipb` is the generated package for 105 `google/protobuf/api.proto`. 106 * [`types/known/typepb`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/types/known/typepb): 107 Package `typepb` is the generated package for 108 `google/protobuf/type.proto`. 109 * [`types/known/sourcecontextpb`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/types/known/sourcecontextpb): 110 Package `sourcecontextpb` is the generated package for 111 `google/protobuf/source_context.proto`. 112 * [`types/known/emptypb`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/types/known/emptypb): 113 Package `emptypb` is the generated package for 114 `google/protobuf/empty.proto`. 115 * [`types/descriptorpb`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/types/descriptorpb): 116 Package `descriptorpb` is the generated package for 117 `google/protobuf/descriptor.proto`. 118 * [`types/pluginpb`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/types/pluginpb): 119 Package `pluginpb` is the generated package for 120 `google/protobuf/compiler/plugin.proto`. 121 * [`compiler/protogen`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/compiler/protogen): 122 Package `protogen` provides support for writing protoc plugins. 123 * [`cmd/protoc-gen-go`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/cmd/protoc-gen-go): 124 The `protoc-gen-go` binary is a protoc plugin to generate a Go protocol 125 buffer package. 126 127 ## Reporting issues 128 129 The issue tracker for this project is currently located at 130 [golang/protobuf](https://github.com/golang/protobuf/issues). 131 132 Please report any issues there with a sufficient description of the bug or 133 feature request. Bug reports should ideally be accompanied by a minimal 134 reproduction of the issue. Irreproducible bugs are difficult to diagnose and fix 135 (and likely to be closed after some period of time). Bug reports must specify 136 the version of the 137 [Go protocol buffer module](https://github.com/protocolbuffers/protobuf-go/releases) 138 and also the version of the 139 [protocol buffer toolchain](https://github.com/protocolbuffers/protobuf/releases) 140 being used. 141 142 ## Contributing 143 144 This project is open-source and accepts contributions. See the 145 [contribution guide](https://github.com/protocolbuffers/protobuf-go/blob/master/CONTRIBUTING.md) 146 for more information. 147 148 ## Compatibility 149 150 This module and the generated code are expected to be stable over time. However, 151 we reserve the right to make breaking changes without notice for the following 152 reasons: 153 154 * **Security:** A security issue in the specification or implementation may 155 come to light whose resolution requires breaking compatibility. We reserve 156 the right to address such issues. 157 * **Unspecified behavior:** There are some aspects of the protocol buffer 158 specification that are undefined. Programs that depend on unspecified 159 behavior may break in future releases. 160 * **Specification changes:** It may become necessary to address an 161 inconsistency, incompleteness, or change in the protocol buffer 162 specification, which may affect the behavior of existing programs. We 163 reserve the right to address such changes. 164 * **Bugs:** If a package has a bug that violates correctness, a program 165 depending on the buggy behavior may break if the bug is fixed. We reserve 166 the right to fix such bugs. 167 * **Generated additions**: We reserve the right to add new declarations to 168 generated Go packages of `.proto` files. This includes declared constants, 169 variables, functions, types, fields in structs, and methods on types. This 170 may break attempts at injecting additional code on top of what is generated 171 by `protoc-gen-go`. Such practice is not supported by this project. 172 * **Internal changes**: We reserve the right to add, modify, and remove 173 internal code, which includes all unexported declarations, the 174 [`protoc-gen-go/internal_gengo`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/cmd/protoc-gen-go/internal_gengo) 175 package, the 176 [`runtime/protoimpl`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/runtime/protoimpl?tab=doc) 177 package, and all packages under 178 [`internal`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/internal). 179 180 Any breaking changes outside of these will be announced 6 months in advance to 181 [protobuf@googlegroups.com](https://groups.google.com/forum/#!forum/protobuf). 182 183 Users should use generated code produced by a version of 184 [`protoc-gen-go`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/cmd/protoc-gen-go) 185 that is identical to the runtime version provided by the 186 [protobuf module](https://pkg.go.dev/mod/github.com/whiteCcinn/protobuf-go). This 187 project promises that the runtime remains compatible with code produced by a 188 version of the generator that is no older than 1 year from the version of the 189 runtime used, according to the release dates of the minor version. Generated 190 code is expected to use a runtime version that is at least as new as the 191 generator used to produce it. Generated code contains references to 192 [`protoimpl.EnforceVersion`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/runtime/protoimpl?tab=doc#EnforceVersion) 193 to statically ensure that the generated code and runtime do not drift 194 sufficiently far apart. 195 196 ## Historical legacy 197 198 This project is the second major revision 199 ([released in 2020](https://blog.golang.org/a-new-go-api-for-protocol-buffers)) 200 of the Go protocol buffer API implemented by the 201 [`github.com/whiteCcinn/protobuf-go`](https://pkg.go.dev/mod/github.com/whiteCcinn/protobuf-go) 202 module. The first major version 203 ([released publicly in 2010](https://blog.golang.org/third-party-libraries-goprotobuf-and)) 204 is implemented by the 205 [`github.com/golang/protobuf`](https://pkg.go.dev/mod/github.com/golang/protobuf) 206 module. 207 208 The first version predates the release of Go 1 by several years. It has a long 209 history as one of the first core pieces of infrastructure software ever written 210 in Go. As such, the Go protobuf project was one of many pioneers for determining 211 what the Go language should even look like and what would eventually be 212 considered good design patterns and “idiomatic” Go (by simultaneously being 213 both positive and negative examples of it). 214 215 Consider the changing signature of the `proto.Unmarshal` function as an example 216 of Go language and library evolution throughout the life of this project: 217 218 ```go 219 // 2007/09/25 - Conception of Go 220 221 // 2008/11/12 222 export func UnMarshal(r io.Read, pb_e reflect.Empty) *os.Error 223 224 // 2008/11/13 225 export func UnMarshal(buf *[]byte, pb_e reflect.Empty) *os.Error 226 227 // 2008/11/24 228 export func UnMarshal(buf *[]byte, pb_e interface{}) *os.Error 229 230 // 2008/12/18 231 export func UnMarshal(buf []byte, pb_e interface{}) *os.Error 232 233 // 2009/01/20 234 func UnMarshal(buf []byte, pb_e interface{}) *os.Error 235 236 // 2009/04/17 237 func UnMarshal(buf []byte, pb_e interface{}) os.Error 238 239 // 2009/05/22 240 func Unmarshal(buf []byte, pb_e interface{}) os.Error 241 242 // 2011/11/03 243 func Unmarshal(buf []byte, pb_e interface{}) error 244 245 // 2012/03/28 - Release of Go 1 246 247 // 2012/06/12 248 func Unmarshal(buf []byte, pb Message) error 249 ``` 250 251 These changes demonstrate the difficulty of determining what the right API is 252 for any new technology. It takes time multiplied by many users to determine what 253 is best; even then, “best” is often still somewhere over the horizon. 254 255 The change on June 6th, 2012 added a degree of type-safety to Go protobufs by 256 declaring a new interface that all protobuf messages were required to implement: 257 258 ```go 259 type Message interface { 260 Reset() 261 String() string 262 ProtoMessage() 263 } 264 ``` 265 266 This interface reduced the set of types that can be passed to `proto.Unmarshal` 267 from the universal set of all possible Go types to those with a special 268 `ProtoMessage` marker method. The intention of this change is to limit the 269 protobuf API to only operate on protobuf data types (i.e., protobuf messages). 270 For example, there is no sensible operation if a Go channel were passed to the 271 protobuf API as a channel cannot be serialized. The restricted interface would 272 prevent that. 273 274 This interface does not behaviorally describe what a protobuf message is, but 275 acts as a marker with an undocumented expectation that protobuf messages must be 276 a Go struct with a specific layout of fields with formatted tags. This 277 expectation is not statically enforced by the Go language, for it is an 278 implementation detail checked dynamically at runtime using Go reflection. Back 279 in 2012, the only types with this marker were those generated by 280 `protoc-gen-go`. Since `protoc-gen-go` would always generate messages with the 281 proper layout of fields, this was deemed an acceptable and dramatic improvement 282 over `interface{}`. 283 284 Over the next 10 years, 285 [use of Go would skyrocket](https://blog.golang.org/10years) and use of 286 protobufs in Go would skyrocket as well. With increased popularity also came 287 more diverse usages and requirements for Go protobufs and an increased number of 288 custom `proto.Message` implementations that were not generated by 289 `protoc-gen-go`. 290 291 The increasingly diverse ecosystem of Go types implementing the `proto.Message` 292 interface led to incompatibilities, which often occurred when: 293 294 * **Passing custom `proto.Message` types to the protobuf APIs**: A concrete 295 message implementation might work with some top-level functions (e.g., 296 `proto.Marshal`), but cause others (e.g., `proto.Equal`) to choke and panic. 297 This occurs because the type only had partial support for being an actual 298 message by only implementing the `proto.Marshaler` interface or having 299 malformed struct field tags that happened to work with one function, but not 300 another. 301 302 * **Using Go reflection on any `proto.Message` types**: A common desire is to 303 write general-purpose code that operates on any protobuf message. For 304 example, a microservice might want to populate a `trace_id` field if it is 305 present in a message. To accomplish this, one would use Go reflection to 306 introspect the message type, and assume it were a pointer to a Go struct 307 with a field named `TraceId` (as would be commonly produced by 308 `protoc-gen-go`). If the concrete message type did not match this 309 expectation, it either failed to work or even resulted in a panic. Such was 310 the case for concrete message types that might be backed by a Go map instead 311 of a Go struct. 312 313 Both of these issues are solved by following the idiom that _interfaces should 314 describe behavior, not data_. This means that the interface itself should 315 provide sufficient functionality through its methods that users can introspect 316 and interact with all aspects of a protobuf message through a principled API. 317 This feature is called _protobuf reflection_. Just as how Go reflection provides 318 an API for programmatically interacting with any arbitrary Go value, protobuf 319 reflection provides an API for programmatically interacting with any arbitrary 320 protobuf message. 321 322 Since an interface cannot be extended in a backwards compatible way, this 323 suggested the need for a new major version that defines a new `proto.Message` 324 interface: 325 326 ```go 327 type Message interface { 328 ProtoReflect() protoreflect.Message 329 } 330 ``` 331 332 The new 333 [`proto.Message`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/proto?tab=doc#Message) 334 interface contains a single `ProtoReflect` method that returns a 335 [`protoreflect.Message`](https://pkg.go.dev/github.com/whiteCcinn/protobuf-go/reflect/protoreflect?tab=doc#Message), 336 which is a reflective view over a protobuf message. In addition to making a 337 breaking change to the `proto.Message` interface, we took this opportunity to 338 cleanup the supporting functionality that operate on a `proto.Message`, split up 339 complicated functionality apart into manageable packages, and to hide 340 implementation details away from the public API. 341 342 The goal for this major revision is to improve upon all the benefits of, while 343 addressing all the shortcomings of the old API. We hope that it will serve the 344 Go ecosystem well for the next 10 years and beyond.