You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			149 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			149 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //------------------------------------------------------------------------------ | |||
|  | // <copyright file="CustomLoaderHelper.cs" company="Microsoft"> | |||
|  | //    Copyright (c) Microsoft Corporation.  All rights reserved. | |||
|  | // </copyright> | |||
|  | //------------------------------------------------------------------------------ | |||
|  | 
 | |||
|  | namespace System.Web.Hosting { | |||
|  |     using System; | |||
|  |     using System.Globalization; | |||
|  |     using System.Reflection; | |||
|  |     using System.Runtime.InteropServices; | |||
|  |     using System.Runtime.Remoting; | |||
|  |     using System.Runtime.Versioning; | |||
|  | 
 | |||
|  |     // Used to locate a custom loader implementation within a bin-deployed assembly. | |||
|  | 
 | |||
|  |     internal sealed class CustomLoaderHelper : MarshalByRefObject { | |||
|  | 
 | |||
|  |         // the first framework version where the custom loader feature was implemented | |||
|  |         private static readonly string _customLoaderTargetFrameworkName = new FrameworkName(".NETFramework", new Version(4, 5, 1)).ToString(); | |||
|  | 
 | |||
|  |         private static readonly string _customLoaderAssemblyName = typeof(CustomLoaderHelper).Assembly.FullName; | |||
|  |         private static readonly string _customLoaderTypeName = typeof(CustomLoaderHelper).FullName; | |||
|  |         private static readonly Guid IID_ICustomLoader = new Guid("50A3CE65-2F9F-44E9-9094-32C6C928F966"); | |||
|  | 
 | |||
|  |         // Instances of this type should only ever be created via reflection (see call to CreateObjectAndUnwrap | |||
|  |         // in GetCustomLoader). | |||
|  |         private CustomLoaderHelper() { } | |||
|  | 
 | |||
|  |         internal static IObjectHandle GetCustomLoader(ICustomLoaderHelperFunctions helperFunctions, string appConfigMetabasePath, string configFilePath, string customLoaderPhysicalPath, out AppDomain newlyCreatedAppDomain) { | |||
|  |             // Step 1: Does the host allow custom loaders? | |||
|  | 
 | |||
|  |             bool? customLoaderIsEnabled = helperFunctions.CustomLoaderIsEnabled; | |||
|  |             if (customLoaderIsEnabled.HasValue) { | |||
|  |                 if ((bool)customLoaderIsEnabled) { | |||
|  |                     // The custom loader is enabled; move on to the next step. | |||
|  |                 } | |||
|  |                 else { | |||
|  |                     // The custom loader is disabled, fail. | |||
|  |                     throw new NotSupportedException(ApplicationServicesStrings.CustomLoader_ForbiddenByHost); | |||
|  |                 } | |||
|  |             } | |||
|  |             else { | |||
|  |                 // The host hasn't set a policy, so we'll fall back to our default logic of checking the application's trust level. | |||
|  |                 if (!IsFullyTrusted(helperFunctions, appConfigMetabasePath)) { | |||
|  |                     throw new NotSupportedException(ApplicationServicesStrings.CustomLoader_NotInFullTrust); | |||
|  |                 } | |||
|  |             } | |||
|  | 
 | |||
|  |             // Step 2: Create the new AD | |||
|  | 
 | |||
|  |             string binFolderPhysicalPath = helperFunctions.MapPath("/bin/"); | |||
|  | 
 | |||
|  |             AppDomainSetup setup = new AppDomainSetup() { | |||
|  |                 PrivateBinPathProbe = "*",  // disable loading from app base | |||
|  |                 PrivateBinPath = binFolderPhysicalPath, | |||
|  |                 ApplicationBase = helperFunctions.AppPhysicalPath, | |||
|  |                 TargetFrameworkName = _customLoaderTargetFrameworkName | |||
|  |             }; | |||
|  | 
 | |||
|  |             if (configFilePath != null) { | |||
|  |                 setup.ConfigurationFile = configFilePath; | |||
|  |             } | |||
|  | 
 | |||
|  |             AppDomain newAppDomainForCustomLoader = AppDomain.CreateDomain("aspnet-custom-loader-" + Guid.NewGuid(), null, setup); | |||
|  |             try { | |||
|  |                 // Step 3: Instantiate helper in new AD so that we can get a reference to the loader | |||
|  |                 CustomLoaderHelper helper = (CustomLoaderHelper)newAppDomainForCustomLoader.CreateInstanceAndUnwrap(_customLoaderAssemblyName, _customLoaderTypeName, | |||
|  |                     ignoreCase: false, | |||
|  |                     bindingAttr: BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.CreateInstance | BindingFlags.Instance, | |||
|  |                     binder: null, | |||
|  |                     args: null, | |||
|  |                     culture: null, | |||
|  |                     activationAttributes: null); | |||
|  |                 ObjectHandle ohCustomLoader = helper.GetCustomLoaderImpl(customLoaderPhysicalPath); | |||
|  | 
 | |||
|  |                 // If we got this far, success! | |||
|  |                 newlyCreatedAppDomain = newAppDomainForCustomLoader; | |||
|  |                 return ohCustomLoader; | |||
|  |             } | |||
|  |             catch { | |||
|  |                 // If something went wrong, kill the new AD. | |||
|  |                 AppDomain.Unload(newAppDomainForCustomLoader); | |||
|  |                 throw; | |||
|  |             } | |||
|  |         } | |||
|  | 
 | |||
|  |         private ObjectHandle GetCustomLoaderImpl(string customLoaderPhysicalPath) { | |||
|  |             // Step 4: Find the implementation in the custom loader assembly | |||
|  | 
 | |||
|  |             // Since we have set the private bin path, we can use this call to Assembly.Load | |||
|  |             // to avoid the load-from context, which has weird behaviors. | |||
|  |             AssemblyName customLoaderAssemblyName = AssemblyName.GetAssemblyName(customLoaderPhysicalPath); | |||
|  |             Assembly customLoaderAssembly = Assembly.Load(customLoaderAssemblyName); | |||
|  |             CustomLoaderAttribute customLoaderAttribute = customLoaderAssembly.GetCustomAttribute<CustomLoaderAttribute>(); | |||
|  | 
 | |||
|  |             if (customLoaderAttribute == null) { | |||
|  |                 throw new InvalidOperationException(String.Format(CultureInfo.CurrentCulture, ApplicationServicesStrings.CustomLoader_NoAttributeFound, customLoaderAssemblyName)); | |||
|  |             } | |||
|  | 
 | |||
|  |             // Step 5: Instantiate the custom loader and return a reference back to native code | |||
|  | 
 | |||
|  |             object customLoader = Activator.CreateInstance(customLoaderAttribute.CustomLoaderType); | |||
|  | 
 | |||
|  |             // This check isn't strictly necessary since the unmanaged layer will handle QueryInterface failures | |||
|  |             // appropriately, but we have an opportunity to provide a better error message at this layer. | |||
|  |             if (!ObjectImplementsComInterface(customLoader, IID_ICustomLoader)) { | |||
|  |                 throw new InvalidOperationException(String.Format(CultureInfo.CurrentCulture, ApplicationServicesStrings.CustomLoader_MustImplementICustomLoader, customLoader.GetType())); | |||
|  |             } | |||
|  | 
 | |||
|  |             return new ObjectHandle(customLoader); | |||
|  |         } | |||
|  | 
 | |||
|  |         private static bool IsFullyTrusted(ICustomLoaderHelperFunctions helperFunctions, string appConfigMetabasePath) { | |||
|  |             // The managed configuration system hasn't yet been instantiated but the IIS native config system understands | |||
|  |             // ASP.NET configuration and honors hierarchy and section locking. | |||
|  | 
 | |||
|  |             try { | |||
|  |                 // Must exactly match <trust level="Full" />, as this is what ApplicationManager expects. | |||
|  |                 string trustLevel = helperFunctions.GetTrustLevel(appConfigMetabasePath); | |||
|  |                 return String.Equals("Full", trustLevel, StringComparison.Ordinal); | |||
|  |             } | |||
|  |             catch { | |||
|  |                 // If any of the sections are locked or there is a config error, bail. | |||
|  |                 return false; | |||
|  |             } | |||
|  |         } | |||
|  | 
 | |||
|  |         private static bool ObjectImplementsComInterface(object o, Guid iid) { | |||
|  |             IntPtr pUnknown = IntPtr.Zero; | |||
|  |             IntPtr pInterface = IntPtr.Zero; | |||
|  | 
 | |||
|  |             try { | |||
|  |                 pUnknown = Marshal.GetIUnknownForObject(o); // AddRef | |||
|  |                 int hr = Marshal.QueryInterface(pUnknown, ref iid, out pInterface); // AddRef | |||
|  |                 return (hr == 0 && pInterface != IntPtr.Zero); | |||
|  |             } | |||
|  |             finally { | |||
|  |                 if (pUnknown != IntPtr.Zero) { | |||
|  |                     Marshal.Release(pUnknown); | |||
|  |                 } | |||
|  |                 if (pInterface != IntPtr.Zero) { | |||
|  |                     Marshal.Release(pInterface); | |||
|  |                 } | |||
|  |             } | |||
|  |         } | |||
|  |     } | |||
|  | } |