/nim-protobuf-serialization

Primary LanguageNimApache License 2.0Apache-2.0

nim-protobuf-serialization

License: Apache License: MIT Stability: experimental Github action

Protobuf implementation compatible with the nim-serialization framework.

Usage

Messages in protobuf are serialized according to a schema found in .proto files. The library requires that types are annotated with schema information - this can be done either directly in Nim or, for some proto3 files, generated using the import_proto3 macro.

Both Protobuf 2 and Protobuf 3 semantics are supported. When declaring an object, add either the proto2 or proto3 pragma to declare which to use, as seen in the syntax element in protobuf.

When using Protobuf 3, a import_proto3 macro is available. Taking in a file path, it can directly parse a Protobuf 3 spec file and generate the matching Nim types, same as if they had been written manually.

Annotating objects

The protobuf schema can be declared using annotations similar to what is found in a typical .proto file - see types for available annotations:

my_protocol.proto3:

syntax = "proto3";

message ExampleMsg {
  int32 a = 1;
  float b = 2;
}

Annotated Nim code

type ExampleMsg {.proto3.} = object
  a {.fieldNumber: 1, pint.}: int32
  b {.fieldNumber: 2.}: float32

Importing proto file:

import protobuf_serialization/proto_parser

# This generates the same definition as above using a compile-time macro / parser
import_proto3 "my_protocol.proto3"

Encoding and decoding

let x = ExampleMsg(a: 10, b: 20.0)
let encoded = Protobuf.encode(x)
...
let decoded = Protobuf.decode(encoded, ExampleMsg)

Both Protobuf 2 and Protobuf 3 objects have the following properties:

  • Every field requires the fieldNumber pragma, which takes in an integer of what field number to encode that field with.
  • Every int/uint must have its bits explicitly specified.
  • int/uint fields require their encoding to be specified. pint is valid for both, and uses VarInt encoding, which only uses the amount of bytes it needs. fixed is also valid for both, and uses the full amount of bytes the number uses, instead of stripping unused bytes. This has performance advantages for large numbers. Finally, sint uses zig-zagged VarInt encoding, which is recommended for numbers which are frequently negative, and is only valid for ints.

Protobuf 2 has the additional properties:

  • A required pragma is enabled, matching the syntax of Protobuf 2's required keyword.
  • Every primitive value, such as a number or string, must have the required pragma or be a PBOption. PBOptions are a generic type instantiated with the default value for that field. They serve as regular Options, except when they're none, they still return a value when get is called (the default value). PBOptions can be constructed using pbSome(PBOption[T], value).

Here is an example demonstrating how the various pragmas can be combined:

type
  X {.proto3.} = object
    a {.fieldNumber: 1, pint.}: int32
    b {.fieldNumber: 2.}: float32

  Y {.proto2.} = object
    a {.fieldNumber: 1.}: seq[string]
    b {.fieldNumber: 2, pint.}: PBOption[int32(2)]
    c {.fieldNumber: 3, required, sint.}: int32

License

Licensed and distributed under either of

or

at your option. These files may not be copied, modified, or distributed except according to those terms.