Errata (34 items)
If you find any mistakes, then please raise an issue in this repository or email me at markjprice (at) gmail.com.
- Page 15 - Central Package Management
- Page 23 - Installing Docker and the Azure SQL Edge container image
- Page 33 - Creating a class library for entity models
- Page 67 - What does UseMigrationsEndPoint do?
- Page 81 - Implementing views
- Page 83 - How cache busting with Tag Helpers works
- Page 97 - Temporarily storing data
- Page 117 - Displaying Northwind suppliers
- Page 118 - Inserting, updating, and deleting suppliers
- Page 129 - Querying a database and using display templates
- Page 143 - Comparing HTML Helpers and Tag Helpers
- Page 154 - Exploring Forms-related Tag Helpers
- Page 158 - Localizing your user interface
- Page 159 - If you are using Visual Studio
- Page 160 - If you are using Visual Studio
- Page 161 - If you are using VS Code
- Page 162 - If you are using VS Code
- Page 163 - Other resource file tools
- Page 167 - Understanding the Accept-Language header
- Page 208 - Exploring in-memory object caching
- Page 211 - Caching objects using distributed caching
- Page 267 - When you cannot use constructor injection
- Page 295 - Configuration validation
- Page 307 - Adding Aspire to an existing solution
- Page 308 - Adding Aspire to an existing solution
- Page 337 - Configuring the customer repository and Web API controller
- Page 346 - Understanding the OpenAPI Specification
- Page 350 - Caching HTTP responses for web services
- Page 366 - Configuring HTTP logging for the web service
- Page 367 - Configuring HTTP logging for the web service
- Page 399 - Creating an HTTP file for making requests
- Page 411 - Calling services in the Northwind MVC website
- Page 460 - Mocking with NSubstitute example
- Page 485 - Creating and initializing a new Umbraco project
Thanks to Rob for raising this issue on December 26, 2024.
Packt publishes Early Access editions of some of their books. They are available on the website via Packt subscription. The content uses preliminary drafts of books that will change during the writing and editing process. Once the final version is signed off by author and publisher, the paperback, Kindle, and other ebook formats are produced. The final step is to update the Early Access online web edition to match the published edition but sometimes this process takes up to a week after publish date.
In the Early Access edition of this book, Chapter 1 included the following package versions for testing in the Directory.Packages.props
file:
<ItemGroup Label="For unit testing.">
<PackageVersion Include="coverlet.collector" Version="6.0.2" />
<PackageVersion Include="Microsoft.NET.Test.Sdk" Version="18.0.0" />
<PackageVersion Include="xunit" Version="2.9.0" />
<PackageVersion Include="xunit.runner.visualstudio" Version="3.0.0" />
</ItemGroup>
But in the final published editions they are as shown in the following markup:
<ItemGroup Label="For testing.">
<PackageVersion Include="coverlet.collector" Version="6.0.2" />
<PackageVersion Include="Microsoft.NET.Test.Sdk" Version="17.11.1" />
<PackageVersion Include="xunit" Version="2.9.2" />
<!--The following package was still a preview on .NET 9 release day.-->
<PackageVersion Include="xunit.runner.visualstudio" Version="3.0.0-pre.49" />
<PackageVersion Include="Microsoft.Playwright" Version="1.49.0" />
<PackageVersion Include="Microsoft.AspNetCore.Mvc.Testing" Version="9.0.0" />
</ItemGroup>
And remember to check if a newer package version has been released. For example, the Microsoft.NET.Test.Sdk
package is already now 17.12.0
, so I updated the solution code in the book's GitHub repository.
Thanks to P9avel for raising this issue on January 2, 2025.
In the paragraph that leads in to the numbered steps, I wrote, "Azure SQL Edge requires a 64-bit processor (either x64 or ARM64), with a minimum of one processor and 1 GB RAM on the host:"
In the next edition, I will simplify that statement to the following, "Azure SQL Edge requires a 64-bit processor (either x64 or ARM64), with a minimum of 1 GB RAM on the host:"
Thanks to P9avel for raising this issue on January 2, 2025.
In the Good Practice box, I use the phrase "data context". In other places, I use the phrase "database context". Both uses refer to a class that derives from DbContext
that represents a combination of the Unit Of Work and Repository patterns such that it can be used to query from a database and group together changes that will then be written back to the store as a unit.
The Microsoft official documentation just uses the word "context" when referring to this class but I feel that loses some meaning. In future editions, I will try to consistently use the phrase "database context".
Thanks to P9avel for raising this issue on January 3, 2025.
This section is about trying to find out what path is configured as an endpoint when you call the UseMigrationsEndPoint
extension method. We already know what calling the UseMigrationsEndPoint
extension method does conceptually: it configures a route that if called, triggers a database migration. What we don't know is what the relative path is for that route.
In the fourth paragraph, I wrote, "You can find the source code for the UseMigrationsEndPoint
extension method at the following link: https://github.com/dotnet/aspnetcore/blob/main/src/Middleware/Diagnostics.EntityFrameworkCore/src/MigrationsEndPointOptions.cs#L18."
But we do not care about the source code for the UseMigrationsEndPoint
extension method itself. We just want to know the relative path that it uses for its endpoint.
So the link is correct, because it shows us what the path is defined by a property on the MigrationsEndPointOptions
class named DefaultPath
. But the description is misleading. In the next edition I will write, "You can find the source code for the relative path used by the UseMigrationsEndPoint
extension method at the following link: https://github.com/dotnet/aspnetcore/blob/main/src/Middleware/Diagnostics.EntityFrameworkCore/src/MigrationsEndPointOptions.cs#L18."
Thanks to P9avel for raising this issue on January 3, 2025.
At the end of Step 9, I wrote, "If you want to navigate to a feature in a Razor class library, like the employees component that you created in the previous chapter, then you use asp-area
to specify the feature name."
I originally planned to have sections about view components and Razor class libraries but they were postponed to the next edition. If I do add those sections and have a task that creates an employees component, then I will be able to leave this sentence unchanged. Otherwise, I will change it to, "If you want to navigate to a feature in a Razor class library then you use asp-area
to specify the feature name."
Thanks to P9avel for raising this issue on January 3, 2025.
The path in the <script>
element has an extra space, as shown in the following markup:
<script src="~/js/site.js? v=Kl_dqr9NVtnMdsM2MUg4qthUnWZm5T1fCEimBPWDNgM"></
script>
In the next edition, I will delete the extra space, as shown in the following markup:
<script src="~/js/site.js?v=Kl_dqr9NVtnMdsM2MUg4qthUnWZm5T1fCEimBPWDNgM"></
script>
Thanks to P9avel for raising this issue on January 3, 2025.
In Step 2, I wrote, "(You will learn more about middleware later in this chapter.)" But during final drafts of the book, I moved this section from Chapter 8 to Chapter 2, so this note is no longer true. In the next edition, I will delete the note, and I will add a new section briefly introducing middleware and other terminology to Chapter 1.
Thanks to P9avel for raising this issue on January 4, 2025.
In Step 5, I show the reader how to render the links to edit and delete a supplier using simple anchor tags (and with the wrong path!), as shown in the following markup:
<a href="edit/@s.SupplierId">Edit</a>
<a href="delete/@s.SupplierId">Delete</a>
But the links should be rendered using the Anchor Tag Helper to make it clearer, as shown in the following markup:
<a asp-controller="Home" asp-action="EditSupplier"
asp-route-id="@s.SupplierId">Edit</a>
<a asp-controller="Home" asp-action="DeleteSupplier"
asp-route-id="@s.SupplierId">Delete</a>
I also forgot to add the markup for the Add New Supplier button that should appear between the Suppliers heading and the table of suppliers, as shown in the following markup:
<h1 class="display-2">Suppliers</h1>
<a class="btn btn-outline-primary"
asp-controller="Home" asp-action="AddSupplier">Add New Supplier</a>
<table class="table">
The markup was already correct in the code solution found here: https://github.com/markjprice/web-dev-net9/blob/main/code/MatureWeb/Northwind.Mvc/Views/Home/Suppliers.cshtml
Thanks to P9avel for raising this issue on January 4, 2025.
In Step 5, the first bullet says, "The <form>
element with a PUT
method is ordinary HTML, so an <input type="submit" />
element inside it will make an HTTP PUT
request back to the current path with values of any other elements inside that form."
But HTML forms inherently support only two HTTP methods: GET
and POST
. They do not natively support PUT
, DELETE
, or other HTTP methods. This limitation means that when you need to use PUT
or DELETE
, you cannot specify these methods directly in the <form>
element.
A convention in ASP.NET Core MVC is to simulate these methods using POST
, often with the help of a hidden input field called _method
or specifying the action in the path, like /home/editsupplier
and /home/deletesupplier
.
In the next edition, I will change PUT
to POST
in that sentence, and I will add a note at the start of this section to explain why we do not use PUT
or DELETE
methods, despite in a RESTful API design:
POST
is typically used for creating resources.PUT
is used for updating resources.DELETE
is used for deleting resources.
Thanks to Paul Marangoni for raising this issue on March 3, 2025.
In Step 3, I tell the reader to modify the contents of CostlyProducts.cshtml
. But two of the column values for category name and supplier company name do not output because of missing @
before Html
. The correct statements should be as shown in the following markup:
@foreach (Product p in Model)
{
<tr>
<td>
@if (p.Category is not null) { @Html.DisplayFor(modelItem => p.Category.CategoryName); }
</td>
<td>
@if (p.Supplier is not null) { @Html.DisplayFor(modelItem => p.Supplier.CompanyName); }
</td>
Or you can use block style, as I use in the code solution for this file, as shown in the following markup:
@foreach (Product p in Model)
{
<tr>
<td>
@if (p.Category is not null)
{
@Html.DisplayFor(modelItem => p.Category.CategoryName);
}
</td>
<td>
@if (p.Supplier is not null)
{
@Html.DisplayFor(modelItem => p.Supplier.CompanyName);
}
</td>
Thanks to P9avel for raising this issue on January 5, 2025.
I showed two examples of rendering a hyperlink:
@Html.ActionLink("View our privacy policy.", "Privacy", "Index")
@Html.ActionLink(linkText: "View our privacy policy.",
action: "Privacy", controller: "Index")
But the controller name is Home
, not Index
, so the markup should be:
@Html.ActionLink("View our privacy policy.", "Privacy", "Home")
@Html.ActionLink(linkText: "View our privacy policy.",
action: "Privacy", controller: "Home")
Thanks to a reader who emailed a question about this issue to Packt.
In Step 1, I told the reader to enter markup for two forms. The second form uses the Label Tag Helper, but the <label>
elements I used were self-closing, as shown in the following markup:
<label asp-for="ShipperId" class="form-label" />
Self-closing tags are not supported by the Label Tag Helper. You must use a pair of open and close tags, as shown in the following markup:
<label asp-for="ShipperId" class="form-label"></label>
The complete corrected second form is shown in the following markup:
<h2>With Form Tag Helper</h2>
<form asp-controller="Home" asp-action="ProcessShipper"
class="form-horizontal" role="form">
<div>
<div class="mb-3">
<label asp-for="ShipperId" class="form-label"></label>
<input asp-for="ShipperId" class="form-control">
</div>
<div class="mb-3">
<label asp-for="CompanyName" class="form-label"></label>
<input asp-for="CompanyName" class="form-control">
</div>
<div class="mb-3">
<label asp-for="Phone" class="form-label"></label>
<input asp-for="Phone" class="form-control">
</div>
<div class="mb-3">
<input type="submit" class="form-control">
</div>
</div>
</form>
Also, the Label Tag Helper will use the property names from the model as the labels in the form, so it will use ShipperId and CompanyName by default.
To override these values, in the Northwind.EntityModels
project, in the Shipper.cs
class, you can decorate the properties with the [Display]
attribute, as shown in the following code:
using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;
using System.ComponentModel.DataAnnotations.Schema;
using Microsoft.EntityFrameworkCore;
namespace Northwind.EntityModels;
public partial class Shipper
{
[Key]
[Display(Name = "Shipper ID")] // Used by the Label Tag Helper.
public int ShipperId { get; set; }
[StringLength(40)]
[Display(Name = "Company Name")] // Used by the Label Tag Helper.
public string CompanyName { get; set; } = null!;
[StringLength(24)]
public string? Phone { get; set; }
[InverseProperty("ShipViaNavigation")]
public virtual ICollection<Order> Orders { get; set; } = new List<Order>();
}
Thanks to P9avel for raising this issue on January 5, 2025.
I wrote, "In the coding task, you will create a console app with an embedded invariant culture and satellite assemblies for Danish, French, French-Canadian, Polish, and Iranian (Persian)."
I should have written, "In the coding task, you will add resources to the ASP.NET Core MVC project that compile to an embedded invariant culture (English) and satellite assemblies for French (Neutral), French (France), and English (British)."
Thanks to P9avel for raising this issue on January 5, 2025.
Figure 4.6 to Figure 4.9 all show the Orders.resx
file in a folder structure: Resources\Views\Orders
but they should be Resources\Views\Home
.
In the next edition, I will retake or edit the screenshots to show the correct folder name Home
.
Also, in Step 1, the path says Resource\Views\Home
when it should be Resources\Views\Home
.
Thanks to Paul Marangoni for raising this issue on March 5, 2025.
Throughout this section, I repeatedly write Index
when I should have written Orders
. For example, in Step 5, I wrote, "Copy and paste the file and rename it Index.en-GB.resx
." I should have written, "Copy and paste the file and rename it Orders.en-GB.resx
."
Similarly in Steps 6, 8, 9, 10, and 11.
Thanks to P9avel for raising this issue on January 5, 2025.
In Step 1, I wrote, "In Resources\Views\Orders
, add a new file named Index.resx
." This should be, "In Resources\Views\Home
, add a new file named Orders.resx
."
Similarly in Steps 3, 4, 5, 6, 7, and 8.
Thanks to P9avel for raising this issue on January 5, 2025.
In Step 6, I wrote, "modify the value
column", when I should have written, "modify the value
element".
Thanks to P9avel for raising this issue on January 5, 2025.
I wrote, "...without needing to recompile the original console app." I should have written, "...without needing to recompile the original project."
Thanks to Paul Marangoni for raising this issue on March 5, 2025.
In the third bullet point, I wrote, "fr-FR;q=0.8
: English language in France ranked at 0.8"
I should have written, "fr-FR;q=0.8
: French language in France ranked at 0.8"
Thanks to P9avel for raising this issue on January 6, 2025.
In Step 5, I wrote, "add statements to try to get the product from the cache, and if it is not cached, get it from the database and set it in the cache, using a sliding expiration of one hour". The sliding expiration should be five seconds, to match the code: SlidingExpiration = TimeSpan.FromSeconds(5)
.
In Step 9, I wrote, "Click Reload this page within 30 seconds", but this should be "5 seconds".
In Step 10, I wrote, "Wait at least 30 seconds", but this should be "5 seconds".
In Step 11, I wrote, "within the 30-second sliding expiration window", but this should be "5-second".
In the next edition, I might change the duration to ten seconds to give readers more time to review the console output and click reload.
Thanks to Paul Marangoni for raising this issue on March 11, 2025.
In Step 3, the code shows a constructor named ProductsController
with a logger
parameter of type ILogger<ProductsController>
, as shown in the following code:
public ProductsController(ILogger<ProductsController> logger,
It should be a constructor named HomeController
with a logger
parameter of type ILogger<HomeController>
, as shown in the following code:
public HomeController(ILogger<HomeController> logger,
Thanks to P9avel for raising this issue on January 7, 2025.
For the Background services bullet, I ended with the sentence, "The solution is to use method injection by resolving services within the ExecuteAsync
method."
I should have written, "The solution is to create a scope with the IServiceScopeFactory.CreateScope()
API, as described in the following link: https://learn.microsoft.com/en-us/dotnet/core/extensions/scoped-service."
Thanks to P9avel for raising this issue on January 9, 2025.
In this section, I wrote, "ASP.NET Core allows you to validate configuration options. This can be done by implementing the IValidateOptions<T>
interface or using the Validate
extension method, as shown in the following code:
builder.Services
.Configure<NorthwindOptions>(builder.Configuration.GetSection("Northwind"))
.Validate(settings => settings.PagerSize > 0,
"PagerSize must be greater than zero.");
But to use the Validate
extension method, you must call the AddOptions<T>
method, as shown in the following code:
builder.Services
.AddOptions<NorthwindOptions>("Northwind")
.Validate(settings => settings.PagerSize > 0,
"PagerSize must be greater than zero.");
Thanks to P9avel for raising this issue on January 9, 2025.
In Step 11, I wrote that I had highlighted the code but the highlight is missing. In the next edition, I will highlight the following code statement:
IResourceBuilder<ContainerResource> sqlServer = builder
.AddContainer(name: "azuresqledge",
image: "mcr.microsoft.com/azure-sql-edge")
.WithLifetime(ContainerLifetime.Persistent);
And the following additional method call after adding the project: .WaitFor(sqlServer)
.
Thanks to P9avel for raising this issue on January 9, 2025.
In Step 14, the output shown is from a different project solution. You can tell because: (1) the version of Aspire is 8.0, (2) the host directory is the Microsoft eShop sample web project, and (3) the port number is different.
In the next edition, I will replace this output with the correct output for the MatureWeb
solution.
Thanks to P9avel for raising this issue on January 11, 2025.
In Step 9, the RetrieveAsync
method should have the default
value passed to it, but instead I accidently put that in the GetCustomer
method declaration, as shown in the following code:
public async Task<IActionResult> GetCustomer(string id, default)
{
Customer? c = await _repo.RetrieveAsync(id);
It should be as shown in the following code:
public async Task<IActionResult> GetCustomer(string id)
{
Customer? c = await _repo.RetrieveAsync(id, default);
The code was already correct in the GitHub respository: https://github.com/markjprice/web-dev-net9/blob/main/code/MatureWeb/Northwind.WebApi/Controllers/CustomersController.cs#L47
Thanks to Paul Marangoni for raising this issue on March 31, 2025.
In Step 4, I wrote "using Swagger" when I should have written "mapping an endpoint for an auto-generated OpenAPI document".
Thanks to Paul Marangoni for raising this issue on March 31, 2025.
In Table 9.5, in the no-cache
row, I wrote "A server is telling the client and intermediaries not the cache the response." I should have written "A server is telling the client and intermediaries not to cache the response."
Thanks to Paul Marangoni for raising this issue on April 3, 2025.
In Step 5, I wrote, "In the Views/Tools
folder" when I should have written, "In the Views/Cors
folder".
Thanks to Paul Marangoni for raising this issue on April 3, 2025.
In Step 6, the code for the getCustomersButton_click
event handler function looks in the input box named companyName
, as shown in the following code:
function getCustomersButton_click() {
xhr.open("GET", baseaddress + "api/customers/?country=" +
document.getElementById("companyName").value);
xhr.send();
}
It should look in the input box named country
, as shown in the following code:
function getCustomersButton_click() {
xhr.open("GET", baseaddress + "api/customers/?country=" +
document.getElementById("country").value);
xhr.send();
}
Thanks to P9avel for raising this issue on January 11, 2025.
In Step 4, in Table 10.2, the last row has a column value of products(2)
for the Relative request. It should be products(77)
because the Response column value is product 77.
Thanks to P9avel for raising this issue on January 11, 2025.
In Step 1, I wrote, "add a navigation menu item to go to a CORS controller with a JavaScript action method".
I should have written, "add a navigation menu item to go to a ODataClient
controller with an Index
action method".
Thanks to P9avel for raising this issue on January 13, 2025.
In Step 11, I wrote, "In the BusinessLogicUnitTests
project, add a package reference for NSubstitute
, as shown in the following markup:
<PackageReference Include="NSubstitute" Version="5.1.0" />
But we are using CPM, so you should not specify the version number, as shown in the following markup:
<PackageReference Include="NSubstitute" />
Thanks to P9avel for raising this issue on January 14, 2025.
In the note box at the end of this section, I wrote, "The relative path to the Umbraco backoffice for your website is \umbraco
. So, the absolute link to the website will be https:\\localhost:5131\
and the absolute link to the website will be https:\\localhost:5131\umbraco
."
I mistakenly repeated, "the absolute link to the website", so I should have written, "The relative path to the Umbraco backoffice for your website is \umbraco
. So, the absolute link to the website will be https:\\localhost:5131\
and the absolute link to the Umbraco backoffice will be https:\\localhost:5131\umbraco
."
And I had previously written a note box before Step 17, "The Umbraco backoffice is accessed through the /umbraco
relative path, so for our
project, it is at the following link: https://localhost:5131/umbraco."
In the next edition, I will remove one of these note boxes. I will also add a heading between steps 7 and 8 to break up this long task a bit more.