This repository contains the code samples for ASP.NET Core in Action, Second Edition
No code samples
- WebApplication1 - A sample web application, based on the Visual Studio Razor Pages template
- ErrorHandler - A sample application demonstrating the built-in error handling. The home page (Pages/Index.cshtml.cs throws an exception when executed.
- CreatingAHoldingPage - 3.2.1 Simple pipeline scenario 1: A holding page
- CreatingAStaticFileWebsite - 3.2.2 Simple pipeline scenario 2: Handling static files
- SimpleRazorPagesApplication - 3.2.3 Simple pipeline scenario 3: A Razor Pages application
- SimpleRazorPagesApplicationAndHoldingPage - 3.2.3 Simple pipeline scenario 3: A Razor Pages web application + a holding page for "/"
- DeveloperExceptionPage - 3.3.1 Viewing exceptions in development: the DeveloperExceptionPage
- ExceptionHandlerMiddleware - 3.3.2 Handling exceptions in production: the ExceptionHandlerMiddleware
- StatusCodePages - 3.3.3 Handling other errors: the StatusCodePagesMiddleware
- StatusCodePagesWithRexecute - 3.3.3 Handling other errors: the StatusCodePagesMiddleware. Reexecuting the pipeline to create custom status code pages
- ATypicalRazorPage - 4.1.1 Exploring a Simple Razor Page
- AddingRazorPagesToEmptyProject - 4.1.4 Adding Razor Pages to an "Empty" web application template
- ConvertingToMvc - 4.2 An MVC application, with separate Model, View, and Controller files
- PageHandlers - 4.3 Using different Razor Page handlers to handle different HTTP verbs (
GET
andPOST
)
- RoutingExamples - Multiple examples of routing. In particular, see Search.cshtml, Products.cshtml, and ProductDetails/Index.cshtml. Index.cshtml includes links demonstrating route parameters
- ChangingConventions - 5.7. Customizing the URLs using conventions. See Startup.cs
- ToDoList - Basic application demonstrating the application in Figure 6.1
- ExampleBinding_EditProduct - Binding a custom class using Model Binding
- ExampleBinding_Calculator - Binding a custom class using Model Binding
- SimpleCurrencyConverterBindings - Model binding simple properties. Demonstrates Table 6.1, binding to route parameters, form parameters and the querystring
- ListBinding - Model binding to collections, as shown in Figure 6.5
- ValidatingWithDataAnnotations - A dummy checkout page, demonstrating Model validation using various
DataAnnotations
. Also shows POST-REDIRECT-GET - CurrencyConverter - A dummy currency converter application. Demonstrates model binding,
DataAnnotations
, and a custom validation attribute - RazorPageFormLayout - Organising a Razor Page for model binding, as in section 6.4
- ManageUsers - Display a list of users, and allow adding new users, as shown in figure 7.3
- DynamicHtml - Example of creating dynamic HTML by using C# in Razor templates.
- ToDoList - Example of writing model values to HTML in Razor templates, as shown in section 7.2
- NestedLayouts - Demonstrates using a nested layout, where a two column layout,
_TwoColumn.cshtml
is nested in_Layout.cshtml
. - PartialViews - Demonstrates extracting common code into a partial view, as in section 7.4.3. Also shows adding additional namespace to _ViewImports for
PartialViews.Models
namespace. - DefaultMVCProject - Default MVC template, showing the default conventions for finding a view. Also shows how you can specify the template to Render - the
HomeController
specifies alternative view to render at the URL/Home/IndexThatRendersPrivacy
- CurrencyConverter - A demo currency converter application using TagHelpers to generate form elements.
- TagHelpers - Demonstrates the input types generated for various property types and
DataAnnotations
, as described table 8.1. - SelectLists - Generating a variety of select lists, as shown in section 8.2.4
- EnvironmentTag - Using the environment tag to conditionally render content, as shown in section 8.5
- BlazorWebAssemblyProject - A basic Blazor WebAssembly web application, as shown in Figure 9.1, created using the Blazor WebAssembly template. Follow the instructions at https://docs.microsoft.com/en-gb/aspnet/core/blazor/get-started to try it for yourself.
- DefaultWebApiProject - The default Web API project, created using the Visual Studio API template, as in section 9.2.
- BasicWebApiProject - A basic Web API project, returning a list of fruit, as demonstrated in section 9.2.
- UsingApiControllerAttribute - A project containing 2 controllers, demonstrating the additional code required if you don't use the
[ApiController]
attribute, as in section 9.5. - ProblemDetailsExample - A simple API controller that demonstrates automatically returning a
ValidationProblemDetails
object when the binding model (themyValue
route parameter) is empty. - CarsWebApi - A Web API controller that demonstrates generating various different response types. Is configured to allow XML output in Startup.cs Use https://www.getpostman.com to make requests to the API. Also configured to use the Newtonsoft.Json formatter instead of the System.Text.Json formatter.
- SendingAnEmailWithoutDI - An example demonstrating a use case where you want to send an email when a user registers. The
EmailSender
class is created in code usingnew
as shown in section 10.1.1 and Listing 10.3. - SendingAnEmailWithDI - A refactoring of the SendingAnEmailWithoutDI project to use DI, showing how the
UserController
has been simplified. - InjectingMultipleImplementations - Example demonstrating the behaviour when registering multiple instances of a service, as in section 10.2.4. Click the buttons shown on the home page and observe the console output to see the effect of the DI configuration
- LifetimeExamples - The effect of lifetime on DI. For details, see section 10.3 - the project broadly follows this outline, with slightly different naming to allow registering all the services in a single project.
- ReplacingDefaultConfigProviders - Demonstrating how you can replace the configuration providers added in
CreateDefaultBuilder
, as shown in section 11.3.1. - StoreViewerApplication - A simple application that uses
IOptions<>
and strongly typed settings to bind configuration to POCOs. Optionally uses Google Maps to demonstrate loading settings from multiple sources. Follow the documentation from Google to obtain an API key. - DesigningForAutomaticBinding - Demonstrating how to create strongly typed settings that can be bound to configuration, and the limitations, as shown in section 11.4.3.
- LifetimeExamples - Demonstrates how to overwrite values based on the environment. In particular, observe how list values are overwritten.
- InstallEFCore - Demonstrating how to install EF Core, as shown in section 12.2. Starts from the Default Web App template from Visual Studio and installs the EF Core packages, adds the
Recipe
andIngredient
entities, configures theAppDbContext
, and registers EF Core with the DI container. Configured by default for Local DB, but demonstrates how to configure the application for SQLite instead. - Migrate_LocalDb - The same code as InstallEfCore, but with migrations added using the
dotnet-ef
global tool. Migrations generated for Local DB. - Migrate_SQLite - The same as Migrate_LocalDb but configured for SQLite, and with SQLite-specific migrations generated by the
dotnet-ef
global tool. - RecipeApplication_LocalDb - The final Razor Page application, using the Local DB database provider.
- RecipeApplication_SQLite - The final Razor Page application, using the SQLite database provider.
- FilterPipelineExample - A sample application with a single API Controller and a single Razor Page that contains one of each filter, and logs when the filter runs. Each filter contains commented out code to short-circuit the pipeline. Uncomment the code from each filter in turn to see the effect.
- RecipeApplication - The RecipeApplication from chapter 12 plus two API controllers. The
NoApiController
includes the code from listing 13.8, while theRecipeApiController
includes the code from listing 13.9 where the code is refactored to use filters.
- DefaultTemplate - The default web app template for ASP.NET Core with Authentication, as discussed in section 14.3.
- RecipeApplication - The recipe application from chapter 13 with authentication added, as described in section 14.4. Also, the register page has been scaffolded to remove the references to external services, as described in section 14.5.
- RecipeApplicationWithNameClaim - The recipe application with an additional field added to the
RegisterModel
to record theFullName
, as described in section 14.6. The field is added as an extra claim when the user registers, and is displayed in the menu bar when a user logs in.
- Airport - An analogy to the airport example presented in section 15.1. There are 4 steps, Home Page, Through security, Enter airport lounge, Board the plane. You can set the claims for a user when you register. Which claims you add will determine how far through the airport you can get.
- RecipeApplication - The recipe application from previous chapters, with authorization to prevent unauthorized users creating recipes, and resoure based authorization to ensure only the user which created a recipe can edit it.
- RecipeApplication - The Recipe application from previous chapters, for hosting in IIS. Changed the LocalDb migrations and connection string to use SqlLite as connecting to a LocalDb instance from IIS can be problematic. Also, added additional JS and CSS files to demonstrate bundling and minifying, and added bundleconfig.json
- RecipeApplication - The recipe application from previous chapters with some additional logging added to some Razor Pages.
- FileLogger - A simple Web API Project configured to write log messages to a rolling file by using a rolling file logging provider, as shown in section 17.3.1. Note that the log levels have been changed from the defaults in appsettings.json and appsettings.Development.json to show more in the logs.
- SerilogLogger - A simple Web API Project configured to write log messages to the console using Serilog, as shown in section 17.3.2.
- LogFiltering - A simple Web API Project configured to use the configuration filters defined in section 17.4.
- SeqLogger - A simple web API project to demonstrate structured logging using Seq, and using scopes to add additional properties to a log, as shown in section 17.5.
- CustomHttpsCertificate - A basic Razor Pages app using Kestrel with a self-signed certificate, similar to the way you would configure a certificate in production. Shows configuring the default certificate used by Kestrel in appsettings.json.
On Windows, you can generate a self-signed certificate using the Install-Certificate.ps1 PowerShell script. This will create a self-signed certificate and add it to Windows' trusted certificate store.
You can generate a certificate on Ubuntu using install_certificate.sh. This uses localhost.conf to create a self signed certificate, and trusts it. On Linux, not all applications use the same store, so you may have to trust it explicitly for those applications. Use password testpassword
to create the certificate.
-
CrossSiteScripting - A simple app to demonstrate XSS attacks due to not encoding user input. The user submits content which is added to an internal list and is later rendered to the page. Using
@Html.Raw
renders the provided input exactly as it was entered - if the content is malicious, e.g. a<script>
tag, then it is written into the page as a script tag, executing any code it contains. Instead, you should render content with the@
symbol alone - that way the content is rendered as a string, and can be displayed safely. -
CrossSiteRequestForgery - A pair of apps to demonstrate a CSRF vulnerability. You can login to the banking application and view your balance. You can 'withdraw' funds using the provided form, and you'll see your balance reduce. The attacker website contains a form that posts to the banking application and withdraws funds for the currently logged in user. In the example you have to click the button to see the vulnerability, but this could easily be automated. To protect the endpoint, add the
[ValidateAntiForgeryToken]
attribute to theBalanceController.Withdraw()
action. Run both applications by selecting "Set Startup Projects" in Visual Studio, or by running both applications usingdotnet run
. -
CorsApplication - A pair of apps to demonstrate CORS. The "shopping.com" site is a Razor Pages application, that loads a product list from a separate app, "api.shopping.com", hosted at a different host. With the default configuration, the request succeeds. Experiment by removing the default CORS policy from the
UseCors()
middleware configuration, and applying[EnableCors]
to theProductsController
instead. Note that it's the API that defines which applications can call it. Run both applications by selecting "Set Startup Projects" in Visual Studio, or by running both applications usingdotnet run
.
- CustomMiddleware - Various custom middleware, using,
Map
,Run
,Use
, and middleware classes, as described in section 19.1. See Startup.cs for a description of how each middleware responds to requests. - CustomEndpoint - The
PingPongMiddleware
,VersionMiddleware
, andCalculatorMiddleware
, from the CustomMiddleware project, exposed as endpoints using endpoint routing, as described in section 19.2. TheCalculatorMiddleware
demonstrates how routing can be used to extract route parameters. The Health Checks endpoint (built-into ASP.NET Core), has authorization applied withRequireAuthorization
, so you must be logged in to access it. This project is a basic RazorPages application with Identity using Sqlite to make it easier to authenticate and test authorization. - CustomConfiguration - Loading configuration in multiple stages, as described in section 19.3.1. The baseconfig.xml file is loaded first, and is used to locate the appsettings.json file, which is added to the configuration.
- ConfigureOptionsExample - Configuring
IOptions
using services as described in section 19.3.2. TheCurrencyOptions
in configured in multiple ways - from configuration values, from static values (using a Lambda), and usingConfigureCurrencyOptions
which uses a service registered in the DI container. - LamarExample - Replacing the default DI container with Lamar (the successor to StructureMap), as in section 19.4. Demonstrates some of the functionality available in Lamar.
- CustomTagHelpers - Creating custom Tag Helpers, an
IfTagHelper
andSystemInfoTagHelper
, as shown in section 20.1. - RecipeApplication - The Recipe Application from previous chapters, this time with a custom view component, as described in section 20.2.
- CurrencyConverter - The demo Currency converter application, containing a custom validation attribute for validating the selected currencies, as in section 20.3.
- FluentValidationConverter - The demo Currency converter application, configured to use the FluentValidation library instead of DataAnnotations. Contains validation extension methods for validating the selected currencies, as in section 20.4.
- SocketExhaustion - A simple application that creates many
HttpClient
s, demonstrating sockets being consumed. Runnetstat
in a separate window, to view sockets stuck in theTIME_WAIT
status, as discussed in section 21.1.1. - ExchangeRateViewer - An API controller that calls a remote exchange rate API, and returns the value. Shows 4 different ways of using
HttpClient
andIHttpClientFactory
:- Singleton
HttpClient
: A singleHttpClient
that lives for the life of the application, as discussed in section 21.1. This client won't respect DNS changes. - Using
IHttpClientFactory
to create anHttpClient
, as described in section 21.2.1. - Using a named
HttpClient
, as described in section 21.2.2 - Creating a typed
HttpClient
, as described in section 21.2.3 - Adding transient error handling using Polly in
Startup.ConfigureServices()
, as described in section 21.3 - Creating a custom
HttpMessageHandler
for adding an API key, as described in section 21.4
- Singleton
- BackgroundServiceCache - An
IHostedService
that downloads exchange rates from a remote API and saves them in a dictionary, for consumption by an API controller, as described in section 22.1.1 - BackgroundServiceDatabaseCache - An
IHostedService
that uses scoped services, downloads exchange rates from a remote API and saves them in a dictionary, as described in section 22.1.2 - SystemdService - A generic
Host
to download exchange rates, configured to executes as a systemd daemon, as described in section 22.2 - WindowsService - A generic
Host
to download exchange rates, configured to executes as a Windows Service, as described in section 22.2. - QuartzHostedService - A generic
Host
that uses Quartz.NET to run background tasks - QuartzClustering - A generic
Host
that uses Quartz.NET to run background tasks configured to use clustering. Note that SQLite is not supported for clustering, so this application uses LocalDB
- ExchangeRates - A basic exchange rate application. Includes unit tests for the
CurrencyConverter
class (section 23.2), for theStatusMiddleware
(section 23.3), and for API controllers (section 23.4). It also includes "Test Host" integration tests for theStatusMiddleware
(section 23.5.1) as well asWebApplicationFactory
-based integration tests for the whole app (section 23.5.2, 23.5.3, 23.5.4). - RecipeApplication - Testing a service that relies on an EF Core
DbContext
, as described in section 23.6. TheRecipeServiceTests
class shows how you can test theRecipeService
using the in-memory SQLite provider. Also shows a customWebApplicationFactory
implementation that uses an in-memory database.