233 lines
8.1 KiB
C#
233 lines
8.1 KiB
C#
// Copyright (c) .NET Foundation. All rights reserved.
|
|
// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Threading.Tasks;
|
|
using Microsoft.AspNetCore.Http;
|
|
using Microsoft.AspNetCore.Routing.Metadata;
|
|
using Microsoft.AspNetCore.Routing.Patterns;
|
|
using Microsoft.Extensions.DependencyInjection;
|
|
using Xunit;
|
|
|
|
namespace Microsoft.AspNetCore.Routing.Matchers
|
|
{
|
|
// End-to-end tests for the HTTP method matching functionality
|
|
public class HttpMethodMatcherPolicyIntegrationTest
|
|
{
|
|
[Fact]
|
|
public async Task Match_HttpMethod()
|
|
{
|
|
// Arrange
|
|
var endpoint = CreateEndpoint("/hello", httpMethods: new string[] { "GET", });
|
|
|
|
var matcher = CreateMatcher(endpoint);
|
|
var (httpContext, feature) = CreateContext("/hello", "GET");
|
|
|
|
// Act
|
|
await matcher.MatchAsync(httpContext, feature);
|
|
|
|
// Assert
|
|
MatcherAssert.AssertMatch(feature, endpoint);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Match_HttpMethod_CaseInsensitive()
|
|
{
|
|
// Arrange
|
|
var endpoint = CreateEndpoint("/hello", httpMethods: new string[] { "GeT", });
|
|
|
|
var matcher = CreateMatcher(endpoint);
|
|
var (httpContext, feature) = CreateContext("/hello", "GET");
|
|
|
|
// Act
|
|
await matcher.MatchAsync(httpContext, feature);
|
|
|
|
// Assert
|
|
MatcherAssert.AssertMatch(feature, endpoint);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Match_NoMetadata_MatchesAnyHttpMethod()
|
|
{
|
|
// Arrange
|
|
var endpoint = CreateEndpoint("/hello");
|
|
|
|
var matcher = CreateMatcher(endpoint);
|
|
var (httpContext, feature) = CreateContext("/hello", "GET");
|
|
|
|
// Act
|
|
await matcher.MatchAsync(httpContext, feature);
|
|
|
|
// Assert
|
|
MatcherAssert.AssertMatch(feature, endpoint);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Match_EmptyMethodList_MatchesAnyHttpMethod()
|
|
{
|
|
// Arrange
|
|
var endpoint = CreateEndpoint("/hello", httpMethods: new string[] { });
|
|
|
|
var matcher = CreateMatcher(endpoint);
|
|
var (httpContext, feature) = CreateContext("/hello", "GET");
|
|
|
|
// Act
|
|
await matcher.MatchAsync(httpContext, feature);
|
|
|
|
// Assert
|
|
MatcherAssert.AssertMatch(feature, endpoint);
|
|
}
|
|
|
|
[Fact] // When all of the candidates handles specific verbs, use a 405 endpoint
|
|
public async Task NotMatch_HttpMethod_Returns405Endpoint()
|
|
{
|
|
// Arrange
|
|
var endpoint1 = CreateEndpoint("/hello", httpMethods: new string[] { "GET", "PUT" });
|
|
var endpoint2 = CreateEndpoint("/hello", httpMethods: new string[] { "DELETE" });
|
|
|
|
var matcher = CreateMatcher(endpoint1, endpoint2);
|
|
var (httpContext, feature) = CreateContext("/hello", "POST");
|
|
|
|
// Act
|
|
await matcher.MatchAsync(httpContext, feature);
|
|
|
|
// Assert
|
|
Assert.NotSame(endpoint1, feature.Endpoint);
|
|
Assert.NotSame(endpoint2, feature.Endpoint);
|
|
|
|
Assert.Same(HttpMethodEndpointSelectorPolicy.Http405EndpointDisplayName, feature.Endpoint.DisplayName);
|
|
|
|
// Invoke the endpoint
|
|
await feature.Invoker((c) => Task.CompletedTask)(httpContext);
|
|
Assert.Equal(405, httpContext.Response.StatusCode);
|
|
Assert.Equal("DELETE, GET, PUT", httpContext.Response.Headers["Allow"]);
|
|
}
|
|
|
|
[Fact] // When one of the candidates handles all verbs, dont use a 405 endpoint
|
|
public async Task NotMatch_HttpMethod_WithAllMethodEndpoint_DoesNotReturn405()
|
|
{
|
|
// Arrange
|
|
var endpoint1 = CreateEndpoint("/{x:int}", httpMethods: new string[] { });
|
|
var endpoint2 = CreateEndpoint("/hello", httpMethods: new string[] { "DELETE" });
|
|
|
|
var matcher = CreateMatcher(endpoint1, endpoint2);
|
|
var (httpContext, feature) = CreateContext("/hello", "POST");
|
|
|
|
// Act
|
|
await matcher.MatchAsync(httpContext, feature);
|
|
|
|
// Assert
|
|
MatcherAssert.AssertNotMatch(feature);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Match_EndpointWithHttpMethodPreferred()
|
|
{
|
|
// Arrange
|
|
var endpoint1 = CreateEndpoint("/hello", httpMethods: new string[] { "GET", });
|
|
var endpoint2 = CreateEndpoint("/bar");
|
|
|
|
var matcher = CreateMatcher(endpoint1, endpoint2);
|
|
var (httpContext, feature) = CreateContext("/hello", "GET");
|
|
|
|
// Act
|
|
await matcher.MatchAsync(httpContext, feature);
|
|
|
|
// Assert
|
|
MatcherAssert.AssertMatch(feature, endpoint1);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Match_EndpointWithHttpMethodPreferred_EmptyList()
|
|
{
|
|
// Arrange
|
|
var endpoint1 = CreateEndpoint("/hello", httpMethods: new string[] { "GET", });
|
|
var endpoint2 = CreateEndpoint("/bar", httpMethods: new string[] { });
|
|
|
|
var matcher = CreateMatcher(endpoint1, endpoint2);
|
|
var (httpContext, feature) = CreateContext("/hello", "GET");
|
|
|
|
// Act
|
|
await matcher.MatchAsync(httpContext, feature);
|
|
|
|
// Assert
|
|
MatcherAssert.AssertMatch(feature, endpoint1);
|
|
}
|
|
|
|
[Fact] // The non-http-method-specific endpoint is part of the same candidate set
|
|
public async Task Match_EndpointWithHttpMethodPreferred_FallsBackToNonSpecific()
|
|
{
|
|
// Arrange
|
|
var endpoint1 = CreateEndpoint("/{x}", httpMethods: new string[] { "GET", });
|
|
var endpoint2 = CreateEndpoint("/{x}", httpMethods: new string[] { });
|
|
|
|
var matcher = CreateMatcher(endpoint1, endpoint2);
|
|
var (httpContext, feature) = CreateContext("/hello", "POST");
|
|
|
|
// Act
|
|
await matcher.MatchAsync(httpContext, feature);
|
|
|
|
// Assert
|
|
MatcherAssert.AssertMatch(feature, endpoint2, ignoreValues: true);
|
|
}
|
|
|
|
private static Matcher CreateMatcher(params MatcherEndpoint[] endpoints)
|
|
{
|
|
var services = new ServiceCollection()
|
|
.AddOptions()
|
|
.AddLogging()
|
|
.AddRouting()
|
|
.BuildServiceProvider();
|
|
|
|
var builder = services.GetRequiredService<DfaMatcherBuilder>();
|
|
for (var i = 0; i < endpoints.Length; i++)
|
|
{
|
|
builder.AddEndpoint(endpoints[i]);
|
|
}
|
|
|
|
return builder.Build();
|
|
}
|
|
|
|
internal static (HttpContext httpContext, IEndpointFeature feature) CreateContext(string path, string httpMethod)
|
|
{
|
|
var httpContext = new DefaultHttpContext();
|
|
httpContext.Request.Method = httpMethod;
|
|
httpContext.Request.Path = path;
|
|
|
|
var feature = new EndpointFeature();
|
|
httpContext.Features.Set<IEndpointFeature>(feature);
|
|
|
|
return (httpContext, feature);
|
|
}
|
|
internal static MatcherEndpoint CreateEndpoint(
|
|
string template,
|
|
object defaults = null,
|
|
object constraints = null,
|
|
int order = 0,
|
|
string[] httpMethods = null)
|
|
{
|
|
var metadata = new List<object>();
|
|
if (httpMethods != null)
|
|
{
|
|
metadata.Add(new HttpMethodMetadata(httpMethods));
|
|
}
|
|
|
|
var displayName = "endpoint: " + template + " " + string.Join(", ", httpMethods ?? new[] { "(any)" });
|
|
return new MatcherEndpoint(
|
|
MatcherEndpoint.EmptyInvoker,
|
|
RoutePatternFactory.Parse(template, defaults, constraints),
|
|
new RouteValueDictionary(),
|
|
order,
|
|
new EndpointMetadataCollection(metadata),
|
|
displayName);
|
|
}
|
|
|
|
internal (Matcher matcher, MatcherEndpoint endpoint) CreateMatcher(string template)
|
|
{
|
|
var endpoint = CreateEndpoint(template);
|
|
return (CreateMatcher(endpoint), endpoint);
|
|
}
|
|
}
|
|
}
|