Migrating from protoc-gen-validate
to protovalidate
should be safe,
incremental, and relatively painless, but it will still require a handful of
operations to get there. To ease this burden, we've provided this documentation
and a migration tool to simplify the process.
Note
protovalidate
does not require any code generation. The plugin protoc-gen-validate
is no longer needed, and shouldn't be used with protovalidate
.
-
[OPTIONAL] Use
protovalidate
in legacy mode. If supported, an implementation may provide optional "legacy support" for theprotoc-gen-validate
annotations. Usingprotovalidate
in this way can allow immediate utilization of the new library without having to make any Protobuf changes. As an example, Go'sprotovalidate
supports opting into legacy support. -
[OPTIONAL] Format the
.proto
files. The migration tool does a best-effort to preserve the formatting of the original file, but in some circumstances may produce valid yet ill-formatted code. To reduce noise related to reformatting, we recommend running a formatter (such asbuf format
) over the corpus of.proto
files and committing these changes to make them consistent before proceeding. -
Run the migration tool. This first run will add matching
protovalidate
annotations alongside any existingprotoc-gen-validate
ones. Optionally run the formatter again after the migration tool to clean up any strange output.go run ./tools/protovalidate-migrate -w /path/to/protos
-
Use
protovalidate
. Update code that usesprotoc-gen-validate
code generation to consume theprotovalidate
library instead (or simultaneously). -
[OPTIONAL] Update migrated annotations. Rerunning the migration tool is a no-op for any
.proto
files that importprotovalidate
. To have the tool replace any existingprotovalidate
annotations, run it with the-replace-protovalidate
flag. This will ensure the annotations matchprotoc-gen-validate
annotations.go run ./tools/protovalidate-migrate -w --replace-protovalidate /path/to/protos
-
Remove
protoc-gen-validate
annotations. Once ready to make the switch, and references to theprotoc-gen-validate
generated code has been removed, the migration tool can be run again with the-remove-legacy
flag to remove the legacy annotations from the.proto
files.go run ./tools/protovalidate-migrate -w --remove-legacy /path/to/protos
go run ./tools/protovalidate-migrate <flags> /path/to/proto
Flag | Description | Default Value |
---|---|---|
--verbose , -v |
Enables verbose logging. | false |
--write , -w |
Overwrites target files in-place. | false |
--output , -o |
Writes output to the given path. | None |
--legacy-import |
Specifies a protoc-gen-validate proto import path. |
"validate/validate.proto" |
--protovalidate-import |
Specifies a protovalidate proto import path. |
"buf/validate/validate.proto" |
--remove-legacy |
Allows the program to remove protoc-gen-validate options. |
false |
--replace-protovalidate |
Replaces protovalidate options to match protoc-gen-validate options (only if present). |
false |
- If neither
-w
nor-o
is specified, the modified proto(s) are emitted to stdout.
As part of the migration process, please note the following changes to the
standard constraints between protoc-gen-validate
and protovalidate
:
-
All message-level constraints have moved into a single option:
(buf.validate.message)
. -
(validate.ignored)
: removed.protovalidate
does not generate code, and as a result, generation does not need to be skipped. -
(validate.disabled)
: moved to(buf.validate.message).disabled
.
-
All oneof-level constraints have moved into a single option:
(buf.validate.oneof)
. -
(validate.required)
: moved to(buf.validate.oneof).required
.
-
All field-level constraints have moved into a single option:
(buf.validate.field)
. -
(validate.rules).<TYPE>.required
: moved to(buf.validate.field).required
. The semantics of required have changed and been clarified. Please consult the documentation to ensure they match expectations compared to its behavior in PGV. -
(validate.rules).message.skip
: moved to(buf.validate.field).skipped
. -
(validate.rules).<TYPE>.ignore_empty
: moved to(buf.validate.field).ignore_empty
.ignore_empty
has also been deprecated in favor of the new, more powerfulignore
enum. Please consult the documentation to see which semantics match expectations. -
(validate.rules).map.no_sparse
: removed. The original rule accommodated for a situation that was only possible in the Go code generation exclusively.protovalidate
now treats a sparse map value as an empty value which matches the semantics of the Go Protobuf runtime.
While the migration tool is designed to simplify the process of migrating
from protoc-gen-validate
to protovalidate
, there may be cases where manual
migration is required or preferred. The following steps outline the process of
manual migration:
-
Understand the changes: The first step to manual migration is understanding the changes between
protoc-gen-validate
andprotovalidate
. Review the "Standard Constraint Changes" section of this guide to understand how constraints have changed. -
Update the imports: Replace the import of
validate/validate.proto
withbuf/validate/validate.proto
in your.proto
files. -
Update message-level constraints: Replace
(validate.ignored)
and(validate.disabled)
with the new(buf.validate.message)
option as described in the "Message Constraints" section. -
Update oneof constraints: Replace
(validate.required)
for oneof fields with the new(buf.validate.oneof)
option as described in the "Oneof Constraints" section. -
Update field-level constraints: Replace all field-level constraints, including
(validate.rules).<TYPE>.required
,(validate.rules).message.skip
, and(validate.rules).<TYPE>.ignore_empty
, with the new(buf.validate.field)
option as described in the "Field Constraints" section. -
Remove unnecessary constraints: Remove the
(validate.rules).map.no_sparse
constraint as it's not supported inprotovalidate
. -
Test and validate: After performing the above steps, test your Protobuf code to ensure it's functioning as expected. Review any warnings or errors, and make corrections as necessary.