You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			651 lines
		
	
	
		
			29 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			651 lines
		
	
	
		
			29 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //------------------------------------------------------------------------------ | ||
|  | // <copyright file="ScriptReference.cs" company="Microsoft"> | ||
|  | //     Copyright (c) Microsoft Corporation.  All rights reserved. | ||
|  | // </copyright> | ||
|  | //------------------------------------------------------------------------------ | ||
|  | 
 | ||
|  | namespace System.Web.UI { | ||
|  |     using System; | ||
|  |     using System.Collections; | ||
|  |     using System.ComponentModel; | ||
|  |     using System.Diagnostics.CodeAnalysis; | ||
|  |     using System.Globalization; | ||
|  |     using System.Reflection; | ||
|  |     using System.Web; | ||
|  |     using System.Web.Handlers; | ||
|  |     using System.Web.Resources; | ||
|  |     using System.Web.Util; | ||
|  |     using Debug = System.Diagnostics.Debug; | ||
|  | 
 | ||
|  |     [ | ||
|  |     DefaultProperty("Path"), | ||
|  |     ] | ||
|  |     public class ScriptReference : ScriptReferenceBase { | ||
|  |         // Maps Tuple<string, Assembly>(resource name, assembly) to string (partial script path) | ||
|  |         private static readonly Hashtable _scriptPathCache = Hashtable.Synchronized(new Hashtable()); | ||
|  | 
 | ||
|  |         private string _assembly; | ||
|  |         private bool _ignoreScriptPath; | ||
|  |         private string _name; | ||
|  |         private ScriptEffectiveInfo _scriptInfo; | ||
|  | 
 | ||
|  |         public ScriptReference() : base() { } | ||
|  | 
 | ||
|  |         public ScriptReference(string name, string assembly) | ||
|  |             : this() { | ||
|  |             Name = name; | ||
|  |             Assembly = assembly; | ||
|  |         } | ||
|  | 
 | ||
|  |         public ScriptReference(string path) | ||
|  |             : this() { | ||
|  |             Path = path; | ||
|  |         } | ||
|  | 
 | ||
|  |         internal ScriptReference(string name, IClientUrlResolver clientUrlResolver, Control containingControl) | ||
|  |             : this() { | ||
|  |             Debug.Assert(!String.IsNullOrEmpty(name), "The script's name must be specified."); | ||
|  |             Debug.Assert(clientUrlResolver != null && clientUrlResolver is ScriptManager, "The clientUrlResolver must be the ScriptManager."); | ||
|  |             Name = name; | ||
|  |             ClientUrlResolver = clientUrlResolver; | ||
|  |             IsStaticReference = true; | ||
|  |             ContainingControl = containingControl; | ||
|  |         } | ||
|  | 
 | ||
|  |         internal bool IsDirectRegistration { | ||
|  |             // set to true internally to disable checking for adding .debug | ||
|  |             // used when registering a script directly through SM.RegisterClientScriptResource | ||
|  |             get; | ||
|  |             set; | ||
|  |         } | ||
|  | 
 | ||
|  |         [ | ||
|  |         Category("Behavior"), | ||
|  |         DefaultValue(""), | ||
|  |         ResourceDescription("ScriptReference_Assembly") | ||
|  |         ] | ||
|  |         public string Assembly { | ||
|  |             get { | ||
|  |                 return (_assembly == null) ? String.Empty : _assembly; | ||
|  |             } | ||
|  |             set { | ||
|  |                 _assembly = value; | ||
|  |                 _scriptInfo = null; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal Assembly EffectiveAssembly { | ||
|  |             get { | ||
|  |                 return ScriptInfo.Assembly; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal string EffectivePath { | ||
|  |             get { | ||
|  |                 return String.IsNullOrEmpty(Path) ? ScriptInfo.Path : Path; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal string EffectiveResourceName { | ||
|  |             get { | ||
|  |                 return ScriptInfo.ResourceName; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal ScriptMode EffectiveScriptMode { | ||
|  |             get { | ||
|  |                 if (ScriptMode == ScriptMode.Auto) { | ||
|  |                     // - When a mapping.DebugPath exists, ScriptMode.Auto is equivilent to ScriptMode.Inherit, | ||
|  |                     //   since a debug path exists, even though it may not be an assembly based script. | ||
|  |                     // - An explicitly set Path on the ScriptReference effectively ignores a DebugPath. | ||
|  |                     // - When only Path is specified, ScriptMode.Auto is equivalent to ScriptMode.Release. | ||
|  |                     // - When only Name is specified, ScriptMode.Auto is equivalent to ScriptMode.Inherit. | ||
|  |                     // - When Name and Path are both specified, the Path is used instead of the Name, but | ||
|  |                     //   ScriptMode.Auto is still equivalent to ScriptMode.Inherit, since the assumption | ||
|  |                     //   is that if the Assembly contains both release and debug scripts, the Path should | ||
|  |                     //   contain both as well. | ||
|  |                     return ((String.IsNullOrEmpty(EffectiveResourceName) && | ||
|  |                         (!String.IsNullOrEmpty(Path) || String.IsNullOrEmpty(ScriptInfo.DebugPath))) ? | ||
|  |                         ScriptMode.Release : ScriptMode.Inherit); | ||
|  |                 } | ||
|  |                 else { | ||
|  |                     return ScriptMode; | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         [ | ||
|  |         Category("Behavior"), | ||
|  |         DefaultValue(false), | ||
|  |         ResourceDescription("ScriptReference_IgnoreScriptPath"), | ||
|  |         Obsolete("This property is obsolete. Instead of using ScriptManager.ScriptPath, set the Path property on each individual ScriptReference.") | ||
|  |         ] | ||
|  |         public bool IgnoreScriptPath { | ||
|  |             get { | ||
|  |                 return _ignoreScriptPath; | ||
|  |             } | ||
|  |             set { | ||
|  |                 _ignoreScriptPath = value; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         [ | ||
|  |         Category("Behavior"), | ||
|  |         DefaultValue(""), | ||
|  |         ResourceDescription("ScriptReference_Name") | ||
|  |         ] | ||
|  |         public string Name { | ||
|  |             get { | ||
|  |                 return (_name == null) ? String.Empty : _name; | ||
|  |             } | ||
|  |             set { | ||
|  |                 _name = value; | ||
|  |                 _scriptInfo = null; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal ScriptEffectiveInfo ScriptInfo { | ||
|  |             get { | ||
|  |                 if (_scriptInfo == null) { | ||
|  |                     _scriptInfo = new ScriptEffectiveInfo(this); | ||
|  |                 } | ||
|  |                 return _scriptInfo; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         private string AddCultureName(ScriptManager scriptManager, string resourceName) { | ||
|  |             Debug.Assert(!String.IsNullOrEmpty(resourceName)); | ||
|  |             CultureInfo culture = (scriptManager.EnableScriptLocalization ? | ||
|  |                 DetermineCulture(scriptManager) : CultureInfo.InvariantCulture); | ||
|  |             if (!culture.Equals(CultureInfo.InvariantCulture)) { | ||
|  |                 return AddCultureName(culture, resourceName); | ||
|  |             } | ||
|  |             else { | ||
|  |                 return resourceName; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         private static string AddCultureName(CultureInfo culture, string resourceName) { | ||
|  |             if (resourceName.EndsWith(".js", StringComparison.OrdinalIgnoreCase)) { | ||
|  |                 resourceName = resourceName.Substring(0, resourceName.Length - 2) + | ||
|  |                     culture.Name + ".js"; | ||
|  |             } | ||
|  |             return resourceName; | ||
|  |         } | ||
|  | 
 | ||
|  |         internal bool DetermineResourceNameAndAssembly(ScriptManager scriptManager, bool isDebuggingEnabled, ref string resourceName, ref Assembly assembly) { | ||
|  |             // If the assembly is the AjaxFrameworkAssembly, the resource may come from that assembly | ||
|  |             // or from the fallback assembly (SWE). | ||
|  |             if (assembly == scriptManager.AjaxFrameworkAssembly) { | ||
|  |                 assembly = ApplyFallbackResource(assembly, resourceName); | ||
|  |             } | ||
|  |             // ShouldUseDebugScript throws exception if the resource name does not exist in the assembly | ||
|  |             bool isDebug = ShouldUseDebugScript(resourceName, assembly, | ||
|  |                 isDebuggingEnabled, scriptManager.AjaxFrameworkAssembly); | ||
|  |             if (isDebug) { | ||
|  |                 resourceName = GetDebugName(resourceName); | ||
|  |             } | ||
|  |             // returning true means the debug version is selected | ||
|  |             return isDebug; | ||
|  |         } | ||
|  | 
 | ||
|  |         internal CultureInfo DetermineCulture(ScriptManager scriptManager) { | ||
|  |             if ((ResourceUICultures == null) || (ResourceUICultures.Length == 0)) { | ||
|  |                 // In this case we want to determine available cultures from assembly info if available | ||
|  |                 if (!String.IsNullOrEmpty(EffectiveResourceName)) { | ||
|  |                     return ScriptResourceHandler | ||
|  |                         .DetermineNearestAvailableCulture(GetAssembly(scriptManager), EffectiveResourceName, CultureInfo.CurrentUICulture); | ||
|  |                 } | ||
|  |                 return CultureInfo.InvariantCulture; | ||
|  |             } | ||
|  |             CultureInfo currentCulture = CultureInfo.CurrentUICulture; | ||
|  |             while (!currentCulture.Equals(CultureInfo.InvariantCulture)) { | ||
|  |                 string cultureName = currentCulture.ToString(); | ||
|  |                 foreach (string uiCulture in ResourceUICultures) { | ||
|  |                     if (String.Equals(cultureName, uiCulture.Trim(), StringComparison.OrdinalIgnoreCase)) { | ||
|  |                         return currentCulture; | ||
|  |                     } | ||
|  |                 } | ||
|  |                 currentCulture = currentCulture.Parent; | ||
|  |             } | ||
|  |             return currentCulture; | ||
|  |         } | ||
|  | 
 | ||
|  |         internal Assembly GetAssembly() { | ||
|  |             return String.IsNullOrEmpty(Assembly) ? null : AssemblyCache.Load(Assembly); | ||
|  |         } | ||
|  | 
 | ||
|  |         internal Assembly GetAssembly(ScriptManager scriptManager) { | ||
|  |             // normalizes the effective assembly by redirecting it to the given scriptmanager's | ||
|  |             // ajax framework assembly when it is set to SWE. | ||
|  |             // EffectiveAssembly can't do this since ScriptReference does not have access by itself | ||
|  |             // to the script manager. | ||
|  |             Debug.Assert(scriptManager != null); | ||
|  |             Assembly assembly = EffectiveAssembly; | ||
|  |             if (assembly == null) { | ||
|  |                 return scriptManager.AjaxFrameworkAssembly; | ||
|  |             } | ||
|  |             else { | ||
|  |                 return ((assembly == AssemblyCache.SystemWebExtensions) ? | ||
|  |                     scriptManager.AjaxFrameworkAssembly : | ||
|  |                     assembly); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         // Release: foo.js | ||
|  |         // Debug:   foo.debug.js | ||
|  |         private static string GetDebugName(string releaseName) { | ||
|  |             // Since System.Web.Handlers.AssemblyResourceLoader treats the resource name as case-sensitive, | ||
|  |             // we must do the same when verifying the extension. | ||
|  |             // Ignore trailing whitespace. For example, "MicrosoftAjax.js " is valid (at least from | ||
|  |             // a debug/release naming perspective). | ||
|  |             if (!releaseName.EndsWith(".js", StringComparison.Ordinal)) { | ||
|  |                 throw new InvalidOperationException( | ||
|  |                     String.Format(CultureInfo.CurrentUICulture, AtlasWeb.ScriptReference_InvalidReleaseScriptName, releaseName)); | ||
|  |             } | ||
|  | 
 | ||
|  |             return ReplaceExtension(releaseName); | ||
|  |         } | ||
|  | 
 | ||
|  |         internal string GetPath(ScriptManager scriptManager, string releasePath, string predeterminedDebugPath, bool isDebuggingEnabled) { | ||
|  |             // convert the release path to a debug path if: | ||
|  |             // isDebuggingEnabled && not resource based | ||
|  |             // isDebuggingEnabled && resource based && debug resource exists | ||
|  |             // ShouldUseDebugScript is called even when isDebuggingEnabled=false as it verfies | ||
|  |             // the existence of the resource. | ||
|  |             // applies the culture name to the path if appropriate | ||
|  |             string path; | ||
|  |             if (!String.IsNullOrEmpty(EffectiveResourceName)) { | ||
|  |                 Assembly assembly = GetAssembly(scriptManager); | ||
|  |                 string resourceName = EffectiveResourceName; | ||
|  |                 isDebuggingEnabled = DetermineResourceNameAndAssembly(scriptManager, isDebuggingEnabled, ref resourceName, ref assembly); | ||
|  |             } | ||
|  | 
 | ||
|  |             if (isDebuggingEnabled) { | ||
|  |                 // Just use predeterminedDebugPath if it is provided. This may be because | ||
|  |                 // a script mapping has DebugPath set. If it is empty or null, then '.debug' is added | ||
|  |                 // to the .js extension of the release path. | ||
|  |                 path = String.IsNullOrEmpty(predeterminedDebugPath) ? GetDebugPath(releasePath) : predeterminedDebugPath; | ||
|  |             } | ||
|  |             else { | ||
|  |                 path = releasePath; | ||
|  |             } | ||
|  |             return AddCultureName(scriptManager, path); | ||
|  |         } | ||
|  | 
 | ||
|  |         internal Assembly ApplyFallbackResource(Assembly assembly, string releaseName) { | ||
|  |             // fall back to SWE if the assembly does not contain the requested resource | ||
|  |             if ((assembly != AssemblyCache.SystemWebExtensions) && | ||
|  |                 !WebResourceUtil.AssemblyContainsWebResource(assembly, releaseName)) { | ||
|  |                 assembly = AssemblyCache.SystemWebExtensions; | ||
|  |             } | ||
|  |             return assembly; | ||
|  |         } | ||
|  | 
 | ||
|  |         // Format: <ScriptPath>/<AssemblyName>/<AssemblyVersion>/<ResourceName> | ||
|  |         // This function does not canonicalize the path in any way (i.e. remove duplicate slashes). | ||
|  |         // You must call ResolveClientUrl() on this path before rendering to the page. | ||
|  |         internal static string GetScriptPath( | ||
|  |             string resourceName, | ||
|  |             Assembly assembly, | ||
|  |             CultureInfo culture, | ||
|  |             string scriptPath) { | ||
|  | 
 | ||
|  |             return scriptPath + "/" + GetScriptPathCached(resourceName, assembly, culture); | ||
|  |         } | ||
|  | 
 | ||
|  |         // Cache partial script path, since Version.ToString() and HttpUtility.UrlEncode() are expensive. | ||
|  |         // Increases requests/second by 50% in ScriptManagerScriptPath.aspx test. | ||
|  |         private static string GetScriptPathCached(string resourceName, Assembly assembly, CultureInfo culture) { | ||
|  |             Tuple<string, Assembly, CultureInfo> key = Tuple.Create(resourceName, assembly, culture); | ||
|  |             string scriptPath = (string)_scriptPathCache[key]; | ||
|  | 
 | ||
|  |             if (scriptPath == null) { | ||
|  |                 // Need to use "new AssemblyName(assembly.FullName)" instead of "assembly.GetName()", | ||
|  |                 // since Assembly.GetName() requires FileIOPermission to the path of the assembly. | ||
|  |                 // In partial trust, we may not have this permission. | ||
|  |                 AssemblyName assemblyName = new AssemblyName(assembly.FullName); | ||
|  | 
 | ||
|  |                 string name = assemblyName.Name; | ||
|  |                 string version = assemblyName.Version.ToString(); | ||
|  |                 string fileVersion = AssemblyUtil.GetAssemblyFileVersion(assembly); | ||
|  | 
 | ||
|  |                 if (!culture.Equals(CultureInfo.InvariantCulture)) { | ||
|  |                     resourceName = AddCultureName(culture, resourceName); | ||
|  |                 } | ||
|  | 
 | ||
|  |                 // Assembly name, fileVersion, and resource name may contain invalid URL characters (like '#' or '/'), | ||
|  |                 // so they must be url-encoded. | ||
|  |                 scriptPath = String.Join("/", new string[] { | ||
|  |                     HttpUtility.UrlEncode(name), version, HttpUtility.UrlEncode(fileVersion), HttpUtility.UrlEncode(resourceName) | ||
|  |                 }); | ||
|  | 
 | ||
|  |                 _scriptPathCache[key] = scriptPath; | ||
|  |             } | ||
|  | 
 | ||
|  |             return scriptPath; | ||
|  |         } | ||
|  | 
 | ||
|  |         [SuppressMessage("Microsoft.Design", "CA1055", Justification = "Consistent with other URL properties in ASP.NET.")] | ||
|  |         protected internal override string GetUrl(ScriptManager scriptManager, bool zip) { | ||
|  |             bool hasName = !String.IsNullOrEmpty(Name); | ||
|  |             bool hasAssembly = !String.IsNullOrEmpty(Assembly); | ||
|  | 
 | ||
|  |             if (!hasName && String.IsNullOrEmpty(Path)) { | ||
|  |                 throw new InvalidOperationException(AtlasWeb.ScriptReference_NameAndPathCannotBeEmpty); | ||
|  |             } | ||
|  | 
 | ||
|  |             if (hasAssembly && !hasName) { | ||
|  |                 throw new InvalidOperationException(AtlasWeb.ScriptReference_AssemblyRequiresName); | ||
|  |             } | ||
|  | 
 | ||
|  |             return GetUrlInternal(scriptManager, zip); | ||
|  |         } | ||
|  | 
 | ||
|  |         internal string GetUrlInternal(ScriptManager scriptManager, bool zip) { | ||
|  |             bool enableCdn = scriptManager != null && scriptManager.EnableCdn; | ||
|  |             return GetUrlInternal(scriptManager, zip, useCdnPath: enableCdn); | ||
|  |         } | ||
|  | 
 | ||
|  |         internal string GetUrlInternal(ScriptManager scriptManager, bool zip, bool useCdnPath) { | ||
|  |             if (!String.IsNullOrEmpty(EffectiveResourceName) && !IsAjaxFrameworkScript(scriptManager) && | ||
|  |                 AssemblyCache.IsAjaxFrameworkAssembly(GetAssembly(scriptManager))) { | ||
|  |                 // it isnt an AjaxFrameworkScript but it might be from an assembly that is meant to | ||
|  |                 // be an ajax script assembly, in which case we should throw an error. | ||
|  |                 throw new InvalidOperationException(String.Format(CultureInfo.CurrentUICulture, | ||
|  |                     AtlasWeb.ScriptReference_ResourceRequiresAjaxAssembly, EffectiveResourceName, GetAssembly(scriptManager))); | ||
|  |             } | ||
|  | 
 | ||
|  |             if (!String.IsNullOrEmpty(Path)) { | ||
|  |                 // if an explicit path is set on the SR (not on a mapping) it always | ||
|  |                 // takes precedence, even when EnableCdn=true. | ||
|  |                 // Also, even if a script mapping has a DebugPath, the explicitly set path | ||
|  |                 // overrides all -- so path.debug.js is used instead of the mapping's DebugPath, | ||
|  |                 // hence the null 3rd parameter. | ||
|  |                 return GetUrlFromPath(scriptManager, Path, null); | ||
|  |             } | ||
|  |             else if (!String.IsNullOrEmpty(ScriptInfo.Path)) { | ||
|  |                 // when only the mapping has a path, CDN takes first priority | ||
|  |                 if (useCdnPath) { | ||
|  |                     // first determine the actual resource name and assembly to be used | ||
|  |                     // This is so we can (1) apply fallback logic, where ajax fx scripts can come from the  | ||
|  |                     // current ajax assembly or from SWE, whichever is first, and (2) the .debug resource | ||
|  |                     // name is applied if appropriate. | ||
|  |                     string resourceName = EffectiveResourceName; | ||
|  |                     Assembly assembly = null; | ||
|  |                     bool hasDebugResource = false; | ||
|  |                     if (!String.IsNullOrEmpty(resourceName)) { | ||
|  |                         assembly = GetAssembly(scriptManager); | ||
|  |                         hasDebugResource = DetermineResourceNameAndAssembly(scriptManager, | ||
|  |                             IsDebuggingEnabled(scriptManager), ref resourceName, ref assembly); | ||
|  |                     } | ||
|  |                     string cdnPath = GetUrlForCdn(scriptManager, resourceName, assembly, hasDebugResource); | ||
|  |                     if (!String.IsNullOrEmpty(cdnPath)) { | ||
|  |                         return cdnPath; | ||
|  |                     } | ||
|  |                 } | ||
|  |                 // the mapping's DebugPath applies if it exists | ||
|  |                 return GetUrlFromPath(scriptManager, ScriptInfo.Path, ScriptInfo.DebugPath); | ||
|  |             } | ||
|  | 
 | ||
|  |             Debug.Assert(!String.IsNullOrEmpty(EffectiveResourceName)); | ||
|  |             return GetUrlFromName(scriptManager, scriptManager.Control, zip, useCdnPath); | ||
|  |         } | ||
|  | 
 | ||
|  |         private string GetUrlForCdn(ScriptManager scriptManager, string resourceName, Assembly assembly, bool hasDebugResource) { | ||
|  |             // if EnableCdn, then url always comes from mapping.Cdn[Debug]Path or WRA.CdnPath, if available. | ||
|  |             // first see if the script description mapping has a cdn path defined | ||
|  |             bool isDebuggingEnabled = IsDebuggingEnabled(scriptManager); | ||
|  |             bool isAssemblyResource = !String.IsNullOrEmpty(resourceName); | ||
|  |             bool secureConnection = scriptManager.IsSecureConnection; | ||
|  |             isDebuggingEnabled = isDebuggingEnabled && (hasDebugResource || !isAssemblyResource); | ||
|  |             string cdnPath = isDebuggingEnabled ? | ||
|  |                 (secureConnection ? ScriptInfo.CdnDebugPathSecureConnection : ScriptInfo.CdnDebugPath) : | ||
|  |                 (secureConnection ? ScriptInfo.CdnPathSecureConnection : ScriptInfo.CdnPath); | ||
|  | 
 | ||
|  |             // then see if the WebResourceAttribute for the resource has one | ||
|  |             // EXCEPT when the ScriptInfo has a cdnpath but it wasn't selected due to this being a secure connection | ||
|  |             // and it does not support secure connections. Avoid having the HTTP cdn path come from the mapping and the | ||
|  |             // HTTPS path come from the WRA. | ||
|  |             if (isAssemblyResource && String.IsNullOrEmpty(cdnPath) && String.IsNullOrEmpty(isDebuggingEnabled ? ScriptInfo.CdnDebugPath : ScriptInfo.CdnPath)) { | ||
|  |                 ScriptResourceInfo scriptResourceInfo = ScriptResourceInfo.GetInstance(assembly, resourceName); | ||
|  |                 if (scriptResourceInfo != null) { | ||
|  |                     cdnPath = secureConnection ? scriptResourceInfo.CdnPathSecureConnection : scriptResourceInfo.CdnPath; | ||
|  |                 } | ||
|  |             } | ||
|  |             return String.IsNullOrEmpty(cdnPath) ? null : ClientUrlResolver.ResolveClientUrl(AddCultureName(scriptManager, cdnPath)); | ||
|  |         } | ||
|  | 
 | ||
|  |         private string GetUrlFromName(ScriptManager scriptManager, IControl scriptManagerControl, bool zip, bool useCdnPath) { | ||
|  |             string resourceName = EffectiveResourceName; | ||
|  |             Assembly assembly = GetAssembly(scriptManager); | ||
|  |             bool hasDebugResource = DetermineResourceNameAndAssembly(scriptManager, IsDebuggingEnabled(scriptManager), | ||
|  |                 ref resourceName, ref assembly); | ||
|  | 
 | ||
|  |             if (useCdnPath) { | ||
|  |                 string cdnPath = GetUrlForCdn(scriptManager, resourceName, assembly, hasDebugResource); | ||
|  |                 if (!String.IsNullOrEmpty(cdnPath)) { | ||
|  |                     return cdnPath; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             CultureInfo culture = (scriptManager.EnableScriptLocalization ? | ||
|  |                 DetermineCulture(scriptManager) : CultureInfo.InvariantCulture); | ||
|  | #pragma warning disable 618 | ||
|  |             // ScriptPath is obsolete but still functional | ||
|  |             if (IgnoreScriptPath || String.IsNullOrEmpty(scriptManager.ScriptPath)) { | ||
|  |                 return ScriptResourceHandler.GetScriptResourceUrl(assembly, resourceName, culture, zip); | ||
|  |             } | ||
|  |             else { | ||
|  |                 string path = GetScriptPath(resourceName, assembly, culture, scriptManager.ScriptPath); | ||
|  | 
 | ||
|  |                 if (IsBundleReference) { | ||
|  |                     return scriptManager.BundleReflectionHelper.GetBundleUrl(path); | ||
|  |                 } | ||
|  | 
 | ||
|  |                 // Always want to resolve ScriptPath urls against the ScriptManager itself, | ||
|  |                 // regardless of whether the ScriptReference was declared on the ScriptManager | ||
|  |                 // or a ScriptManagerProxy. | ||
|  |                 return scriptManagerControl.ResolveClientUrl(path); | ||
|  |             } | ||
|  | #pragma warning restore 618 | ||
|  |         } | ||
|  | 
 | ||
|  |         private string GetUrlFromPath(ScriptManager scriptManager, string releasePath, string predeterminedDebugPath) { | ||
|  |             string path = GetPath(scriptManager, releasePath, predeterminedDebugPath, IsDebuggingEnabled(scriptManager)); | ||
|  |             if (IsBundleReference) { | ||
|  |                 return scriptManager.BundleReflectionHelper.GetBundleUrl(path); | ||
|  |             } | ||
|  | 
 | ||
|  |             return ClientUrlResolver.ResolveClientUrl(path); | ||
|  |         } | ||
|  | 
 | ||
|  |         private bool IsDebuggingEnabled(ScriptManager scriptManager) { | ||
|  |             // Deployment mode retail overrides all values of ScriptReference.ScriptMode. | ||
|  |             if (IsDirectRegistration || scriptManager.DeploymentSectionRetail) { | ||
|  |                 return false; | ||
|  |             } | ||
|  | 
 | ||
|  |             switch (EffectiveScriptMode) { | ||
|  |                 case ScriptMode.Inherit: | ||
|  |                     return scriptManager.IsDebuggingEnabled; | ||
|  |                 case ScriptMode.Debug: | ||
|  |                     return true; | ||
|  |                 case ScriptMode.Release: | ||
|  |                     return false; | ||
|  |                 default: | ||
|  |                     Debug.Fail("Invalid value for ScriptReference.EffectiveScriptMode"); | ||
|  |                     return false; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         protected internal override bool IsAjaxFrameworkScript(ScriptManager scriptManager) { | ||
|  |             return (GetAssembly(scriptManager) == scriptManager.AjaxFrameworkAssembly); | ||
|  |         } | ||
|  | 
 | ||
|  |         [Obsolete("This method is obsolete. Use IsAjaxFrameworkScript(ScriptManager) instead.")] | ||
|  |         protected internal override bool IsFromSystemWebExtensions() { | ||
|  |             return (EffectiveAssembly == AssemblyCache.SystemWebExtensions); | ||
|  |         } | ||
|  | 
 | ||
|  |         internal bool IsFromSystemWeb() { | ||
|  |             return (EffectiveAssembly == AssemblyCache.SystemWeb); | ||
|  |         } | ||
|  | 
 | ||
|  |         internal bool ShouldUseDebugScript(string releaseName, Assembly assembly, | ||
|  |             bool isDebuggingEnabled, Assembly currentAjaxAssembly) { | ||
|  |             bool useDebugScript; | ||
|  |             string debugName = null; | ||
|  | 
 | ||
|  |             if (isDebuggingEnabled) { | ||
|  |                 debugName = GetDebugName(releaseName); | ||
|  | 
 | ||
|  |                 // If an assembly contains a release script but not a corresponding debug script, and we | ||
|  |                 // need to register the debug script, we normally throw an exception.  However, we automatically | ||
|  |                 // use the release script if ScriptReference.ScriptMode is Auto.  This improves the developer | ||
|  |                 // experience when ScriptMode is Auto, yet still gives the developer full control with the | ||
|  |                 // other ScriptModes. | ||
|  |                 if (ScriptMode == ScriptMode.Auto && !WebResourceUtil.AssemblyContainsWebResource(assembly, debugName)) { | ||
|  |                     useDebugScript = false; | ||
|  |                 } | ||
|  |                 else { | ||
|  |                     useDebugScript = true; | ||
|  |                 } | ||
|  |             } | ||
|  |             else { | ||
|  |                 useDebugScript = false; | ||
|  |             } | ||
|  | 
 | ||
|  |             // Verify that assembly contains required web resources.  Always check for release | ||
|  |             // script before debug script. | ||
|  |             if (!IsDirectRegistration) { | ||
|  |                 // Don't check if direct registration, because calls to ScriptManager.RegisterClientScriptResource | ||
|  |                 // with resources that do not exist does not throw an exception until the resource is served. | ||
|  |                 // This was the existing behavior and matches the behavior with ClientScriptManager.GetWebResourceUrl. | ||
|  |                 WebResourceUtil.VerifyAssemblyContainsReleaseWebResource(assembly, releaseName, currentAjaxAssembly); | ||
|  |                 if (useDebugScript) { | ||
|  |                     Debug.Assert(debugName != null); | ||
|  |                     WebResourceUtil.VerifyAssemblyContainsDebugWebResource(assembly, debugName); | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             return useDebugScript; | ||
|  |         } | ||
|  | 
 | ||
|  |         // Improves the UI in the VS collection editor, by displaying the Name or Path (if available), or | ||
|  |         // the short type name. | ||
|  |         [SuppressMessage("Microsoft.Security", "CA2123:OverrideLinkDemandsShouldBeIdenticalToBase")] | ||
|  |         public override string ToString() { | ||
|  |             if (!String.IsNullOrEmpty(Name)) { | ||
|  |                 return Name; | ||
|  |             } | ||
|  |             else if (!String.IsNullOrEmpty(Path)) { | ||
|  |                 return Path; | ||
|  |             } | ||
|  |             else { | ||
|  |                 return GetType().Name; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal class ScriptEffectiveInfo { | ||
|  |             private string _resourceName; | ||
|  |             private Assembly _assembly; | ||
|  |             private string _path; | ||
|  |             private string _debugPath; | ||
|  |             private string _cdnPath; | ||
|  |             private string _cdnDebugPath; | ||
|  |             private string _cdnPathSecureConnection; | ||
|  |             private string _cdnDebugPathSecureConnection; | ||
|  | 
 | ||
|  |             public ScriptEffectiveInfo(ScriptReference scriptReference) { | ||
|  |                 ScriptResourceDefinition definition = | ||
|  |                     ScriptManager.ScriptResourceMapping.GetDefinition(scriptReference); | ||
|  |                 string name = scriptReference.Name; | ||
|  |                 string path = scriptReference.Path; | ||
|  |                 Assembly assembly = scriptReference.GetAssembly(); | ||
|  |                 if (definition != null) { | ||
|  |                     if (String.IsNullOrEmpty(path)) { | ||
|  |                         // only when the SR has no path, the mapping's path and debug path, if any, apply | ||
|  |                         path = definition.Path; | ||
|  |                         _debugPath = definition.DebugPath; | ||
|  |                     } | ||
|  |                     name = definition.ResourceName; | ||
|  |                     assembly = definition.ResourceAssembly; | ||
|  |                     _cdnPath = definition.CdnPath; | ||
|  |                     _cdnDebugPath = definition.CdnDebugPath; | ||
|  |                     _cdnPathSecureConnection = definition.CdnPathSecureConnection; | ||
|  |                     _cdnDebugPathSecureConnection = definition.CdnDebugPathSecureConnection; | ||
|  |                     LoadSuccessExpression = definition.LoadSuccessExpression; | ||
|  |                 } | ||
|  |                 else if ((assembly == null) && !String.IsNullOrEmpty(name)) { | ||
|  |                     // name is set and there is no mapping, default to SWE for assembly | ||
|  |                     assembly = AssemblyCache.SystemWebExtensions; | ||
|  |                 } | ||
|  |                 _resourceName = name; | ||
|  |                 _assembly = assembly; | ||
|  |                 _path = path; | ||
|  | 
 | ||
|  |                 if (assembly != null && !String.IsNullOrEmpty(name) && String.IsNullOrEmpty(LoadSuccessExpression)) { | ||
|  |                     var scriptResourceInfo = ScriptResourceInfo.GetInstance(assembly, name); | ||
|  |                     if (scriptResourceInfo != null) { | ||
|  |                         LoadSuccessExpression = scriptResourceInfo.LoadSuccessExpression; | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             public Assembly Assembly { | ||
|  |                 get { | ||
|  |                     return _assembly; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             public string CdnDebugPath { | ||
|  |                 get { | ||
|  |                     return _cdnDebugPath; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             public string CdnPath { | ||
|  |                 get { | ||
|  |                     return _cdnPath; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             public string CdnDebugPathSecureConnection { | ||
|  |                 get { | ||
|  |                     return _cdnDebugPathSecureConnection; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             public string CdnPathSecureConnection { | ||
|  |                 get { | ||
|  |                     return _cdnPathSecureConnection; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             public string LoadSuccessExpression { | ||
|  |                 get; | ||
|  |                 private set; | ||
|  |             } | ||
|  | 
 | ||
|  |             public string DebugPath { | ||
|  |                 get { | ||
|  |                     return _debugPath; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             public string Path { | ||
|  |                 get { | ||
|  |                     return _path; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             public string ResourceName { | ||
|  |                 get { | ||
|  |                     return _resourceName; | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  |     } | ||
|  | } |