AspNetCore.Docs/aspnetcore/mvc/advanced/app-parts.md

108 lines
6.8 KiB
Markdown
Raw Normal View History

---
title: Application Parts in ASP.NET Core
author: ardalis
description: Learn how to use application parts, which are abstractions over the resources of an app, to discover or avoid loading features from an assembly.
manager: wpickett
2018-01-29 23:21:31 +08:00
ms.author: riande
2017-09-20 02:44:04 +08:00
ms.date: 01/04/2017
ms.prod: asp.net-core
2018-01-29 23:21:31 +08:00
ms.technology: aspnet
ms.topic: article
uid: mvc/extensibility/app-parts
---
# Application Parts in ASP.NET Core
2017-01-31 15:14:17 +08:00
[View or download sample code](https://github.com/aspnet/Docs/tree/master/aspnetcore/mvc/advanced/app-parts/sample) ([how to download](xref:tutorials/index#how-to-download-a-sample))
2017-01-31 15:14:17 +08:00
An *Application Part* is an abstraction over the resources of an application, from which MVC features like controllers, view components, or tag helpers may be discovered. One example of an application part is an AssemblyPart, which encapsulates an assembly reference and exposes types and compilation references. *Feature providers* work with application parts to populate the features of an ASP.NET Core MVC app. The main use case for application parts is to allow you to configure your app to discover (or avoid loading) MVC features from an assembly.
2017-01-31 15:14:17 +08:00
## Introducing Application Parts
MVC apps load their features from [application parts](/aspnet/core/api/microsoft.aspnetcore.mvc.applicationparts.applicationpart). In particular, the [AssemblyPart](/aspnet/core/api/microsoft.aspnetcore.mvc.applicationparts.assemblypart#Microsoft_AspNetCore_Mvc_ApplicationParts_AssemblyPart) class represents an application part that's backed by an assembly. You can use these classes to discover and load MVC features, such as controllers, view components, tag helpers, and razor compilation sources. The [ApplicationPartManager](/aspnet/core/api/microsoft.aspnetcore.mvc.applicationparts.applicationpartmanager) is responsible for tracking the application parts and feature providers available to the MVC app. You can interact with the `ApplicationPartManager` in `Startup` when you configure MVC:
2017-01-31 15:14:17 +08:00
```csharp
// create an assembly part from a class's assembly
var assembly = typeof(Startup).GetTypeInfo().Assembly;
services.AddMvc()
.AddApplicationPart(assembly);
2017-01-31 15:14:17 +08:00
// OR
var assembly = typeof(Startup).GetTypeInfo().Assembly;
var part = new AssemblyPart(assembly);
services.AddMvc()
2018-03-29 21:30:35 +08:00
.ConfigureApplicationPartManager(apm => apm.ApplicationParts.Add(part));
2017-01-31 15:14:17 +08:00
```
By default MVC will search the dependency tree and find controllers (even in other assemblies). To load an arbitrary assembly (for instance, from a plugin that isn't referenced at compile time), you can use an application part.
You can use application parts to *avoid* looking for controllers in a particular assembly or location. You can control which parts (or assemblies) are available to the app by modifying the `ApplicationParts` collection of the `ApplicationPartManager`. The order of the entries in the `ApplicationParts` collection isn't important. It's important to fully configure the `ApplicationPartManager` before using it to configure services in the container. For example, you should fully configure the `ApplicationPartManager` before invoking `AddControllersAsServices`. Failing to do so, will mean that controllers in application parts added after that method call won't be affected (won't get registered as services) which might result in incorrect bevavior of your application.
2017-01-31 15:14:17 +08:00
If you have an assembly that contains controllers you don't want to be used, remove it from the `ApplicationPartManager`:
2017-01-31 15:14:17 +08:00
```csharp
services.AddMvc()
2018-03-29 21:30:35 +08:00
.ConfigureApplicationPartManager(apm =>
2017-01-31 15:14:17 +08:00
{
2018-03-29 21:30:35 +08:00
var dependentLibrary = apm.ApplicationParts
2017-01-31 15:14:17 +08:00
.FirstOrDefault(part => part.Name == "DependentLibrary");
2017-01-31 15:14:17 +08:00
if (dependentLibrary != null)
{
p.ApplicationParts.Remove(dependentLibrary);
}
})
```
In addition to your project's assembly and its dependent assemblies, the `ApplicationPartManager` will include parts for `Microsoft.AspNetCore.Mvc.TagHelpers` and `Microsoft.AspNetCore.Mvc.Razor` by default.
## Application Feature Providers
Application Feature Providers examine application parts and provide features for those parts. There are built-in feature providers for the following MVC features:
* [Controllers](https://docs.microsoft.com/aspnet/core/api/microsoft.aspnetcore.mvc.controllers.controllerfeatureprovider)
* [Metadata Reference](https://docs.microsoft.com/aspnet/core/api/microsoft.aspnetcore.mvc.razor.compilation.metadatareferencefeatureprovider)
* [Tag Helpers](https://docs.microsoft.com/aspnet/core/api/microsoft.aspnetcore.mvc.razor.taghelpers.taghelperfeatureprovider)
* [View Components](https://docs.microsoft.com/aspnet/core/api/microsoft.aspnetcore.mvc.viewcomponents.viewcomponentfeatureprovider)
2017-01-31 15:14:17 +08:00
Feature providers inherit from `IApplicationFeatureProvider<T>`, where `T` is the type of the feature. You can implement your own feature providers for any of MVC's feature types listed above. The order of feature providers in the `ApplicationPartManager.FeatureProviders` collection can be important, since later providers can react to actions taken by previous providers.
### Sample: Generic controller feature
2017-01-31 15:14:17 +08:00
By default, ASP.NET Core MVC ignores generic controllers (for example, `SomeController<T>`). This sample uses a controller feature provider that runs after the default provider and adds generic controller instances for a specified list of types (defined in `EntityTypes.Types`):
[!code-csharp[](./app-parts/sample/AppPartsSample/GenericControllerFeatureProvider.cs?highlight=13&range=18-36)]
2017-01-31 15:14:17 +08:00
The entity types:
[!code-csharp[](./app-parts/sample/AppPartsSample/Model/EntityTypes.cs?range=6-16)]
2017-01-31 15:14:17 +08:00
The feature provider is added in `Startup`:
```csharp
services.AddMvc()
2018-03-29 21:30:35 +08:00
.ConfigureApplicationPartManager(apm =>
apm.FeatureProviders.Add(new GenericControllerFeatureProvider()));
2017-01-31 15:14:17 +08:00
```
By default, the generic controller names used for routing would be of the form *GenericController`1[Widget]* instead of *Widget*. The following attribute is used to modify the name to correspond to the generic type used by the controller:
[!code-csharp[](./app-parts/sample/AppPartsSample/GenericControllerNameConvention.cs)]
2017-01-31 15:14:17 +08:00
The `GenericController` class:
[!code-csharp[](./app-parts/sample/AppPartsSample/GenericController.cs?highlight=5-6)]
2017-01-31 15:14:17 +08:00
The result, when a matching route is requested:
![Example output from the sample app reads, 'Hello from a generic Sproket controller.'](app-parts/_static/generic-controller.png)
2017-01-31 15:14:17 +08:00
### Sample: Display available features
2017-01-31 15:14:17 +08:00
You can iterate through the populated features available to your app by requesting an `ApplicationPartManager` through [dependency injection](../../fundamentals/dependency-injection.md) and using it to populate instances of the appropriate features:
[!code-csharp[](./app-parts/sample/AppPartsSample/Controllers/FeaturesController.cs?highlight=16,25-27)]
2017-01-31 15:14:17 +08:00
Example output:
![Example output from the sample app](app-parts/_static/available-features.png)