In previous labs, we have created a web site that shoppers can use to browser a through pages of products, optionally with filtering by brand or type, and added the ability for users to create an account and sign in. In this lab, we will add the capability to add products to a shopping basket. The shopping basket will be stored in Redis, and exposed via a new gRPC service that the web site will communicate with.
-
Add a new project called
Basket.API
to the solution using the ASP.NET Core gRPC Service template. Ensure that the following template options are configured:- Framework: .NET 8.0 (Long Term Support)
- Enable container support: disabled
- Do not use top-level statements: disabled
- Enable native AOT publish: disabled
- Enlist in .NET Aspire orchestration: enabled
In the .NET CLI, we need to do a few steps manually to configure .NET Aspire orchestration.
-
Run the following commands in the
src
folder to create theBasket.API
gRPC project.dotnet new grpc -n Basket.API dotnet sln add Basket.API
-
Add a reference to the
eShop.AppHost
from theBasket.API
project:cd eShop.AppHost dotnet add reference ..\Basket.API
-
Add a reference from the
Basket.API
project to theeShop.ServiceDefaults
project:cd ..\Basket.API dotnet add reference ..\ServiceDefaults
-
Open the
Program.cs
file in theAppHost
project and add the following code to create a new resource for theBasket.API
project:var basketApi = builder.AddProject<Projects.Basket_API>("basket-api");
-
Open the
Program.cs
file in theBasket.API
and add a line at the top to add the service defaults:builder.AddServiceDefaults();
-
Open the
Program.cs
file in the AppHost project and add the following code:var builder = DistributedApplication.CreateBuilder(args); builder.AddProject<Projects.Catalog_Data_Manager>("catalog-db-mgr"); builder.Build().Run();
-
Open the
Basket.Api.csproj
file and add a line to thePropertyGroup
at the top change the default root namespace of the project toeShop.Basket.API
:<RootNamespace>eShop.Basket.API</RootNamespace>
-
Install the
Aspire.Hosting.Redis
package in theeShop.AppHost
project using either of the following:dotnet add package Aspire.Hosting.Redis
<PackageReference Include="Aspire.Hosting.Redis" Version="8.2.0" />
-
Open the
Program.cs
file in theeShop.AppHost
project and add a line to create a new Redis resource named"BasketStore"
and configure it to host a Redis Commander instance too (this will make it easier to inspect the Redis database during development). Capture the resource in abasketStore
variable:var basketStore = builder.AddRedis("BasketStore").WithRedisCommander();
Aspire will automatically create containers for both Redis and Redis Commander when the application is run.
-
Find the line that was added to add the
Basket.API
gRPC project to the AppHost as a resource. Update the code to name the resource"basket-api"
, make it reference theidp
andBasketStore
resources, and capture it in abasketApi
variable:var basketApi = builder.AddProject<Projects.Basket_API>("basket-api") .WithReference(idp) .WithReference(basketStore);
The
Basket.API
will require calls to be authenticated by the IdP, and will need to access the Redis database to store and retrieve shopping baskets. -
Update the
webapp
resource to reference thebasket-api
resource so the web site can communicate with the Basket API:var webApp = builder.AddProject<Projects.WebApp>("webapp") .WithReference(catalogApi) .WithReference(basketApi) // <--- Add this line .WithReference(idp, env: "Identity__ClientSecret");
-
Run the AppHost project and verify that the containers for Redis and Redis Commander are created and running by using the dashboard. Also verify that the
Basket.API
project is running and that it's environment variables contain the configuration values to communicate with the IdP and Redis.
-
Following the pattern we've used in our other projects, create a new file
HostingExtensions.cs
in anExtensions
directory in theBasket.API
project and add a class to it calledHostingExtensions
. Add a method to the class calledAddApplicationServices
:namespace Microsoft.Extensions.Hosting; public static class HostingExtensions { public static IHostApplicationBuilder AddApplicationServices(this IHostApplicationBuilder builder) { return builder; } }
We'll add code here later to configure the services in the application's DI container as we build out the Basket API.
-
In the
Program.cs
file, add a call to theAddApplicationServices
method after the call toAddServiceDefaults
:builder.AddServiceDefaults(); builder.AddApplicationServices();
-
To communicate with Redis we'll use the Aspire StackExchange Redis component. Aspire components are NuGet packages that integrate common client libraries with the Aspire stack, ensuring that they're configured with the application's DI container and setup for observability, reliability, and configurability.
Add the
Aspire.StackExchange.Redis
component NuGet package to theBasket.API
project. You can use the Add > .NET Aspire Compoenent... project menu item in Visual Studio, thedotnet add package
command at the command line, or by editing theBasket.API.csproj
file directly:<PackageReference Include="Aspire.StackExchange.Redis" Version="8.2.0" />
-
In the
AddApplicationServices
method inHostingExtensions.cs
, add a call toAddRedis
to configure the Redis client in the application's DI container. Pass the name"BasketStore"
to the method to indicate that the client should be configured to connect to the Redis resource with that name in the AppHost:public static IHostApplicationBuilder AddApplicationServices(this IHostApplicationBuilder builder) { builder.AddRedisClient("BasketStore"); return builder; }
Now, to use the Redis client in the application, we simply need to accept a constructor parameter of type
IConnectionMultiplexer
-
gRPC services are defined using Protocol Buffers (protobuf) files. Add a new file called
basket.proto
to theProtos
directory in theBasket.API
project. Add the following content to the file:syntax = "proto3"; option csharp_namespace = "eShop.Basket.API.Grpc"; package BasketApi; service Basket { rpc GetBasket(GetBasketRequest) returns (CustomerBasketResponse) {} rpc UpdateBasket(UpdateBasketRequest) returns (CustomerBasketResponse) {} } message GetBasketRequest { } message UpdateBasketRequest { repeated BasketItem items = 1; } message BasketItem { int32 product_id = 1; int32 quantity = 2; } message CustomerBasketResponse { repeated BasketItem items = 1; }
This file defines a gRPC service called
Basket
with two methods:GetBasket
andUpdateBasket
. TheGetBasket
method takes aGetBasketRequest
message and returns aCustomerBasketResponse
message with a repeatedBasketItem
field. TheUpdateBasket
method takes anUpdateBasketRequest
message with a repeatedBasketItem
field and also returns aCustomerBasketResponse
. -
Delete the
greeter.proto
file that was included with the template. -
Update the
Basket.API.csproj
file to add thebasket.proto
file as aProtobuf
item in the project and set theGrpcServices
metadata to"Server"
to indicate that it should be used to generate server-side code for the gRPC service:<ItemGroup> <Protobuf Include="Protos\basket.proto" GrpcServices="Server" /> </ItemGroup>
The project system will automatically generate code behind the scenes to represent the messages and service defined in the
basket.proto
file. We'll use these generated types in our service implementation. -
In the
Basket.API
project, create aGrpc
directory and add a new file to it namedBasketService.cs
. Use the following code to define a class in it namedBasketService
that derives fromBasket.BasketBase
:namespace eShop.Basket.API.Grpc; public class BasketService : Basket.BasketBase { }
-
In the
Program.cs
file, update the line that maps theGreeterService
gRPC service so that it maps theBasketService
instead:app.MapGrpcService<BasketService>();
-
Delete the
Services/GreeterService.cs
file that was included with the template, including theServices
directory.
-
In the
Basket.API
project, create a fileRedisBasketStore.cs
in aStorage
directory and define a class in it namedRedisBasketStore
with a constructor that accepts a single parameter of typeIConnectionMultiplexer
:namespace eShop.Basket.API.Storage; public class RedisBasketStore(IConnectionMultiplexer redis) { }
The
RedisBasketStore
class will be responsible for storing and retrieving shopping baskets from the Redis database. -
Add a field of type
IDatabase
and initialize it by callingredis.GetDatabase()
:private readonly IDatabase _database = redis.GetDatabase();
-
Our shopping baskets will be stored in Redis with a key like
/basket/123
where123
is the user's ID. Define a field of typeRedisKey
namedBasketKeyPrefix
to store the prefix of the key, and a method namedGetBaksetKey
that accepts astring
parameteruserId
and returns aRedisKey
:private static readonly RedisKey BasketKeyPrefix = "/basket/"; private static RedisKey GetBasketKey(string userId) => BasketKeyPrefix.Append(userId);
-
Before we can add our methods to get and update the basket, we need to define the types that will be used to represent the shopping basket in Redis. Instances of these types will be serialized and deserialized to and from JSON.
Add a new file
BasketItem.cs
in aModels
directory and define a class in it namedBasketItem
with properties for the product ID, product name, unit price, and quantity:namespace eShop.Basket.API.Models; public class BasketItem { public int ProductId { get; set; } public int Quantity { get; set; } }
Add another file
CustomerBasket.cs
in theModels
directory and define a class in it namedCustomerBasket
with properties for the buyer ID and items in the basket:namespace eShop.Basket.API.Models; public class CustomerBasket { public required string BuyerId { get; set; } public List<BasketItem> Items { get; set; } = []; }
-
In
RedisBasketStore.cs
, add an async method namedGetBasketAsync
that accepts a parameter for the buyer ID and returns theCustomerBasket
from Redis if it exists:public async Task<CustomerBasket?> GetBasketAsync(string customerId) { var key = GetBasketKey(customerId); using var data = await _database.StringGetLeaseAsync(key); return data is { Length: > 0 } ? JsonSerializer.Deserialize<CustomerBasket>(data.Span) : null; }
Now add a method to update the basket:
public async Task<CustomerBasket?> UpdateBasketAsync(CustomerBasket basket) { var json = JsonSerializer.SerializeToUtf8Bytes(basket); var key = GetBasketKey(basket.BuyerId); var created = await _database.StringSetAsync(key, json); return created ? await GetBasketAsync(basket.BuyerId) : null; }
-
In
HostingExtensions.cs
, add a call in theAddApplicationServices
method toAddSingleton
to register theRedisBasketStore
class in the application's DI container:builder.Services.AddSingleton<RedisBasketStore>();
The
RedisBasketStore
class is now ready to be used by our gRPCBasketService
class.
-
Back in the
BasketService.cs
file, update the constructor to accept aRedisBasketStore
parameter. This will be populated from the application's DI container when the service is created:public class BasketService(RedisBasketStore basketStore) : Basket.BasketBase { }
-
Add methods to convert between our model types (those serialized to JSON by
BasketService
to store in Redis) and our gRPC message types:private static CustomerBasketResponse MapToCustomerBasketResponse(CustomerBasket customerBasket) { var response = new CustomerBasketResponse(); foreach (var item in customerBasket.Items) { response.Items.Add(new BasketItem { ProductId = item.ProductId, Quantity = item.Quantity, }); } return response; } private static CustomerBasket MapToCustomerBasket(string userId, UpdateBasketRequest customerBasketRequest) { var response = new CustomerBasket { BuyerId = userId }; foreach (var item in customerBasketRequest.Items) { response.Items.Add(new() { ProductId = item.ProductId, Quantity = item.Quantity, }); } return response; }
-
Add an async method named
GetBasket
that overrides the method of the same name in the base class. The method should acceptGetBasketRequest
andServerCallContext
parameters, and return aCustomerBasketResponse
:public override async Task<CustomerBasketResponse> GetBasket(GetBasketRequest request, ServerCallContext context) { return new(); }
This method needs to extract the user ID from the passed
ServerCallContext
and use it to call theGetBasketAsync
method of theRedisBasketStore
class. Let's add an extensions class to help with extracting the user ID from theServerCallContext
. -
Create a new file
GrpcExtensions.cs
in theExtensions
directory and add an extension method that extracts the user ID from theServerCallContext
:using System.Security.Claims; namespace Grpc.Core; internal static class GrpcExtensions { public static string? GetUserIdentity(this ServerCallContext context) => context.GetHttpContext().User.GetUserId(); }
-
Back in
BasketService.cs
, update theGetBasket
method to use theGetUserIdentity
extension method to extract the user ID and call theGetBasketAsync
method of theRedisBasketStore
class, before returning the result as aCustomerBasketResponse
. If the user ID is not found, the method should throw anRpcException
with a status ofUnauthenticated
:public override async Task<CustomerBasketResponse> GetBasket(GetBasketRequest request, ServerCallContext context) { var userId = context.GetUserIdentity(); if (string.IsNullOrEmpty(userId)) { ThrowNotAuthenticated(); } var data = await basketStore.GetBasketAsync(userId); return data is not null ? MapToCustomerBasketResponse(data) : new(); } [DoesNotReturn] private static void ThrowNotAuthenticated() => throw new RpcException(new Status(StatusCode.Unauthenticated, "The caller is not authenticated."));
-
We've added code that requires that the calling client be authenticated, but haven't yet configured the application to support authentication.
In the
HostingExtensions.cs
file, add a call to theAddDefaultAuthentication
method (already defined in theeShop.ServiceDefaults
project) to setup the application to use JWT Bearer authentication for tokens issued by theidp
resource:public static IHostApplicationBuilder AddApplicationServices(this IHostApplicationBuilder builder) { builder.AddDefaultAuthentication(); // <-- Add this line builder.AddRedisClient("BasketStore"); builder.Services.AddSingleton<RedisBasketStore>(); return builder; }
-
Navigate to the definition of the
AddDefaultAuthentication
method and take a moment to read through the code. This code looks similar in places to the authentication configuration code in theWebApp
project added in lab 3, but is slightly simpler as it doesn't need to configure the application to instigate login flows via OpenID Connect. It just needs to validate tokens issued by the IdP.Note that the
ConfigureDefaultJwtBearer
method expects the application configuration to have a section named Identity from which it retrieves a required value named Audience. This code will throw a runtime extepsion if it doesn't exist, so let's add that now. -
Open the
appsettings.json
file and add a new section named Identity with a property named Audience set to a value of"basket"
:{ // Add the following section to the existing file "Identity": { "Audience": "basket" } }
-
Back in
BasketService.cs
, add an async method namedUpdateBasket
that overrides the method of the same name in the base class. The method should acceptUpdateBasketRequest
andServerCallContext
parameters, and return aCustomerBasketResponse
. If the user ID is not found, the method should throw anRpcException
with a status ofUnauthenticated
. If the basket does not exist, the method should throw anRpcException
with a status ofNotFound
:public override async Task<CustomerBasketResponse> UpdateBasket(UpdateBasketRequest request, ServerCallContext context) { var userId = context.GetUserIdentity(); if (string.IsNullOrEmpty(userId)) { ThrowNotAuthenticated(); } var customerBasket = MapToCustomerBasket(userId, request); var response = await basketStore.UpdateBasketAsync(customerBasket); if (response is null) { ThrowBasketDoesNotExist(userId); } return MapToCustomerBasketResponse(response); } [DoesNotReturn] private static void ThrowBasketDoesNotExist(string userId) => throw new RpcException(new Status(StatusCode.NotFound, $"Basket with buyer ID {userId} does not exist"));
-
Run the
AppHost
project and verify that theBasket.API
project is running without any startup errors by using the dashboard to view its logs.At this point our Basket API is ready to be used by the web site. In the next section, we'll update the web site to use the new gRPC service so that shoppers can manage items in their basket.
The starting point for this lab already includes updates to the web site to provide UI for the shopping basket. The UI includes a shopping basket icon in the header that displays the number of items in the basket, and a shopping basket page that displays the items in the basket and allows the user to update the quantity of items in the basket. The item page has also been updated to display a button that allows the user to add the item to the basket, or sign in if they are not already signed in.
-
Open the
BasketState.cs
file in theWebApp
project and add take a moment to read through the code. This class is used to manage the current state of the shopping bag during a web request. Blazor components run their logic individually as the component tree is rendered, so the state of the shopping basket needs to be managed in a way that is shared between components during a single request. -
Open the
CartPage.razor
file and take a moment to read through the code. This component displays the items in the shopping basket and allows the user to update the quantity of items in the basket. The component uses theBasketState
class to manage the state of the shopping basket during the request.Pay special note to the logic in the
CurrentOrPendingQuantity
method that ensures the quantity rendered in the UI matches what the user has requested if they have requested a change, or the current quantity if they have not. This is required due to the way Blazor components are rendered and re-rendered during a request when using Streaming Rendering. -
Open the
ItemPage.razor
file and take a moment to read through the code. This component has been updated with a form that the user can use to add the item to their cart, or sign in if required. If the item is already in their basket, it displays the quantity. -
Run the AppHost project and navigate to the web site homepage. Verify that the shopping basket icon is in the header and that after signing-in in it links to the cart page, and on the item pages an Add to shopping bag button is shown:
Next we'll make the changes required to make the web site shopping bag UI elements actually work.
-
In the
WebApp
project, add a reference to the following NuGet packages:Grpc.AspNetCore.Server.ClientFactory
Grpc.Tools
<PackageReference Include="Grpc.AspNetCore.Server.ClientFactory" Version="2.65.0" /> <PackageReference Include="Grpc.Tools" PrivateAssets="All" Version="2.65.0" />
-
In the
WebApp.csproj
project file, add thebasket.proto
file from theBasket.API
project as aProtobuf
item in the project and set theGrpcServices
metadata to"Client"
to indicate that it should be used to generate client-side code for the gRPC service:<ItemGroup> <Protobuf Include="..\Basket.API\Protos\basket.proto" GrpcServices="Client" /> </ItemGroup>
Similar to the
Basket.API
project, the project system will automatically generate code behind the scenes to represent the messages and service defined in thebasket.proto
file. We'll use these generated types in our client service implementation. -
Open the
BasketService.cs
file in theWebApp
project and add someusing
statements to import the namespace and create some type aliases for the generated gRPC client types. This will make it easier to refer to them in the rest of the code:using eShop.Basket.API.Grpc; using GrpcBasketItem = eShop.Basket.API.Grpc.BasketItem; using GrpcBasketClient = eShop.Basket.API.Grpc.Basket.BasketClient;
-
Update the
BasketService
constructor to accept a parameter of typeGrpcBasketClient
:public class BasketService(GrpcBasketClient basketClient) { }
-
Add a private method to convert the
CustomerBasketResponse
message returned by theBasket.API
service, to a list of theBasketQuantity
model used by the web site:private static List<BasketQuantity> MapToBasket(CustomerBasketResponse response) { var result = new List<BasketQuantity>(); foreach (var item in response.Items) { result.Add(new BasketQuantity(item.ProductId, item.Quantity)); } return result; }
The
BasketQuantity
type should already be defined at the bottom of theBasketService.cs
file. -
Update the
GetBasketAsync
method so that it actually calls theBasket.API
service to get the customer basket items:public async Task<IReadOnlyCollection<BasketQuantity>> GetBasketAsync() { var result = await basketClient.GetBasketAsync(new()); return MapToBasket(result); }
-
Update the
UpdateBasketAsync
method so that it actually calls theBasket.API
service to update the customer basket:public async Task UpdateBasketAsync(IReadOnlyCollection<BasketQuantity> basket) { var updatePayload = new UpdateBasketRequest(); foreach (var item in basket) { var updateItem = new GrpcBasketItem { ProductId = item.ProductId, Quantity = item.Quantity, }; updatePayload.Items.Add(updateItem); } await basketClient.UpdateBasketAsync(updatePayload); }
-
In the
HostingExtensions.cs
file, add a line to register theBasket.BasketClient
gRPC client in the application's DI container, and configure itsAddress
property to point to thebasket-api
resource in the AppHost (adding the necessaryusing eShop.Basket.API.Grpc
statement). Make sure you include a call to.AddAuthToken()
to add the current user's access token to outgoing requests:// HTTP and gRPC client registrations builder.Services.AddGrpcClient<Basket.BasketClient>(o => o.Address = new("http://basket-api")) .AddAuthToken(); builder.Services.AddHttpClient<CatalogService>(o => o.BaseAddress = new("http://catalog-api")); builder.Services.AddHttpClient(OpenIdConnectBackchannel, o => o.BaseAddress = new("http://idp"));
The code for
AddAuthToken
is already defined in theeShop.ServiceDefaults
project. Take a moment to read through the code to understand how it works. -
Run the AppHost project and load the web site home page. Ten seconds after the home page is initially displayed, you might see an error displayed with the message:
Grpc.Core.RpcException: Status(StatusCode="Unauthenticated", Detail="The caller is not authenticated.")
Use the dashboard page to observe more details about the error. The error is being thrown because the web site thinks the user is authenticated thanks to the authentication session cookie it manages (discussed in lab 3) and is trying to call the
Basket.API
service to get the customer basket items, but the access token inside the cookie is no longer valid due to our IdP state being discarded after every run, including the cryptographic keys that were used to sign the token.To workaround this for now, use the browser developer tools to clear the cookies for the web site and then refresh the page. This will cause the user to be signed out so that you can sign in again and get a new access token. In a later lab we'll update the web site to handle this situation more gracefully.
-
At this point, you should be able to sign in to the web site, add items to the shopping basket, and view the shopping basket page. The shopping basket icon in the header should display the number of items in the basket, and the shopping basket page should display the items in the basket and allow the user to adjust the quantity of items: