Enable Tag Helper initializers:

- Any Action<ITagHelper, ViewContext> can be registered in DI to allow setting of tag helper properties after any [Activate] properties are set but before their bound from the source document and the tag helper is executed
- Removed previous tag helper options as initializers is the way to do this now
- #1689
- #2162
This commit is contained in:
damianedwards 2015-03-13 12:31:02 -07:00
parent 00feaaac29
commit 7d1c1ed8eb
16 changed files with 296 additions and 240 deletions

View File

@ -3,6 +3,7 @@
using System;
using System.Collections.Concurrent;
using System.Collections.Generic;
using System.Reflection;
using Microsoft.AspNet.Mvc.Rendering;
using Microsoft.AspNet.Razor.Runtime.TagHelpers;
@ -29,7 +30,8 @@ namespace Microsoft.AspNet.Mvc.Razor
}
/// <inheritdoc />
public void Activate([NotNull] ITagHelper tagHelper, [NotNull] ViewContext context)
public void Activate<TTagHelper>([NotNull] TTagHelper tagHelper, [NotNull] ViewContext context)
where TTagHelper : ITagHelper
{
var propertiesToActivate = _injectActions.GetOrAdd(tagHelper.GetType(),
_getPropertiesToActivate);
@ -39,6 +41,21 @@ namespace Microsoft.AspNet.Mvc.Razor
var activateInfo = propertiesToActivate[i];
activateInfo.Activate(tagHelper, context);
}
InitializeTagHelper(tagHelper, context);
}
private static void InitializeTagHelper<TTagHelper>(TTagHelper tagHelper, ViewContext context)
where TTagHelper : ITagHelper
{
// Run any tag helper initializers in the container
var serviceProvider = context.HttpContext.RequestServices;
var initializers = serviceProvider.GetService<IEnumerable<ITagHelperInitializer<TTagHelper>>>();
foreach (var initializer in initializers)
{
initializer.Initialize(tagHelper, context);
}
}
private static PropertyActivator<ViewContext> CreateActivateInfo(PropertyInfo property)

View File

@ -13,8 +13,9 @@ namespace Microsoft.AspNet.Mvc.Razor
/// <summary>
/// When implemented in a type, activates an instantiated <see cref="ITagHelper"/>.
/// </summary>
/// <param name="tagHelper">The <see cref="ITagHelper"/> to activate.</param>
/// <typeparam name="TTagHelper">The <see cref="ITagHelper"/> type.</typeparam>
/// <param name="tagHelper">The <typeparamref name="TTagHelper"/> to activate.</param>
/// <param name="context">The <see cref="ViewContext"/> for the executing view.</param>
void Activate(ITagHelper tagHelper, ViewContext context);
void Activate<TTagHelper>(TTagHelper tagHelper, ViewContext context) where TTagHelper : ITagHelper;
}
}

View File

@ -0,0 +1,24 @@
// Copyright (c) Microsoft Open Technologies, Inc. All rights reserved.
// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
using Microsoft.AspNet.Mvc.Rendering;
using Microsoft.AspNet.Razor.Runtime.TagHelpers;
using Microsoft.Framework.Internal;
namespace Microsoft.AspNet.Mvc.Razor
{
/// <summary>
/// Initializes an <see cref="ITagHelper"/> before it's executed.
/// </summary>
/// <typeparam name="TTagHelper">The <see cref="ITagHelper"/> type.</typeparam>
public interface ITagHelperInitializer<TTagHelper>
where TTagHelper : ITagHelper
{
/// <summary>
/// Initializes the <typeparamref name="TTagHelper"/>.
/// </summary>
/// <param name="helper">The <typeparamref name="TTagHelper"/> to initialize.</param>
/// <param name="context">The <see cref="ViewContext"/> for the executing view.</param>
void Initialize([NotNull] TTagHelper helper, [NotNull] ViewContext context);
}
}

View File

@ -4,6 +4,7 @@
using System;
using Microsoft.AspNet.Mvc;
using Microsoft.AspNet.Mvc.Razor;
using Microsoft.AspNet.Razor.Runtime.TagHelpers;
using Microsoft.Framework.Internal;
namespace Microsoft.Framework.DependencyInjection
@ -24,5 +25,28 @@ namespace Microsoft.Framework.DependencyInjection
{
services.Configure(setupAction);
}
/// <summary>
/// Adds an initialization callback for a given <typeparamref name="TTagHelper"/>.
/// </summary>
/// <remarks>
/// The callback will be invoked on any <typeparamref name="TTagHelper"/> instance before the
/// <see cref="ITagHelper.ProcessAsync(TagHelperContext, TagHelperOutput)"/> method is called.
/// </remarks>
/// <typeparam name="TTagHelper">The type of <see cref="ITagHelper"/> being initialized.</typeparam>
/// <param name="services">The <see cref="IServiceCollection"/> instance this method extends.</param>
/// <param name="initialize">An action to initialize the <typeparamref name="TTagHelper"/>.</param>
/// <returns>The <see cref="IServiceCollection"/> instance this method extends.</returns>
public static IServiceCollection InitializeTagHelper<TTagHelper>(
[NotNull] this IServiceCollection services,
[NotNull] Action<TTagHelper, ViewContext> initialize)
where TTagHelper : ITagHelper
{
var initializer = new TagHelperInitializer<TTagHelper>(initialize);
services.AddInstance(typeof(ITagHelperInitializer<TTagHelper>), initializer);
return services;
}
}
}

View File

@ -0,0 +1,31 @@
// Copyright (c) Microsoft Open Technologies, Inc. All rights reserved.
// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
using System;
using Microsoft.AspNet.Razor.Runtime.TagHelpers;
using Microsoft.Framework.Internal;
namespace Microsoft.AspNet.Mvc.Razor
{
/// <inheritdoc />
public class TagHelperInitializer<TTagHelper> : ITagHelperInitializer<TTagHelper>
where TTagHelper : ITagHelper
{
private readonly Action<TTagHelper, ViewContext> _initializeDelegate;
/// <summary>
/// Creates a <see cref="TagHelperInitializer{TTagHelper}"/>.
/// </summary>
/// <param name="action">The initialization delegate.</param>
public TagHelperInitializer([NotNull] Action<TTagHelper, ViewContext> action)
{
_initializeDelegate = action;
}
/// <inheritdoc />
public void Initialize([NotNull] TTagHelper helper, [NotNull] ViewContext context)
{
_initializeDelegate(helper, context);
}
}
}

View File

@ -6,7 +6,6 @@ using System.Collections.Generic;
using System.Linq;
using Microsoft.AspNet.Mvc.Rendering;
using Microsoft.AspNet.Razor.Runtime.TagHelpers;
using Microsoft.Framework.OptionsModel;
namespace Microsoft.AspNet.Mvc.TagHelpers
{
@ -28,11 +27,7 @@ namespace Microsoft.AspNet.Mvc.TagHelpers
// Protected to ensure subclasses are correctly activated. Internal for ease of use when testing.
[Activate]
protected internal IHtmlGenerator Generator { get; set; }
// Protected to ensure subclasses are correctly activated. Internal for ease of use when testing.
[Activate]
protected internal IOptions<FormTagHelperOptions> Options { get; set; }
/// <summary>
/// The name of the action method.
/// </summary>
@ -113,7 +108,7 @@ namespace Microsoft.AspNet.Mvc.TagHelpers
}
}
if (AntiForgery ?? Options.Options.GenerateAntiForgeryToken ?? antiForgeryDefault)
if (AntiForgery ?? antiForgeryDefault)
{
var antiForgeryTagBuilder = Generator.GenerateAntiForgery(ViewContext);
if (antiForgeryTagBuilder != null)

View File

@ -1,21 +0,0 @@
// Copyright (c) Microsoft Open Technologies, Inc. All rights reserved.
// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
namespace Microsoft.AspNet.Mvc.TagHelpers
{
/// <summary>
/// Options pertaining to the default behavior of <see cref="FormTagHelper"/> instances.
/// </summary>
public class FormTagHelperOptions
{
/// <summary>
/// Whether the anti-forgery token should be generated by default for all instances of
/// <see cref="FormTagHelper"/>. Can be overridden on any given instance.
/// </summary>
/// <value>
/// Defaults to <c>null</c>, which indicates a token will only be generated if the <c>action</c>
/// attribute was not explicitly defined.
/// </value>
public bool? GenerateAntiForgeryToken { get; set; }
}
}

View File

@ -1,18 +0,0 @@
// Copyright (c) Microsoft Open Technologies, Inc. All rights reserved.
// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
using Microsoft.AspNet.Razor.Runtime.TagHelpers;
namespace Microsoft.Framework.DependencyInjection
{
/// <summary>
/// Used for adding options pertaining to <see cref="ITagHelper"/>s to an <see cref="IServiceCollection"/>.
/// </summary>
public interface ITagHelperOptionsCollection
{
/// <summary>
/// The <see cref="IServiceCollection"/>.
/// </summary>
IServiceCollection Services { get; }
}
}

View File

@ -1,29 +0,0 @@
// Copyright (c) Microsoft Open Technologies, Inc. All rights reserved.
// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
using Microsoft.AspNet.Razor.Runtime.TagHelpers;
using Microsoft.Framework.DependencyInjection;
using Microsoft.Framework.Internal;
namespace Microsoft.AspNet.Mvc.TagHelpers
{
/// <summary>
/// Used for adding options pertaining to <see cref="ITagHelper"/>s to an <see cref="IServiceCollection"/>.
/// </summary>
public class TagHelperOptionsCollection : ITagHelperOptionsCollection
{
/// <summary>
/// Creates a new <see cref="TagHelperOptionsCollection"/>;
/// </summary>
/// <param name="serviceCollection">The <see cref="IServiceCollection"/> instance to add the options to.</param>
public TagHelperOptionsCollection([NotNull] IServiceCollection serviceCollection)
{
Services = serviceCollection;
}
/// <summary>
/// The <see cref="IServiceCollection"/>.
/// </summary>
public IServiceCollection Services { get; }
}
}

View File

@ -1,46 +0,0 @@
// Copyright (c) Microsoft Open Technologies, Inc. All rights reserved.
// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
using System;
using Microsoft.AspNet.Mvc.TagHelpers;
using Microsoft.Framework.ConfigurationModel;
using Microsoft.Framework.Internal;
namespace Microsoft.Framework.DependencyInjection
{
/// <summary>
/// Extension methods for <see cref="ITagHelperOptionsCollection"/>.
/// </summary>
public static class TagHelperOptionsCollectionExtensions
{
/// <summary>
/// Configures options for the <see cref="FormTagHelper"/> from an <see cref="IConfiguration"/>.
/// </summary>
/// <param name="collection">The <see cref="ITagHelperOptionsCollection"/> instance this method extends.</param>
/// <param name="configuration">An <see cref="IConfiguration"/> to get the options from.</param>
/// <returns>The <see cref="ITagHelperOptionsCollection"/>.</returns>
public static ITagHelperOptionsCollection ConfigureForm(
[NotNull] this ITagHelperOptionsCollection collection,
[NotNull] IConfiguration configuration)
{
collection.Services.Configure<FormTagHelperOptions>(configuration);
return collection;
}
/// <summary>
/// Configures options for the <see cref="FormTagHelper"/> using a delegate.
/// </summary>
/// <param name="collection">The <see cref="ITagHelperOptionsCollection"/> instance this method extends.</param>
/// <param name="setupAction">The options setup delegate.</param>
/// <returns>The <see cref="ITagHelperOptionsCollection"/>.</returns>
public static ITagHelperOptionsCollection ConfigureForm(
[NotNull] this ITagHelperOptionsCollection collection,
[NotNull] Action<FormTagHelperOptions> setupAction)
{
collection.Services.Configure(setupAction);
return collection;
}
}
}

View File

@ -1,27 +0,0 @@
// Copyright (c) Microsoft Open Technologies, Inc. All rights reserved.
// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
using Microsoft.AspNet.Mvc.TagHelpers;
using Microsoft.AspNet.Razor.Runtime.TagHelpers;
using Microsoft.Framework.Internal;
namespace Microsoft.Framework.DependencyInjection
{
/// <summary>
/// Extension methods for <see cref="IServiceCollection"/>.
/// </summary>
public static class TagHelperServiceCollectionExtensions
{
/// <summary>
/// Creates an <see cref="ITagHelperOptionsCollection"/> which can be used to add options pertaining to
/// <see cref="ITagHelper"/>s to the <see cref="IServiceCollection"/>.
/// </summary>
/// <param name="serviceCollection">The <see cref="IServiceCollection"/> instance this method extends.</param>
/// <returns>The <see cref="ITagHelperOptionsCollection"/>.</returns>
public static ITagHelperOptionsCollection ConfigureTagHelpers(
[NotNull] this IServiceCollection serviceCollection)
{
return new TagHelperOptionsCollection(serviceCollection);
}
}
}

View File

@ -10,6 +10,7 @@ using System.Net.Http.Headers;
using System.Reflection;
using System.Threading.Tasks;
using Microsoft.AspNet.Builder;
using Microsoft.AspNet.Mvc.TagHelpers;
using Microsoft.Framework.DependencyInjection;
using MvcTagHelpersWebSite;
using Xunit;
@ -344,7 +345,7 @@ Products: Laptops (3)";
{
// Arrange
var newServices = new ServiceCollection();
newServices.ConfigureTagHelpers().ConfigureForm(options => options.GenerateAntiForgeryToken = optionsAntiForgery);
newServices.InitializeTagHelper<FormTagHelper>((helper, _) => helper.AntiForgery = optionsAntiForgery);
var server = TestHelper.CreateServer(_app, SiteName, services => services.Add(newServices));
var client = server.CreateClient();
var expectedMediaType = MediaTypeHeaderValue.Parse("text/html; charset=utf-8");

View File

@ -0,0 +1,176 @@
// Copyright (c) Microsoft Open Technologies, Inc. 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.IO;
using Microsoft.AspNet.Http;
using Microsoft.AspNet.Http.Core;
using Microsoft.AspNet.Mvc.ModelBinding;
using Microsoft.AspNet.Mvc.Rendering;
using Microsoft.AspNet.Razor.Runtime.TagHelpers;
using Microsoft.AspNet.Routing;
using Microsoft.Framework.DependencyInjection;
using Moq;
using Xunit;
namespace Microsoft.AspNet.Mvc.Razor
{
public class DefaultTagHelperActivatorTest
{
[Theory]
[InlineData("test", 100)]
[InlineData(null, -1)]
public void Activate_InitializesTagHelpers(string name, int number)
{
// Arrange
var services = new ServiceCollection();
services.InitializeTagHelper<TestTagHelper>((h, vc) =>
{
h.Name = name;
h.Number = number;
h.ViewDataValue = vc.ViewData["TestData"];
});
var httpContext = MakeHttpContext(services.BuildServiceProvider());
var viewContext = MakeViewContext(httpContext);
var viewDataValue = new object();
viewContext.ViewData.Add("TestData", viewDataValue);
var activator = new DefaultTagHelperActivator();
var helper = new TestTagHelper();
// Act
activator.Activate(helper, viewContext);
// Assert
Assert.Equal(name, helper.Name);
Assert.Equal(number, helper.Number);
Assert.Same(viewDataValue, helper.ViewDataValue);
}
[Fact]
public void Activate_InitializesTagHelpersAfterActivatingProperties()
{
// Arrange
var services = new ServiceCollection();
services.InitializeTagHelper<TestTagHelper>((h, _) => h.ViewContext = MakeViewContext(MakeHttpContext()));
var httpContext = MakeHttpContext(services.BuildServiceProvider());
var viewContext = MakeViewContext(httpContext);
var activator = new DefaultTagHelperActivator();
var helper = new TestTagHelper();
// Act
activator.Activate(helper, viewContext);
// Assert
Assert.NotSame(viewContext, helper.ViewContext);
}
[Fact]
public void Activate_InitializesTagHelpersWithMultipleInitializers()
{
// Arrange
var services = new ServiceCollection();
services.InitializeTagHelper<TestTagHelper>((h, vc) =>
{
h.Name = "Test 1";
h.Number = 100;
});
services.InitializeTagHelper<TestTagHelper>((h, vc) =>
{
h.Name += ", Test 2";
h.Number += 100;
});
var httpContext = MakeHttpContext(services.BuildServiceProvider());
var viewContext = MakeViewContext(httpContext);
var activator = new DefaultTagHelperActivator();
var helper = new TestTagHelper();
// Act
activator.Activate(helper, viewContext);
// Assert
Assert.Equal("Test 1, Test 2", helper.Name);
Assert.Equal(200, helper.Number);
}
[Fact]
public void Activate_InitializesTagHelpersWithCorrectInitializers()
{
// Arrange
var services = new ServiceCollection();
services.InitializeTagHelper<TestTagHelper>((h, vc) =>
{
h.Name = "Test 1";
h.Number = 100;
});
services.InitializeTagHelper<AnotherTestTagHelper>((h, vc) =>
{
h.Name = "Test 2";
h.Number = 102;
});
var httpContext = MakeHttpContext(services.BuildServiceProvider());
var viewContext = MakeViewContext(httpContext);
var activator = new DefaultTagHelperActivator();
var testTagHelper = new TestTagHelper();
var anotherTestTagHelper = new AnotherTestTagHelper();
// Act
activator.Activate(testTagHelper, viewContext);
activator.Activate(anotherTestTagHelper, viewContext);
// Assert
Assert.Equal("Test 1", testTagHelper.Name);
Assert.Equal(100, testTagHelper.Number);
Assert.Equal("Test 2", anotherTestTagHelper.Name);
Assert.Equal(102, anotherTestTagHelper.Number);
}
private static HttpContext MakeHttpContext(IServiceProvider services = null)
{
var httpContext = new DefaultHttpContext();
if (services != null)
{
httpContext.RequestServices = services;
}
return httpContext;
}
private static ViewContext MakeViewContext(HttpContext httpContext)
{
var actionContext = new ActionContext(httpContext, new RouteData(), new ActionDescriptor());
var metadataProvider = new EmptyModelMetadataProvider();
var viewData = new ViewDataDictionary(metadataProvider);
var viewContext = new ViewContext(
actionContext,
Mock.Of<IView>(),
viewData,
Mock.Of<ITempDataDictionary>(),
TextWriter.Null);
return viewContext;
}
private class TestTagHelper : TagHelper
{
public string Name { get; set; } = "Initial Name";
public int Number { get; set; } = 1000;
public object ViewDataValue { get; set; } = new object();
[Activate]
public ViewContext ViewContext { get; set; }
}
private class AnotherTestTagHelper : TagHelper
{
public string Name { get; set; } = "Initial Name";
public int Number { get; set; } = 1000;
public object ViewDataValue { get; set; } = new object();
[Activate]
public ViewContext ViewContext { get; set; }
}
}
}

View File

@ -2,7 +2,9 @@
// 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.IO;
using System.Linq;
using System.Threading.Tasks;
using Microsoft.AspNet.Http;
using Microsoft.AspNet.Mvc.ModelBinding;
@ -106,6 +108,13 @@ namespace Microsoft.AspNet.Mvc.Razor
.Returns(myService);
serviceProvider.Setup(mock => mock.GetService(typeof(ITagHelperActivator)))
.Returns(new DefaultTagHelperActivator());
serviceProvider.Setup(mock => mock.GetService(It.Is<Type>(serviceType =>
serviceType.IsGenericType && serviceType.GetGenericTypeDefinition() == typeof(IEnumerable<>))))
.Returns<Type>(serviceType =>
{
var enumerableType = serviceType.GetGenericArguments().First();
return typeof(Enumerable).GetMethod("Empty").MakeGenericMethod(enumerableType).Invoke(null, null);
});
var httpContext = new Mock<HttpContext>();
httpContext.SetupGet(c => c.RequestServices)
.Returns(serviceProvider.Object);

View File

@ -90,22 +90,14 @@ namespace Microsoft.AspNet.Mvc.TagHelpers
}
[Theory]
[InlineData(null, true, "<input />")]
[InlineData(null, false, "")]
[InlineData(null, null, "<input />")]
[InlineData(true, true, "<input />")]
[InlineData(true, false, "")]
[InlineData(true, null, "<input />")]
[InlineData(false, true, "<input />")]
[InlineData(false, false, "")]
[InlineData(false, null, "")]
[InlineData(null, "<input />")]
[InlineData(true, "<input />")]
[InlineData(false, "")]
public async Task ProcessAsync_GeneratesAntiForgeryCorrectly(
bool? optionsAntiForgery,
bool? antiForgery,
string expectedPostContent)
{
// Arrange
var options = MakeOptions(new FormTagHelperOptions { GenerateAntiForgeryToken = optionsAntiForgery });
var viewContext = CreateViewContext();
var context = new TagHelperContext(
allAttributes: new Dictionary<string, object>(),
@ -139,7 +131,6 @@ namespace Microsoft.AspNet.Mvc.TagHelpers
Action = "Index",
AntiForgery = antiForgery,
Generator = generator.Object,
Options = options,
ViewContext = viewContext,
};
@ -280,11 +271,9 @@ namespace Microsoft.AspNet.Mvc.TagHelpers
public async Task ProcessAsync_RestoresBoundAttributesIfActionIsSpecified(string htmlAction)
{
// Arrange
var options = MakeOptions<FormTagHelperOptions>();
var formTagHelper = new FormTagHelper
{
Method = "POST",
Options = options
Method = "POST"
};
var output = new TagHelperOutput("form",
attributes: new Dictionary<string, string>
@ -323,22 +312,14 @@ namespace Microsoft.AspNet.Mvc.TagHelpers
}
[Theory]
[InlineData(null, true, "<input />")]
[InlineData(null, false, "")]
[InlineData(null, null, "")]
[InlineData(true, true, "<input />")]
[InlineData(true, false, "")]
[InlineData(true, null, "<input />")]
[InlineData(false, true, "<input />")]
[InlineData(false, false, "")]
[InlineData(false, null, "")]
[InlineData(true, "<input />")]
[InlineData(false, "")]
[InlineData(null, "")]
public async Task ProcessAsync_SupportsAntiForgeryIfActionIsSpecified(
bool? optionsAntiForgery,
bool? antiForgery,
string expectedPostContent)
{
// Arrange
var options = MakeOptions(new FormTagHelperOptions { GenerateAntiForgeryToken = optionsAntiForgery });
var viewContext = CreateViewContext();
var generator = new Mock<IHtmlGenerator>();
@ -348,7 +329,6 @@ namespace Microsoft.AspNet.Mvc.TagHelpers
{
AntiForgery = antiForgery,
Generator = generator.Object,
Options = options,
ViewContext = viewContext,
};
@ -421,15 +401,6 @@ namespace Microsoft.AspNet.Mvc.TagHelpers
Assert.Equal(expectedErrorMessage, ex.Message);
}
private static IOptions<TOptions> MakeOptions<TOptions>(TOptions options = null)
where TOptions : class, new()
{
var optionsAccessor = new Mock<IOptions<TOptions>>();
optionsAccessor.Setup(o => o.Options).Returns(options ?? new TOptions());
return optionsAccessor.Object;
}
private static ViewContext CreateViewContext()
{
var actionContext = new ActionContext(

View File

@ -1,52 +0,0 @@
// Copyright (c) Microsoft Open Technologies, Inc. All rights reserved.
// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
using System.Collections.Generic;
using Microsoft.Framework.ConfigurationModel;
using Microsoft.Framework.DependencyInjection;
using Microsoft.Framework.OptionsModel;
using Xunit;
namespace Microsoft.AspNet.Mvc.TagHelpers.Test
{
public class TagHelperOptionsCollectionExtensionsTest
{
public static TheoryData ConfigureForm_GetsOptionsFromConfigurationCorrectly_Data
{
get
{
return new TheoryData<string, bool?>
{
{ "true", true },
{ "false", false },
{ "True", true },
{ "False", false },
{ "TRue", true },
{ "FAlse", false },
{ null, null }
};
}
}
[Theory]
[MemberData(nameof(ConfigureForm_GetsOptionsFromConfigurationCorrectly_Data))]
public void ConfigureForm_GetsOptionsFromConfigurationCorrectly(string configValue, bool? expectedValue)
{
// Arrange
var configValues = new Dictionary<string, string>
{
{ $"{nameof(FormTagHelperOptions.GenerateAntiForgeryToken)}", configValue }
};
var config = new Configuration(new MemoryConfigurationSource(configValues));
var services = new ServiceCollection().AddOptions();
services.ConfigureTagHelpers().ConfigureForm(config);
var serviceProvider = services.BuildServiceProvider();
// Act
var options = serviceProvider.GetService<IOptions<FormTagHelperOptions>>().Options;
// Assert
Assert.Equal(expectedValue, options.GenerateAntiForgeryToken);
}
}
}