diff --git a/src/GlobalSuppressions.cs b/src/GlobalSuppressions.cs index f4741ddc0..46a66f80e 100644 --- a/src/GlobalSuppressions.cs +++ b/src/GlobalSuppressions.cs @@ -4,6 +4,7 @@ using System.Diagnostics.CodeAnalysis; #region Permanent Exclusions +[assembly: SuppressMessage("Microsoft.Design", "CA1000:DoNotDeclareStaticMembersOnGenericTypes", Scope = "member", Target = "Microsoft.Restier.Providers.EntityFramework.EntityFrameworkApi`1.#ConfigureApi(System.Type,Microsoft.Extensions.DependencyInjection.IServiceCollection)")] [assembly: SuppressMessage("Microsoft.Design", "CA1011:ConsiderPassingBaseTypesAsParameters", Scope = "member", Target = "Microsoft.Restier.Core.Query.QueryRequest.#Create`2(System.Linq.IQueryable`1,System.Linq.Expressions.Expression`1,!!1>>,System.Nullable`1)")] [assembly: SuppressMessage("Microsoft.Design", "CA1032:ImplementStandardExceptionConstructors", Scope = "type", Target = "Microsoft.Restier.Core.Submit.ChangeSetValidationException", Justification = "We do not intend to support serialization of this exception yet")] [assembly: SuppressMessage("Microsoft.Design", "CA1061:DoNotHideBaseClassMethods", Scope = "member", Target = "Microsoft.Restier.Publishers.OData.Batch.RestierBatchChangeSetRequestItem.#DisposeResponses(System.Collections.Generic.IEnumerable`1)")] diff --git a/src/Microsoft.Restier.Core/ApiBase.cs b/src/Microsoft.Restier.Core/ApiBase.cs index b92a446f2..535bdc4b1 100644 --- a/src/Microsoft.Restier.Core/ApiBase.cs +++ b/src/Microsoft.Restier.Core/ApiBase.cs @@ -78,48 +78,23 @@ public ApiConfiguration Configuration set { this.apiConfiguration = value; - bool isSuccess = Configurations.TryAdd(GetType(), apiConfiguration); - if (isSuccess) - { - UpdateApiConfiguration(this.apiConfiguration); - } - } - } - - /// - /// Performs application-defined tasks associated with - /// freeing, releasing, or resetting unmanaged resources. - /// - public void Dispose() - { - if (this.IsDisposed) - { - return; - } - - this.IsDisposed = true; - - if (this.apiContext != null) - { - this.apiContext.DisposeScope(); - this.apiContext = null; + Configurations.TryAdd(GetType(), apiConfiguration); } - - GC.SuppressFinalize(this); } /// /// Configure services for this API. /// + /// + /// The Api type. + /// /// - /// The with which to create an . + /// The with which is used to store all services. /// /// The . [CLSCompliant(false)] - public virtual IServiceCollection ConfigureApi(IServiceCollection services) + public static IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { - Type apiType = this.GetType(); - // Add core and convention's services services = services.AddCoreServices(apiType) .AddConventionBasedServices(apiType); @@ -131,13 +106,25 @@ public virtual IServiceCollection ConfigureApi(IServiceCollection services) } /// - /// Allow user to update the ApiConfiguration - /// . + /// Performs application-defined tasks associated with + /// freeing, releasing, or resetting unmanaged resources. /// - /// The for the Api instance. - [CLSCompliant(false)] - protected virtual void UpdateApiConfiguration(ApiConfiguration configuration) + public void Dispose() { + if (this.IsDisposed) + { + return; + } + + this.IsDisposed = true; + + if (this.apiContext != null) + { + this.apiContext.DisposeScope(); + this.apiContext = null; + } + + GC.SuppressFinalize(this); } /// diff --git a/src/Microsoft.Restier.Core/ApiConfiguration.cs b/src/Microsoft.Restier.Core/ApiConfiguration.cs index 681816838..fad124c06 100644 --- a/src/Microsoft.Restier.Core/ApiConfiguration.cs +++ b/src/Microsoft.Restier.Core/ApiConfiguration.cs @@ -56,19 +56,20 @@ internal IServiceProvider ServiceProvider internal IEdmModel Model { get; private set; } /// - /// Adds a configuration procedure for API type . + /// Adds a configuration procedure for apiType. /// This is expected to be called by publisher like WebApi to add services. /// - /// The API type. + /// + /// The Api Type. + /// /// - /// An action that will be called during the configuration of . + /// An action that will be called during the configuration of apiType. /// [CLSCompliant(false)] - public static void AddPublisherServices(Action configurationCallback) - where TApi : ApiBase + public static void AddPublisherServices(Type apiType, Action configurationCallback) { publisherServicesCallback.AddOrUpdate( - typeof(TApi), + apiType, configurationCallback, (type, existing) => existing + configurationCallback); } diff --git a/src/Microsoft.Restier.Core/RestierContainerBuilder.cs b/src/Microsoft.Restier.Core/RestierContainerBuilder.cs index f913606d2..4160fc0ae 100644 --- a/src/Microsoft.Restier.Core/RestierContainerBuilder.cs +++ b/src/Microsoft.Restier.Core/RestierContainerBuilder.cs @@ -3,6 +3,7 @@ using System; using System.Globalization; +using System.Reflection; using System.Threading; using Microsoft.Extensions.DependencyInjection; using Microsoft.OData; @@ -17,15 +18,15 @@ namespace Microsoft.Restier.Core public class RestierContainerBuilder : IContainerBuilder { private readonly IServiceCollection services = new ServiceCollection(); - private Func apiFactory; + private Type apiType; /// /// Initializes a new instance of the class. /// - /// The Api factory to create the Api instance - public RestierContainerBuilder(Func apiFactory) + /// The Api Type + public RestierContainerBuilder(Type apiType) { - this.apiFactory = apiFactory; + this.apiType = apiType; } /// @@ -103,24 +104,29 @@ internal IContainerBuilder AddRestierService() { Func modelFactory = sp => { - using (var api = apiFactory()) - { - var configuation = sp.GetService(); - if (api.Configuration == null) - { - api.Configuration = configuation; - } - - var model = api.Context.GetModelAsync(default(CancellationToken)).Result; - return model; - } + var context = sp.GetService(); + var model = context.GetModelAsync(default(CancellationToken)).Result; + return model; }; - using (var api = apiFactory()) + // Configure the API via reflection call + var methodDeclaredType = apiType; + + MethodInfo method = null; + while (method == null && methodDeclaredType != null) { - api.ConfigureApi(services); + // In case the subclass does not override the method, call super class method + method = methodDeclaredType.GetMethod("ConfigureApi"); + methodDeclaredType = methodDeclaredType.BaseType; } + var parameters = new object[] + { + apiType, services + }; + + method.Invoke(null, parameters); + services.AddSingleton(modelFactory); return this; } diff --git a/src/Microsoft.Restier.Core/ServiceCollectionExtensions.cs b/src/Microsoft.Restier.Core/ServiceCollectionExtensions.cs index 452699136..3e498df92 100644 --- a/src/Microsoft.Restier.Core/ServiceCollectionExtensions.cs +++ b/src/Microsoft.Restier.Core/ServiceCollectionExtensions.cs @@ -211,13 +211,11 @@ public static IServiceCollection MakeTransient(this IServiceCollection /// Current public static IServiceCollection AddCoreServices(this IServiceCollection services, Type apiType) { - if (!services.HasService()) - { - services.AddScoped() - .AddScoped(apiType, sp => sp.GetService().Api) - .AddScoped(sp => sp.GetService().Api) - .AddScoped(sp => sp.GetService().Api.Context); - } + Ensure.NotNull(apiType, "apiType"); + + services.AddScoped(apiType, apiType) + .AddScoped(typeof(ApiBase), apiType) + .AddScoped(); services.TryAddSingleton(); diff --git a/src/Microsoft.Restier.Providers.EntityFramework/EntityFrameworkApi.cs b/src/Microsoft.Restier.Providers.EntityFramework/EntityFrameworkApi.cs index 48294fc27..46456252c 100644 --- a/src/Microsoft.Restier.Providers.EntityFramework/EntityFrameworkApi.cs +++ b/src/Microsoft.Restier.Providers.EntityFramework/EntityFrameworkApi.cs @@ -45,6 +45,9 @@ protected T DbContext /// Configures the API services for this API. Descendants may override this method to register /// as a scoped service. /// + /// + /// The Api type. + /// /// /// The with which to create an . /// @@ -52,10 +55,8 @@ protected T DbContext /// The . /// [CLSCompliant(false)] - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { - Type apiType = this.GetType(); - // Add core and convention's services services = services.AddCoreServices(apiType) .AddConventionBasedServices(apiType); diff --git a/src/Microsoft.Restier.Publishers.OData/Microsoft.Restier.Publishers.OData.csproj b/src/Microsoft.Restier.Publishers.OData/Microsoft.Restier.Publishers.OData.csproj index 2378dbd87..9aed1bfdf 100644 --- a/src/Microsoft.Restier.Publishers.OData/Microsoft.Restier.Publishers.OData.csproj +++ b/src/Microsoft.Restier.Publishers.OData/Microsoft.Restier.Publishers.OData.csproj @@ -86,8 +86,8 @@ - + diff --git a/src/Microsoft.Restier.Publishers.OData/Model/ApiConfigurationExtensions.cs b/src/Microsoft.Restier.Publishers.OData/Model/ApiConfigurationExtensions.cs deleted file mode 100644 index 56bf26d3e..000000000 --- a/src/Microsoft.Restier.Publishers.OData/Model/ApiConfigurationExtensions.cs +++ /dev/null @@ -1,65 +0,0 @@ -// Copyright (c) Microsoft Corporation. All rights reserved. -// Licensed under the MIT License. See License.txt in the project root for license information. - -using System; -using System.Collections.Generic; -using Microsoft.Restier.Core; - -namespace Microsoft.Restier.Publishers.OData.Model -{ - /// - /// Offers a collection of extension methods to . - /// - public static class ApiConfigurationExtensions - { - private const string IgnoredPropertiesKey = "Microsoft.Restier.Publishers.OData.IgnoredProperties"; - - #region IgnoreProperty - - /// - /// Ignores the given property in ApiBase or sub-class when building the model. - /// - /// An API configuration. - /// The name of the property to be ignored. - /// The current API configuration instance. - public static ApiConfiguration IgnoreProperty( - this ApiConfiguration configuration, - string propertyName) - { - Ensure.NotNull(configuration, "configuration"); - Ensure.NotNull(propertyName, "propertyName"); - - configuration.GetIgnoredPropertiesImplementation().Add(propertyName); - return configuration; - } - - #endregion - - #region IgnoreProperty Internal - - internal static bool IsPropertyIgnored(this ApiConfiguration configuration, string propertyName) - { - Ensure.NotNull(configuration, "configuration"); - - return configuration.GetIgnoredPropertiesImplementation().Contains(propertyName); - } - - #endregion - - #region IgnoreProperty Private - - private static ICollection GetIgnoredPropertiesImplementation(this ApiConfiguration configuration) - { - var ignoredProperties = configuration.GetProperty>(IgnoredPropertiesKey); - if (ignoredProperties == null) - { - ignoredProperties = new HashSet(); - configuration.SetProperty(IgnoredPropertiesKey, ignoredProperties); - } - - return ignoredProperties; - } - - #endregion - } -} \ No newline at end of file diff --git a/src/Microsoft.Restier.Publishers.OData/Model/ResourceAttribute.cs b/src/Microsoft.Restier.Publishers.OData/Model/ResourceAttribute.cs new file mode 100644 index 000000000..5c79ce966 --- /dev/null +++ b/src/Microsoft.Restier.Publishers.OData/Model/ResourceAttribute.cs @@ -0,0 +1,21 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. See License.txt in the project root for license information. + +using System; + +namespace Microsoft.Restier.Publishers.OData.Model +{ + /// + /// Attribute that indicates a property is an entity set or singleton. + /// The name will be same as property name. + /// + [AttributeUsage(AttributeTargets.Property)] + public sealed class ResourceAttribute : Attribute + { + /// + /// Gets or sets a value indicating whether it is singleton or entity set. + /// The default value is false means it is an entity set + /// + public bool IsSingleton { get; set; } + } +} diff --git a/src/Microsoft.Restier.Publishers.OData/Model/RestierModelExtender.cs b/src/Microsoft.Restier.Publishers.OData/Model/RestierModelExtender.cs index df0d43658..f689458b8 100644 --- a/src/Microsoft.Restier.Publishers.OData/Model/RestierModelExtender.cs +++ b/src/Microsoft.Restier.Publishers.OData/Model/RestierModelExtender.cs @@ -176,27 +176,27 @@ private void ScanForDeclaredPublicProperties() } } - private void BuildEntitySetsAndSingletons(ModelContext context, EdmModel model) + private void BuildEntitySetsAndSingletons(EdmModel model) { - var configuration = context.ServiceProvider.GetService(); foreach (var property in this.publicProperties) { - if (configuration.IsPropertyIgnored(property.Name)) + var resourceAttribute = property.GetCustomAttributes(true).FirstOrDefault(); + if (resourceAttribute == null) { continue; } - var isEntitySet = IsEntitySetProperty(property); - if (!isEntitySet) + bool isSingleton = resourceAttribute.IsSingleton; + if ((!isSingleton && !IsEntitySetProperty(property)) + || (isSingleton && !IsSingletonProperty(property))) { - if (!IsSingletonProperty(property)) - { - continue; - } + // This means property type is not IQueryable when indicating an entityset + // or not non-generic type when indicating a singleton + continue; } var propertyType = property.PropertyType; - if (isEntitySet) + if (!isSingleton) { propertyType = propertyType.GetGenericArguments()[0]; } @@ -209,7 +209,7 @@ private void BuildEntitySetsAndSingletons(ModelContext context, EdmModel model) } var container = model.EnsureEntityContainer(this.targetType); - if (isEntitySet) + if (!isSingleton) { if (container.FindEntitySet(property.Name) == null) { @@ -341,7 +341,7 @@ public async Task GetModelAsync(ModelContext context, CancellationTok } ModelCache.ScanForDeclaredPublicProperties(); - ModelCache.BuildEntitySetsAndSingletons(context, edmModel); + ModelCache.BuildEntitySetsAndSingletons(edmModel); ModelCache.AddNavigationPropertyBindings(edmModel); return edmModel; } diff --git a/src/Microsoft.Restier.Publishers.OData/Routing/HttpConfigurationExtensions.cs b/src/Microsoft.Restier.Publishers.OData/Routing/HttpConfigurationExtensions.cs index 48b72f1b7..8017a127a 100644 --- a/src/Microsoft.Restier.Publishers.OData/Routing/HttpConfigurationExtensions.cs +++ b/src/Microsoft.Restier.Publishers.OData/Routing/HttpConfigurationExtensions.cs @@ -53,14 +53,16 @@ public static Task MapRestierRoute( // This will be added a service to callback stored in ApiConfiguration // Callback is called by ApiBase.AddApiServices method to add real services. - ApiConfiguration.AddPublisherServices(services => + ApiConfiguration.AddPublisherServices( + typeof(TApi), + services => { services.AddODataServices(); }); using (var api = apiFactory()) { - Func func = () => new RestierContainerBuilder(apiFactory); + Func func = () => new RestierContainerBuilder(api.GetType()); config.UseCustomContainerBuilder(func); var conventions = CreateRestierRoutingConventions(config, routeName, apiFactory); diff --git a/src/Microsoft.Restier.Publishers.OData/Routing/RestierRoutingConvention.cs b/src/Microsoft.Restier.Publishers.OData/Routing/RestierRoutingConvention.cs index dfaeb5237..dc2fcb005 100644 --- a/src/Microsoft.Restier.Publishers.OData/Routing/RestierRoutingConvention.cs +++ b/src/Microsoft.Restier.Publishers.OData/Routing/RestierRoutingConvention.cs @@ -7,6 +7,7 @@ using System.Web.Http; using System.Web.Http.Controllers; using System.Web.Http.Routing; +using System.Web.OData.Extensions; using System.Web.OData.Routing.Conventions; using Microsoft.OData.Edm; using Microsoft.OData.UriParser; @@ -67,6 +68,8 @@ public string SelectController(ODataPath odataPath, HttpRequestMessage request) } // Create ApiBase instance + // TODO need to change the way to create ApiBase + // var provider = request.GetRequestContainer(); request.SetApiInstance(apiFactory.Invoke()); return RestierControllerName; } diff --git a/test/Microsoft.Restier.Core.Tests/Api.Tests.cs b/test/Microsoft.Restier.Core.Tests/Api.Tests.cs index ed0d375de..ef638af4e 100644 --- a/test/Microsoft.Restier.Core.Tests/Api.Tests.cs +++ b/test/Microsoft.Restier.Core.Tests/Api.Tests.cs @@ -5,6 +5,7 @@ using System.Collections; using System.Linq; using System.Linq.Expressions; +using System.Net; using System.Threading; using System.Threading.Tasks; using Microsoft.Extensions.DependencyInjection; @@ -79,7 +80,7 @@ public Task ExecuteSubmitAsync(SubmitContext context, Cancellation private class TestApi : ApiBase { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public new static IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { var modelBuilder = new TestModelBuilder(); var modelMapper = new TestModelMapper(); @@ -87,7 +88,7 @@ public override IServiceCollection ConfigureApi(IServiceCollection services) var changeSetPreparer = new TestChangeSetInitializer(); var submitExecutor = new TestSubmitExecutor(); - services.AddCoreServices(this.GetType()); + services.AddCoreServices(apiType); services.AddService((sp, next) => modelBuilder); services.AddService((sp, next) => modelMapper); services.AddService((sp, next) => querySourcer); @@ -106,7 +107,7 @@ private class TestApiEmpty : ApiBase public void ApiSourceOfEntityContainerElementIsCorrect() { var api = new TestApi(); - var container = new RestierContainerBuilder(() => new TestApi()); + var container = new RestierContainerBuilder(typeof(TestApi)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var arguments = new object[0]; @@ -131,7 +132,7 @@ public void ApiSourceOfEntityContainerElementIsCorrect() public void SourceOfEntityContainerElementThrowsIfNotMapped() { var api = new TestApiEmpty(); - var container = new RestierContainerBuilder(() => new TestApiEmpty()); + var container = new RestierContainerBuilder(typeof(TestApiEmpty)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var context = api.Context; var arguments = new object[0]; @@ -166,7 +167,7 @@ public void SourceOfEntityContainerElementIsCorrect() public void ApiSourceOfComposableFunctionIsCorrect() { var api = new TestApi(); - var container = new RestierContainerBuilder(() => new TestApi()); + var container = new RestierContainerBuilder(typeof(TestApi)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var arguments = new object[0]; @@ -193,7 +194,7 @@ public void ApiSourceOfComposableFunctionIsCorrect() public void SourceOfComposableFunctionThrowsIfNotMapped() { var api = new TestApiEmpty(); - var container = new RestierContainerBuilder(() => new TestApiEmpty()); + var container = new RestierContainerBuilder(typeof(TestApiEmpty)); var serviceProvider = container.BuildContainer(); api.Configuration = serviceProvider.GetService(); var context = api.Context; @@ -206,7 +207,7 @@ public void SourceOfComposableFunctionThrowsIfNotMapped() public void SourceOfComposableFunctionIsCorrect() { var api = new TestApi(); - var container = new RestierContainerBuilder(() => new TestApi()); + var container = new RestierContainerBuilder(typeof(TestApi)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var context = api.Context; @@ -266,7 +267,7 @@ public void GenericSourceOfEntityContainerElementThrowsIfWrongType() public void GenericSourceOfEntityContainerElementIsCorrect() { var api = new TestApi(); - var container = new RestierContainerBuilder(() => new TestApi()); + var container = new RestierContainerBuilder(typeof(TestApi)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var context = api.Context; @@ -317,7 +318,7 @@ public void GenericApiSourceOfComposableFunctionIsCorrect() public void GenericSourceOfComposableFunctionThrowsIfWrongType() { var api = new TestApi(); - var container = new RestierContainerBuilder(() => new TestApi()); + var container = new RestierContainerBuilder(typeof(TestApi)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var context = api.Context; @@ -385,7 +386,7 @@ public void SourceQueryProviderCannotGenericExecute() public void SourceQueryProviderCannotExecute() { var api = new TestApi(); - var container = new RestierContainerBuilder(() => new TestApi()); + var container = new RestierContainerBuilder(typeof(TestApi)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var context = api.Context; @@ -397,7 +398,7 @@ public void SourceQueryProviderCannotExecute() public async Task ApiQueryAsyncWithQueryReturnsResults() { var api = new TestApi(); - var container = new RestierContainerBuilder(() => new TestApi()); + var container = new RestierContainerBuilder(typeof(TestApi)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var request = new QueryRequest(api.GetQueryableSource("Test")); @@ -411,7 +412,7 @@ public async Task ApiQueryAsyncWithQueryReturnsResults() public async Task ApiQueryAsyncCorrectlyForwardsCall() { var api = new TestApi(); - var container = new RestierContainerBuilder(() => new TestApi()); + var container = new RestierContainerBuilder(typeof(TestApi)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var queryRequest = new QueryRequest( diff --git a/test/Microsoft.Restier.Core.Tests/ApiBase.Tests.cs b/test/Microsoft.Restier.Core.Tests/ApiBase.Tests.cs index 0827a48cc..53c5be1ce 100644 --- a/test/Microsoft.Restier.Core.Tests/ApiBase.Tests.cs +++ b/test/Microsoft.Restier.Core.Tests/ApiBase.Tests.cs @@ -11,9 +11,9 @@ public class ApiBaseTests { private class TestApi : ApiBase { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { - return base.ConfigureApi(services) + return ApiBase.ConfigureApi(apiType, services) .MakeScoped() .AddService(); } @@ -53,7 +53,7 @@ public void ApiAndApiContextCanBeInjectedByDI() { using (var api = new TestApi()) { - var container = new RestierContainerBuilder(() => new TestApi()); + var container = new RestierContainerBuilder(typeof(TestApi)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var context = api.Context; diff --git a/test/Microsoft.Restier.Core.Tests/ApiConfiguration.Tests.cs b/test/Microsoft.Restier.Core.Tests/ApiConfiguration.Tests.cs index 57d7a7a12..eca3ff2fe 100644 --- a/test/Microsoft.Restier.Core.Tests/ApiConfiguration.Tests.cs +++ b/test/Microsoft.Restier.Core.Tests/ApiConfiguration.Tests.cs @@ -18,7 +18,7 @@ public class ApiConfigurationTests public void CachedConfigurationIsCachedCorrectly() { ApiBase api = new TestApiA(); - var container = new RestierContainerBuilder(() => new TestApiA()); + var container = new RestierContainerBuilder(typeof(TestApiA)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var configuration = api.Context.Configuration; @@ -32,18 +32,17 @@ public void CachedConfigurationIsCachedCorrectly() public void ConfigurationRegistersApiServicesCorrectly() { var api = new TestApiA(); - var container = new RestierContainerBuilder(() => new TestApiA()); + var container = new RestierContainerBuilder(typeof(TestApiA)); api.Configuration = new ApiConfiguration(container.BuildContainer()); Assert.Null(api.Context.GetApiService()); Assert.Null(api.Context.GetApiService()); var apiB = new TestApiB(); - container = new RestierContainerBuilder(() => new TestApiB()); + container = new RestierContainerBuilder(typeof(TestApiB)); apiB.Configuration = new ApiConfiguration(container.BuildContainer()); - // This is not same as during configure Api, a new APi is created which has new Service A registered. - Assert.NotSame(apiB.serviceA, apiB.Context.GetApiService()); + Assert.Same(TestApiB.serviceA, apiB.Context.GetApiService()); var serviceBInstance = apiB.Context.GetApiService(); var serviceBInterface = apiB.Context.GetApiService(); @@ -55,15 +54,14 @@ public void ConfigurationRegistersApiServicesCorrectly() var serviceBFirst = serviceBInterface as ServiceB; Assert.NotNull(serviceBFirst); - // This is not same as during configure Api, a new APi is created which has new Service B registered. - Assert.NotSame(apiB.serviceB, serviceBFirst.InnerHandler); + Assert.Same(TestApiB.serviceB, serviceBFirst.InnerHandler); } [Fact] public void ServiceChainTest() { var api = new TestApiC(); - var container = new RestierContainerBuilder(() => new TestApiC()); + var container = new RestierContainerBuilder(typeof(TestApiC)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var handler = api.Context.GetApiService(); @@ -76,11 +74,11 @@ private class TestApiA : ApiBase private class TestApiB : ApiBase { - private ServiceA _serviceA; + private static ServiceA _serviceA; - private ServiceB _serviceB; + private static ServiceB _serviceB; - public ServiceA serviceA + public static ServiceA serviceA { get { @@ -92,7 +90,7 @@ public ServiceA serviceA } } - public ServiceB serviceB + public static ServiceB serviceB { get { @@ -104,7 +102,7 @@ public ServiceB serviceB } } - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { services.AddService((sp, next) => serviceA); services.AddService((sp, next) => serviceB); @@ -116,7 +114,7 @@ public override IServiceCollection ConfigureApi(IServiceCollection services) } private class TestApiC : ApiBase { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { var q1 = new ServiceB("q1Pre", "q1Post"); var q2 = new ServiceB("q2Pre", "q2Post"); diff --git a/test/Microsoft.Restier.Core.Tests/ApiContext.Tests.cs b/test/Microsoft.Restier.Core.Tests/ApiContext.Tests.cs index ab8412b86..d43bfb008 100644 --- a/test/Microsoft.Restier.Core.Tests/ApiContext.Tests.cs +++ b/test/Microsoft.Restier.Core.Tests/ApiContext.Tests.cs @@ -17,7 +17,7 @@ private class TestApi : ApiBase public void NewApiContextIsConfiguredCorrectly() { var api = new TestApi(); - var container = new RestierContainerBuilder(() => new TestApi()); + var container = new RestierContainerBuilder(typeof(TestApi)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var context = api.Context; diff --git a/test/Microsoft.Restier.Core.Tests/InvocationContext.Tests.cs b/test/Microsoft.Restier.Core.Tests/InvocationContext.Tests.cs index 9562c251a..a4108eeda 100644 --- a/test/Microsoft.Restier.Core.Tests/InvocationContext.Tests.cs +++ b/test/Microsoft.Restier.Core.Tests/InvocationContext.Tests.cs @@ -1,6 +1,7 @@ // Copyright (c) Microsoft Corporation. All rights reserved. // Licensed under the MIT License. See License.txt in the project root for license information. +using System; using Microsoft.Extensions.DependencyInjection; using Xunit; @@ -12,7 +13,7 @@ private class TestApi : ApiBase { private static ApiServiceA _service; - public ApiServiceA ApiService + public static ApiServiceA ApiService { get { @@ -23,7 +24,8 @@ public ApiServiceA ApiService return _service; } } - public override IServiceCollection ConfigureApi(IServiceCollection services) + + public new static IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { services.AddService((sp, next) => ApiService); @@ -35,7 +37,7 @@ public override IServiceCollection ConfigureApi(IServiceCollection services) public void NewInvocationContextIsConfiguredCorrectly() { var api = new TestApi(); - var container = new RestierContainerBuilder(() => new TestApi()); + var container = new RestierContainerBuilder(typeof(TestApi)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var apiContext = api.Context; var context = new InvocationContext(apiContext); @@ -46,11 +48,11 @@ public void NewInvocationContextIsConfiguredCorrectly() public void InvocationContextGetsApiServicesCorrectly() { var api = new TestApi(); - var container = new RestierContainerBuilder(() => new TestApi()); + var container = new RestierContainerBuilder(typeof(TestApi)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var apiContext = api.Context; var context = new InvocationContext(apiContext); - Assert.Same(api.ApiService, context.GetApiService()); + Assert.Same(TestApi.ApiService, context.GetApiService()); } private interface IServiceA diff --git a/test/Microsoft.Restier.Core.Tests/Model/DefaultModelHandler.Tests.cs b/test/Microsoft.Restier.Core.Tests/Model/DefaultModelHandler.Tests.cs index 3d64b3374..e4538ae9e 100644 --- a/test/Microsoft.Restier.Core.Tests/Model/DefaultModelHandler.Tests.cs +++ b/test/Microsoft.Restier.Core.Tests/Model/DefaultModelHandler.Tests.cs @@ -16,7 +16,7 @@ public class DefaultModelHandlerTests { private class TestApiA : ApiBase { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { services.AddService((sp, next) => new TestModelProducer()); services.AddService((sp, next) => new TestModelExtender(2) @@ -33,7 +33,7 @@ public override IServiceCollection ConfigureApi(IServiceCollection services) private class TestApiB : ApiBase { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { var service = new TestSingleCallModelBuilder(); services.AddService((sp, next) => service); @@ -43,7 +43,7 @@ public override IServiceCollection ConfigureApi(IServiceCollection services) private class TestApiC : ApiBase { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { var service = new TestRetryModelBuilder(); services.AddService((sp, next) => service); @@ -105,7 +105,7 @@ public async Task GetModelAsync(ModelContext context, CancellationTok public async Task GetModelUsingDefaultModelHandler() { var api = new TestApiA(); - var container = new RestierContainerBuilder(() => new TestApiA()); + var container = new RestierContainerBuilder(typeof(TestApiA)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var context = api.Context; @@ -142,7 +142,7 @@ public async Task GetModelAsync(ModelContext context, CancellationTok private static Task[] PrepareThreads(int count, Type apiType, ManualResetEventSlim wait) { var api2 = (ApiBase)Activator.CreateInstance(apiType); - var container = new RestierContainerBuilder(() => (ApiBase)Activator.CreateInstance(apiType)); + var container = new RestierContainerBuilder(apiType); api2.Configuration = new ApiConfiguration(container.BuildContainer()); var tasks = new Task[count]; diff --git a/test/Microsoft.Restier.Core.Tests/PropertyBag.Tests.cs b/test/Microsoft.Restier.Core.Tests/PropertyBag.Tests.cs index b5c4cfe9e..40399a948 100644 --- a/test/Microsoft.Restier.Core.Tests/PropertyBag.Tests.cs +++ b/test/Microsoft.Restier.Core.Tests/PropertyBag.Tests.cs @@ -13,7 +13,7 @@ public class PropertyBagTests public void PropertyBagManipulatesPropertiesCorrectly() { var api = new TestApi(); - var container = new RestierContainerBuilder(() => new TestApi()); + var container = new RestierContainerBuilder(typeof(TestApi)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var context =api.Context; @@ -38,7 +38,7 @@ public void PropertyBagManipulatesPropertiesCorrectly() public void DifferentPropertyBagsDoNotConflict() { var api = new TestApi(); - var container = new RestierContainerBuilder(() => new TestApi()); + var container = new RestierContainerBuilder(typeof(TestApi)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var context = api.Context; @@ -55,7 +55,7 @@ public void DifferentPropertyBagsDoNotConflict() public void PropertyBagsAreDisposedCorrectly() { var api = new TestApi(); - var container = new RestierContainerBuilder(() => new TestApi()); + var container = new RestierContainerBuilder(typeof(TestApi)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var context = api.Context; var configuration = context.Configuration; @@ -96,9 +96,9 @@ public void Dispose() private class TestApi : ApiBase { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { - return base.ConfigureApi(services).AddScoped(); + return ApiBase.ConfigureApi(apiType, services).AddScoped(); } } } diff --git a/test/Microsoft.Restier.Core.Tests/ServiceConfiguration.Tests.cs b/test/Microsoft.Restier.Core.Tests/ServiceConfiguration.Tests.cs index ea7cbbb82..b1eb580f9 100644 --- a/test/Microsoft.Restier.Core.Tests/ServiceConfiguration.Tests.cs +++ b/test/Microsoft.Restier.Core.Tests/ServiceConfiguration.Tests.cs @@ -11,7 +11,7 @@ public class ServiceConfigurationTests { private class TestApiA : ApiBase { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { var i = 0; services.AddService((sp, next) => new SomeService @@ -42,7 +42,7 @@ public override IServiceCollection ConfigureApi(IServiceCollection services) private class TestApiB : ApiBase { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { services.AddService((sp, next) => new SomeService { @@ -58,7 +58,7 @@ public override IServiceCollection ConfigureApi(IServiceCollection services) private class TestApiC : ApiBase { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { services.MakeScoped() .AddService((sp, next) => new SomeService()); @@ -68,7 +68,7 @@ public override IServiceCollection ConfigureApi(IServiceCollection services) private class TestApiD : ApiBase { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { services.AddService((sp, next) => new SomeService { @@ -83,7 +83,7 @@ public override IServiceCollection ConfigureApi(IServiceCollection services) private class TestApiE : ApiBase { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { var first = new SomeService { @@ -100,7 +100,7 @@ public override IServiceCollection ConfigureApi(IServiceCollection services) private class TestApiF : ApiBase { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { services.AddService((sp, next) => new SomeService { @@ -115,7 +115,7 @@ public override IServiceCollection ConfigureApi(IServiceCollection services) private class TestApiG : ApiBase { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { services.AddService((sp, next) => new SomeService { @@ -136,7 +136,7 @@ public override IServiceCollection ConfigureApi(IServiceCollection services) private class TestApiH : ApiBase { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { services.AddService((sp, next) => new SomeService { @@ -153,7 +153,7 @@ public override IServiceCollection ConfigureApi(IServiceCollection services) private class TestApiI : ApiBase { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { services.MakeTransient() .AddService((sp, next) => new SomeService @@ -297,7 +297,7 @@ public string Call() public void ContributorsAreCalledCorrectly() { var api = new TestApiA(); - var container = new RestierContainerBuilder(() => new TestApiA()); + var container = new RestierContainerBuilder(typeof(TestApiA)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var value = api.Context.GetApiService().Call(); Assert.Equal("03210", value); @@ -307,7 +307,7 @@ public void ContributorsAreCalledCorrectly() public void NextInjectedViaProperty() { var api = new TestApiB(); - var container = new RestierContainerBuilder(() => new TestApiB()); + var container = new RestierContainerBuilder(typeof(TestApiB)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var value = api.Context.GetApiService().Call(); Assert.Equal("01", value); @@ -321,13 +321,13 @@ public void NextInjectedViaProperty() public void ContextApiScopeWorksCorrectly() { var api = new TestApiC(); - var container = new RestierContainerBuilder(() => new TestApiC()); + var container = new RestierContainerBuilder(typeof(TestApiC)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var service1 = api.Context.GetApiService(); var api2 = new TestApiC(); - container = new RestierContainerBuilder(() => new TestApiC()); + container = new RestierContainerBuilder(typeof(TestApiC)); api2.Configuration = new ApiConfiguration(container.BuildContainer()); var service2 = api2.Context.GetApiService(); @@ -345,7 +345,7 @@ public void NothingInjectedStillWorks() { // Outmost service does not call inner service var api = new TestApiD(); - var container = new RestierContainerBuilder(() => new TestApiD()); + var container = new RestierContainerBuilder(typeof(TestApiD)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var value = api.Context.GetApiService().Call(); @@ -362,7 +362,7 @@ public void NothingInjectedStillWorks() public void ServiceInjectedViaProperty() { var api = new TestApiE(); - var container = new RestierContainerBuilder(() => new TestApiE()); + var container = new RestierContainerBuilder(typeof(TestApiE)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var expected = "Text42"; @@ -384,7 +384,7 @@ public void ServiceInjectedViaProperty() public void DefaultValueInConstructorUsedIfNoService() { var api = new TestApiF(); - var container = new RestierContainerBuilder(() => new TestApiF()); + var container = new RestierContainerBuilder(typeof(TestApiF)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var value = api.Context.GetApiService().Call(); @@ -401,7 +401,7 @@ public void DefaultValueInConstructorUsedIfNoService() public void MultiInjectionViaConstructor() { var api = new TestApiG(); - var container = new RestierContainerBuilder(() => new TestApiG()); + var container = new RestierContainerBuilder(typeof(TestApiG)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var value = api.Context.GetApiService().Call(); @@ -418,7 +418,7 @@ public void MultiInjectionViaConstructor() public void ThrowOnNoServiceFound() { var api = new TestApiH(); - var container = new RestierContainerBuilder(() => new TestApiH()); + var container = new RestierContainerBuilder(typeof(TestApiH)); api.Configuration = new ApiConfiguration(container.BuildContainer()); Assert.Throws(() => { api.Context.GetApiService(); }); @@ -428,7 +428,7 @@ public void ThrowOnNoServiceFound() public void NextInjectedWithInheritedField() { var api = new TestApiI(); - var container = new RestierContainerBuilder(() => new TestApiI()); + var container = new RestierContainerBuilder(typeof(TestApiI)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var value = api.Context.GetApiService().Call(); diff --git a/test/Microsoft.Restier.Providers.EntityFramework.Tests/ChangeSetPreparerTests.cs b/test/Microsoft.Restier.Providers.EntityFramework.Tests/ChangeSetPreparerTests.cs index 958cba164..015343479 100644 --- a/test/Microsoft.Restier.Providers.EntityFramework.Tests/ChangeSetPreparerTests.cs +++ b/test/Microsoft.Restier.Providers.EntityFramework.Tests/ChangeSetPreparerTests.cs @@ -19,7 +19,7 @@ public async Task ComplexTypeUpdate() { // Arrange var libraryApi = new LibraryApi(); - var container = new RestierContainerBuilder(() => new LibraryApi()); + var container = new RestierContainerBuilder(typeof(LibraryApi)); libraryApi.Configuration = new ApiConfiguration(container.BuildContainer()); var item = new DataModificationItem( diff --git a/test/Microsoft.Restier.Providers.EntityFramework.Tests/Models/Library/LibraryApi.cs b/test/Microsoft.Restier.Providers.EntityFramework.Tests/Models/Library/LibraryApi.cs index 5dc766b98..72e70edb1 100644 --- a/test/Microsoft.Restier.Providers.EntityFramework.Tests/Models/Library/LibraryApi.cs +++ b/test/Microsoft.Restier.Providers.EntityFramework.Tests/Models/Library/LibraryApi.cs @@ -1,6 +1,7 @@ // Copyright (c) Microsoft Corporation. All rights reserved. // Licensed under the MIT License. See License.txt in the project root for license information. +using System; using System.Threading; using System.Threading.Tasks; using System.Web.OData.Builder; @@ -14,9 +15,9 @@ namespace Microsoft.Restier.Providers.EntityFramework.Tests.Models.Library class LibraryApi : EntityFrameworkApi { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { - base.ConfigureApi(services); + EntityFrameworkApi.ConfigureApi(apiType, services); services.AddService(); return services; diff --git a/test/Microsoft.Restier.Publishers.OData.Test/ExceptionHandlerTests.cs b/test/Microsoft.Restier.Publishers.OData.Test/ExceptionHandlerTests.cs index 380ba1ddb..ded9f2cfe 100644 --- a/test/Microsoft.Restier.Publishers.OData.Test/ExceptionHandlerTests.cs +++ b/test/Microsoft.Restier.Publishers.OData.Test/ExceptionHandlerTests.cs @@ -1,4 +1,5 @@ -using System.Linq.Expressions; +using System; +using System.Linq.Expressions; using System.Net; using System.Net.Http; using System.Security; @@ -32,9 +33,9 @@ public async Task ShouldReturn403HandlerThrowsSecurityException() private class ExcApi : StoreApi { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { - return base.ConfigureApi(services) + return StoreApi.ConfigureApi(apiType, services) .AddService((sp, next) => new FakeSourcer()); } } diff --git a/test/Microsoft.Restier.Publishers.OData.Test/FallbackTests.cs b/test/Microsoft.Restier.Publishers.OData.Test/FallbackTests.cs index af0b224f7..a2920f820 100644 --- a/test/Microsoft.Restier.Publishers.OData.Test/FallbackTests.cs +++ b/test/Microsoft.Restier.Publishers.OData.Test/FallbackTests.cs @@ -17,6 +17,7 @@ using Microsoft.Restier.Core; using Microsoft.Restier.Core.Model; using Microsoft.Restier.Core.Query; +using Microsoft.Restier.Publishers.OData.Model; using Xunit; namespace Microsoft.Restier.Publishers.OData.Test @@ -95,15 +96,16 @@ static FallbackModel() internal class FallbackApi : ApiBase { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { services.AddService((sp, next) => new TestModelProducer(FallbackModel.Model)); services.AddService((sp, next) => new FallbackModelMapper()); services.AddService((sp, next) => new FallbackQueryExpressionSourcer()); - services = base.ConfigureApi(services); + services = ApiBase.ConfigureApi(apiType, services); return services; } + [Resource] public IQueryable PreservedOrders { get { return this.GetQueryableSource("Orders").Where(o => o.Id > 123); } diff --git a/test/Microsoft.Restier.Publishers.OData.Test/Model/LibraryApi.cs b/test/Microsoft.Restier.Publishers.OData.Test/Model/LibraryApi.cs index 45f0c64b9..13773a73d 100644 --- a/test/Microsoft.Restier.Publishers.OData.Test/Model/LibraryApi.cs +++ b/test/Microsoft.Restier.Publishers.OData.Test/Model/LibraryApi.cs @@ -1,6 +1,7 @@ // Copyright (c) Microsoft Corporation. All rights reserved. // Licensed under the MIT License. See License.txt in the project root for license information. +using System; using Microsoft.Extensions.DependencyInjection; using Microsoft.Restier.Providers.EntityFramework; @@ -9,9 +10,9 @@ namespace Microsoft.Restier.Publishers.OData.Test.Model class LibraryApi : EntityFrameworkApi { // Need to register publisher services as MapRestierRoute is not called - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { - base.ConfigureApi(services); + EntityFrameworkApi.ConfigureApi(apiType, services); services.AddODataServices(); return services; } diff --git a/test/Microsoft.Restier.Publishers.OData.Test/Model/RestierModelBuilderTests.cs b/test/Microsoft.Restier.Publishers.OData.Test/Model/RestierModelBuilderTests.cs index 36a833f9b..c7aaabfce 100644 --- a/test/Microsoft.Restier.Publishers.OData.Test/Model/RestierModelBuilderTests.cs +++ b/test/Microsoft.Restier.Publishers.OData.Test/Model/RestierModelBuilderTests.cs @@ -16,7 +16,7 @@ public class RestierModelBuilderTests public void ComplexTypeShoudWork() { var api = new LibraryApi(); - var container = new RestierContainerBuilder(() => new LibraryApi()); + var container = new RestierContainerBuilder(typeof(LibraryApi)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var model = api.Context.GetModelAsync().Result; @@ -34,7 +34,7 @@ public void ComplexTypeShoudWork() public void PrimitiveTypesShouldWork() { var api = new LibraryApi(); - var container = new RestierContainerBuilder(() => new LibraryApi()); + var container = new RestierContainerBuilder(typeof(LibraryApi)); api.Configuration = new ApiConfiguration(container.BuildContainer()); var model = api.Context.GetModelAsync().Result; diff --git a/test/Microsoft.Restier.Publishers.OData.Test/Model/RestierModelExtender.Tests.cs b/test/Microsoft.Restier.Publishers.OData.Test/Model/RestierModelExtender.Tests.cs index 3220fc92a..a44460c38 100644 --- a/test/Microsoft.Restier.Publishers.OData.Test/Model/RestierModelExtender.Tests.cs +++ b/test/Microsoft.Restier.Publishers.OData.Test/Model/RestierModelExtender.Tests.cs @@ -65,7 +65,6 @@ public async Task ApiModelBuilderShouldProduceCorrectModelForIgnoringInheritedPr Assert.DoesNotContain("ApiConfiguration", model.EntityContainer.Elements.Select(e => e.Name)); Assert.DoesNotContain("ApiContext", model.EntityContainer.Elements.Select(e => e.Name)); Assert.DoesNotContain("Invisible", model.EntityContainer.Elements.Select(e => e.Name)); - Assert.DoesNotContain("People", model.EntityContainer.Elements.Select(e => e.Name)); Assert.Equal("Customer", model.EntityContainer.FindEntitySet("Customers").EntityType().Name); Assert.Equal("Customer", model.EntityContainer.FindSingleton("Me").EntityType().Name); } @@ -179,12 +178,6 @@ public ApiConfiguration ApiConfiguration { get { return base.Configuration; } } - - protected override void UpdateApiConfiguration(ApiConfiguration config) - { - config.IgnoreProperty("ApiConfiguration") - .IgnoreProperty("ApiContext"); - } } public class EmptyApi : BaseApi @@ -198,25 +191,22 @@ public class Person public class ApiA : BaseApi { + [Resource] public IQueryable People { get; set; } + [Resource(IsSingleton = true)] public Person Me { get; set; } public IQueryable Invisible { get; set; } - protected override void UpdateApiConfiguration(ApiConfiguration config) - { - base.UpdateApiConfiguration(config); - config.IgnoreProperty("Invisible"); - } - - public override IServiceCollection ConfigureApi(IServiceCollection services) + public new static IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { services.AddService((sp, next) => new TestModelBuilder()); - return base.ConfigureApi(services); + return BaseApi.ConfigureApi(apiType, services); } } public class ApiB : ApiA { + [Resource] public IQueryable Customers { get; set; } } @@ -233,17 +223,14 @@ public class VipCustomer : Customer public class ApiC : ApiB { + [Resource] public new IQueryable Customers { get; set; } + [Resource(IsSingleton = true)] public new Customer Me { get; set; } } public class ApiD : ApiC { - protected override void UpdateApiConfiguration(ApiConfiguration config) - { - base.UpdateApiConfiguration(config); - config.IgnoreProperty("People"); - } } public class Order @@ -253,13 +240,15 @@ public class Order public class ApiE : BaseApi { + [Resource] public IQueryable People { get; set; } + [Resource] public IQueryable Orders { get; set; } - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { services.AddService((sp, next) => new TestModelBuilder()); - return base.ConfigureApi(services); + return BaseApi.ConfigureApi(apiType, services); } } @@ -267,28 +256,32 @@ public class ApiF : BaseApi { public IQueryable VipCustomers { get; set; } - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { services.AddService((sp, next) => new TestModelBuilder()); - return base.ConfigureApi(services); + return BaseApi.ConfigureApi(apiType, services); } } public class ApiG : ApiC { + [Resource] public IQueryable Employees { get; set; } } public class ApiH : BaseApi { + [Resource(IsSingleton = true)] public Person Me { get; set; } + [Resource] public IQueryable Customers { get; set; } + [Resource(IsSingleton = true)] public Customer Me2 { get; set; } - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { services.AddService((sp, next) => new TestModelBuilder()); - return base.ConfigureApi(services); + return BaseApi.ConfigureApi(apiType, services); } } } \ No newline at end of file diff --git a/test/Microsoft.Restier.Publishers.OData.Test/StoreApi.cs b/test/Microsoft.Restier.Publishers.OData.Test/StoreApi.cs index 5c521efe0..e3903241c 100644 --- a/test/Microsoft.Restier.Publishers.OData.Test/StoreApi.cs +++ b/test/Microsoft.Restier.Publishers.OData.Test/StoreApi.cs @@ -36,9 +36,9 @@ static StoreModel() internal class StoreApi : ApiBase { - public override IServiceCollection ConfigureApi(IServiceCollection services) + public new static IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { - services = base.ConfigureApi(services); + services = ApiBase.ConfigureApi(apiType, services); services.AddService((sp, next) => new TestModelProducer(StoreModel.Model)); services.AddService((sp, next) => new TestModelMapper()); services.AddService((sp, next) => new TestQueryExpressionSourcer()); diff --git a/test/Microsoft.Restier.TestCommon/PublicApi.bsl b/test/Microsoft.Restier.TestCommon/PublicApi.bsl index 314d87dfe..9b1fb977b 100644 --- a/test/Microsoft.Restier.TestCommon/PublicApi.bsl +++ b/test/Microsoft.Restier.TestCommon/PublicApi.bsl @@ -8,14 +8,10 @@ public abstract class Microsoft.Restier.Core.ApiBase : IDisposable { [ CLSCompliantAttribute(), ] - public virtual Microsoft.Extensions.DependencyInjection.IServiceCollection ConfigureApi (Microsoft.Extensions.DependencyInjection.IServiceCollection services) + public static Microsoft.Extensions.DependencyInjection.IServiceCollection ConfigureApi (System.Type apiType, Microsoft.Extensions.DependencyInjection.IServiceCollection services) protected virtual Microsoft.Restier.Core.ApiContext CreateApiContext (Microsoft.Restier.Core.ApiConfiguration configuration) public virtual void Dispose () - [ - CLSCompliantAttribute(), - ] - protected virtual void UpdateApiConfiguration (Microsoft.Restier.Core.ApiConfiguration configuration) } [ @@ -244,7 +240,7 @@ public class Microsoft.Restier.Core.ApiConfiguration { [ CLSCompliantAttribute(), ] - public static void AddPublisherServices (System.Action`1[[Microsoft.Extensions.DependencyInjection.IServiceCollection]] configurationCallback) + public static void AddPublisherServices (System.Type apiType, System.Action`1[[Microsoft.Extensions.DependencyInjection.IServiceCollection]] configurationCallback) [ CLSCompliantAttribute(), @@ -572,7 +568,7 @@ public class Microsoft.Restier.Providers.EntityFramework.EntityFrameworkApi`1 : [ CLSCompliantAttribute(), ] - public virtual Microsoft.Extensions.DependencyInjection.IServiceCollection ConfigureApi (Microsoft.Extensions.DependencyInjection.IServiceCollection services) + public static Microsoft.Extensions.DependencyInjection.IServiceCollection ConfigureApi (System.Type apiType, Microsoft.Extensions.DependencyInjection.IServiceCollection services) } [ @@ -728,16 +724,6 @@ public class Microsoft.Restier.Publishers.OData.Formatter.RestierResourceSetSeri public virtual void WriteObject (object graph, System.Type type, Microsoft.OData.ODataMessageWriter messageWriter, System.Web.OData.Formatter.Serialization.ODataSerializerContext writeContext) } -[ -ExtensionAttribute(), -] -public sealed class Microsoft.Restier.Publishers.OData.Model.ApiConfigurationExtensions { - [ - ExtensionAttribute(), - ] - public static Microsoft.Restier.Core.ApiConfiguration IgnoreProperty (Microsoft.Restier.Core.ApiConfiguration configuration, string propertyName) -} - [ AttributeUsageAttribute(), ] @@ -752,3 +738,12 @@ public sealed class Microsoft.Restier.Publishers.OData.Model.OperationAttribute string Namespace { [CompilerGeneratedAttribute(),]public get; [CompilerGeneratedAttribute(),]public set; } } +[ +AttributeUsageAttribute(), +] +public sealed class Microsoft.Restier.Publishers.OData.Model.ResourceAttribute : System.Attribute, _Attribute { + public ResourceAttribute () + + bool IsSingleton { [CompilerGeneratedAttribute(),]public get; [CompilerGeneratedAttribute(),]public set; } +} + diff --git a/test/ODataEndToEnd/Microsoft.OData.Service.Sample.Northwind.Tests/SaveTests.cs b/test/ODataEndToEnd/Microsoft.OData.Service.Sample.Northwind.Tests/SaveTests.cs index a6170ddaa..3fea4bf5e 100644 --- a/test/ODataEndToEnd/Microsoft.OData.Service.Sample.Northwind.Tests/SaveTests.cs +++ b/test/ODataEndToEnd/Microsoft.OData.Service.Sample.Northwind.Tests/SaveTests.cs @@ -27,9 +27,8 @@ private class TestEntityFilterReturnsTaskApi : NorthwindApi /// /// /// - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { - Type apiType = this.GetType(); // Add core and convention's services services = services.AddCoreServices(apiType) .AddConventionBasedServices(apiType); @@ -55,7 +54,7 @@ protected async Task OnInsertingCustomers(Customer customer) public async Task TestEntityFilterReturnsTask() { TestEntityFilterReturnsTaskApi api = new TestEntityFilterReturnsTaskApi(); - var container = new RestierContainerBuilder(() => new TestEntityFilterReturnsTaskApi()); + var container = new RestierContainerBuilder(typeof(TestEntityFilterReturnsTaskApi)); api.Configuration = new ApiConfiguration(container.BuildContainer()); DataModificationItem createCustomer = new DataModificationItem( "Customers", diff --git a/test/ODataEndToEnd/Microsoft.OData.Service.Sample.Northwind/Models/NorthwindApi.cs b/test/ODataEndToEnd/Microsoft.OData.Service.Sample.Northwind/Models/NorthwindApi.cs index 16af6c00c..31d93cd58 100644 --- a/test/ODataEndToEnd/Microsoft.OData.Service.Sample.Northwind/Models/NorthwindApi.cs +++ b/test/ODataEndToEnd/Microsoft.OData.Service.Sample.Northwind/Models/NorthwindApi.cs @@ -22,6 +22,7 @@ public class NorthwindApi : EntityFrameworkApi public new NorthwindContext Context { get { return DbContext; } } // Imperative views. Currently CUD operations not supported + [Resource] public IQueryable ExpensiveProducts { get @@ -31,6 +32,7 @@ public IQueryable ExpensiveProducts } } + [Resource] public IQueryable CurrentOrders { get @@ -56,9 +58,9 @@ public double MostExpensive(IEnumerable bindingParameter) return 0.0; } - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { - return base.ConfigureApi(services) + return EntityFrameworkApi.ConfigureApi(apiType, services) .AddService(); } diff --git a/test/ODataEndToEnd/Microsoft.OData.Service.Sample.Trippin/Api/TrippinApi.cs b/test/ODataEndToEnd/Microsoft.OData.Service.Sample.Trippin/Api/TrippinApi.cs index a4bc7fb79..556f4a409 100644 --- a/test/ODataEndToEnd/Microsoft.OData.Service.Sample.Trippin/Api/TrippinApi.cs +++ b/test/ODataEndToEnd/Microsoft.OData.Service.Sample.Trippin/Api/TrippinApi.cs @@ -30,6 +30,7 @@ public TrippinModel ModelContext get { return DbContext; } } + [Resource(IsSingleton = true)] public Person Me { get @@ -42,16 +43,19 @@ public Person Me } } + [Resource] public IQueryable Flights1 { get { return DbContext.Flights; } } + [Resource] public IQueryable Flights2 { get { return this.GetQueryableSource("Flights"); } } + [Resource] public IQueryable PeopleWithAge { get @@ -66,6 +70,7 @@ public IQueryable PeopleWithAge } } + [Resource] public IQueryable PeopleWithAge1 { get @@ -80,6 +85,7 @@ public IQueryable PeopleWithAge1 } } + [Resource(IsSingleton = true)] public PersonWithAge PeopleWithAgeMe { get @@ -561,7 +567,7 @@ protected bool CanDeleteTrips() return false; } - public override IServiceCollection ConfigureApi(IServiceCollection services) + public static new IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { // Add customized OData validation settings Func validationSettingFactory = sp => new ODataValidationSettings @@ -572,7 +578,7 @@ public override IServiceCollection ConfigureApi(IServiceCollection services) services.AddService(); - return base.ConfigureApi(services) + return EntityFrameworkApi.ConfigureApi(apiType, services) .AddSingleton() .AddSingleton(validationSettingFactory) .AddSingleton() diff --git a/test/ODataEndToEnd/Microsoft.OData.Service.Sample.TrippinInMemory/Api/TrippinApi.cs b/test/ODataEndToEnd/Microsoft.OData.Service.Sample.TrippinInMemory/Api/TrippinApi.cs index 78b2fc0fb..744ef5eab 100644 --- a/test/ODataEndToEnd/Microsoft.OData.Service.Sample.TrippinInMemory/Api/TrippinApi.cs +++ b/test/ODataEndToEnd/Microsoft.OData.Service.Sample.TrippinInMemory/Api/TrippinApi.cs @@ -36,6 +36,7 @@ private string Key #region Entity Set + [Resource] public IQueryable People { get @@ -50,6 +51,7 @@ public IQueryable People } } + [Resource] public IQueryable NewComePeople { get @@ -64,6 +66,7 @@ public IQueryable NewComePeople } } + [Resource(IsSingleton = true)] public Person Me { get @@ -78,6 +81,7 @@ public Person Me } } + [Resource] public IQueryable Airlines { get @@ -92,6 +96,7 @@ public IQueryable Airlines } } + [Resource] public IQueryable Airports { get @@ -332,7 +337,7 @@ private static double CalculateDistance(GeographyPoint p1, GeographyPoint p2) #endregion - public override IServiceCollection ConfigureApi(IServiceCollection services) + public new static IServiceCollection ConfigureApi(Type apiType, IServiceCollection services) { Func> defaultDataStoreManager = sp => new DefaultDataStoreManager() @@ -345,7 +350,7 @@ public override IServiceCollection ConfigureApi(IServiceCollection services) services.AddService((sp, next) => new ChangeSetInitializer()); services.AddService((sp, next) => new SubmitExecutor()); services.AddSingleton(defaultDataStoreManager); - return base.ConfigureApi(services); + return ApiBase.ConfigureApi(apiType, services); } private class ModelBuilder : IModelBuilder