Quickstart
This quickstart steps through adding Protovalidate to existing Protobuf projects using the Buf CLI:
- Adding Protovalidate rules to schemas.
- Using CEL to add domain-specific validation logic.
- Enabling server-side validation.
Download the code (optional)
If you'd like to code along in Go, Java, or Python, complete the following steps. If you're only here for a quick tour, feel free to skip ahead.
- Install the Buf CLI. If you already have, run
buf --version
to verify that you're using at least1.32.0
. - Have
git
and your choice ofgo
,Java 17+
, orPython 3.7+
installed. -
Clone the
buf-examples
repository: -
Open a terminal to the repository and navigate to the
protovalidate/quickstart-go/start
,protovalidate/quickstart-java/start
, orprotovalidate/quickstart-python/start
directory.
Each language's quickstart code contains Buf CLI configuration files (buf.yaml
, buf.gen.yaml
), a simple weather_service.proto
, and an idiomatic unit test.
Adding Protovalidate to schemas
Depending on Protovalidate
Published publicly on the Buf Schema Registry, the Protovalidate module provides the Protobuf extensions, options, and messages powering validation.
Add it as a dependency in buf.yaml
:
version: v2
modules:
- path: proto
+ deps:
+ - buf.build/bufbuild/protovalidate
lint:
use:
- STANDARD
breaking:
use:
- FILE
Next, update dependencies. You may see a warning that Protovalidate hasn't yet been used. That's fine.
$ buf dep update
WARN Module buf.build/bufbuild/protovalidate is declared in your buf.yaml deps but is unused...
If you're using Go or Java, update managed mode options in buf.gen.yaml
:
version: v2
inputs:
- directory: proto
plugins:
- remote: buf.build/protocolbuffers/go:v1.36.5
out: gen
opt:
- paths=source_relative
managed:
enabled: true
override:
- file_option: go_package_prefix
value: github.com/bufbuild/buf-examples/protovalidate/quickstart-go/start/gen
+ # Don't modify any file option or field option for protovalidate. Without
+ # this, generated Go will fail to compile.
+ disable:
+ - file_option: go_package
+ module: buf.build/bufbuild/protovalidate
Adding rules to a message
To add rules to a message, you'll first import Protovalidate and then add Protovalidate annotations.
Make the following changes to proto/bufbuild/weather/v1/weather_service.proto
to add rules to a GetWeatherRequest
message. (Java note: this directory is relative to src/main
.)
syntax = "proto3";
package bufbuild.weather.v1;
+ import "buf/validate/validate.proto";
import "google/protobuf/timestamp.proto";
// GetWeatherRequest is a request for weather at a point on Earth.
message GetWeatherRequest {
// latitude must be between -90 and 90, inclusive, to be valid. Use of a
// float allows precision to about one square meter.
- float latitude = 1;
+ float latitude = 1 [
+ (buf.validate.field).float.gte = -90,
+ (buf.validate.field).float.lte = 90
+ ];
// longitude must be between -180 and 180, inclusive, to be valid. Use of a
// float allows precision to about one square meter.
- float longitude = 2;
+ float longitude = 2 [
+ (buf.validate.field).float.gte = -180,
+ (buf.validate.field).float.lte = 180
+ ];
// forecast_date for the weather request. It must be within the next
// three days.
google.protobuf.Timestamp forecast_date = 3;
}
Generating code
Protovalidate doesn't introduce any new code generation plugins because its rules are compiled as part of your service and message descriptors—buf generate
works without any changes.
Run it to include your new rules in the GetWeatherRequest
descriptor:
To learn more about generating code with the Buf CLI, read the code generation overview.
Adding business logic with CEL
If Protovalidate only provided logical validations on known types, such as maximum and minimum values or verifying required fields were provided, it'd be an incomplete library. Real world validation rules are often more complicated:
- A
BuyMovieTicketsRequest
request must be for ashowtime
in the future but no more than two weeks in the future. - A
SaveBlogEntryRequest
must have astatus
ofDRAFT
,PUBLISHED
, orARCHIVED
. - An
AddProductToInventoryRequest
must have a serial number starting with a constant prefix and matching a complicated regular expression.
Protovalidate can meet all of these requirements because all Protovalidate rules are defined in Common Expression Language (CEL).
CEL is a lightweight, high-performance expression language that allows expressions like this.first_flight_duration + this.second_flight_duration < duration('48h')
to evaluate consistently across languages.
Adding a CEL-based rule to a field is straightforward.
Instead of a providing a static value, you provide a unique identifier (id
), an error message, and a CEL expression.
Building on the prior GetWeatherRequest
example, add a custom rule stating that users must ask for weather forecasts within the next 72 hours:
syntax = "proto3";
package bufbuild.weather.v1;
import "buf/validate/validate.proto";
import "google/protobuf/timestamp.proto";
// GetWeatherRequest is a request for weather at a point on Earth.
message GetWeatherRequest {
// latitude must be between -90 and 90, inclusive, to be valid. Use of a
// float allows precision to about one square meter.
float latitude = 1 [
(buf.validate.field).float.gte = -90,
(buf.validate.field).float.lte = 90
];
// longitude must be between -180 and 180, inclusive, to be valid. Use of a
// float allows precision to about one square meter.
float longitude = 2 [
(buf.validate.field).float.gte = -180,
(buf.validate.field).float.lte = 180
];
// forecast_date for the weather request. It must be within the next
// three days.
- google.protobuf.Timestamp forecast_date = 3;
+ google.protobuf.Timestamp forecast_date = 3 [(buf.validate.field).cel = {
+ id: "forecast_date.within_72_hours"
+ message: "Forecast date must be in the next 72 hours."
+ expression: "this >= now && this <= now + duration('72h')"
+ }];
}
Remember to recompile and regenerate code:
Running validations
All Protovalidate languages provide an idiomatic API for validating a Protobuf message.
In the final code exercise, you'll use it directly, checking enforcement of GetWeatherRequest
's validation rules.
-
Make sure you've navigated to
protovalidate/quickstart-go/start
within thebuf-examples
repository. -
Install Protovalidate using
go get
: -
Run
weather/weather_test.go
withgo test
. It should fail—it expects invalid latitudes and longitudes to be rejected, but you haven't yet added any validation.$ go test ./weather --- FAIL: TestRequests (0.00s) --- FAIL: TestRequests/latitude_too_low (0.00s) weather_test.go:65: Error Trace: /Users/janedoe/dev/bufbuild/buf-examples/protovalidate/quickstart-go/start/weather/weather_test.go:65 Error: An error is expected but got nil. Test: TestRequests/latitude_too_low --- FAIL: TestRequests/latitude_too_high (0.00s) weather_test.go:65: Error Trace: /Users/janedoe/dev/bufbuild/buf-examples/protovalidate/quickstart-go/start/weather/weather_test.go:65 Error: An error is expected but got nil. Test: TestRequests/latitude_too_high FAIL FAIL github.com/bufbuild/buf-examples/protovalidate/quickstart-go/start/weather 0.244s FAIL
-
Open
weather/weather.go
. Update thevalidateWeather
function to return the result ofprotovalidate.Validate()
:weather/weather.gopackage weather import ( weatherv1 "github.com/bufbuild/buf-examples/protovalidate/quickstart-go/start/gen/bufbuild/weather/v1" + "github.com/bufbuild/protovalidate-go" ) - func validateWeather(_ *weatherv1.GetWeatherRequest) error { - // TODO: validate the request - return nil - } + func validateWeather(req *weatherv1.GetWeatherRequest) error { + return protovalidate.Validate(req) + }
-
Run
go test
. Now that you've added validation, all tests should pass.
-
Make sure you've navigated to
protovalidate/quickstart-java/start
within thebuf-examples
repository. -
Open
build.gradle.kts
and verify thatlibs.protovalidate
has already been added as a dependency. (In your own projects, you'd need to add it.) -
Run
WeatherTest
with./gradlew test
. It should fail—it expects invalid latitudes and longitudes to be rejected, but you haven't yet added any validation.$ ./gradlew test > Task :test FAILED WeatherTest > TestBadLatitude() FAILED org.opentest4j.AssertionFailedError at WeatherTest.java:56 WeatherTest > TestValidRequest() PASSED WeatherTest > TestBadLongitude() FAILED org.opentest4j.AssertionFailedError at WeatherTest.java:73 WeatherTest > TestBadForecastDate() FAILED org.opentest4j.AssertionFailedError at WeatherTest.java:90
-
Open
WeatherService
(src/main/java/bufbuild/weather
). Update thevalidateGetWeatherRequest
function to return the result ofvalidator.validate()
:WeatherServicepublic class WeatherService { private static final Validator validator = new Validator(); public ValidationResult validateGetWeatherRequest(GetWeatherRequest request) throws ValidationException { - return new ValidationResult(Collections.emptyList()); + return validator.validate(request); } }
-
Run
./gradlew test
. Now that you've added validation, all tests should pass.
-
Make sure you've navigated to
protovalidate/quickstart-python/start
within thebuf-examples
repository. -
Using a virtual environment, install dependencies.
-
Run
weather_test.py
. It should fail—it expects invalid latitudes and longitudes to be rejected, but you haven't yet added any validation.$ python3 -m unittest -v weather_test.py test_bad_forecast_date (weather_test.WeatherTest.test_bad_forecast_date) ... FAIL test_bad_latitude (weather_test.WeatherTest.test_bad_latitude) ... FAIL test_bad_longitude (weather_test.WeatherTest.test_bad_longitude) ... FAIL test_valid_request (weather_test.WeatherTest.test_valid_request) ... ok
-
Open
weather.py
. Update thevalidateWeather
function to return the result ofprotovalidate.validate()
: -
Run
weather_test.py
. Now that you've added validation, all tests should pass.
You've now walked through the basic steps for using Protovalidate: adding it as a dependency, annotating your schemas with rules, and validating Protobuf messages.
Validating API requests
One of Protovalidate's most common use cases is for validating requests made to RPC APIs. Though it's possible to use the above examples to add a validation request at the start of every request handler, it's not efficient. Instead, use Protovalidate within a ConnectRPC or gRPC interceptor, providing global input validation.
Open-source Protovalidate interceptors are available for Connect Go and gRPC-Go. In the quickstarts for specific languages and gRPC frameworks, you'll also find example interceptors for Java, and Python.
Adding these interceptors is no different from configuring any other RPC interceptor:
// Create the validation interceptor provided by connectrpc.com/validate.
interceptor, err := validate.NewInterceptor()
if err != nil {
log.Fatal(err)
}
// Include the interceptor when adding handlers.
path, handler := weatherv1connect.NewWeatherServiceHandler(
weatherServer,
connect.WithInterceptors(interceptor),
)
// Create a Protovalidate Validator
validator, err := protovalidate.New()
assert.Nil(t, err)
// Use the protovalidate_middleware interceptor provided by grpc-ecosystem
interceptor := protovalidate_middleware.UnaryServerInterceptor(validator)
// Include the interceptor when configuring the gRPC server.
grpcServer := grpc.NewServer(
grpc.UnaryInterceptor(interceptor),
)
For a deep dive into using Protovalidate for RPC APIs with executable tutorials, explore one of these example Protovalidate integrations:
Validating Kafka messages
In traditional Kafka, brokers are simple data pipes—they have no understanding of what data traverses them. Though this simplicity helped Kafka gain ubiquity, most data sent through Kafka topics is structured and should follow a schema.
Using Bufstream—the Kafka-compatible message queue built for the data lakehouse era—you can add Protovalidate rule enforcement to broker-side schema awareness.
With a Bufstream broker already using the Buf Schema Registry's Confluent Schema Registry support, enabling Protovalidate is a two-line configuration change within data_enforcement
:
data_enforcement:
produce:
- topics: { all: true }
values:
on_parse_error: REJECT_BATCH
+ validation:
+ on_error: REJECT_BATCH
For a deep dive into using Protovalidate with Bufstream, follow the Protovalidate in Kafka quickstart.
Next steps
Read on to learn more about enabling schema-first validation with Protovalidate:
- Discover how to add Protovalidate to Protobuf schemas.
- Learn how to write custom validation rules with Common Expression Language.
- Explore how Protovalidate works in advanced CEL topics.