//------------------------------------------------------------------------------
//
// Copyright (c) Microsoft Corporation. All rights reserved.
//
//------------------------------------------------------------------------------
/************************************************************************************************************/
namespace System.Web.Compilation {
using System;
using System.CodeDom;
using System.CodeDom.Compiler;
using System.Collections;
using System.Collections.Generic;
using System.Collections.Specialized;
using System.Globalization;
using System.IO;
using System.Reflection;
using System.Security.Permissions;
using System.Text;
using System.Threading;
using System.Web;
using System.Web.Caching;
using System.Web.Configuration;
using System.Web.Hosting;
using System.Web.UI;
using System.Web.Util;
using Debug = System.Web.Util.Debug;
//
// Instances of this class are created in the ASP.NET app domain. The class
// methods are called by ClientBuildManager (cross app domain)
//
internal class BuildManagerHost : MarshalByRefObject, IRegisteredObject {
private ClientBuildManager _client;
private BuildManager _buildManager;
private int _pendingCallsCount;
private EventHandler _onAppDomainUnload;
private bool _ignorePendingCalls;
private IDictionary _assemblyCollection;
private object _lock = new object();
private static bool _inClientBuildManager;
internal static bool InClientBuildManager {
get { return _inClientBuildManager; }
set { _inClientBuildManager = true; }
}
///
/// Returns whether there is support for multitargeting. This is usually only true in VS as they will supply a
/// TypeDescriptionProvider that is required for correctly reflecting over types in the target framework.
/// When running in the 4.0 runtime application pool, or when running from aspnet_compiler, this value
/// will be false.
///
internal static bool SupportsMultiTargeting {
get; set;
}
private ClientVirtualPathProvider _virtualPathProvider;
public BuildManagerHost() {
HostingEnvironment.RegisterObject(this);
AppDomain.CurrentDomain.AssemblyResolve += new ResolveEventHandler(this.ResolveAssembly);
}
void IRegisteredObject.Stop(bool immediate) {
// Make sure all the pending calls complete
WaitForPendingCallsToFinish();
HostingEnvironment.UnregisterObject(this);
if (_client != null) {
_client.ResetHost();
}
}
internal IApplicationHost ApplicationHost {
get {
return HostingEnvironment.ApplicationHostInternal;
}
}
internal string CodeGenDir {
get {
// Add a pending call to make sure our thread doesn't get killed
AddPendingCall();
try {
return HttpRuntime.CodegenDirInternal;
}
finally {
RemovePendingCall();
}
}
}
internal void RegisterAssembly(String assemblyName, String assemblyLocation) {
Debug.Trace("BuildManagerHost", "RegisterAssembly '" + assemblyName + "','" + assemblyLocation + "'");
if (_assemblyCollection == null) {
lock (_lock) {
if (_assemblyCollection == null) {
_assemblyCollection = Hashtable.Synchronized(new Hashtable());
}
}
}
AssemblyName asmName = new AssemblyName(assemblyName);
_assemblyCollection[asmName.FullName] = assemblyLocation;
}
[PermissionSet(SecurityAction.Assert, Unrestricted = true)]
private Assembly ResolveAssembly(object sender, ResolveEventArgs e) {
Debug.Trace("BuildManagerHost", "ResolveAssembly '" + e.Name + "'");
if (_assemblyCollection == null)
return null;
String assemblyLocation = (String)_assemblyCollection[e.Name];
if (assemblyLocation == null)
return null;
Debug.Trace("BuildManagerHost", "ResolveAssembly: found");
return Assembly.LoadFrom(assemblyLocation);
}
/*
* Make sure all the (non-request based) pending calls complete
*/
private void WaitForPendingCallsToFinish() {
for (;;) {
if (_pendingCallsCount <= 0 || _ignorePendingCalls)
break;
Thread.Sleep(250);
}
}
internal void AddPendingCall() {
Interlocked.Increment(ref _pendingCallsCount);
}
internal void RemovePendingCall() {
Interlocked.Decrement(ref _pendingCallsCount);
}
private void OnAppDomainShutdown(object o, BuildManagerHostUnloadEventArgs args) {
_client.OnAppDomainShutdown(args.Reason);
}
internal void CompileApplicationDependencies() {
// Add a pending call to make sure our thread doesn't get killed
AddPendingCall();
try {
_buildManager.EnsureTopLevelFilesCompiled();
}
finally {
RemovePendingCall();
}
}
internal void PrecompileApp(ClientBuildManagerCallback callback, List excludedVirtualPaths) {
// Add a pending call to make sure our thread doesn't get killed
AddPendingCall();
try {
_buildManager.PrecompileApp(callback, excludedVirtualPaths);
}
finally {
RemovePendingCall();
}
}
internal IDictionary GetBrowserDefinitions() {
// Add a pending call to make sure our thread doesn't get killed
AddPendingCall();
try {
return BrowserCapabilitiesCompiler.BrowserCapabilitiesFactory.InternalGetBrowserElements();
}
finally {
RemovePendingCall();
}
}
internal string[] GetVirtualCodeDirectories() {
// Add a pending call to make sure our thread doesn't get killed
AddPendingCall();
try {
return _buildManager.GetCodeDirectories();
}
finally {
RemovePendingCall();
}
}
[PermissionSet(SecurityAction.Assert, Unrestricted = true)]
internal void GetCodeDirectoryInformation(VirtualPath virtualCodeDir,
out Type codeDomProviderType, out CompilerParameters compParams,
out string generatedFilesDir) {
// Add a pending call to make sure our thread doesn't get killed
AddPendingCall();
try {
BuildManager.SkipTopLevelCompilationExceptions = true;
_buildManager.EnsureTopLevelFilesCompiled();
// Treat it as relative to the app root
virtualCodeDir = virtualCodeDir.CombineWithAppRoot();
_buildManager.GetCodeDirectoryInformation(virtualCodeDir,
out codeDomProviderType, out compParams, out generatedFilesDir);
}
finally {
BuildManager.SkipTopLevelCompilationExceptions = false;
RemovePendingCall();
}
}
internal void GetCompilerParams(VirtualPath virtualPath, out Type codeDomProviderType,
out CompilerParameters compParams) {
// Add a pending call to make sure our thread doesn't get killed
AddPendingCall();
try {
BuildManager.SkipTopLevelCompilationExceptions = true;
_buildManager.EnsureTopLevelFilesCompiled();
// Ignore the BuildProvider return value
GetCompilerParamsAndBuildProvider(virtualPath, out codeDomProviderType, out compParams);
// This is the no-compile case
if (compParams == null)
return;
FixupReferencedAssemblies(virtualPath, compParams);
}
finally {
BuildManager.SkipTopLevelCompilationExceptions = false;
RemovePendingCall();
}
}
internal string[] GetCompiledTypeAndAssemblyName(VirtualPath virtualPath, ClientBuildManagerCallback callback) {
// Add a pending call to make sure our thread doesn't get killed
AddPendingCall();
try {
// Treat it as relative to the app root
virtualPath.CombineWithAppRoot();
Type t = BuildManager.GetCompiledType(virtualPath, callback);
if (t == null) return null;
string assemblyPath = Util.GetAssemblyPathFromType(t);
return new string[] { t.FullName, assemblyPath };
}
finally {
RemovePendingCall();
}
}
internal string GetGeneratedSourceFile(VirtualPath virtualPath) {
// Add a pending call to make sure our thread doesn't get killed
AddPendingCall();
Type codeDomProviderType;
CompilerParameters compilerParameters;
string generatedFilesDir;
try {
if (!virtualPath.DirectoryExists()) {
throw new ArgumentException(SR.GetString(SR.GetGeneratedSourceFile_Directory_Only,
virtualPath.VirtualPathString), "virtualPath");
}
// Calls GetCodeDirectoryInformation to ensure the source files are created for the
// directory specified by virtualPath
GetCodeDirectoryInformation(virtualPath,
out codeDomProviderType, out compilerParameters,
out generatedFilesDir);
return BuildManager.GenerateFileTable[virtualPath.VirtualPathStringNoTrailingSlash];
}
finally {
RemovePendingCall();
}
}
internal string GetGeneratedFileVirtualPath(string filePath) {
// Add a pending call to make sure our thread doesn't get killed
AddPendingCall();
try {
// Performs reverse hashtable lookup to find the filePath in the Value collection.
Dictionary.Enumerator e = BuildManager.GenerateFileTable.GetEnumerator();
while (e.MoveNext()) {
KeyValuePair pair = e.Current;
if (filePath.Equals(pair.Value, StringComparison.Ordinal)) {
return pair.Key;
}
}
return null;
}
finally {
RemovePendingCall();
}
}
/*
* Returns an array of the assemblies defined in the bin and assembly reference config section
*/
internal String[] GetTopLevelAssemblyReferences(VirtualPath virtualPath) {
// Add a pending call to make sure our thread doesn't get killed
AddPendingCall();
List assemblyList = new List();
try {
// Treat it as relative to the app root
virtualPath.CombineWithAppRoot();
CompilationSection compConfig = MTConfigUtil.GetCompilationConfig(virtualPath);
// Add all the config assemblies to the list
foreach (AssemblyInfo assemblyInfo in compConfig.Assemblies) {
Assembly[] assemblies = assemblyInfo.AssemblyInternal;
for (int i = 0; i < assemblies.Length; i++) {
if (assemblies[i] != null) {
assemblyList.Add(assemblies[i]);
}
}
}
} finally {
RemovePendingCall();
}
StringCollection paths = new StringCollection();
Util.AddAssembliesToStringCollection(assemblyList, paths);
string[] references = new string[paths.Count];
paths.CopyTo(references, 0);
return references;
}
[PermissionSet(SecurityAction.Assert, Unrestricted = true)]
internal string GenerateCode(
VirtualPath virtualPath, string virtualFileString, out IDictionary linePragmasTable) {
AddPendingCall();
try {
string code = null;
Type codeDomProviderType;
CompilerParameters compilerParameters;
CodeCompileUnit ccu = GenerateCodeCompileUnit(virtualPath, virtualFileString, out codeDomProviderType,
out compilerParameters, out linePragmasTable);
if (ccu != null && codeDomProviderType != null) {
CodeDomProvider codeProvider = CompilationUtil.CreateCodeDomProvider(codeDomProviderType);
CodeGeneratorOptions codeGeneratorOptions = new CodeGeneratorOptions();
codeGeneratorOptions.BlankLinesBetweenMembers = false;
codeGeneratorOptions.IndentString = string.Empty;
StringWriter sw = new StringWriter(CultureInfo.InvariantCulture);
codeProvider.GenerateCodeFromCompileUnit(ccu, sw, codeGeneratorOptions);
code = sw.ToString();
}
return code;
}
finally {
RemovePendingCall();
}
}
[PermissionSet(SecurityAction.Assert, Unrestricted = true)]
internal CodeCompileUnit GenerateCodeCompileUnit(
VirtualPath virtualPath, string virtualFileString, out Type codeDomProviderType,
out CompilerParameters compilerParameters, out IDictionary linePragmasTable) {
// Add a pending call to make sure our thread doesn't get killed
AddPendingCall();
try {
BuildManager.SkipTopLevelCompilationExceptions = true;
_buildManager.EnsureTopLevelFilesCompiled();
// Get the virtual file content so that we can use the correct hash code.
if (virtualFileString == null) {
using (Stream stream = virtualPath.OpenFile()) {
TextReader reader = Util.ReaderFromStream(stream, virtualPath);
virtualFileString = reader.ReadToEnd();
}
}
_virtualPathProvider.RegisterVirtualFile(virtualPath, virtualFileString);
string cacheKey = BuildManager.GetCacheKeyFromVirtualPath(virtualPath) + "_CBMResult";
BuildResultCodeCompileUnit result = (BuildResultCodeCompileUnit)BuildManager.GetBuildResultFromCache(cacheKey, virtualPath);
if (result == null) {
lock (_lock) {
// Don't need to check the result again since it's very unlikely in CBM scenarios.
DateTime utcStart = DateTime.UtcNow;
BuildProvider internalBuildProvider = GetCompilerParamsAndBuildProvider(
virtualPath, out codeDomProviderType, out compilerParameters);
// This is the no-compile case
if (internalBuildProvider == null) {
linePragmasTable = null;
return null;
}
CodeCompileUnit ccu = internalBuildProvider.GetCodeCompileUnit(out linePragmasTable);
result = new BuildResultCodeCompileUnit(codeDomProviderType, ccu, compilerParameters, linePragmasTable);
result.VirtualPath = virtualPath;
result.SetCacheKey(cacheKey);
FixupReferencedAssemblies(virtualPath, compilerParameters);
// CodeCompileUnit could be null, do not try to fix referenced assemblies.
// This happens for example when an .asmx file does not contain any code.
if (ccu != null) {
// VSWhidbey 501260 Add all the referenced assemblies to the CodeCompileUnit
// in case the CodeDom provider needs them for code generation
foreach (String assemblyString in compilerParameters.ReferencedAssemblies) {
ccu.ReferencedAssemblies.Add(assemblyString);
}
}
// Add all the dependencies, so that the ccu gets cached correctly (VSWhidbey 275091)
ICollection dependencies = internalBuildProvider.VirtualPathDependencies;
if (dependencies != null)
result.AddVirtualPathDependencies(dependencies);
BuildManager.CacheBuildResult(cacheKey, result, utcStart);
return ccu;
}
}
codeDomProviderType = result.CodeDomProviderType;
compilerParameters = result.CompilerParameters;
linePragmasTable = result.LinePragmasTable;
FixupReferencedAssemblies(virtualPath, compilerParameters);
return result.CodeCompileUnit;
}
finally {
if (virtualFileString != null) {
_virtualPathProvider.RevertVirtualFile(virtualPath);
}
BuildManager.SkipTopLevelCompilationExceptions = false;
RemovePendingCall();
}
}
internal bool IsCodeAssembly(string assemblyName) {
return BuildManager.GetNormalizedCodeAssemblyName(assemblyName) != null;
}
// Add the referenced assemblies into the compileParameters. Notice that buildProviders do not have
// the correct referenced assemblies and we don't cache them since the assemblies could change
// between appdomains. (removing assemblies from bin, etc)
private void FixupReferencedAssemblies(VirtualPath virtualPath, CompilerParameters compilerParameters) {
CompilationSection compConfig = MTConfigUtil.GetCompilationConfig(virtualPath);
ICollection referencedAssemblies = BuildManager.GetReferencedAssemblies(compConfig);
Util.AddAssembliesToStringCollection(referencedAssemblies, compilerParameters.ReferencedAssemblies);
}
private BuildProvider GetCompilerParamsAndBuildProvider(VirtualPath virtualPath,
out Type codeDomProviderType, out CompilerParameters compilerParameters) {
virtualPath.CombineWithAppRoot();
CompilationSection compConfig = MTConfigUtil.GetCompilationConfig(virtualPath);
ICollection referencedAssemblies = BuildManager.GetReferencedAssemblies(compConfig);
// Create the buildprovider for the passed in virtualPath
BuildProvider buildProvider = null;
// Special case global asax build provider here since we do not want to compile every files with ".asax" extension.
if (StringUtil.EqualsIgnoreCase(virtualPath.VirtualPathString, BuildManager.GlobalAsaxVirtualPath.VirtualPathString)) {
ApplicationBuildProvider provider = new ApplicationBuildProvider();
provider.SetVirtualPath(virtualPath);
provider.SetReferencedAssemblies(referencedAssemblies);
buildProvider = provider;
}
else {
buildProvider = BuildManager.CreateBuildProvider(virtualPath, compConfig,
referencedAssemblies, true /*failIfUnknown*/);
}
// DevDiv 69017
// The methods restricted to internalBuildProvider have been moved up to BuildProvider
// to allow WCFBuildProvider to support .svc syntax highlighting.
// Ignore parse errors, since they should not break the designer
buildProvider.IgnoreParseErrors = true;
// Ignore all control properties, since we do not generate code for the properties
buildProvider.IgnoreControlProperties = true;
// Process as many errors as possible, do not rethrow on first error
buildProvider.ThrowOnFirstParseError = false;
// Get the language (causes the file to be parsed)
CompilerType compilerType = buildProvider.CodeCompilerType;
// compilerType could be null in the no-compile case (VSWhidbey 221749)
if (compilerType == null) {
codeDomProviderType = null;
compilerParameters = null;
return null;
}
// Return the provider type and compiler params
codeDomProviderType = compilerType.CodeDomProviderType;
compilerParameters = compilerType.CompilerParameters;
IAssemblyDependencyParser parser = buildProvider.AssemblyDependencyParser;
// Add all the assemblies that the page depends on (e.g. user controls)
if (parser != null && parser.AssemblyDependencies != null) {
Util.AddAssembliesToStringCollection(parser.AssemblyDependencies,
compilerParameters.ReferencedAssemblies);
}
// Make any fix up adjustments to the CompilerParameters to work around some issues
AssemblyBuilder.FixUpCompilerParameters(compConfig, codeDomProviderType, compilerParameters);
return buildProvider;
}
public override Object InitializeLifetimeService() {
return null; // never expire lease
}
internal void Configure(ClientBuildManager client) {
// Add a pending call to make sure our thread doesn't get killed
AddPendingCall();
try {
_virtualPathProvider = new ClientVirtualPathProvider();
HostingEnvironment.RegisterVirtualPathProviderInternal(_virtualPathProvider);
_client = client;
// Type description provider required for multitargeting compilation support in VS.
if (_client.CBMTypeDescriptionProviderBridge != null) {
TargetFrameworkUtil.CBMTypeDescriptionProviderBridge = _client.CBMTypeDescriptionProviderBridge;
}
// start watching for app domain unloading
_onAppDomainUnload = new EventHandler(OnAppDomainUnload);
Thread.GetDomain().DomainUnload += _onAppDomainUnload;
_buildManager = BuildManager.TheBuildManager;
// Listen to appdomain shutdown.
HttpRuntime.AppDomainShutdown += new BuildManagerHostUnloadEventHandler(this.OnAppDomainShutdown);
}
finally {
RemovePendingCall();
}
}
internal Exception InitializationException {
get {
return HostingEnvironment.InitializationException;
}
}
private void OnAppDomainUnload(Object unusedObject, EventArgs unusedEventArgs) {
Thread.GetDomain().DomainUnload -= _onAppDomainUnload;
if (_client != null) {
_client.OnAppDomainUnloaded(HttpRuntime.ShutdownReason);
_client = null;
}
}
internal bool UnloadAppDomain() {
_ignorePendingCalls = true;
// Make sure HttpRuntime does not ignore the appdomain shutdown.
HttpRuntime.SetUserForcedShutdown();
// Force unload the appdomain when called from client
return HttpRuntime.ShutdownAppDomain(ApplicationShutdownReason.UnloadAppDomainCalled, "CBM called UnloadAppDomain");
}
// This provider is created in the hosted appdomain for faster access of the virtual file content.
// Note this is used both in CBM and the aspnet precompilation tool
internal class ClientVirtualPathProvider : VirtualPathProvider {
private IDictionary _stringDictionary;
internal ClientVirtualPathProvider() {
_stringDictionary = new HybridDictionary(true);
}
public override bool FileExists(string virtualPath) {
if (_stringDictionary.Contains(virtualPath)) {
return true;
}
return base.FileExists(virtualPath);
}
public override CacheDependency GetCacheDependency(string virtualPath,
IEnumerable virtualPathDependencies, DateTime utcStart) {
if (virtualPath != null) {
virtualPath = UrlPath.MakeVirtualPathAppAbsolute(virtualPath);
// Return now so the build result will be invalidated based on hashcode.
// This is for the case that Venus passed in the file content so we don't
// get file change notification
if (_stringDictionary.Contains(virtualPath)) {
return null;
}
}
// otherwise creates a cachedependency using MapPathBasedVirtualPathProvider
return base.GetCacheDependency(virtualPath, virtualPathDependencies, utcStart);
}
public override VirtualFile GetFile(string virtualPath) {
String _virtualFileString = (String)_stringDictionary[virtualPath];
return _virtualFileString != null? new ClientVirtualFile(virtualPath, _virtualFileString) : base.GetFile(virtualPath);
}
public override string GetFileHash(string virtualPath, IEnumerable virtualPathDependencies) {
HashCodeCombiner hashCodeCombiner = null;
ArrayList clonedPaths = new ArrayList();
foreach (string virtualDependency in virtualPathDependencies) {
// if the virtual path is previously cached, add the actual content to
// the hash code combiner.
if (_stringDictionary.Contains(virtualDependency)) {
if (hashCodeCombiner == null) {
hashCodeCombiner = new HashCodeCombiner();
}
hashCodeCombiner.AddInt(StringUtil.GetNonRandomizedHashCode((string)_stringDictionary[virtualDependency]));
continue;
}
// Otherwise move it to the cloned collection and use the base class (previous provider)
// to get the hash code.
clonedPaths.Add(virtualDependency);
}
if (hashCodeCombiner == null) {
return base.GetFileHash(virtualPath, virtualPathDependencies);
}
hashCodeCombiner.AddObject(base.GetFileHash(virtualPath, clonedPaths));
return hashCodeCombiner.CombinedHashString;
}
internal void RegisterVirtualFile(VirtualPath virtualPath, String virtualFileString) {
_stringDictionary[virtualPath.VirtualPathString] = virtualFileString;
}
internal void RevertVirtualFile(VirtualPath virtualPath) {
_stringDictionary.Remove(virtualPath.VirtualPathString);
}
internal class ClientVirtualFile : VirtualFile {
String _virtualFileString;
internal ClientVirtualFile(string virtualPath, String virtualFileString) : base(virtualPath) {
_virtualFileString = virtualFileString;
}
public override Stream Open() {
Stream stream = new MemoryStream();
StreamWriter writer = new StreamWriter(stream, Encoding.Unicode);
writer.Write(_virtualFileString);
writer.Flush();
stream.Seek(0, SeekOrigin.Begin);
return stream;
}
}
}
}
}