From 438f8292cd36cf8e2d142b390fbc554feb50272c Mon Sep 17 00:00:00 2001 From: Rick Anderson Date: Tue, 16 Jan 2018 11:17:15 -1000 Subject: [PATCH 1/3] CSAT HPV config doc (#5169) * CSAT HPV config doc * work * work * acrylynx --- .../fundamentals/configuration/index.md | 58 ++++++++++--------- .../index/sample/CommandLine/Program.cs | 2 +- .../index/sample/ConfigJson/Program.cs | 6 +- .../index/sample/InMemory/MyWindow.cs | 7 --- .../index/sample/InMemory/Program.cs | 14 ++++- .../index/sample/InMemoryGetValue/MyWindow.cs | 7 --- .../index/sample/InMemoryGetValue/Program.cs | 12 +++- .../sample_snapshot/CommandLine/Program.cs | 2 +- 8 files changed, 62 insertions(+), 46 deletions(-) delete mode 100644 aspnetcore/fundamentals/configuration/index/sample/InMemory/MyWindow.cs delete mode 100644 aspnetcore/fundamentals/configuration/index/sample/InMemoryGetValue/MyWindow.cs diff --git a/aspnetcore/fundamentals/configuration/index.md b/aspnetcore/fundamentals/configuration/index.md index e7b3ec3a6bb5..5446e6199aa5 100644 --- a/aspnetcore/fundamentals/configuration/index.md +++ b/aspnetcore/fundamentals/configuration/index.md @@ -5,7 +5,7 @@ description: Use the Configuration API to configure an ASP.NET Core app by multi manager: wpickett ms.author: riande ms.custom: mvc -ms.date: 11/01/2017 +ms.date: 1/11/2018 ms.topic: article ms.technology: aspnet ms.prod: asp.net-core @@ -15,7 +15,7 @@ uid: fundamentals/configuration/index By [Rick Anderson](https://twitter.com/RickAndMSFT), [Mark Michaelis](http://intellitect.com/author/mark-michaelis/), [Steve Smith](https://ardalis.com/), [Daniel Roth](https://github.com/danroth27), and [Luke Latham](https://github.com/guardrex) -The Configuration API provides a way to configure an ASP.NET Core web app based on a list of name-value pairs. Configuration is read at runtime from multiple sources. You can group these name-value pairs into a multi-level hierarchy. +The Configuration API provides a way to configure an ASP.NET Core web app based on a list of name-value pairs. Configuration is read at runtime from multiple sources. You can group these name-value pairs into a multi-level hierarchy. There are configuration providers for: @@ -45,10 +45,7 @@ The app reads and displays the following configuration settings: Configuration consists of a hierarchical list of name-value pairs in which the nodes are separated by a colon. To retrieve a value, access the `Configuration` indexer with the corresponding item's key: -```csharp -Console.WriteLine($"option1 = {Configuration["subsection:suboption1"]}"); -// Output: option1 = subvalue1_from_json -``` +[!code-csharp[Main](index/sample/ConfigJson/Program.cs?range=24-24)] To work with arrays in JSON-formatted configuration sources, use an array index as part of the colon-separated string. The following example gets the name of the first item in the preceding `wizards` array: @@ -57,10 +54,11 @@ Console.Write($"{Configuration["wizards:0:Name"]}"); // Output: Gandalf ``` -Name-value pairs written to the built-in `Configuration` providers are **not** persisted. However, you can create a custom provider that saves values. See [custom configuration provider](xref:fundamentals/configuration/index#custom-config-providers). +Name-value pairs written to the built-in [Configuration](https://docs.microsoft.com/ dotnet/api/microsoft.extensions.configuration) providers are **not** persisted. However, you can create a custom provider that saves values. See [custom configuration provider](xref:fundamentals/configuration/index#custom-config-providers). The preceding sample uses the configuration indexer to read values. To access configuration outside of `Startup`, use the *options pattern*. For more information, see the [Options](xref:fundamentals/configuration/options) topic. + ## Configuration by environment It's typical to have different configuration settings for different environments, for example, development, testing, and production. The `CreateDefaultBuilder` extension method in an ASP.NET Core 2.x app (or using `AddJsonFile` and `AddEnvironmentVariables` directly in an ASP.NET Core 1.x app) adds configuration providers for reading JSON files and system configuration sources: @@ -69,7 +67,9 @@ It's typical to have different configuration settings for different environments * *appsettings.\.json* * Environment variables -See [AddJsonFile](/dotnet/api/microsoft.extensions.configuration.jsonconfigurationextensions) for an explanation of the parameters. `reloadOnChange` is only supported in ASP.NET Core 1.1 and later. +ASP.NET Core 1.x apps need to call `AddJsonFile` and [AddEnvironmentVariables](https://docs.microsoft.com/ dotnet/api/microsoft.extensions.configuration.environmentvariablesextensions.addenvironmentvariables #Microsoft_Extensions_Configuration_EnvironmentVariablesExtensions_AddEnvironmentVariables_Microsoft_Extensions_Configuration_IConfigurationBuilder_System_String_). + +See [AddJsonFile](/dotnet/api/microsoft.extensions.configuration.jsonconfigurationextensions) for an explanation of the parameters. `reloadOnChange` is only supported in ASP.NET Core 1.1 and later. Configuration sources are read in the order that they're specified. In the preceding code, the environment variables are read last. Any configuration values set through the environment replace those set in the two previous providers. @@ -81,14 +81,14 @@ When the environment is set to `Staging`, the following `Configure` method reads [!code-csharp[Main](index/sample/StartupConfig.cs?name=snippet&highlight=3,4)] -The environment is typically set to `Development`, `Staging`, or `Production`. See [Working with multiple environments](xref:fundamentals/environments) for more information. + +The environment is typically set to `Development`, `Staging`, or `Production`. For more information, see [Working with multiple environments](xref:fundamentals/environments). Configuration considerations: -* `IOptionsSnapshot` can reload configuration data when it changes. See [IOptionsSnapshot](xref:fundamentals/configuration/options#reload-configuration-data-with-ioptionssnapshot) for more information. -* Configuration keys are case insensitive. -* Specify environment variables last so that the local environment can override settings in deployed configuration files. -* **Never** store passwords or other sensitive data in configuration provider code or in plain text configuration files. Don't use production secrets in your development or test environments. Instead, specify secrets outside of the project so that they can't be accidentally committed to your repository. Learn more about [working with multiple environments](xref:fundamentals/environments) and managing [safe storage of app secrets during development](xref:security/app-secrets). +* `IOptionsSnapshot` can reload configuration data when it changes. For more information, see [IOptionsSnapshot](xref:fundamentals/configuration/options#reload-configuration-data-with-ioptionssnapshot)., +* Configuration keys are **not** case-sensitive. +* **Never** store passwords or other sensitive data in configuration provider code or in plain text configuration files. Don't use production secrets in your development or test environments. Specify secrets outside of the project so that they can't be accidentally committed to your repository. Learn more about [working with multiple environments](xref:fundamentals/environments) and managing [safe storage of app secrets during development](xref:security/app-secrets). * If a colon (`:`) can't be used in environment variables on your system, replace the colon (`:`) with a double-underscore (`__`). ## In-memory provider and binding to a POCO class @@ -103,7 +103,7 @@ Configuration values are returned as strings, but binding enables the constructi The following sample demonstrates the [GetValue<T>](https://docs.microsoft.com/aspnet/core/api/microsoft.extensions.configuration.configurationbinder#Microsoft_Extensions_Configuration_ConfigurationBinder_GetValue_Microsoft_Extensions_Configuration_IConfiguration_System_Type_System_String_System_Object_) extension method: -[!code-csharp[Main](index/sample/InMemoryGetValue/Program.cs?highlight=27-29)] +[!code-csharp[Main](index/sample/InMemoryGetValue/Program.cs?highlight=31)] The ConfigurationBinder's `GetValue` method allows you to specify a default value (80 in the sample). `GetValue` is for simple scenarios and does not bind to entire sections. `GetValue` gets scalar values from `GetSection(key).Value` converted to a specific type. @@ -117,7 +117,7 @@ The following sample binds to the `AppSettings` class: [!code-csharp[Main](index/sample/ObjectGraph/Program.cs?highlight=15-16)] -**ASP.NET Core 1.1** and higher can use `Get`, which works with entire sections. `Get` can be more convenient than using `Bind`. The following code shows how to use `Get` with the sample above: +**ASP.NET Core 1.1** and higher can use `Get`, which works with entire sections. `Get` can be more convenient than using `Bind`. The following code shows how to use `Get` with the preceding sample: ```csharp var appConfig = config.GetSection("App").Get(); @@ -160,7 +160,7 @@ public void CanBindObjectTree() ## Create an Entity Framework custom provider -In this section, a basic configuration provider that reads name-value pairs from a database using EF is created. +In this section, a basic configuration provider that reads name-value pairs from a database using EF is created. Define a `ConfigurationValue` entity for storing configuration values in the database: @@ -170,11 +170,11 @@ Add a `ConfigurationContext` to store and access the configured values: [!code-csharp[Main](index/sample/CustomConfigurationProvider/ConfigurationContext.cs?name=snippet1)] -Create an class that implements [IConfigurationSource](https://docs.microsoft.com/aspnet/core/api/microsoft.extensions.configuration.iconfigurationsource): +Create a class that implements [IConfigurationSource](https://docs.microsoft.com/aspnet/core/api/microsoft.extensions.configuration.iconfigurationsource): [!code-csharp[Main](index/sample/CustomConfigurationProvider/EntityFrameworkConfigurationSource.cs?highlight=7)] -Create the custom configuration provider by inheriting from [ConfigurationProvider](https://docs.microsoft.com/aspnet/core/api/microsoft.extensions.configuration.configurationprovider). The configuration provider initializes the database when it's empty: +Create the custom configuration provider by inheriting from [ConfigurationProvider](https://docs.microsoft.com/aspnet/core/api/microsoft.extensions.configuration.configurationprovider). The configuration provider initializes the database when it's empty: [!code-csharp[Main](index/sample/CustomConfigurationProvider/EntityFrameworkConfigurationProvider.cs?highlight=9,18-31,38-39)] @@ -194,7 +194,7 @@ Using the following *appsettings.json* file: [!code-json[Main](index/sample/CustomConfigurationProvider/appsettings.json)] -The following is displayed: +The following output is displayed: ```console key1=value_from_ef_1 @@ -248,10 +248,15 @@ Typical ASP.NET Core 2.x apps use the static convenience method `CreateDefaultBu `CreateDefaultBuilder` loads optional configuration from *appsettings.json*, *appsettings.{Environment}.json*, [user secrets](xref:security/app-secrets) (in the `Development` environment), environment variables, and command-line arguments. The CommandLine configuration provider is called last. Calling the provider last allows the command-line arguments passed at runtime to override configuration set by the other configuration providers called earlier. -Note that for *appsettings* files that `reloadOnChange` is enabled. Command-line arguments are overridden if a matching configuration value in an *appsettings* file is changed after the app starts. +For *appsettings* files where: + +* `reloadOnChange` is enabled. +* Contain the same setting in the command-line arguments and an *appsettings* file. +* The *appsettings* file containing the matching command-line argument is changed after the app starts. + +If all the preceding conditions are true, the command-line arguments are overridden. -> [!NOTE] -> As an alternative to using the `CreateDefaultBuilder` method, creating a host using [WebHostBuilder](/dotnet/api/microsoft.aspnetcore.hosting.webhostbuilder) and manually building configuration with [ConfigurationBuilder](/api/microsoft.extensions.configuration.configurationbuilder) is supported in ASP.NET Core 2.x. See the ASP.NET Core 1.x tab for more information. +ASP.NET Core 2.x app can use WebHostBuilder](/dotnet/api/microsoft.aspnetcore.hosting.webhostbuilder) instead of ``CreateDefaultBuilder`. When using `WebHostBuilder`, manually set configuration with [ConfigurationBuilder](/api/microsoft.extensions.configuration.configurationbuilder). See the ASP.NET Core 1.x tab for more information. # [ASP.NET Core 1.x](#tab/aspnetcore1x) @@ -263,7 +268,7 @@ Create a [ConfigurationBuilder](/api/microsoft.extensions.configuration.configur ### Arguments -Arguments passed on the command line must conform to one of two formats shown in the following table. +Arguments passed on the command line must conform to one of two formats shown in the following table: | Argument format | Example | | ------------------------------------------------------------------- | :------------: | @@ -360,7 +365,7 @@ MachineName: DahliaPC Left: 1984 ``` -After the switch mappings dictionary is created, it contains the data shown in the following table. +After the switch mappings dictionary is created, it contains the data shown in the following table: | Key | Value | | -------------- | --------------------- | @@ -389,8 +394,9 @@ A *web.config* file is required when hosting the app in IIS or IIS Express. Sett * Dependency Injection (DI) is not set up until after `ConfigureServices` is invoked. * The configuration system is not DI aware. * `IConfiguration` has two specializations: - * `IConfigurationRoot` Used for the root node. Can trigger a reload. - * `IConfigurationSection` Represents a section of configuration values. The `GetSection` and `GetChildren` methods return an `IConfigurationSection`. + * `IConfigurationRoot` Used for the root node. Can trigger a reload. + * `IConfigurationSection` Represents a section of configuration values. The `GetSection` and `GetChildren` methods return an `IConfigurationSection`. + * Use [IConfigurationRoot](https://docs.microsoft.com/ dotnet/api/microsoft.extensions.configuration.iconfigurationroot) when reloading configuration or need access to each provider. Neither of these situations are common. ## Additional resources diff --git a/aspnetcore/fundamentals/configuration/index/sample/CommandLine/Program.cs b/aspnetcore/fundamentals/configuration/index/sample/CommandLine/Program.cs index ae83d35c7921..91e3f0ec8939 100644 --- a/aspnetcore/fundamentals/configuration/index/sample/CommandLine/Program.cs +++ b/aspnetcore/fundamentals/configuration/index/sample/CommandLine/Program.cs @@ -5,7 +5,7 @@ public class Program { - public static IConfigurationRoot Configuration { get; set; } + public static IConfiguration Configuration { get; set; } public static Dictionary GetSwitchMappings( IReadOnlyDictionary configurationStrings) diff --git a/aspnetcore/fundamentals/configuration/index/sample/ConfigJson/Program.cs b/aspnetcore/fundamentals/configuration/index/sample/ConfigJson/Program.cs index 55ff84644671..d97f4fec787f 100644 --- a/aspnetcore/fundamentals/configuration/index/sample/ConfigJson/Program.cs +++ b/aspnetcore/fundamentals/configuration/index/sample/ConfigJson/Program.cs @@ -1,10 +1,12 @@ using System; using System.IO; +// Requires NuGet package +// Microsoft.Extensions.Configuration.Json using Microsoft.Extensions.Configuration; public class Program { - public static IConfigurationRoot Configuration { get; set; } + public static IConfiguration Configuration { get; set; } public static void Main(string[] args = null) { @@ -14,7 +16,7 @@ public static void Main(string[] args = null) Configuration = builder.Build(); - Console.WriteLine($"option1 = {Configuration["option1"]}"); + Console.WriteLine($"option1 = {Configuration["Option1"]}"); Console.WriteLine($"option2 = {Configuration["option2"]}"); Console.WriteLine( $"suboption1 = {Configuration["subsection:suboption1"]}"); diff --git a/aspnetcore/fundamentals/configuration/index/sample/InMemory/MyWindow.cs b/aspnetcore/fundamentals/configuration/index/sample/InMemory/MyWindow.cs deleted file mode 100644 index c09769764a23..000000000000 --- a/aspnetcore/fundamentals/configuration/index/sample/InMemory/MyWindow.cs +++ /dev/null @@ -1,7 +0,0 @@ -public class MyWindow -{ - public int Height { get; set; } - public int Width { get; set; } - public int Top { get; set; } - public int Left { get; set; } -} \ No newline at end of file diff --git a/aspnetcore/fundamentals/configuration/index/sample/InMemory/Program.cs b/aspnetcore/fundamentals/configuration/index/sample/InMemory/Program.cs index 0b03897ccf68..b56a093f4df7 100644 --- a/aspnetcore/fundamentals/configuration/index/sample/InMemory/Program.cs +++ b/aspnetcore/fundamentals/configuration/index/sample/InMemory/Program.cs @@ -1,10 +1,12 @@ using System; using System.Collections.Generic; +// Requires NuGet package +// Microsoft.Extensions.Configuration.Json using Microsoft.Extensions.Configuration; public class Program { - public static IConfigurationRoot Configuration { get; set; } + public static IConfiguration Configuration { get; set; } public static void Main(string[] args = null) { @@ -25,6 +27,8 @@ public static void Main(string[] args = null) Console.WriteLine($"Hello {Configuration["Profile:MachineName"]}"); var window = new MyWindow(); + // Bind requrires NuGet package + // Microsoft.Extensions.Configuration.Binder Configuration.GetSection("App:MainWindow").Bind(window); Console.WriteLine($"Left {window.Left}"); Console.WriteLine(); @@ -32,4 +36,12 @@ public static void Main(string[] args = null) Console.WriteLine("Press any key..."); Console.ReadKey(); } +} + +public class MyWindow +{ + public int Height { get; set; } + public int Width { get; set; } + public int Top { get; set; } + public int Left { get; set; } } \ No newline at end of file diff --git a/aspnetcore/fundamentals/configuration/index/sample/InMemoryGetValue/MyWindow.cs b/aspnetcore/fundamentals/configuration/index/sample/InMemoryGetValue/MyWindow.cs deleted file mode 100644 index c09769764a23..000000000000 --- a/aspnetcore/fundamentals/configuration/index/sample/InMemoryGetValue/MyWindow.cs +++ /dev/null @@ -1,7 +0,0 @@ -public class MyWindow -{ - public int Height { get; set; } - public int Width { get; set; } - public int Top { get; set; } - public int Left { get; set; } -} \ No newline at end of file diff --git a/aspnetcore/fundamentals/configuration/index/sample/InMemoryGetValue/Program.cs b/aspnetcore/fundamentals/configuration/index/sample/InMemoryGetValue/Program.cs index b062acfdeb7d..961f7c2ce273 100644 --- a/aspnetcore/fundamentals/configuration/index/sample/InMemoryGetValue/Program.cs +++ b/aspnetcore/fundamentals/configuration/index/sample/InMemoryGetValue/Program.cs @@ -1,10 +1,12 @@ using System; using System.Collections.Generic; +// Requires NuGet package +// Microsoft.Extensions.Configuration.Json using Microsoft.Extensions.Configuration; public class Program { - public static IConfigurationRoot Configuration { get; set; } + public static IConfiguration Configuration { get; set; } public static void Main(string[] args = null) { @@ -37,4 +39,12 @@ public static void Main(string[] args = null) Console.WriteLine("Press a key..."); Console.ReadKey(); } +} + +public class MyWindow +{ + public int Height { get; set; } + public int Width { get; set; } + public int Top { get; set; } + public int Left { get; set; } } \ No newline at end of file diff --git a/aspnetcore/fundamentals/configuration/index/sample_snapshot/CommandLine/Program.cs b/aspnetcore/fundamentals/configuration/index/sample_snapshot/CommandLine/Program.cs index 6f9b16f084f4..5cc5c7debaf4 100644 --- a/aspnetcore/fundamentals/configuration/index/sample_snapshot/CommandLine/Program.cs +++ b/aspnetcore/fundamentals/configuration/index/sample_snapshot/CommandLine/Program.cs @@ -5,7 +5,7 @@ public class Program { - public static IConfigurationRoot Configuration { get; set; } + public static IConfiguration Configuration { get; set; } public static void Main(string[] args = null) { From 4b91aa433f09297e6404424420a2cb8f841562d0 Mon Sep 17 00:00:00 2001 From: Platon Efimov Date: Wed, 17 Jan 2018 08:17:56 +0700 Subject: [PATCH 2/3] Highlight (#5188) Highlight update --- aspnetcore/tutorials/razor-pages/model.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/aspnetcore/tutorials/razor-pages/model.md b/aspnetcore/tutorials/razor-pages/model.md index 72e257784b53..21cc465221fd 100644 --- a/aspnetcore/tutorials/razor-pages/model.md +++ b/aspnetcore/tutorials/razor-pages/model.md @@ -35,7 +35,7 @@ Add a connection string to the *appsettings.json* file. Register the database context with the [dependency injection](xref:fundamentals/dependency-injection) container in the *Startup.cs* file. -[!code-csharp[Main](../../tutorials/razor-pages/razor-pages-start/sample/RazorPagesMovie/Startup.cs?name=snippet_ConfigureServices&highlight=3-6)] +[!code-csharp[Main](../../tutorials/razor-pages/razor-pages-start/sample/RazorPagesMovie/Startup.cs?name=snippet_ConfigureServices&highlight=3-5,7-9)] Build the project to verify you don't have any errors. From 23f0eaf80fbc979969f84f78b0145ddea714e639 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9bastien=20Putier?= Date: Wed, 17 Jan 2018 05:57:57 +0100 Subject: [PATCH 3/3] Razor Pages - Added explanations about the use of `return Page();` (#5187) * Razor Pages - Added explanations about the use of `return Page();` * Update page.md * Update page.md --- aspnetcore/tutorials/razor-pages/page.md | 24 ++++++++++++++++++++++-- 1 file changed, 22 insertions(+), 2 deletions(-) diff --git a/aspnetcore/tutorials/razor-pages/page.md b/aspnetcore/tutorials/razor-pages/page.md index 010bd5367a4a..048891ce16e4 100644 --- a/aspnetcore/tutorials/razor-pages/page.md +++ b/aspnetcore/tutorials/razor-pages/page.md @@ -21,13 +21,33 @@ This tutorial examines the Razor Pages created by scaffolding in the previous tu ## The Create, Delete, Details, and Edit pages. -Examine the *Pages/Movies/Index.cshtml.cs* code-behind file: +Examine the *Pages/Movies/Index.cshtml.cs* Page Model: [!code-csharp[Main](razor-pages-start/snapshot_sample/RazorPagesMovie/Pages/Movies/Index.cshtml.cs)] Razor Pages are derived from `PageModel`. By convention, the `PageModel`-derived class is called `Model`. The constructor uses [dependency injection](xref:fundamentals/dependency-injection) to add the `MovieContext` to the page. All the scaffolded pages follow this pattern. See [Asynchronous code](xref:data/ef-rp/intro#asynchronous-code) for more information on asynchronous programing with Entity Framework. -When a request is made for the page, the `OnGetAsync` method returns a list of movies to the Razor Page. `OnGetAsync` or `OnGet` is called on a Razor Page to initialize the state for the page. In this case, `OnGetAsync` gets a list of movies to display. +When a request is made for the page, the `OnGetAsync` method returns a list of movies to the Razor Page. `OnGetAsync` or `OnGet` is called on a Razor Page to initialize the state for the page. In this case, `OnGetAsync` gets a list of movies and displays them. +When `OnGet` returns `void` or `OnGetAsync` returns`Task`, no return method is used. When the return type is `IActionResult` or `Task`, a return statement must be provided. For example, the *Pages/Movies/Create.cshtml.cs* `OnPostAsync` method: + + + +```csharp +public async Task OnPostAsync() +{ + if (!ModelState.IsValid) + { + return Page(); + } + + _context.Movie.Add(Movie); + await _context.SaveChangesAsync(); + + return RedirectToPage("./Index"); +} +``` Examine the *Pages/Movies/Index.cshtml* Razor Page: [!code-cshtml[Main](razor-pages-start/snapshot_sample/RazorPagesMovie/Pages/Movies/Index.cshtml)]