Protovalidate in Connect and Go
This tutorial shows how to add Protovalidate to a Go RPC powered by Connect:
- Adding the Protovalidate dependency.
- Annotating Protobuf files and regenerating code.
- Adding a Connect interceptor.
- Testing your validation logic.
Just need an example?
There's an example of Protovalidate for Connect and Go in GitHub.
Prerequisites
- Install the Buf CLI. If you already have, run
buf --version
to verify that you're using at least1.32.0
. - Have
git
andgo
installed and in your$PATH
. -
Clone the
buf-examples
repo and navigate to theprotovaldiate/connect-go/start
directory:
Goal
This tutorial's CreateInvoice
RPC doesn't have any input validation. Your goal is to pass a unit test verifying that you've added two validation rules using Protovalidate:
- Requests must provide an
Invoice
with a UUIDinvoice_id
. - Within the
Invoice
message, all of itsrepeated LineItem line_items
must have unique combinations ofproduct_id
andunit_price
.
Run the test now, and you can see that it fails:
$ go test -v ./...
--- PASS: TestCreateInvoice/A_valid_invoice_passes_validation (0.00s)
--- FAIL: TestCreateInvoice/InvoiceId_is_required (0.00s)
--- FAIL: TestCreateInvoice/Two_line_items_cannot_have_the_same_product_id_and_unit_price (0.00s)
When this test passes, you've met your goal.
Run the server
Before you begin to code, verify that the example is working. Compile and start the included server:
After a few seconds, you should see that it has started:
In a second terminal window, use buf curl
to send an invalid CreateInvoiceRequest
:
$ buf curl \
--data '{ "invoice": { "invoice_id": "" } }' \
--schema . \
--http2-prior-knowledge \
http://localhost:8080/invoice.v1.InvoiceService/CreateInvoice
The server should respond with the version number of the invoice that was created, despite the invalid request. That's what you're here to fix.
Before you start coding, take a few minutes to explore the code in the example.
Explore tutorial code
This tutorial uses the example in connect-go/start
. All filenames are relative to this directory.
Protobuf
The project provides a single unary RPC:
// InvoiceService is a simple CRUD service for managing invoices.
service InvoiceService {
// CreateInvoice creates a new invoice.
rpc CreateInvoice(CreateInvoiceRequest) returns (CreateInvoiceResponse);
}
CreateInvoiceRequest
includes an invoice
field that's an Invoice
message. An Invoice
has a repeated field of type LineItem
:
message Invoice {
// invoice_id is a unique identifier for this invoice.
string invoice_id = 1;
// line_items represent individual items on this invoice.
repeated LineItem line_items = 4;
}
// LineItem is an individual good or service added to an invoice.
message LineItem {
// product_id is the unique identifier for the good or service on this line.
string product_id = 2;
// quantity is the unit count of the good or service provided.
uint64 quantity = 3;
}
YAML
When you add Protovalidate, you'll update the following files:
buf.yaml
: Protovalidate must be added as a dependency.buf.gen.yaml
: To avoid a common Go issue in projects using the Buf CLI's managed mode, you'll see how to exclude Protovalidate from package renaming.
Go
You'll be working in cmd/server.go
. It's an executable that runs a server on port 8080. You'll edit it to add a Protovalidate interceptor to Connect.
Where's the service itself?
internal/invoice/invoice.go
provides invoice.Service
, a handler for the InvoiceService
. Its CreateInvoice
function sends back a static response.
Now that you know your way around the example code, it's time to integrate Protovalidate.
Integrate Protovalidate
It's time to add Protovalidate to your project. It may be useful to read the Protovalidate overview and its quickstart before continuing.
Add Protovalidate dependency
Because Protovalidate is a publicly available Buf Schema Registry (BSR) module, it's simple to add it to any Buf CLI project.
-
Add Protovalidate to your Go project:
-
Add Protovalidate as a dependency to
buf.yaml
. -
Update dependencies with the Buf CLI. You'll be warned that Protovalidate is declared but unused. That's fine.
-
Because this example uses managed mode, exclude Protovalidate from any updates to
go_package
.buf.gen.yamlversion: v2 inputs: - directory: proto plugins: - remote: buf.build/protocolbuffers/go out: gen opt: paths=source_relative - remote: buf.build/connectrpc/go out: gen opt: paths=source_relative managed: enabled: true override: - file_option: go_package_prefix value: github.com/bufbuild/buf-examples/protovalidate/connect-go/start/gen + disable: + - file_option: go_package + module: buf.build/bufbuild/protovalidate
-
Verify that configuration is complete by running
buf generate
. It should complete with no error.
Further reading and protoc
Learn more about incorporating Protovalidate and protoc
support in the Adding protovalidate page.
Add a standard rule
You'll now add a standard rule to proto/invoice.proto
to require that the invoice_id
field is a UUID. Start by importing Protovalidate:
syntax = "proto3";
package invoice.v1;
+ import "buf/validate/validate.proto";
import "google/protobuf/timestamp.proto";
You could use the required
rule to verify that requests provide this field, but Protovalidate makes it easy to do more specific validations.
Use string.uuid
to declare that invoice_id
must be present and a valid UUID.
// Invoice is a collection of goods or services sold to a customer.
message Invoice {
// invoice_id is a unique identifier for this invoice.
- string invoice_id = 1;
+ string invoice_id = 1 [
+ (buf.validate.field).string.uuid = true
+ ];
// account_id is the unique identifier for the account purchasing goods.
string account_id = 2;
// invoice_date is the date for an invoice. It should represent a date and
// have no values for time components.
google.protobuf.Timestamp invoice_date = 3;
// line_items represent individual items on this invoice.
repeated LineItem line_items = 4;
}
Enforce complex rules
In Invoice
, the line_items
field needs to meet two business rules:
- There should always be at least one
LineItem
. - No two
LineItems
should ever share the sameproduct_id
andprice
.
Protovalidate can enforce both of these rules by combining a standard rule with a custom rule written in Common Expression Language (CEL).
First, use the min_items
standard rule to require at least one LineItem
:
// Invoice is a collection of goods or services sold to a customer.
message Invoice {
// invoice_id is a unique identifier for this invoice.
string invoice_id = 1 [
(buf.validate.field).string.uuid = true
];
// account_id is the unique identifier for the account purchasing goods.
string account_id = 2;
// invoice_date is the date for an invoice. It should represent a date and
// have no values for time components.
google.protobuf.Timestamp invoice_date = 3;
// line_items represent individual items on this invoice.
- repeated LineItem line_items = 4;
+ repeated LineItem line_items = 4 [
+ (buf.validate.field).repeated.min_items = 1
+ ];
}
Next, use a CEL expression to add a custom rule. Use the map
, string
, and unique
CEL functions to check that no combination of product_id
and unit_price
appears twice within the array of LineItems
:
// Invoice is a collection of goods or services sold to a customer.
message Invoice {
// invoice_id is a unique identifier for this invoice.
string invoice_id = 1 [
(buf.validate.field).string.uuid = true
];
// account_id is the unique identifier for the account purchasing goods.
string account_id = 2;
// invoice_date is the date for an invoice. It should represent a date and
// have no values for time components.
google.protobuf.Timestamp invoice_date = 3;
// line_items represent individual items on this invoice.
repeated LineItem line_items = 4 [
- (buf.validate.field).repeated.min_items = 1
+ (buf.validate.field).repeated.min_items = 1,
+
+ (buf.validate.field).cel = {
+ id: "line_items.logically_unique"
+ message: "line items must be unique combinations of product_id and unit_price"
+ expression: "this.map( it, it.product_id + '-' + string(it.unit_price) ).unique()"
+ }
];
}
You've added validation rules to your Protobuf. To enforce them, you still need to regenerate code and add a Protovalidate interceptor to your server.
Learn more about custom rules.
Compile Protobuf and Go
Next, compile your Protobuf and regenerate code, adding the Protovalidate options to all of your message descriptors:
If you're still running the server, stop it with Ctrl-c
. Compile the new generated code and restart:
After a few seconds, you should see that it has started:
In a second terminal window, use buf curl
to send the same invalid CreateInvoiceRequest
:
$ buf curl \
--data '{ "invoice": { "invoice_id": "" } }' \
--schema . \
--http2-prior-knowledge \
http://localhost:8080/invoice.v1.InvoiceService/CreateInvoice
The response may be a surprise: the server still considers the request valid and returns a version number for the new invoice:
The RPC is still successful because Connect hasn't been told to validate inbound requests.
Note
No Connect or gRPC implementations automatically enforce Protovalidate rules. To enforce your validation rules, you'll need to add an interceptor.
Add a Protovalidate interceptor
Thanks to Connect RPC's prebuilt Protovalidate interceptor, adding a Protovalidate interceptor for Connect-Go is just a few lines of code.
Follow these steps to begin enforcing Protovalidate rules with connectrpc.com/validate
:
- In your first terminal window, use
Ctrl-c
to stop your server. -
Install the interceptor.
-
Import
connectprc.com/connect
andconnectrpc.com/validate
incmd/server.go
.cmd/server.goimport ( "log/slog" "net/http" "os" "time" + "connectrpc.com/connect" + "connectrpc.com/validate" "github.com/bufbuild/buf-examples/protovalidate/connect-go/finish/gen/invoice/v1/invoicev1connect" "github.com/bufbuild/buf-examples/protovalidate/connect-go/finish/internal/invoice" "golang.org/x/net/http2" "golang.org/x/net/http2/h2c" )
-
Add the interceptor to any handler definitions that should use Protovalidate:
cmd/server.gofunc main() { // Code omitted for brevity + // Create the validation interceptor provided by connectrpc.com/validate. + validateInterceptor, err := validate.NewInterceptor() + if err != nil { + slog.Error("error creating interceptor", + slog.String("error", err.Error()), + ) + os.Exit(1) + } + // Include the interceptor when adding handlers. mux.Handle(invoicev1connect.NewInvoiceServiceHandler( invoiceServer, + connect.WithInterceptors(validateInterceptor), ) // Code omitted for brevity }
-
Stop (
Ctrl-c
) and restart your server:After a few seconds, you should see that it has started:
Now that you've added the Connect interceptor and restarted your server, try the buf curl
command again:
$ buf curl \
--data '{ "invoice": { "invoice_id": "" } }' \
--schema . \
--http2-prior-knowledge \
http://localhost:8080/invoice.v1.InvoiceService/CreateInvoice
This time, you should receive a block of JSON representing Protovalidate's enforcement of your rules. In the abridged excerpt below, you can see that it contains details about every field that violated Protovalidate rules:
{
"violations": [
{
"fieldPath": "invoice.invoice_id",
"constraintId": "string.uuid_empty",
"message": "value is empty, which is not a valid UUID"
},
{
"fieldPath": "invoice.line_items",
"constraintId": "repeated.min_items",
"message": "value must contain at least 1 item(s)"
}
]
}
Running the buf curl
command again with the --verbose
flag shows that the interceptor also sends the correct HTTP status code:
$ buf curl \
--data '{ "invoice": { "invoice_id": "" } }' \
--schema . \
--http2-prior-knowledge \
--verbose \
http://localhost:8080/invoice.v1.InvoiceService/CreateInvoice
Last, use buf curl
to test the custom rule that checks for logically unique LineItems
:
$ buf curl \
--data '{ "invoice": { "invoice_id": "079a91c2-cb8b-4f01-9cf9-1b9c0abdd6d2", "line_items": [{"product_id": "A", "unit_price": "1" }, {"product_id": "A", "unit_price": "1" }] } }' \
--schema . \
--http2-prior-knowledge \
http://localhost:8080/invoice.v1.InvoiceService/CreateInvoice
You can see that this more complex expression is enforced at runtime:
{
"violations": [
{
"fieldPath": "invoice.line_items",
"constraintId": "line_items.logically_unique",
"message": "line items must be unique combinations of product_id and unit_price"
}
]
}
You've now added Protovalidate to a Connect RPC in Go, but buf curl
isn't a great way to make sure you're meeting all of your requirements.
Next, you'll see one way to verify Protovalidate rules in tests.
Test Protovalidate errors
The starting code for this tutorial contains a table test in internal/invoice/invoice_test.go
. It starts a server with a Protovalidate interceptor and iterates through a series of test cases.
In the prior section, you saw that the violations
list returned by Protovalidate follows a predictable structure. Each violation in the list is a Protobuf message named Violation
, defined within Protovalidate itself.
The test already provides a convenient way to declare expected violations through a violationSpec
type:
// violationSpec is a simple representation of fields tested when inspecting
// a connect.Error that we expect to contain Protovalidate validate.Violations
// messages.
type violationSpec struct {
constraintID string
fieldPath string
message string
}
Examine the highlighted lines in invoice_test.go
, noting that the tests check for specific expected violations:
func TestCreateInvoice(t *testing.T) {
// Code omitted for brevity...
testCases := map[string]struct {
producer func(invoice *invoicev1.Invoice) *invoicev1.Invoice
violations []violationSpec
}{
"An invoice can be created": {
producer: func(invoice *invoicev1.Invoice) *invoicev1.Invoice {
return invoice
},
},
"InvoiceId is required": {
producer: func(invoice *invoicev1.Invoice) *invoicev1.Invoice {
invoice.InvoiceId = ""
return invoice
},
violations: []violationSpec{
{
constraintID: "string.uuid_empty",
fieldPath: "invoice.invoice_id",
message: "value is empty, which is not a valid UUID",
},
},
},
"Two line items cannot have the same product_id and unit price": {
producer: func(invoice *invoicev1.Invoice) *invoicev1.Invoice {
invoice.GetLineItems()[0].ProductId = invoice.GetLineItems()[1].GetProductId()
invoice.GetLineItems()[0].UnitPrice = invoice.GetLineItems()[1].GetUnitPrice()
return invoice
},
violations: []violationSpec{
{
constraintID: "line_items.logically_unique",
fieldPath: "invoice.line_items",
message: "line items must be unique combinations of product_id and unit_price",
},
},
},
}
// Code omitted for brevity
}
To check your work, run all tests with the -v
flag.
If all tests pass, you've met your goal:
--- PASS: TestCreateInvoice (0.00s)
--- PASS: TestCreateInvoice/InvoiceId_is_required (0.01s)
--- PASS: TestCreateInvoice/InvoiceId_must_be_a_UUID (0.00s)
--- PASS: TestCreateInvoice/Two_line_items_cannot_have_the_same_identifier (0.01s)
PASS
ok github.com/bufbuild/buf-examples/protovalidate/connect-go/finish/internal/invoice 0.485s
More testing examples
The finish
directory contains a thorough test that you can use as an example for your own tests. Its invoice.proto
file also contains extensive Protovalidate rules.
Wrapping up
In this tutorial, you've learned the basics of working with Protovalidate:
- Adding Protovalidate to your project.
- Declaring validation rules in your Protobuf files.
- Enabing their enforcement within an RPC API.
- Testing their functionality.
Further reading
- Integrate Protovalidate validation within Kafka brokers
- Add Protovalidate's standard rules to schemas
- Use CEL expressions to declare field and message-level custom rules
- Reuse logic with predefined rules
- Learn more about Protovalidate's relationship with CEL.