From 2b18a3d89b0a66a0b6014f85a9215b54cff7b469 Mon Sep 17 00:00:00 2001 From: James Newton-King Date: Tue, 7 Jan 2020 14:26:33 +1300 Subject: [PATCH] Add gRPC versioning article (#16404) --- aspnetcore/grpc/versioning.md | 98 +++++++++++++++++++ .../versioning/sample/GreeterServiceV1.cs | 23 +++++ .../grpc/versioning/sample/greet.v1.proto | 15 +++ aspnetcore/toc.yml | 2 + 4 files changed, 138 insertions(+) create mode 100644 aspnetcore/grpc/versioning.md create mode 100644 aspnetcore/grpc/versioning/sample/GreeterServiceV1.cs create mode 100644 aspnetcore/grpc/versioning/sample/greet.v1.proto diff --git a/aspnetcore/grpc/versioning.md b/aspnetcore/grpc/versioning.md new file mode 100644 index 0000000000..cc8bea16c0 --- /dev/null +++ b/aspnetcore/grpc/versioning.md @@ -0,0 +1,98 @@ +--- +title: Versioning gRPC services +author: jamesnk +description: Learn how to version gRPC services. +monikerRange: '>= aspnetcore-3.0' +ms.author: jamesnk +ms.date: 01/09/2020 +uid: grpc/versioning +--- +# Versioning gRPC services + +By [James Newton-King](https://twitter.com/jamesnk) + +New features added to an app can require gRPC services provided to clients to change, sometimes in unexpected and breaking ways. When gRPC services change: + +* Consideration should be given on how changes impact clients. +* A versioning strategy to support changes should be implemented. + +## Backwards compatibility + +The gRPC protocol is designed to support services that change over time. Generally additions to gRPC services and methods are non-breaking. Non-breaking means existing clients continue to work. Changing or deleting gRPC services are breaking changes. Breaking changes mean existing clients fail. + +Making non-breaking changes to a service has a number of benefits: + +- Existing clients continue to run. +- Avoids work involved with notifying clients of breaking changes, and updating them. +- Only one version of the service needs to be documented and maintained. + +This content focuses on whether changes are **breaking at a gRPC protocol and .NET binary compatibility level**. When making changes, consider whether older clients can logically continue working. For example, adding a new field to a request message: + +* Is not a protocol breaking change. +* Returning an error status on the server if the new field is not set makes it a breaking change for old clients. + +### Non-breaking changes + +These changes are non-breaking at a gRPC protocol level, and .NET binary level. + +- **Adding a new service** +- **Adding a new method to a service** +- **Adding a field to a request message** - Fields added to a request message are deserialized with the [default value](https://developers.google.com/protocol-buffers/docs/proto3#default) on the server when not set. To be non-breaking the service will need to succeed when it is not set by older clients. +- **Adding a field to a response message** - Fields added to a response message are deserialized into the message's [unknown fields](https://developers.google.com/protocol-buffers/docs/proto3#unknowns) collection on the client. +- **Adding a value to an enum** - Enums are serialized as a numeric value. New enum values are deserialized on the client to the enum value without an enum name. To be non-breaking older clients will need to run correctly when they receive and unexcepted value. + +### Binary breaking changes + +The following changes are non-breaking at a gRPC protocol level, but the client needs to be updated if it upgrades to the latest *.proto* contract or client .NET assembly. Binary compatibility is important if you plan to publish a gRPC library to NuGet. + +- **Removing a field** - Values from a removed field are deserialized to a message's [unknown fields](https://developers.google.com/protocol-buffers/docs/proto3#unknowns). This isn't a gRPC protocol breaking change, but the client needs to be updated if it upgrades to the latest contract. It is important that a removed field number isn't accidentally reused in the future. One way to make sure this doesn't happen is to specify deleted field numbers and names on the message using Protobuf's [`reserved`](https://developers.google.com/protocol-buffers/docs/proto3#reserved) keyword. +- **Renaming a field** - Field names are only used in generated code. The field number is used to identify fields on the network. The client will need to be updated if it upgrades to the latest contract. +- **Renaming a message** - Message names are not sent on the network so this isn't a gRPC protocol breaking change, but the client will need to be updated if it upgrades to the latest contract. +- **Changing csharp_namespace** - Changing `csharp_namespace` will change the namespace of generated .NET types. This isn't a gRPC protocol breaking change, but the client needs to be updated if it upgrades to the latest contract. + +### Breaking changes + +These are protocol and binary breaking changes. + +- **Changing a field data type** - Changing a field's data type to an [incompatible type](https://developers.google.com/protocol-buffers/docs/proto3#updating) will cause errors when deserializing the message. Even if the new data type is compatible, it is likely the client will need to be updated to support the new type if it upgrades to the latest contract. +- **Changing a field number** - The field number is used to identify fields on the network. +- **Renaming a package, service or method** - gRPC uses the package name, service name and method name to build the URL. The client gets an *UNIMPLEMENTED* status from the server. +- **Removing a service or method** - The client gets an *UNIMPLEMENTED* status from the server when calling the removed method. + +## Version number services + +Services should strive to remain backwards compatible with old clients. Eventually changes to your app may require breaking changes. Breaking old clients and forcing them to be updated along with your service isn't a good user experience. A way to maintain backwards compatibility while making breaking changes is to publish multiple versions of a service. + +gRPC supports an optional [`package`](https://developers.google.com/protocol-buffers/docs/proto3#packages) specifier, which functions much like a .NET namespace. In fact the `package` will be used as the .NET namespace for generated .NET types if `option csharp_namespace` is not set in the *.proto* file. The package can be used to specify a version number for your service and its messages: + +[!code-protobuf[](versioning/sample/greet.v1.proto?highlight=3)] + +The package name is combined with the service name to identify a service address. A service address allows multiple versions of a service to be hosted side-by-side: + +* `greet.v1.Greeter` +* `greet.v2.Greeter` + +Implementations of the versioned service are registered in *Startup.cs*: + +```csharp +app.UseEndpoints(endpoints => +{ + // Implements greet.v1.Greeter + endpoints.MapGrpcService(); + + // Implements greet.v2.Greeter + endpoints.MapGrpcService(); +}); +``` + +Including a version number in the package name gives you the opportunity to publish a *v2* version of your service with breaking changes, while continuing to support older clients who call the *v1* version. Once clients have updated to use the *v2* service you can choose to remove the old version. When planning to publish multiple versions of a service: + +- Avoid breaking changes if reasonable. +- Don't update the version number unless making breaking changes. +- Do update the version number when you make breaking changes. + +Publishing multiple versions of a service duplicates it. To reduce duplication, consider moving business logic from the service implementations to a centralized location that can be reused by the old and new implementations: + +[!code-csharp[](versioning/sample/GreeterServiceV1.cs?highlight=10,19)] + +Services and messages generated with different package names are **different .NET types**. Moving business logic to a centralized location requires mapping messages to common types. diff --git a/aspnetcore/grpc/versioning/sample/GreeterServiceV1.cs b/aspnetcore/grpc/versioning/sample/GreeterServiceV1.cs new file mode 100644 index 0000000000..8ad2a1a940 --- /dev/null +++ b/aspnetcore/grpc/versioning/sample/GreeterServiceV1.cs @@ -0,0 +1,23 @@ +using Greet.V1; +using Grpc.Core; +using System.Threading.Tasks; + +namespace Services +{ + public class GreeterServiceV1 : Greeter.GreeterBase + { + private readonly IGreeter _greeter; + public GreeterServiceV1(IGreeter greeter) + { + _greeter = greeter; + } + + public override Task SayHello(HelloRequest request, ServerCallContext context) + { + return Task.FromResult(new HelloReply + { + Message = _greeter.GetHelloMessage(request.Name) + }); + } + } +} \ No newline at end of file diff --git a/aspnetcore/grpc/versioning/sample/greet.v1.proto b/aspnetcore/grpc/versioning/sample/greet.v1.proto new file mode 100644 index 0000000000..36c875f056 --- /dev/null +++ b/aspnetcore/grpc/versioning/sample/greet.v1.proto @@ -0,0 +1,15 @@ +syntax = "proto3"; + +package greet.v1; + +service Greeter { + rpc SayHello (HelloRequest) returns (HelloReply); +} + +message HelloRequest { + string name = 1; +} + +message HelloReply { + string message = 1; +} \ No newline at end of file diff --git a/aspnetcore/toc.yml b/aspnetcore/toc.yml index d308984ee9..6e25fcd6bb 100644 --- a/aspnetcore/toc.yml +++ b/aspnetcore/toc.yml @@ -584,6 +584,8 @@ uid: grpc/diagnostics - name: Security considerations uid: grpc/security + - name: Versioning gRPC services + uid: grpc/versioning - name: Manage Protobuf references with dotnet-grpc uid: grpc/dotnet-grpc - name: Migrate gRPC services from C-core