//------------------------------------------------------------------------------ // // Copyright (c) Microsoft Corporation. All rights reserved. // //------------------------------------------------------------------------------ namespace System.Web.UI { using System; using System.Collections; using System.Collections.Generic; using System.Diagnostics; using System.Linq; using System.Globalization; using System.Reflection; using System.Web.UI; using System.Web.Resources; using System.Web.Util; internal static class WebResourceUtil { // Maps Tuple(resourceName, assembly) to bool private static readonly Hashtable _assemblyContainsWebResourceCache = Hashtable.Synchronized(new Hashtable()); private static readonly FastStringLookupTable _systemWebExtensionsCache = CreateSystemWebExtensionsCache(); // Returns true if the assembly contains a Web resource and an embedded resource with // the sepecified name. Throws exception if assembly contains Web resource but no // embedded resource, since this is always an error. public static bool AssemblyContainsWebResource(Assembly assembly, string resourceName) { // PERF: Special-case known resources in our own assembly if (assembly == AssemblyCache.SystemWebExtensions) { return _systemWebExtensionsCache.Contains(resourceName); } // Getting and checking the custom attributes is expensive, so we cache the result // of the lookup. Tuple key = new Tuple(resourceName, assembly); object assemblyContainsWebResource = _assemblyContainsWebResourceCache[key]; if (assemblyContainsWebResource == null) { assemblyContainsWebResource = false; object[] attrs = assembly.GetCustomAttributes(typeof(WebResourceAttribute), false); foreach (WebResourceAttribute attr in attrs) { // Resource names are always case-sensitive if (String.Equals(attr.WebResource, resourceName, StringComparison.Ordinal)) { if (assembly.GetManifestResourceStream(resourceName) != null) { assemblyContainsWebResource = true; break; } else { // Always an error to contain Web resource but not embedded resource. throw new InvalidOperationException(String.Format( CultureInfo.CurrentUICulture, AtlasWeb.WebResourceUtil_AssemblyDoesNotContainEmbeddedResource, assembly, resourceName)); } } } _assemblyContainsWebResourceCache[key] = assemblyContainsWebResource; } return (bool)assemblyContainsWebResource; } private static FastStringLookupTable CreateSystemWebExtensionsCache() { Assembly assembly = AssemblyCache.SystemWebExtensions; object[] attrs = assembly.GetCustomAttributes(typeof(WebResourceAttribute), false); var resourceNames = from WebResourceAttribute attr in attrs select attr.WebResource; return new FastStringLookupTable(resourceNames); } // Throws exception if the assembly does not contain a Web resource and an embedded resource // with the specified name. public static void VerifyAssemblyContainsReleaseWebResource(Assembly assembly, string releaseResourceName, Assembly currentAjaxAssembly) { if (!AssemblyContainsWebResource(assembly, releaseResourceName)) { string errorMessage; if (assembly == AssemblyCache.SystemWebExtensions) { errorMessage = String.Format( CultureInfo.CurrentUICulture, AtlasWeb.WebResourceUtil_SystemWebExtensionsDoesNotContainReleaseWebResource, currentAjaxAssembly ?? assembly, releaseResourceName); } else { errorMessage = String.Format( CultureInfo.CurrentUICulture, AtlasWeb.WebResourceUtil_AssemblyDoesNotContainReleaseWebResource, assembly, releaseResourceName); } throw new InvalidOperationException(errorMessage); } } // Throws exception if the assembly does not contain a Web resource and an embedded resource // with the specified name. public static void VerifyAssemblyContainsDebugWebResource(Assembly assembly, string debugResourceName) { if (!AssemblyContainsWebResource(assembly, debugResourceName)) { throw new InvalidOperationException(String.Format( CultureInfo.CurrentUICulture, AtlasWeb.WebResourceUtil_AssemblyDoesNotContainDebugWebResource, assembly, debugResourceName)); } } private class FastStringLookupTable { // PERF: Switching over the length is more performant than switching over the string itself // or checking equality against each string. When switching over the string itself, the switch // is compiled to a lookup in a static Dictionary, which is 5-10 times slower than // switching over the length. Checking equality against each string ranges from equal performance // to 10 times slower, depending on how early a match is found. private readonly string[][] _table; public FastStringLookupTable(IEnumerable strings) { int longest = (from s in strings orderby s.Length descending select s.Length).First(); _table = new string[longest + 1][]; var groups = from s in strings group s by s.Length into g select g; foreach (var g in groups) { _table[g.Key] = g.ToArray(); } } public bool Contains(string s) { if (String.IsNullOrEmpty(s)) { return false; } if (s.Length >= _table.Length) { return false; } string[] strings = _table[s.Length]; if (strings == null) { return false; } for (int i = 0; i < strings.Length; i++) { if (s == strings[i]) { return true; } } return false; } } } }