Skip to content

well-known-components/proto-compatibility-tool

Repository files navigation

proto-compatibility-tool

proto-compatibility-tool is an utility that verifies .proto files and ensures backwards compatibility to prevent breaking changes on exposed APIs.

It runs as part of the CI process, and runs verifications against published versions of .proto files in NPM packages, S3, static URL or local files.

This tool does generate any kind of code, it only performs validations on .proto files.

Usage

npm i -g proto-compatibility-tool

# proto-compatibility-tool <local file> <remote file>

# compare two local files
proto-compatibility-tool \
  ./api-v0.proto \
  ./api-v1.proto

Design

Types are evaluated nominally and then structurally. That means, this tool not only checks the structure and serialization of each message, it also checks that the name of the types matches to not break any generated code.

Valdiations

Removing required fields throws an exception

message MessageName {
  int32 message_id = 1;
- required string message_payload = 2;
}
// 🚨 Throws

Removing optional fields throws if the number of the field is not reserved

message MessageName {
  int32 message_id = 1;
- string message_payload = 2;
+ reserved 2;
}
// ✅ Passes

Changing type throws

message MessageName {
  int32 message_id = 1;
- string message_payload = 2;
+ int32 message_payload = 2;
}
// 🚨 Throws

Removing reserved fields throws an exception.

Fields are optional by default, field removal should be performed very few times during the development lifecycle. Once you remove a field there is no looking back, because this tool will verify against the previous published version, re-validating the field original type is not possible.

message MessageName {
  int32 message_id = 1;
- reserved 2;
}
// 🚨 Throws

Removing methods from services throws

service Haberdasher {
-  rpc MakeHat(Size) returns (Hat);
}
// 🚨 Throws

Removing deprecated methods from services does not throw

service Haberdasher {
-  rpc MakeHat(Size) returns (Hat) [deprecated = true];
}
// ✅ Passes

Renaming methods from services throws

service Haberdasher {
-  rpc MakeHat(Size) returns (Hat);
+  rpc MakeTophat(Size) returns (Hat);
}
// 🚨 Throws

Changing signatures in services methods throws

If a function receives different input or produces a different result, it is a different function, therefore it must have a different name.

service Haberdasher {
-  rpc MakeHat(Size) returns (Hat);
+  rpc MakeHat(HatOptions) returns (Hat);
}
// 🚨 Throws

Changing or removing package throws

-package twirp.example.haberdasher;
+package twirp.example.hats;

service Haberdasher {
...
}
// 🚨 Throws

Renaming services throw

-service Haberdasher {
+service Hats {
  rpc MakeHat(Size) returns (Hat);
}
// 🚨 Throws