// Copyright (c) Microsoft Open Technologies, Inc. // All Rights Reserved // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // THIS CODE IS PROVIDED *AS IS* BASIS, WITHOUT WARRANTIES OR // CONDITIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING // WITHOUT LIMITATION ANY IMPLIED WARRANTIES OR CONDITIONS OF // TITLE, FITNESS FOR A PARTICULAR PURPOSE, MERCHANTABLITY OR // NON-INFRINGEMENT. // See the Apache 2 License for the specific language governing // permissions and limitations under the License. using System; #if NET45 using System.Runtime.Remoting.Messaging; #endif using System.Threading.Tasks; using Microsoft.AspNet.Http; using Microsoft.AspNet.DependencyInjection; namespace Microsoft.AspNet.RequestContainer { public class ContainerMiddleware { private const string LogicalDataKey = "__HttpContext_Current__"; private readonly RequestDelegate _next; private readonly IServiceProvider _rootServiceProvider; private readonly IContextAccessor _rootHttpContextAccessor; private readonly IServiceScopeFactory _rootServiceScopeFactory; public ContainerMiddleware( RequestDelegate next, IServiceProvider rootServiceProvider, IContextAccessor rootHttpContextAccessor, IServiceScopeFactory rootServiceScopeFactory) { if (rootServiceProvider == null) { throw new ArgumentNullException("rootServiceProvider"); } if (rootHttpContextAccessor == null) { throw new ArgumentNullException("rootHttpContextAccessor"); } if (rootServiceScopeFactory == null) { throw new ArgumentNullException("rootServiceScopeFactory"); } _next = next; _rootServiceProvider = rootServiceProvider; _rootServiceScopeFactory = rootServiceScopeFactory; _rootHttpContextAccessor = rootHttpContextAccessor; _rootHttpContextAccessor.SetContextSource(AccessRootHttpContext, ExchangeRootHttpContext); } private HttpContext AccessRootHttpContext() { #if NET45 return CallContext.LogicalGetData(LogicalDataKey) as HttpContext; #else throw new Exception("TODO: CallContext not available"); #endif } private HttpContext ExchangeRootHttpContext(HttpContext httpContext) { #if NET45 var prior = CallContext.LogicalGetData(LogicalDataKey) as HttpContext; CallContext.LogicalSetData(LogicalDataKey, httpContext); return prior; #else return null; #endif } public async Task Invoke(HttpContext httpContext) { if (httpContext.RequestServices != null) { throw new Exception("TODO: nested request container scope? this is probably a mistake on your part?"); } var priorApplicationServices = httpContext.ApplicationServices; var priorRequestServices = httpContext.RequestServices; var appServiceProvider = _rootServiceProvider; var appServiceScopeFactory = _rootServiceScopeFactory; var appHttpContextAccessor = _rootHttpContextAccessor; if (priorApplicationServices != null && priorApplicationServices != appServiceProvider) { appServiceProvider = priorApplicationServices; appServiceScopeFactory = priorApplicationServices.GetService(); appHttpContextAccessor = priorApplicationServices.GetService>(); } using (var scope = appServiceScopeFactory.CreateScope()) { var scopeServiceProvider = scope.ServiceProvider; var scopeHttpContextAccessor = scopeServiceProvider.GetService>(); httpContext.ApplicationServices = appServiceProvider; httpContext.RequestServices = scopeServiceProvider; var priorAppHttpContext = appHttpContextAccessor.SetValue(httpContext); var priorScopeHttpContext = scopeHttpContextAccessor.SetValue(httpContext); try { await _next.Invoke(httpContext); } finally { scopeHttpContextAccessor.SetValue(priorScopeHttpContext); appHttpContextAccessor.SetValue(priorAppHttpContext); httpContext.RequestServices = priorRequestServices; httpContext.ApplicationServices = priorApplicationServices; } } } } }