* Rename Umbraco.Core namespace to Umbraco.Cms.Core * Move extension methods in core project to Umbraco.Extensions * Move extension methods in core project to Umbraco.Extensions * Rename Umbraco.Examine namespace to Umbraco.Cms.Examine * Move examine extensions to Umbraco.Extensions namespace * Reflect changed namespaces in Builder and fix unit tests * Adjust namespace in Umbraco.ModelsBuilder.Embedded * Adjust namespace in Umbraco.Persistence.SqlCe * Adjust namespace in Umbraco.PublishedCache.NuCache * Align namespaces in Umbraco.Web.BackOffice * Align namespaces in Umbraco.Web.Common * Ensure that SqlCeSupport is still enabled after changing the namespace * Align namespaces in Umbraco.Web.Website * Align namespaces in Umbraco.Web.UI.NetCore * Align namespaces in Umbraco.Tests.Common * Align namespaces in Umbraco.Tests.UnitTests * Align namespaces in Umbraco.Tests.Integration * Fix errors caused by changed namespaces * Fix integration tests * Undo the Umbraco.Examine.Lucene namespace change This breaks integration tests on linux, since the namespace wont exists there because it's only used on windows. * Fix merge * Fix Merge
93 lines
4.0 KiB
C#
93 lines
4.0 KiB
C#
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using Microsoft.AspNetCore.Mvc;
|
|
using Microsoft.AspNetCore.Mvc.ApplicationModels;
|
|
using Microsoft.AspNetCore.Mvc.ModelBinding;
|
|
using Umbraco.Cms.Web.Common.Attributes;
|
|
|
|
namespace Umbraco.Cms.Web.Common.ApplicationModels
|
|
{
|
|
|
|
/// <summary>
|
|
/// An application model provider for Umbraco API controllers to behave like WebApi controllers
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// <para>
|
|
/// Conventions will be applied to controllers attributed with <see cref="UmbracoApiControllerAttribute"/>
|
|
/// </para>
|
|
/// <para>
|
|
/// This is nearly a copy of aspnetcore's ApiBehaviorApplicationModelProvider which supplies a convention for the
|
|
/// [ApiController] attribute, however that convention is too strict for our purposes so we will have our own.
|
|
/// </para>
|
|
/// <para>
|
|
/// See https://shazwazza.com/post/custom-body-model-binding-per-controller-in-asp-net-core/
|
|
/// and https://github.com/dotnet/aspnetcore/issues/21724
|
|
/// </para>
|
|
/// </remarks>
|
|
public class UmbracoApiBehaviorApplicationModelProvider : IApplicationModelProvider
|
|
{
|
|
private readonly List<IActionModelConvention> _actionModelConventions;
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="UmbracoApiBehaviorApplicationModelProvider"/> class.
|
|
/// </summary>
|
|
public UmbracoApiBehaviorApplicationModelProvider(IModelMetadataProvider modelMetadataProvider)
|
|
{
|
|
// see see https://docs.microsoft.com/en-us/aspnet/core/web-api/?view=aspnetcore-3.1#apicontroller-attribute
|
|
// for what these things actually do
|
|
// NOTE: we don't have attribute routing requirements and we cannot use ApiVisibilityConvention without attribute routing
|
|
|
|
_actionModelConventions = new List<IActionModelConvention>()
|
|
{
|
|
new ClientErrorResultFilterConvention(), // Ensures the responses without any body is converted into a simple json object with info instead of a string like "Status Code: 404; Not Found"
|
|
new ConsumesConstraintForFormFileParameterConvention(), // If an controller accepts files, it must accept multipart/form-data.
|
|
new InferParameterBindingInfoConvention(modelMetadataProvider), // no need for [FromBody] everywhere, A complex type parameter is assigned to FromBody
|
|
|
|
// This ensures that all parameters of type BindingSource.Body (based on the above InferParameterBindingInfoConvention) are bound
|
|
// using our own UmbracoJsonModelBinder
|
|
new UmbracoJsonModelBinderConvention()
|
|
};
|
|
|
|
// TODO: Need to determine exactly how this affects errors
|
|
var defaultErrorType = typeof(ProblemDetails);
|
|
var defaultErrorTypeAttribute = new ProducesErrorResponseTypeAttribute(defaultErrorType);
|
|
_actionModelConventions.Add(new ApiConventionApplicationModelConvention(defaultErrorTypeAttribute));
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
/// <summary>
|
|
/// Will execute after <see cref="DefaultApplicationModelProvider"/>
|
|
/// </summary>
|
|
public int Order => 0;
|
|
|
|
/// <inheritdoc/>
|
|
public void OnProvidersExecuted(ApplicationModelProviderContext context)
|
|
{
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public void OnProvidersExecuting(ApplicationModelProviderContext context)
|
|
{
|
|
foreach (var controller in context.Result.Controllers)
|
|
{
|
|
if (!IsUmbracoApiController(controller))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
foreach (var action in controller.Actions)
|
|
{
|
|
foreach (var convention in _actionModelConventions)
|
|
{
|
|
convention.Apply(action);
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
private bool IsUmbracoApiController(ControllerModel controller)
|
|
=> controller.Attributes.OfType<UmbracoApiControllerAttribute>().Any();
|
|
}
|
|
}
|