Skip to content

Commit f470910

Browse files
authored
Support input validation for minimal APIs via generic resolver model (#60724)
* Add generic implementation for validations source generator * Make validate methods async and fix AoT suppressions * Update emitted code * Remove deadcode * Clean up API shapes a bit * Fix up RequiredAttribute handling * Add ValidatableContext and simplify API signature * Support ValidationOptions and multiple resolvers * Make ValidatableContext a setup entry * Move registration of filter to route handlers * Fix async for tests and IValidatableObject * Add doc comments * Add more tests * Enable PublicAPI analyzers and update public API * Add MaxDepth handling * Docs tweaks and package generator in shared framework * Clean up tests * Update for trimming * Harden parameter resolution check * Switch to runtime-based resolution for ParameterInfo validations * Prune out uneeded types * Fix up ValidatableParameterInfo signature * Make Validate methods virtual and support CustomValidationAttribute * Fix up emitted code and use explicit namespaces * Fix up suppression for ValidationContext trimming * Actually use attribute-based suppression * Fix up suppression for trimming warnings * Benchmarks, more tests, some tweaks * More tests and add DisableValidationFilter * Update API and add sample app * Tweak more APIs * Tweaks after API review * Fix nullability handling in generator and add sample Http file * Harden nullability and index checks in generator * Don't generate TypeInfo for invalidatable types * No-op code gen under more cases * Address feedback * Address feedback * Scrub out InterceptsLocationAttribute lines * Exempt more types, add no-op tests * Use List<Type> directly for method
1 parent 652bc2b commit f470910

File tree

59 files changed

+6816
-38
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

59 files changed

+6816
-38
lines changed

src/Framework/App.Ref/src/Microsoft.AspNetCore.App.Ref.sfxproj

+5
Original file line numberDiff line numberDiff line change
@@ -76,6 +76,11 @@
7676
Private="false"
7777
OutputItemType="AspNetCoreAnalyzer"
7878
ReferenceOutputAssembly="false" />
79+
80+
<ProjectReference Include="$(RepoRoot)src\Http\Http.Extensions\gen\Microsoft.AspNetCore.Http.ValidationsGenerator\Microsoft.AspNetCore.Http.ValidationsGenerator.csproj"
81+
Private="false"
82+
OutputItemType="AspNetCoreAnalyzer"
83+
ReferenceOutputAssembly="false" />
7984
</ItemGroup>
8085

8186
<ItemGroup>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
// Licensed to the .NET Foundation under one or more agreements.
2+
// The .NET Foundation licenses this file to you under the MIT license.
3+
4+
namespace Microsoft.AspNetCore.Http.Metadata;
5+
6+
/// <summary>
7+
/// A marker interface which can be used to identify metadata that disables validation
8+
/// on a given endpoint.
9+
/// </summary>
10+
public interface IDisableValidationMetadata
11+
{
12+
}
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,44 @@
11
#nullable enable
2+
abstract Microsoft.AspNetCore.Http.Validation.ValidatableParameterInfo.GetValidationAttributes() -> System.ComponentModel.DataAnnotations.ValidationAttribute![]!
3+
abstract Microsoft.AspNetCore.Http.Validation.ValidatablePropertyInfo.GetValidationAttributes() -> System.ComponentModel.DataAnnotations.ValidationAttribute![]!
4+
Microsoft.AspNetCore.Http.Metadata.IDisableValidationMetadata
25
Microsoft.AspNetCore.Http.ProducesResponseTypeMetadata.Description.get -> string?
36
Microsoft.AspNetCore.Http.ProducesResponseTypeMetadata.Description.set -> void
47
Microsoft.AspNetCore.Http.Metadata.IProducesResponseTypeMetadata.Description.get -> string?
8+
Microsoft.AspNetCore.Http.Validation.IValidatableInfo
9+
Microsoft.AspNetCore.Http.Validation.IValidatableInfo.ValidateAsync(object? value, Microsoft.AspNetCore.Http.Validation.ValidateContext! context, System.Threading.CancellationToken cancellationToken) -> System.Threading.Tasks.Task!
10+
Microsoft.AspNetCore.Http.Validation.IValidatableInfoResolver
11+
Microsoft.AspNetCore.Http.Validation.IValidatableInfoResolver.TryGetValidatableParameterInfo(System.Reflection.ParameterInfo! parameterInfo, out Microsoft.AspNetCore.Http.Validation.IValidatableInfo? validatableInfo) -> bool
12+
Microsoft.AspNetCore.Http.Validation.IValidatableInfoResolver.TryGetValidatableTypeInfo(System.Type! type, out Microsoft.AspNetCore.Http.Validation.IValidatableInfo? validatableInfo) -> bool
13+
Microsoft.AspNetCore.Http.Validation.ValidatableParameterInfo
14+
Microsoft.AspNetCore.Http.Validation.ValidatableParameterInfo.ValidatableParameterInfo(System.Type! parameterType, string! name, string! displayName) -> void
15+
Microsoft.AspNetCore.Http.Validation.ValidatablePropertyInfo
16+
Microsoft.AspNetCore.Http.Validation.ValidatablePropertyInfo.ValidatablePropertyInfo(System.Type! declaringType, System.Type! propertyType, string! name, string! displayName) -> void
17+
Microsoft.AspNetCore.Http.Validation.ValidatableTypeAttribute
18+
Microsoft.AspNetCore.Http.Validation.ValidatableTypeAttribute.ValidatableTypeAttribute() -> void
19+
Microsoft.AspNetCore.Http.Validation.ValidatableTypeInfo
20+
Microsoft.AspNetCore.Http.Validation.ValidatableTypeInfo.ValidatableTypeInfo(System.Type! type, System.Collections.Generic.IReadOnlyList<Microsoft.AspNetCore.Http.Validation.ValidatablePropertyInfo!>! members) -> void
21+
Microsoft.AspNetCore.Http.Validation.ValidateContext
22+
Microsoft.AspNetCore.Http.Validation.ValidateContext.CurrentDepth.get -> int
23+
Microsoft.AspNetCore.Http.Validation.ValidateContext.CurrentDepth.set -> void
24+
Microsoft.AspNetCore.Http.Validation.ValidateContext.CurrentValidationPath.get -> string!
25+
Microsoft.AspNetCore.Http.Validation.ValidateContext.CurrentValidationPath.set -> void
26+
Microsoft.AspNetCore.Http.Validation.ValidateContext.ValidateContext() -> void
27+
Microsoft.AspNetCore.Http.Validation.ValidateContext.ValidationContext.get -> System.ComponentModel.DataAnnotations.ValidationContext?
28+
Microsoft.AspNetCore.Http.Validation.ValidateContext.ValidationContext.set -> void
29+
Microsoft.AspNetCore.Http.Validation.ValidateContext.ValidationErrors.get -> System.Collections.Generic.Dictionary<string!, string![]!>?
30+
Microsoft.AspNetCore.Http.Validation.ValidateContext.ValidationErrors.set -> void
31+
Microsoft.AspNetCore.Http.Validation.ValidateContext.ValidationOptions.get -> Microsoft.AspNetCore.Http.Validation.ValidationOptions!
32+
Microsoft.AspNetCore.Http.Validation.ValidateContext.ValidationOptions.set -> void
33+
Microsoft.AspNetCore.Http.Validation.ValidationOptions
34+
Microsoft.AspNetCore.Http.Validation.ValidationOptions.MaxDepth.get -> int
35+
Microsoft.AspNetCore.Http.Validation.ValidationOptions.MaxDepth.set -> void
36+
Microsoft.AspNetCore.Http.Validation.ValidationOptions.Resolvers.get -> System.Collections.Generic.IList<Microsoft.AspNetCore.Http.Validation.IValidatableInfoResolver!>!
37+
Microsoft.AspNetCore.Http.Validation.ValidationOptions.TryGetValidatableParameterInfo(System.Reflection.ParameterInfo! parameterInfo, out Microsoft.AspNetCore.Http.Validation.IValidatableInfo? validatableInfo) -> bool
38+
Microsoft.AspNetCore.Http.Validation.ValidationOptions.TryGetValidatableTypeInfo(System.Type! type, out Microsoft.AspNetCore.Http.Validation.IValidatableInfo? validatableTypeInfo) -> bool
39+
Microsoft.AspNetCore.Http.Validation.ValidationOptions.ValidationOptions() -> void
40+
Microsoft.Extensions.DependencyInjection.ValidationServiceCollectionExtensions
41+
static Microsoft.Extensions.DependencyInjection.ValidationServiceCollectionExtensions.AddValidation(this Microsoft.Extensions.DependencyInjection.IServiceCollection! services, System.Action<Microsoft.AspNetCore.Http.Validation.ValidationOptions!>? configureOptions = null) -> Microsoft.Extensions.DependencyInjection.IServiceCollection!
42+
virtual Microsoft.AspNetCore.Http.Validation.ValidatableParameterInfo.ValidateAsync(object? value, Microsoft.AspNetCore.Http.Validation.ValidateContext! context, System.Threading.CancellationToken cancellationToken) -> System.Threading.Tasks.Task!
43+
virtual Microsoft.AspNetCore.Http.Validation.ValidatablePropertyInfo.ValidateAsync(object? value, Microsoft.AspNetCore.Http.Validation.ValidateContext! context, System.Threading.CancellationToken cancellationToken) -> System.Threading.Tasks.Task!
44+
virtual Microsoft.AspNetCore.Http.Validation.ValidatableTypeInfo.ValidateAsync(object? value, Microsoft.AspNetCore.Http.Validation.ValidateContext! context, System.Threading.CancellationToken cancellationToken) -> System.Threading.Tasks.Task!
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
// Licensed to the .NET Foundation under one or more agreements.
2+
// The .NET Foundation licenses this file to you under the MIT license.
3+
4+
namespace Microsoft.AspNetCore.Http.Validation;
5+
6+
/// <summary>
7+
/// Represents an interface for validating a value.
8+
/// </summary>
9+
public interface IValidatableInfo
10+
{
11+
/// <summary>
12+
/// Validates the specified <paramref name="value"/>.
13+
/// </summary>
14+
/// <param name="value">The value to validate.</param>
15+
/// <param name="context">The validation context.</param>
16+
/// <param name="cancellationToken">A cancellation token to support cancellation of the validation.</param>
17+
/// <returns>A <see cref="Task"/> representing the asynchronous operation.</returns>
18+
Task ValidateAsync(object? value, ValidateContext context, CancellationToken cancellationToken);
19+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
// Licensed to the .NET Foundation under one or more agreements.
2+
// The .NET Foundation licenses this file to you under the MIT license.
3+
4+
using System.Diagnostics.CodeAnalysis;
5+
using System.Reflection;
6+
7+
namespace Microsoft.AspNetCore.Http.Validation;
8+
9+
/// <summary>
10+
/// Provides an interface for resolving the validation information associated
11+
/// with a given <seealso cref="Type"/> or <seealso cref="ParameterInfo"/>.
12+
/// </summary>
13+
public interface IValidatableInfoResolver
14+
{
15+
/// <summary>
16+
/// Gets validation information for the specified type.
17+
/// </summary>
18+
/// <param name="type">The type to get validation information for.</param>
19+
/// <param name="validatableInfo">
20+
/// The output parameter that will contain the validatable information if found.
21+
/// </param>
22+
/// <returns><see langword="true" /> if the validatable type information was found; otherwise, false.</returns>
23+
bool TryGetValidatableTypeInfo(Type type, [NotNullWhen(true)] out IValidatableInfo? validatableInfo);
24+
25+
/// <summary>
26+
/// Gets validation information for the specified parameter.
27+
/// </summary>
28+
/// <param name="parameterInfo">The parameter to get validation information for.</param>
29+
/// <param name="validatableInfo">The output parameter that will contain the validatable information if found.</param>
30+
/// <returns><see langword="true" /> if the validatable parameter information was found; otherwise, false.</returns>
31+
bool TryGetValidatableParameterInfo(ParameterInfo parameterInfo, [NotNullWhen(true)] out IValidatableInfo? validatableInfo);
32+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
// Licensed to the .NET Foundation under one or more agreements.
2+
// The .NET Foundation licenses this file to you under the MIT license.
3+
4+
using System.ComponentModel.DataAnnotations;
5+
using System.Diagnostics.CodeAnalysis;
6+
using System.Linq;
7+
using System.Reflection;
8+
9+
namespace Microsoft.AspNetCore.Http.Validation;
10+
11+
internal sealed class RuntimeValidatableParameterInfoResolver : IValidatableInfoResolver
12+
{
13+
// TODO: the implementation currently relies on static discovery of types.
14+
public bool TryGetValidatableTypeInfo(Type type, [NotNullWhen(true)] out IValidatableInfo? validatableInfo)
15+
{
16+
validatableInfo = null;
17+
return false;
18+
}
19+
20+
public bool TryGetValidatableParameterInfo(ParameterInfo parameterInfo, [NotNullWhen(true)] out IValidatableInfo? validatableInfo)
21+
{
22+
if (parameterInfo.Name == null)
23+
{
24+
throw new InvalidOperationException($"Encountered a parameter of type '{parameterInfo.ParameterType}' without a name. Parameters must have a name.");
25+
}
26+
27+
var validationAttributes = parameterInfo
28+
.GetCustomAttributes<ValidationAttribute>()
29+
.ToArray();
30+
validatableInfo = new RuntimeValidatableParameterInfo(
31+
parameterType: parameterInfo.ParameterType,
32+
name: parameterInfo.Name,
33+
displayName: GetDisplayName(parameterInfo),
34+
validationAttributes: validationAttributes
35+
);
36+
return true;
37+
}
38+
39+
private static string GetDisplayName(ParameterInfo parameterInfo)
40+
{
41+
var displayAttribute = parameterInfo.GetCustomAttribute<DisplayAttribute>();
42+
if (displayAttribute != null)
43+
{
44+
return displayAttribute.Name ?? parameterInfo.Name!;
45+
}
46+
47+
return parameterInfo.Name!;
48+
}
49+
50+
private sealed class RuntimeValidatableParameterInfo(
51+
Type parameterType,
52+
string name,
53+
string displayName,
54+
ValidationAttribute[] validationAttributes) :
55+
ValidatableParameterInfo(parameterType, name, displayName)
56+
{
57+
protected override ValidationAttribute[] GetValidationAttributes() => _validationAttributes;
58+
59+
private readonly ValidationAttribute[] _validationAttributes = validationAttributes;
60+
}
61+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,134 @@
1+
// Licensed to the .NET Foundation under one or more agreements.
2+
// The .NET Foundation licenses this file to you under the MIT license.
3+
4+
using System.Collections;
5+
using System.ComponentModel.DataAnnotations;
6+
using System.Diagnostics.CodeAnalysis;
7+
8+
namespace Microsoft.AspNetCore.Http.Validation;
9+
10+
internal static class TypeExtensions
11+
{
12+
/// <summary>
13+
/// Determines whether the specified type is an enumerable type.
14+
/// </summary>
15+
/// <param name="type">The type to check.</param>
16+
/// <returns><see langword="true"/> if the type is enumerable; otherwise, <see langword="false"/>.</returns>
17+
public static bool IsEnumerable(this Type type)
18+
{
19+
// Check if type itself is an IEnumerable
20+
if (type.IsGenericType &&
21+
(type.GetGenericTypeDefinition() == typeof(IEnumerable<>) ||
22+
type.GetGenericTypeDefinition() == typeof(ICollection<>) ||
23+
type.GetGenericTypeDefinition() == typeof(List<>) ||
24+
type.GetGenericTypeDefinition() == typeof(IList<>)))
25+
{
26+
return true;
27+
}
28+
29+
// Or an array
30+
if (type.IsArray)
31+
{
32+
return true;
33+
}
34+
35+
// Then evaluate if it implements IEnumerable and is not a string
36+
if (typeof(IEnumerable).IsAssignableFrom(type) &&
37+
type != typeof(string))
38+
{
39+
return true;
40+
}
41+
42+
return false;
43+
}
44+
45+
/// <summary>
46+
/// Determines whether the specified type is a nullable type.
47+
/// </summary>
48+
/// <param name="type">The type to check.</param>
49+
/// <returns><see langword="true"/> if the type is nullable; otherwise, <see langword="false"/>.</returns>
50+
public static bool IsNullable(this Type type)
51+
{
52+
if (type.IsValueType)
53+
{
54+
return false;
55+
}
56+
57+
if (type.IsGenericType &&
58+
type.GetGenericTypeDefinition() == typeof(Nullable<>))
59+
{
60+
return true;
61+
}
62+
63+
return false;
64+
}
65+
66+
/// <summary>
67+
/// Tries to get the <see cref="RequiredAttribute"/> from the specified array of validation attributes.
68+
/// </summary>
69+
/// <param name="attributes">The array of <see cref="ValidationAttribute"/> to search.</param>
70+
/// <param name="requiredAttribute">The found <see cref="RequiredAttribute"/> if available, otherwise null.</param>
71+
/// <returns><see langword="true"/> if a <see cref="RequiredAttribute"/> is found; otherwise, <see langword="false"/>.</returns>
72+
public static bool TryGetRequiredAttribute(this ValidationAttribute[] attributes, [NotNullWhen(true)] out RequiredAttribute? requiredAttribute)
73+
{
74+
foreach (var attribute in attributes)
75+
{
76+
if (attribute is RequiredAttribute requiredAttr)
77+
{
78+
requiredAttribute = requiredAttr;
79+
return true;
80+
}
81+
}
82+
83+
requiredAttribute = null;
84+
return false;
85+
}
86+
87+
/// <summary>
88+
/// Gets all types that the specified type implements or inherits from.
89+
/// </summary>
90+
/// <param name="type">The type to analyze.</param>
91+
/// <returns>A collection containing all implemented interfaces and all base types of the given type.</returns>
92+
public static List<Type> GetAllImplementedTypes([DynamicallyAccessedMembers(DynamicallyAccessedMemberTypes.Interfaces)] this Type type)
93+
{
94+
ArgumentNullException.ThrowIfNull(type);
95+
96+
var implementedTypes = new List<Type>();
97+
98+
// Yield all interfaces directly and indirectly implemented by this type
99+
foreach (var interfaceType in type.GetInterfaces())
100+
{
101+
implementedTypes.Add(interfaceType);
102+
}
103+
104+
// Finally, walk up the inheritance chain
105+
var baseType = type.BaseType;
106+
while (baseType != null && baseType != typeof(object))
107+
{
108+
implementedTypes.Add(baseType);
109+
baseType = baseType.BaseType;
110+
}
111+
112+
return implementedTypes;
113+
}
114+
115+
/// <summary>
116+
/// Determines whether the specified type implements the given interface.
117+
/// </summary>
118+
/// <param name="type">The type to check.</param>
119+
/// <param name="interfaceType">The interface type to check for.</param>
120+
/// <returns>True if the type implements the specified interface; otherwise, false.</returns>
121+
public static bool ImplementsInterface(this Type type, Type interfaceType)
122+
{
123+
ArgumentNullException.ThrowIfNull(type);
124+
ArgumentNullException.ThrowIfNull(interfaceType);
125+
126+
// Check if interfaceType is actually an interface
127+
if (!interfaceType.IsInterface)
128+
{
129+
throw new ArgumentException($"Type {interfaceType.FullName} is not an interface.", nameof(interfaceType));
130+
}
131+
132+
return interfaceType.IsAssignableFrom(type);
133+
}
134+
}

0 commit comments

Comments
 (0)