//------------------------------------------------------------------------------ // // Copyright (c) Microsoft Corporation. All rights reserved. // //------------------------------------------------------------------------------ namespace System.Web.Configuration { using System; using System.CodeDom; using System.CodeDom.Compiler; using System.Configuration; using System.Collections; using System.Collections.Specialized; using System.IO; #if !FEATURE_PAL using System.ServiceProcess; #endif // !FEATURE_PAL using System.Linq; using System.Reflection; using System.Security; using System.Security.Permissions; using System.Text; using System.Text.RegularExpressions; using System.Web; using System.Web.Compilation; using System.Web.Configuration; using System.Web.Hosting; using System.Web.UI; using System.Web.Util; using System.Xml; using System.Xml.Schema; using Microsoft.Build.Utilities; using Microsoft.CSharp; using System.Diagnostics.CodeAnalysis; [PermissionSet(SecurityAction.LinkDemand, Unrestricted = true)] [PermissionSet(SecurityAction.InheritanceDemand, Unrestricted = true)] public class BrowserCapabilitiesCodeGenerator { private static readonly string _browsersDirectory; private static readonly string _publicKeyTokenFile; private static object _staticLock = new object(); private BrowserTree _browserTree; private BrowserTree _defaultTree; private BrowserDefinitionCollection _browserDefinitionCollection; internal const string browserCapsVariable = "browserCaps"; internal const string IgnoreApplicationBrowserVariableName = "ignoreApplicationBrowsers"; private const string _factoryTypeName = "BrowserCapabilitiesFactory"; private const string _headerDictionaryVarName = "_headerDictionary"; private const string _disableOptimizedCacheKeyMethodName = "DisableOptimizedCacheKey"; private const string _matchedHeadersMethodName = "PopulateMatchedHeaders"; private const string _browserElementsMethodName = "PopulateBrowserElements"; private const string _dictionaryRefName = "dictionary"; private const string _regexWorkerRefName = "regexWorker"; private const string _headersRefName = "headers"; private const string _resultVarName = "result"; private const string _processRegexMethod = "ProcessRegex"; private static readonly string _strongNameKeyFileName = browserCapsVariable + ".snk"; private static readonly string _publicKeyTokenFileName = browserCapsVariable + ".token"; private static bool _publicKeyTokenLoaded; private static string _publicKeyToken; private CodeVariableReferenceExpression _dictionaryRefExpr = new CodeVariableReferenceExpression(_dictionaryRefName); private CodeVariableReferenceExpression _regexWorkerRefExpr = new CodeVariableReferenceExpression(_regexWorkerRefName); private CodeVariableReferenceExpression _headersRefExpr = new CodeVariableReferenceExpression(_headersRefName); private CodeVariableReferenceExpression _browserCapsRefExpr = new CodeVariableReferenceExpression(browserCapsVariable); private ArrayList _browserFileList; private ArrayList _customBrowserFileLists; private ArrayList _customTreeList; private ArrayList _customTreeNames; private ArrayList _customBrowserDefinitionCollections; private CaseInsensitiveStringSet _headers; static BrowserCapabilitiesCodeGenerator() { #if !PLATFORM_UNIX // File system paths must account for UNIX _browsersDirectory = HttpRuntime.ClrInstallDirectoryInternal + "\\config\\browsers"; _publicKeyTokenFile = _browsersDirectory + "\\" + _publicKeyTokenFileName; #else // !PLATFORM_UNIX _browsersDirectory = HttpRuntime.ClrInstallDirectoryInternal + "/config/browsers"; _publicKeyTokenFile = _browsersDirectory + "/" + _publicKeyTokenFileName; #endif // !PLATFORM_UNIX } public BrowserCapabilitiesCodeGenerator() { _headers = new CaseInsensitiveStringSet(); } internal BrowserTree BrowserTree { get { return _browserTree; } } internal BrowserTree DefaultTree { get { return _defaultTree; } } internal ArrayList CustomTreeList { get { return _customTreeList; } } internal ArrayList CustomTreeNames { get { return _customTreeNames; } } internal static string BrowserCapAssemblyPublicKeyToken { get { if (_publicKeyTokenLoaded) { return _publicKeyToken; } lock (_staticLock) { if (_publicKeyTokenLoaded) { return _publicKeyToken; } string publicKeyTokenFile; if (MultiTargetingUtil.IsTargetFramework40OrAbove) { publicKeyTokenFile = _publicKeyTokenFile; } else { // If we are targeting pre-4.0, we should be using version 2.0 of the assembly // ASP.BrowserCapsFactory, so we need to read the token file from the 2.0 path. // (Dev10 bug 795509) string subPath = @"config\browsers\" + _publicKeyTokenFileName; publicKeyTokenFile = ToolLocationHelper.GetPathToDotNetFrameworkFile(subPath, TargetDotNetFrameworkVersion.Version20); } _publicKeyToken = LoadPublicKeyTokenFromFile(publicKeyTokenFile); _publicKeyTokenLoaded = true; return _publicKeyToken; } } } internal virtual bool GenerateOverrides { get { return true; } } internal virtual string TypeName { get { return _factoryTypeName; } } internal void AddFile(string filePath) { if (_browserFileList == null) _browserFileList = new ArrayList(); _browserFileList.Add(filePath); } internal void AddCustomFile(string filePath) { if (_customBrowserFileLists == null) { _customBrowserFileLists = new ArrayList(); } _customBrowserFileLists.Add(filePath); } //parse the config info and create BrowserTree //then generate code for, compile, and gac the object [SecurityPermission(SecurityAction.Demand, Unrestricted=true)] public virtual void Create() { DirectoryInfo browserDirInfo = new DirectoryInfo(_browsersDirectory); //get all the browser files and put them in the "tree" FileInfo[] browserFiles = browserDirInfo.GetFiles("*.browser"); if (browserFiles == null || browserFiles.Length == 0) { return; } foreach(FileInfo browserFile in browserFiles) { AddFile(browserFile.FullName); } // First parse the browser files. ProcessBrowserFiles(); // Then parse custom browser files. ProcessCustomBrowserFiles(); // Uninstall previously installed generated assembly. Uninstall(); //generate the source code, compile it, and gac it GenerateAssembly(); // Restart w3svc service RestartW3SVCIfNecessary(); } internal bool UninstallInternal() { // Remove existing strong name public token file if (File.Exists(_publicKeyTokenFile)) { File.Delete(_publicKeyTokenFile); } // Removing existing copy from GAC GacUtil gacutil = new GacUtil(); bool assemblyRemoved = gacutil.GacUnInstall("ASP.BrowserCapsFactory, Version=" + ThisAssembly.Version + ", Culture=neutral"); if (!assemblyRemoved) { return false; } return true; } [SecurityPermission(SecurityAction.Demand, Unrestricted = true)] public bool Uninstall() { // Restart w3svc service RestartW3SVCIfNecessary(); if (!UninstallInternal()) { return false; } // Restart w3svc service again so applications get a fresh copy. RestartW3SVCIfNecessary(); return true; } private void RestartW3SVCIfNecessary() { #if !FEATURE_PAL try { // Dev10 bug 734918 // We should not fail when the w3svc service is not installed. ServiceController[] services = ServiceController.GetServices(); ServiceController controller = services.SingleOrDefault(s => String.Equals(s.ServiceName, "W3SVC", StringComparison.OrdinalIgnoreCase)); if (controller == null) { return; } ServiceControllerStatus status = controller.Status; // Stop the service if it's not currently stopped or pending. if (!status.Equals(ServiceControllerStatus.Stopped) && !status.Equals(ServiceControllerStatus.StopPending) && !status.Equals(ServiceControllerStatus.StartPending)) { controller.Stop(); // Give it 5 minutes to stop controller.WaitForStatus(ServiceControllerStatus.Stopped, new TimeSpan(0, 5, 0)); controller.Start(); // If the service was paused, pause it. if (status.Equals(ServiceControllerStatus.Paused) || status.Equals(ServiceControllerStatus.PausePending)) { controller.Pause(); } } } catch (Exception ex) { throw new InvalidOperationException(SR.GetString(SR.Browser_W3SVC_Failure_Helper_Text, ex)); } #endif // !FEATURE_PAL } internal void ProcessBrowserFiles() { ProcessBrowserFiles(false, String.Empty); } private string NoPathFileName(string fullPath) { int lastSlash = fullPath.LastIndexOf("\\", StringComparison.Ordinal); if(lastSlash > -1) { return fullPath.Substring(lastSlash + 1); } return fullPath; } internal virtual void ProcessBrowserNode(XmlNode node, BrowserTree browserTree) { BrowserDefinition browserInfo = null; if (node.Name == "gateway") { browserInfo = new GatewayDefinition(node); } else if (node.Name == "browser") { browserInfo = new BrowserDefinition(node); } else { Debug.Assert(node.Name == "defaultBrowser"); browserInfo = new BrowserDefinition(node, true); } BrowserDefinition oldNode = (BrowserDefinition)browserTree[browserInfo.Name]; if (oldNode != null) { if (browserInfo.IsRefID) { oldNode.MergeWithDefinition(browserInfo); } else { throw new ConfigurationErrorsException(SR.GetString(SR.Duplicate_browser_id, browserInfo.ID), node); } } else { browserTree[browserInfo.Name] = browserInfo; } } private void NormalizeAndValidateTree(BrowserTree browserTree, bool isDefaultBrowser) { NormalizeAndValidateTree(browserTree, isDefaultBrowser, false); } private void NormalizeAndValidateTree(BrowserTree browserTree, bool isDefaultBrowser, bool isCustomBrowser) { //normalize the tree foreach (DictionaryEntry entry in browserTree) { BrowserDefinition bd = (BrowserDefinition)entry.Value; string parentName = bd.ParentName; BrowserDefinition parentBrowser = null; if (IsRootNode(bd.Name)) { continue; } if (parentName.Length > 0) { parentBrowser = (BrowserDefinition)browserTree[parentName]; } if (parentBrowser != null) { if (bd.IsRefID) { if (bd is GatewayDefinition) { parentBrowser.RefGateways.Add(bd); } else { parentBrowser.RefBrowsers.Add(bd); } } else if (bd is GatewayDefinition) { parentBrowser.Gateways.Add(bd); } else { parentBrowser.Browsers.Add(bd); } } else { if (isCustomBrowser) { throw new ConfigurationErrorsException(SR.GetString(SR.Browser_parentID_Not_Found, bd.ParentID), bd.XmlNode); } else { HandleUnRecognizedParentElement(bd, isDefaultBrowser); } } } //validate the tree //loop check foreach (DictionaryEntry entry in browserTree) { BrowserDefinition bd = (BrowserDefinition)entry.Value; Hashtable loopCheck = new Hashtable(); BrowserDefinition currentBrowser = bd; string currentId = currentBrowser.Name; while (!IsRootNode(currentId)) { if (loopCheck[currentId] != null) { throw new ConfigurationErrorsException(SR.GetString(SR.Browser_Circular_Reference, currentId), currentBrowser.XmlNode); } loopCheck[currentId] = currentId; currentBrowser = (BrowserDefinition)browserTree[currentBrowser.ParentName]; //in app-level, parent can exist in machine level if (currentBrowser == null) { break; } currentId = currentBrowser.Name; } } } private void SetCustomTreeRoots(BrowserTree browserTree, int index) { foreach (DictionaryEntry entry in browserTree) { BrowserDefinition bd = (BrowserDefinition)entry.Value; if (bd.ParentName == null) { _customTreeNames[index] = bd.Name; break; } } } // Now that we support adding custom browser hierarchies, root nodes other than Default are permitted. private bool IsRootNode(string nodeName) { if (String.Compare(nodeName, "Default", StringComparison.OrdinalIgnoreCase) == 0) return true; foreach (string treeRootName in _customTreeNames) { if (String.Compare(nodeName, treeRootName, StringComparison.OrdinalIgnoreCase) == 0) { return true; } } return false; } [SuppressMessage("Microsoft.Security.Xml", "CA3056:UseXmlReaderForLoad", Justification = "Developer-controlled .xml files in application directory are implicitly trusted by ASP.Net.")] protected void ProcessBrowserFiles(bool useVirtualPath, string virtualDir) { _browserTree = new BrowserTree(); _defaultTree = new BrowserTree(); _customTreeNames = new ArrayList(); if (_browserFileList == null) { _browserFileList = new ArrayList(); } _browserFileList.Sort(); //#if OPTIMIZE_FOR_DESKTOP_BROWSER string mozillaFile = null; string ieFile = null; string operaFile = null; // DevDivBugs 180962 // IE, Mozilla and Opera are first-class browsers. Their User-Agent profiles need to be compared to the UA profile // of the HTTP request before other browsers. We put them to the head of the list so that the generated browser capabilities // code will try to match them before other browsers. foreach (String filePath in _browserFileList) { if (filePath.EndsWith("ie.browser", StringComparison.OrdinalIgnoreCase)) { ieFile = filePath; } else if (filePath.EndsWith("mozilla.browser", StringComparison.OrdinalIgnoreCase)) { mozillaFile = filePath; } else if (filePath.EndsWith("opera.browser", StringComparison.OrdinalIgnoreCase)) { operaFile = filePath; break; } } if (ieFile != null) { _browserFileList.Remove(ieFile); _browserFileList.Insert(0, ieFile); } if (mozillaFile != null) { _browserFileList.Remove(mozillaFile); _browserFileList.Insert(1, mozillaFile); } if (operaFile != null) { _browserFileList.Remove(operaFile); _browserFileList.Insert(2, operaFile); } //#endif foreach (string fileName in _browserFileList) { XmlDocument doc = new ConfigXmlDocument(); try { doc.Load(fileName); XmlNode rootNode = doc.DocumentElement; if(rootNode.Name != "browsers") { if(useVirtualPath) { throw new HttpParseException(SR.GetString(SR.Invalid_browser_root), null /*innerException*/, virtualDir + "/" + NoPathFileName(fileName), null /*sourceCode*/, 1); } else { throw new HttpParseException(SR.GetString(SR.Invalid_browser_root), null /*innerException*/, fileName, null /*sourceCode*/, 1); } } foreach (XmlNode node in rootNode.ChildNodes) { if (node.NodeType != XmlNodeType.Element) continue; if (node.Name == "browser" || node.Name == "gateway") { ProcessBrowserNode(node, _browserTree); } else if (node.Name == "defaultBrowser") { ProcessBrowserNode(node, _defaultTree); } else { HandlerBase.ThrowUnrecognizedElement(node); } } } catch (XmlException e) { if(useVirtualPath) { throw new HttpParseException(e.Message, null /*innerException*/, virtualDir + "/" + NoPathFileName(fileName), null /*sourceCode*/, e.LineNumber); } else { throw new HttpParseException(e.Message, null /*innerException*/, fileName, null /*sourceCode*/, e.LineNumber); } } catch (XmlSchemaException e) { if(useVirtualPath) { throw new HttpParseException(e.Message, null /*innerException*/, virtualDir + "/" + NoPathFileName(fileName), null /*sourceCode*/, e.LineNumber); } else { throw new HttpParseException(e.Message, null /*innerException*/, fileName, null /*sourceCode*/, e.LineNumber); } } } NormalizeAndValidateTree(_browserTree, false); NormalizeAndValidateTree(_defaultTree, true); BrowserDefinition defaultBrowser = (BrowserDefinition)_browserTree["Default"]; if (defaultBrowser != null) { AddBrowserToCollectionRecursive(defaultBrowser, 0); } } internal void ProcessCustomBrowserFiles() { ProcessCustomBrowserFiles(false, String.Empty); } [SuppressMessage("Microsoft.Security.Xml", "CA3056:UseXmlReaderForLoad", Justification = "Developer-controlled .xml files in application directory are implicitly trusted by ASP.Net.")] internal void ProcessCustomBrowserFiles(bool useVirtualPath, string virtualDir) { //get all custom browser files and put them in the "tree" DirectoryInfo browserDirInfo = null; DirectoryInfo[] browserSubDirectories = null; DirectoryInfo[] allBrowserSubDirectories = null; ArrayList customBrowserFileNames; _customTreeList = new ArrayList(); _customBrowserFileLists = new ArrayList(); _customBrowserDefinitionCollections = new ArrayList(); /* Machine Level Custom Browsers */ if (useVirtualPath == false) { browserDirInfo = new DirectoryInfo(_browsersDirectory); } /* Application Level Custom Browsers */ else { browserDirInfo = new DirectoryInfo(HostingEnvironment.MapPathInternal(virtualDir)); } allBrowserSubDirectories = browserDirInfo.GetDirectories(); int j = 0; int length = allBrowserSubDirectories.Length; browserSubDirectories = new DirectoryInfo[length]; for (int i = 0; i < length; i++) { if ((allBrowserSubDirectories[i].Attributes & FileAttributes.Hidden) != FileAttributes.Hidden) { browserSubDirectories[j] = allBrowserSubDirectories[i]; j++; } } Array.Resize(ref browserSubDirectories, j); for (int i = 0; i < browserSubDirectories.Length; i++) { /* Recursively Into Subdirectories */ FileInfo[] browserFiles = GetFilesNotHidden(browserSubDirectories[i], browserDirInfo); if (browserFiles == null || browserFiles.Length == 0) { continue; } BrowserTree customTree = new BrowserTree(); _customTreeList.Add(customTree); _customTreeNames.Add(browserSubDirectories[i].Name); customBrowserFileNames = new ArrayList(); foreach (FileInfo browserFile in browserFiles) { customBrowserFileNames.Add(browserFile.FullName); } _customBrowserFileLists.Add(customBrowserFileNames); } for (int i = 0; i < _customBrowserFileLists.Count; i++) { ArrayList fileNames = (ArrayList)_customBrowserFileLists[i]; foreach (string fileName in fileNames) { XmlDocument doc = new ConfigXmlDocument(); try { doc.Load(fileName); XmlNode rootNode = doc.DocumentElement; if (rootNode.Name != "browsers") { if (useVirtualPath) { throw new HttpParseException(SR.GetString(SR.Invalid_browser_root), null /*innerException*/, virtualDir + "/" + NoPathFileName(fileName), null /*sourceCode*/, 1); } else { throw new HttpParseException(SR.GetString(SR.Invalid_browser_root), null /*innerException*/, fileName, null /*sourceCode*/, 1); } } foreach (XmlNode node in rootNode.ChildNodes) { if (node.NodeType != XmlNodeType.Element) { continue; } if (node.Name == "browser" || node.Name == "gateway") { ProcessBrowserNode(node, (BrowserTree)_customTreeList[i]); } else { HandlerBase.ThrowUnrecognizedElement(node); } } } catch (XmlException e) { if (useVirtualPath) { throw new HttpParseException(e.Message, null /*innerException*/, virtualDir + "/" + NoPathFileName(fileName), null /*sourceCode*/, e.LineNumber); } else { throw new HttpParseException(e.Message, null /*innerException*/, fileName, null /*sourceCode*/, e.LineNumber); } } catch (XmlSchemaException e) { if (useVirtualPath) { throw new HttpParseException(e.Message, null /*innerException*/, virtualDir + "/" + NoPathFileName(fileName), null /*sourceCode*/, e.LineNumber); } else { throw new HttpParseException(e.Message, null /*innerException*/, fileName, null /*sourceCode*/, e.LineNumber); } } } SetCustomTreeRoots((BrowserTree)_customTreeList[i], i); NormalizeAndValidateTree((BrowserTree)_customTreeList[i], false, true); _customBrowserDefinitionCollections.Add(new BrowserDefinitionCollection()); AddCustomBrowserToCollectionRecursive((BrowserDefinition)(((BrowserTree)_customTreeList[i])[_customTreeNames[i]]), 0, i); } } internal void AddCustomBrowserToCollectionRecursive(BrowserDefinition bd, int depth, int index) { if(_customBrowserDefinitionCollections[index] == null) { _customBrowserDefinitionCollections[index] = new BrowserDefinitionCollection(); } bd.Depth = depth; bd.IsDeviceNode = true; ((BrowserDefinitionCollection)_customBrowserDefinitionCollections[index]).Add(bd); foreach (BrowserDefinition childBrowser in bd.Browsers) { AddCustomBrowserToCollectionRecursive(childBrowser, depth + 1, index); } } internal void AddBrowserToCollectionRecursive(BrowserDefinition bd, int depth) { if (_browserDefinitionCollection == null) { _browserDefinitionCollection = new BrowserDefinitionCollection(); } bd.Depth = depth; bd.IsDeviceNode = true; _browserDefinitionCollection.Add(bd); foreach(BrowserDefinition childBrowser in bd.Browsers) { AddBrowserToCollectionRecursive(childBrowser, depth + 1); } } internal virtual void HandleUnRecognizedParentElement(BrowserDefinition bd, bool isDefault) { throw new ConfigurationErrorsException(SR.GetString(SR.Browser_parentID_Not_Found, bd.ParentID), bd.XmlNode); } private static FileInfo[] GetFilesNotHidden(DirectoryInfo rootDirectory, DirectoryInfo browserDirInfo) { ArrayList fileList = new ArrayList(); FileInfo[] files; DirectoryInfo[] subDirectories = rootDirectory.GetDirectories("*", SearchOption.AllDirectories); files = rootDirectory.GetFiles("*.browser", SearchOption.TopDirectoryOnly); fileList.AddRange(files); for (int i = 0; i < subDirectories.Length; i++) { if ((HasHiddenParent(subDirectories[i], browserDirInfo) == false)) { files = subDirectories[i].GetFiles("*.browser", SearchOption.TopDirectoryOnly); fileList.AddRange(files); } } return ((FileInfo [])fileList.ToArray(typeof(FileInfo))); } private static bool HasHiddenParent(DirectoryInfo directory, DirectoryInfo browserDirInfo) { while(!String.Equals(directory.Parent.Name, browserDirInfo.Name)) { if ((directory.Attributes & FileAttributes.Hidden) == FileAttributes.Hidden) { return true; } directory = directory.Parent; } return false; } //generate the code from the parsed BrowserDefinitionTree //compile it, and install it in the gac private void GenerateAssembly() { Debug.Assert(_browserTree != null); BrowserDefinition root = (BrowserDefinition)_browserTree["Default"]; BrowserDefinition defaultRoot = (BrowserDefinition)_defaultTree["Default"]; ArrayList customTreeRoots = new ArrayList(); for (int i = 0; i < _customTreeNames.Count; i++) { customTreeRoots.Add((BrowserDefinition)(((BrowserTree)_customTreeList[i])[_customTreeNames[i]])); } //create a CodeCompileUnit //add a CodeNamespace object to the CodeCompileUnit //add a CodeTypeDeclaration to the CodeNamespace //add all the members of the type/class to the CodeTypeDeclaration //use a CodeGenerator to generate code from the CodeCompileUnit //a CodeDomProvider can provide a CodeGenerator //translate the BrowserDefinition tree to code CSharpCodeProvider cscp = new CSharpCodeProvider(); // namespace System.Web.BrowserCapsFactory CodeCompileUnit ccu = new CodeCompileUnit(); //add strong-name key pair for CodeAttributeDeclaration declaration = new CodeAttributeDeclaration( "System.Reflection.AssemblyKeyFile", new CodeAttributeArgument[] { new CodeAttributeArgument(new CodePrimitiveExpression(_strongNameKeyFileName))}); CodeAttributeDeclaration aptca = new CodeAttributeDeclaration( "System.Security.AllowPartiallyTrustedCallers"); ccu.AssemblyCustomAttributes.Add(aptca); ccu.AssemblyCustomAttributes.Add(declaration); //add version number for it so it can distinguished in future versions declaration = new CodeAttributeDeclaration( "System.Reflection.AssemblyVersion", new CodeAttributeArgument[] { new CodeAttributeArgument(new CodePrimitiveExpression(ThisAssembly.Version))}); ccu.AssemblyCustomAttributes.Add(declaration); CodeNamespace cnamespace = new CodeNamespace("ASP"); //GEN: using System; cnamespace.Imports.Add(new CodeNamespaceImport("System")); //GEN: using System.Web; cnamespace.Imports.Add(new CodeNamespaceImport("System.Web")); //GEN: using System.Web.Configuration; cnamespace.Imports.Add(new CodeNamespaceImport("System.Web.Configuration")); //GEN: using System.Reflection; cnamespace.Imports.Add(new CodeNamespaceImport("System.Reflection")); //GEN: class BrowserCapabilitiesFactory ccu.Namespaces.Add(cnamespace); CodeTypeDeclaration factoryType = new CodeTypeDeclaration("BrowserCapabilitiesFactory"); factoryType.Attributes = MemberAttributes.Private; factoryType.IsClass = true; factoryType.Name = TypeName; factoryType.BaseTypes.Add(new CodeTypeReference("System.Web.Configuration.BrowserCapabilitiesFactoryBase")); cnamespace.Types.Add(factoryType); //GEN: protected override object ConfigureBrowserCapabilities(NameValueCollection headers, HttpBrowserCapabilities browserCaps) CodeMemberMethod method = new CodeMemberMethod(); method.Attributes = MemberAttributes.Override | MemberAttributes.Public; method.ReturnType = new CodeTypeReference(typeof(void)); method.Name = "ConfigureBrowserCapabilities"; CodeParameterDeclarationExpression cpde = new CodeParameterDeclarationExpression(typeof(NameValueCollection), _headersRefName); method.Parameters.Add(cpde); cpde = new CodeParameterDeclarationExpression(typeof(HttpBrowserCapabilities), browserCapsVariable); method.Parameters.Add(cpde); factoryType.Members.Add(method); GenerateSingleProcessCall(root, method); for (int i = 0; i < customTreeRoots.Count; i++) { GenerateSingleProcessCall((BrowserDefinition)customTreeRoots[i], method); } //GEN: if(this.IsBrowserUnknown(browserCaps) == false) return; CodeConditionStatement istatement = new CodeConditionStatement(); CodeMethodInvokeExpression cmie = new CodeMethodInvokeExpression(new CodeThisReferenceExpression(), "IsBrowserUnknown"); cmie.Parameters.Add(_browserCapsRefExpr); istatement.Condition = new CodeBinaryOperatorExpression(cmie, CodeBinaryOperatorType.ValueEquality, new CodePrimitiveExpression(false)); istatement.TrueStatements.Add(new CodeMethodReturnStatement()); method.Statements.Add(istatement); if(defaultRoot != null) { GenerateSingleProcessCall(defaultRoot, method, "Default"); } for (int i = 0; i < customTreeRoots.Count; i++) { foreach (DictionaryEntry entry in (BrowserTree)_customTreeList[i]) { BrowserDefinition bd = entry.Value as BrowserDefinition; Debug.Assert(bd != null); GenerateProcessMethod(bd, factoryType); } } //GenerateCallsToProcessMethods(root, method); foreach (DictionaryEntry entry in _browserTree) { BrowserDefinition bd = entry.Value as BrowserDefinition; Debug.Assert(bd != null); GenerateProcessMethod(bd, factoryType); } foreach (DictionaryEntry entry in _defaultTree) { BrowserDefinition bd = entry.Value as BrowserDefinition; Debug.Assert(bd != null); GenerateProcessMethod(bd, factoryType, "Default"); } GenerateOverrideMatchedHeaders(factoryType); GenerateOverrideBrowserElements(factoryType); //TODO: don't actually generate the code, just compile it in memory TextWriter twriter = new StreamWriter(new FileStream(_browsersDirectory + "\\BrowserCapsFactory.cs", FileMode.Create)); try { cscp.GenerateCodeFromCompileUnit(ccu, twriter, null); } finally { if(twriter != null) twriter.Close(); } CompilationSection compConfig = MTConfigUtil.GetCompilationAppConfig(); bool debug = compConfig.Debug; #if !PLATFORM_UNIX // File system paths must account for UNIX string strongNameFile = _browsersDirectory + "\\" + _strongNameKeyFileName; #else // !PLATFORM_UNIX string strongNameFile = _browsersDirectory + "/" + _strongNameKeyFileName; #endif // !PLATFORM_UNIX // Generate strong name file StrongNameUtility.GenerateStrongNameFile(strongNameFile); //TODO: do not use interim file: CompileAssemblyFromDom instead string[] referencedAssemblies = new string[2] { "System.dll", "System.Web.dll" }; CompilerParameters compilerParameters = new CompilerParameters(referencedAssemblies, "ASP.BrowserCapsFactory", debug /* includeDebugInformation */ ); compilerParameters.GenerateInMemory = false; compilerParameters.OutputAssembly = _browsersDirectory + "\\ASP.BrowserCapsFactory.dll"; CompilerResults results = null; try { results = cscp.CompileAssemblyFromFile(compilerParameters, _browsersDirectory + "\\BrowserCapsFactory.cs"); } finally { if (File.Exists(strongNameFile)) { File.Delete(strongNameFile); } } if (results.NativeCompilerReturnValue != 0 || results.Errors.HasErrors) { foreach (CompilerError error in results.Errors) { if (!error.IsWarning) { throw new HttpCompileException(error.ErrorText); } } throw new HttpCompileException(SR.GetString(SR.Browser_compile_error)); } Assembly resultAssembly = results.CompiledAssembly; GacUtil gacutil = new GacUtil(); gacutil.GacInstall(resultAssembly.Location); SavePublicKeyTokenFile(_publicKeyTokenFile, resultAssembly.GetName().GetPublicKeyToken()); } private void SavePublicKeyTokenFile(string filename, byte[] publicKeyToken) { using (FileStream pktStream = new FileStream(filename, FileMode.Create, FileAccess.Write)) { using (StreamWriter pktWriter = new StreamWriter(pktStream)) { foreach (byte b in publicKeyToken) { pktWriter.Write("{0:X2}", b); } } } } private static string LoadPublicKeyTokenFromFile(string filename) { IStackWalk fileReadAccess = InternalSecurityPermissions.FileReadAccess(filename); Debug.Assert(fileReadAccess != null); fileReadAccess.Assert(); if (!File.Exists(filename)) { return null; } try { using (FileStream pktStream = new FileStream(filename, FileMode.Open, FileAccess.Read)) { using (StreamReader pktReader = new StreamReader(pktStream)) { return pktReader.ReadLine(); } } } catch (IOException) { if (HttpRuntime.HasFilePermission(filename)) { throw; } // Don't throw exception if we don't have permission to the file. return null; } finally { CodeAccessPermission.RevertAssert(); } } internal void GenerateOverrideBrowserElements(CodeTypeDeclaration typeDeclaration) { // Don't generate the property if there's nothing to override. if (_browserDefinitionCollection == null) { return; } // GEN: // protected override void PopulateBrowserElements(IDictionary dictionary) { // dictionary["Default"] = new Triplet(null, "default description", 0 /*depth_of_node */); // dictionary["Up"] = new Triplet("Default", "up Description", 1 /*depth_of_node */); // } CodeMemberMethod method = new CodeMemberMethod(); method.Name = _browserElementsMethodName; method.Attributes = MemberAttributes.Override | MemberAttributes.Family; method.ReturnType = new CodeTypeReference(typeof(void)); CodeParameterDeclarationExpression parameter = new CodeParameterDeclarationExpression(new CodeTypeReference(typeof(IDictionary)), _dictionaryRefName); method.Parameters.Add(parameter); typeDeclaration.Members.Add(method); CodeMethodReferenceExpression baseMethod = new CodeMethodReferenceExpression(new CodeBaseReferenceExpression(), _browserElementsMethodName); CodeMethodInvokeExpression baseInvoke = new CodeMethodInvokeExpression(baseMethod, new CodeExpression[] { _dictionaryRefExpr }); method.Statements.Add(baseInvoke); foreach(BrowserDefinition bd in _browserDefinitionCollection) { if (!bd.IsDeviceNode) continue; Debug.Assert(!(bd is GatewayDefinition)); CodeAssignStatement cas = new CodeAssignStatement(); cas.Left = new CodeIndexerExpression(_dictionaryRefExpr, new CodeExpression[] { new CodePrimitiveExpression(bd.ID) }); cas.Right = new CodeObjectCreateExpression(typeof(Triplet), new CodeExpression[] { new CodePrimitiveExpression(bd.ParentName), new CodePropertyReferenceExpression(new CodeTypeReferenceExpression(typeof(String)), "Empty"), new CodePrimitiveExpression(bd.Depth)}); method.Statements.Add(cas); } for (int i = 0; i < _customTreeNames.Count; i++) { foreach (BrowserDefinition bd in (BrowserDefinitionCollection)_customBrowserDefinitionCollections[i]) { if (!bd.IsDeviceNode) continue; Debug.Assert(!(bd is GatewayDefinition)); CodeAssignStatement cas = new CodeAssignStatement(); cas.Left = new CodeIndexerExpression(_dictionaryRefExpr, new CodeExpression[] { new CodePrimitiveExpression(bd.ID) }); cas.Right = new CodeObjectCreateExpression(typeof(Triplet), new CodeExpression[] { new CodePrimitiveExpression(bd.ParentName), new CodePropertyReferenceExpression(new CodeTypeReferenceExpression(typeof(String)), "Empty"), new CodePrimitiveExpression(bd.Depth)}); method.Statements.Add(cas); } } } internal void GenerateOverrideMatchedHeaders(CodeTypeDeclaration typeDeclaration) { // GEN: // protected override void PopulateMatchedHeaders(IDictionary dictionary) { // base.PopulateMatchedHeaders(dictionary); // // dictionary["header0"] = null; // dictionary["header1"] = null; // } CodeMemberMethod method = new CodeMemberMethod(); method.Name = _matchedHeadersMethodName; method.Attributes = MemberAttributes.Override | MemberAttributes.Family; method.ReturnType = new CodeTypeReference(typeof(void)); CodeParameterDeclarationExpression parameter = new CodeParameterDeclarationExpression(new CodeTypeReference(typeof(IDictionary)), _dictionaryRefName); method.Parameters.Add(parameter); typeDeclaration.Members.Add(method); CodeMethodReferenceExpression baseMethod = new CodeMethodReferenceExpression(new CodeBaseReferenceExpression(), _matchedHeadersMethodName); CodeMethodInvokeExpression baseInvoke = new CodeMethodInvokeExpression(baseMethod, new CodeExpression[] { _dictionaryRefExpr }); method.Statements.Add(baseInvoke); foreach(String header in _headers) { CodeAssignStatement cas = new CodeAssignStatement(); cas.Left = new CodeIndexerExpression(_dictionaryRefExpr, new CodeExpression[] { new CodePrimitiveExpression(header) }); cas.Right = new CodePrimitiveExpression(null); method.Statements.Add(cas); } } internal void GenerateProcessMethod(BrowserDefinition bd, CodeTypeDeclaration ctd) { GenerateProcessMethod(bd, ctd, String.Empty); } //generate the xxxProcess method for an individual BrowserDefinition internal void GenerateProcessMethod(BrowserDefinition bd, CodeTypeDeclaration ctd, string prefix) { //GEN: internal bool XxxProcess(NameValueCollection headers, HttpBrowserCapabilities browserCaps) CodeMemberMethod cmm = new CodeMemberMethod(); cmm.Name = prefix + bd.Name + "Process"; cmm.ReturnType = new CodeTypeReference(typeof(bool)); cmm.Attributes = MemberAttributes.Private; CodeParameterDeclarationExpression cpde = new CodeParameterDeclarationExpression(typeof(NameValueCollection), _headersRefName); cmm.Parameters.Add(cpde); cpde = new CodeParameterDeclarationExpression(typeof(HttpBrowserCapabilities), browserCapsVariable); cmm.Parameters.Add(cpde); bool regexWorkerGenerated = false; GenerateIdentificationCode(bd, cmm, ref regexWorkerGenerated); GenerateCapturesCode(bd, cmm, ref regexWorkerGenerated); GenerateSetCapabilitiesCode(bd, cmm, ref regexWorkerGenerated); GenerateSetAdaptersCode(bd, cmm); // Only add the browser node to the browser collection if it represents a device. if (bd.IsDeviceNode) { Debug.Assert(!(bd is GatewayDefinition)); //GEN: browserCaps.AddBrowser("xxx"); CodeMethodInvokeExpression cmie = new CodeMethodInvokeExpression(new CodeVariableReferenceExpression(browserCapsVariable), "AddBrowser"); cmie.Parameters.Add(new CodePrimitiveExpression(bd.ID)); cmm.Statements.Add(cmie); } // Generate ref gateway elements foreach (BrowserDefinition b in bd.RefGateways) { AddComment("ref gateways, parent=" + bd.ID, cmm); GenerateSingleProcessCall(b, cmm); } if ((GenerateOverrides) && (prefix.Length == 0)) { //Gen: protected virtual void XxxProcessGateways(NameValueCollection headers, HttpBrowserCapabilities browserCaps) ; string methodName = prefix + bd.Name + "ProcessGateways"; GenerateChildProcessMethod(methodName, ctd, false); //Gen: XxxProcessGateways(headers, browserCaps) ; GenerateChildProcessInvokeExpression(methodName, cmm, false); } foreach(BrowserDefinition b in bd.Gateways) { AddComment("gateway, parent=" + bd.ID, cmm); GenerateSingleProcessCall(b, cmm); } if (GenerateOverrides) { //GEN: bool ignoreApplicationBrowsers = true | false; //bd.Browsers.Count != 0 CodeVariableDeclarationStatement cvds = new CodeVariableDeclarationStatement(typeof(bool), IgnoreApplicationBrowserVariableName, new CodePrimitiveExpression(bd.Browsers.Count != 0)); cmm.Statements.Add(cvds); } if (bd.Browsers.Count > 0) { CodeStatementCollection statements = cmm.Statements; AddComment("browser, parent=" + bd.ID, cmm); foreach (BrowserDefinition b in bd.Browsers) { statements = GenerateTrackedSingleProcessCall(statements, b, cmm, prefix); } if (GenerateOverrides) { //GEN: ignoreApplicationBrowsers = false; CodeAssignStatement codeAssignStmt = new CodeAssignStatement(); codeAssignStmt.Left = new CodeVariableReferenceExpression(IgnoreApplicationBrowserVariableName); codeAssignStmt.Right = new CodePrimitiveExpression(false); statements.Add(codeAssignStmt); } } // Generate ref browser foreach (BrowserDefinition b in bd.RefBrowsers) { AddComment("ref browsers, parent=" + bd.ID, cmm); if (b.IsDefaultBrowser) { GenerateSingleProcessCall(b, cmm, "Default"); } else { GenerateSingleProcessCall(b, cmm); } } if (GenerateOverrides) { //Gen: protected virtual void XxxProcessBrowsers(bool ignoreApplicationBrowsers, NameValueCollection headers, HttpBrowserCapabilities browserCaps) ; string methodName = prefix + bd.Name + "ProcessBrowsers"; GenerateChildProcessMethod(methodName, ctd, true); //Gen: XxxProcessBrowsers(ignoreApplicationBrowsers, headers, browserCaps); GenerateChildProcessInvokeExpression(methodName, cmm, true); } //GEN: return true; CodeMethodReturnStatement cmrs = new CodeMethodReturnStatement(new CodePrimitiveExpression(true)); cmm.Statements.Add(cmrs); ctd.Members.Add(cmm); } private void GenerateChildProcessInvokeExpression(string methodName, CodeMemberMethod cmm, bool generateTracker) { //Gen: XxxProcessBrowsers(ignoreApplicationBrowsers, headers, browserCaps) ; CodeMethodInvokeExpression expr = new CodeMethodInvokeExpression(new CodeThisReferenceExpression(), methodName); if (generateTracker) { expr.Parameters.Add(new CodeVariableReferenceExpression(IgnoreApplicationBrowserVariableName)); } expr.Parameters.Add(new CodeVariableReferenceExpression(_headersRefName)); expr.Parameters.Add(new CodeVariableReferenceExpression(browserCapsVariable)); cmm.Statements.Add(expr); } private void GenerateChildProcessMethod(string methodName, CodeTypeDeclaration ctd, bool generateTracker) { //Gen: protected virtual void XxxProcessBrowsers(bool ignoreApplicationBrowsers, NameValueCollection headers, HttpBrowserCapabilities browserCaps) ; CodeMemberMethod cmm= new CodeMemberMethod(); cmm.Name = methodName; cmm.ReturnType = new CodeTypeReference(typeof(void)); cmm.Attributes = MemberAttributes.Family; CodeParameterDeclarationExpression cpde = null; if (generateTracker) { cpde = new CodeParameterDeclarationExpression(typeof(bool), IgnoreApplicationBrowserVariableName); cmm.Parameters.Add(cpde); } cpde = new CodeParameterDeclarationExpression(typeof(NameValueCollection), _headersRefName); cmm.Parameters.Add(cpde); cpde = new CodeParameterDeclarationExpression(typeof(HttpBrowserCapabilities), browserCapsVariable); cmm.Parameters.Add(cpde); ctd.Members.Add(cmm); } private void GenerateRegexWorkerIfNecessary(CodeMemberMethod cmm, ref bool regexWorkerGenerated) { if (regexWorkerGenerated) { return; } regexWorkerGenerated = true; //GEN: RegexWorker regexWorker; cmm.Statements.Add(new CodeVariableDeclarationStatement("RegexWorker", _regexWorkerRefName)); //GEN: regexWorker = new RegexWorker(browserCaps); cmm.Statements.Add(new CodeAssignStatement(_regexWorkerRefExpr, new CodeObjectCreateExpression("RegexWorker", _browserCapsRefExpr))); } private void ReturnIfHeaderValueEmpty(CodeMemberMethod cmm, CodeVariableReferenceExpression varExpr) { // GEN: if(String.IsNullOrEmpty(varExpr)) { // GEN: return false; // GEN: } CodeConditionStatement emptyCheckStmt = new CodeConditionStatement(); CodeMethodReferenceExpression emptyCheckMethod = new CodeMethodReferenceExpression(new CodeTypeReferenceExpression(typeof(String)), "IsNullOrEmpty"); CodeMethodInvokeExpression emptyCheckExpr = new CodeMethodInvokeExpression(emptyCheckMethod, varExpr); emptyCheckStmt.Condition = emptyCheckExpr; emptyCheckStmt.TrueStatements.Add(new CodeMethodReturnStatement(new CodePrimitiveExpression(false))); cmm.Statements.Add(emptyCheckStmt); } //generate part of the xxxProcess method for handling determining if the requesting //browser meets the regexes for this browser private void GenerateIdentificationCode(BrowserDefinition bd, CodeMemberMethod cmm, ref bool regexWorkerGenerated) { //GEN: IDictionary dictionary; cmm.Statements.Add(new CodeVariableDeclarationStatement(typeof(IDictionary), _dictionaryRefName)); //GEN: dictionary = browserCaps.Capabilities; CodeAssignStatement assign = new CodeAssignStatement( _dictionaryRefExpr, new CodePropertyReferenceExpression(_browserCapsRefExpr, "Capabilities") ); cmm.Statements.Add(assign); bool disableOptimizedKey = false; CodeVariableReferenceExpression result = null; CodeVariableReferenceExpression headerValue = null; if(bd.IdHeaderChecks.Count > 0) { AddComment("Identification: check header matches", cmm); for (int i = 0; i < bd.IdHeaderChecks.Count; i++) { string matchedString = ((CheckPair)bd.IdHeaderChecks[i]).MatchString; // Skip matching ".*" if (matchedString.Equals(".*")) { continue; } if (headerValue == null) { headerValue = GenerateVarReference(cmm, typeof(string), "headerValue"); } CodeAssignStatement valueAssignment = new CodeAssignStatement(); cmm.Statements.Add(valueAssignment); valueAssignment.Left = headerValue; if (((CheckPair)bd.IdHeaderChecks[i]).Header.Equals("User-Agent")) { _headers.Add(String.Empty); // GEN: headerValue = ((string)(browserCaps[String.Empty])); valueAssignment.Right = new CodeCastExpression(typeof(string), new CodeIndexerExpression( new CodeVariableReferenceExpression(browserCapsVariable), new CodeExpression[] { new CodePropertyReferenceExpression( new CodeTypeReferenceExpression(typeof(String)), "Empty") })); } else { string header = ((CheckPair)bd.IdHeaderChecks[i]).Header; _headers.Add(header); //GEN: headerValue = ((String)headers["xxx"]); valueAssignment.Right = new CodeCastExpression(typeof(string), new CodeIndexerExpression( _headersRefExpr, new CodeExpression[] { new CodePrimitiveExpression(header) } ) ); disableOptimizedKey = true; } // Don't need to use Regex if matching . only. if (matchedString.Equals(".")) { // Simply return if the header exists. ReturnIfHeaderValueEmpty(cmm, headerValue); continue; } if (result == null) { result = GenerateVarReference(cmm, typeof(bool), _resultVarName); } GenerateRegexWorkerIfNecessary(cmm, ref regexWorkerGenerated); CodeMethodInvokeExpression cmie = new CodeMethodInvokeExpression(_regexWorkerRefExpr, _processRegexMethod); cmie.Parameters.Add(headerValue); cmie.Parameters.Add(new CodePrimitiveExpression(matchedString)); //GEN: result = regexWorker.ProcessRegex(headerValue, {matchedString}); assign = new CodeAssignStatement(); assign.Left = result; assign.Right = cmie; cmm.Statements.Add(assign); //GEN: if(result == false) { //GEN: return false; //GEN: } CodeConditionStatement istatement = new CodeConditionStatement(); if(((CheckPair)bd.IdHeaderChecks[i]).NonMatch) { istatement.Condition = new CodeBinaryOperatorExpression(result, CodeBinaryOperatorType.ValueEquality, new CodePrimitiveExpression(true)); } else { istatement.Condition = new CodeBinaryOperatorExpression(result, CodeBinaryOperatorType.ValueEquality, new CodePrimitiveExpression(false)); } istatement.TrueStatements.Add(new CodeMethodReturnStatement(new CodePrimitiveExpression(false))); cmm.Statements.Add(istatement); } } if (bd.IdCapabilityChecks.Count > 0) { AddComment("Identification: check capability matches", cmm); for (int i = 0; i < bd.IdCapabilityChecks.Count; i++) { string matchedString = ((CheckPair)bd.IdCapabilityChecks[i]).MatchString; // Skip matching ".*" if (matchedString.Equals(".*")) { continue; } if (headerValue == null) { headerValue = GenerateVarReference(cmm, typeof(string), "headerValue"); } CodeAssignStatement valueAssignment = new CodeAssignStatement(); cmm.Statements.Add(valueAssignment); valueAssignment.Left = headerValue; valueAssignment.Right = (new CodeCastExpression(typeof(string), new CodeIndexerExpression( _dictionaryRefExpr, new CodeExpression[] { new CodePrimitiveExpression(((CheckPair)bd.IdCapabilityChecks[i]).Header) } ) )); // Don't need to use Regex if matching . only. if (matchedString.Equals(".")) { continue; } if (result == null) { result = GenerateVarReference(cmm, typeof(bool), _resultVarName); } GenerateRegexWorkerIfNecessary(cmm, ref regexWorkerGenerated); //GEN: result = regexWorker.ProcessRegex((string)dictionary["xxxCapability"], "xxxRegexString"); CodeMethodInvokeExpression cmie = new CodeMethodInvokeExpression(_regexWorkerRefExpr, _processRegexMethod); cmie.Parameters.Add(headerValue); cmie.Parameters.Add(new CodePrimitiveExpression(matchedString)); assign = new CodeAssignStatement(); assign.Left = result; assign.Right = cmie; cmm.Statements.Add(assign); //GEN: if(result == false) { //GEN: return false; //GEN: } CodeConditionStatement istatement = new CodeConditionStatement(); if (((CheckPair)bd.IdCapabilityChecks[i]).NonMatch) { istatement.Condition = new CodeBinaryOperatorExpression(result, CodeBinaryOperatorType.ValueEquality, new CodePrimitiveExpression(true)); } else { istatement.Condition = new CodeBinaryOperatorExpression(result, CodeBinaryOperatorType.ValueEquality, new CodePrimitiveExpression(false)); } istatement.TrueStatements.Add(new CodeMethodReturnStatement(new CodePrimitiveExpression(false))); cmm.Statements.Add(istatement); } } //GEN: browserCaps.DisableOptimizedCacheKey(); if (disableOptimizedKey) { CodeMethodInvokeExpression cme = new CodeMethodInvokeExpression(_browserCapsRefExpr, _disableOptimizedCacheKeyMethodName); cmm.Statements.Add(cme); } } private CodeVariableReferenceExpression GenerateVarReference(CodeMemberMethod cmm, Type varType, string varName) { //GEN: {varType} {varName}; cmm.Statements.Add(new CodeVariableDeclarationStatement(varType, varName)); return new CodeVariableReferenceExpression(varName); } //generate part of the xxxProcess method for running and storing the capture regexes private void GenerateCapturesCode(BrowserDefinition bd, CodeMemberMethod cmm, ref bool regexWorkerGenerated) { if ((bd.CaptureHeaderChecks.Count == 0) && (bd.CaptureCapabilityChecks.Count == 0)) { return; } if(bd.CaptureHeaderChecks.Count > 0) { AddComment("Capture: header values", cmm); for(int i = 0; i < bd.CaptureHeaderChecks.Count; i++) { string matchedString = ((CheckPair)bd.CaptureHeaderChecks[i]).MatchString; if (matchedString.Equals(".*")) { continue; } GenerateRegexWorkerIfNecessary(cmm, ref regexWorkerGenerated); CodeMethodInvokeExpression cmie = new CodeMethodInvokeExpression(_regexWorkerRefExpr, _processRegexMethod); if (((CheckPair)bd.CaptureHeaderChecks[i]).Header.Equals("User-Agent")) { _headers.Add(String.Empty); cmie.Parameters.Add(new CodeCastExpression(typeof(string), new CodeIndexerExpression(new CodeVariableReferenceExpression(browserCapsVariable), new CodeExpression[] { new CodePropertyReferenceExpression(new CodeTypeReferenceExpression(typeof(String)), "Empty") }))); } else { string header = ((CheckPair)bd.CaptureHeaderChecks[i]).Header; _headers.Add(header); //GEN: regexWorker.ProcessRegex((string)headers["xxx"], "xxxRegexString"); cmie.Parameters.Add( new CodeCastExpression(typeof(string), new CodeIndexerExpression( _headersRefExpr, new CodeExpression[] { new CodePrimitiveExpression(header) } ) ) ); } cmie.Parameters.Add(new CodePrimitiveExpression(matchedString)); cmm.Statements.Add(cmie); } } if (bd.CaptureCapabilityChecks.Count > 0) { AddComment("Capture: capability values", cmm); for(int i = 0; i < bd.CaptureCapabilityChecks.Count; i++) { string matchedString = ((CheckPair)bd.CaptureCapabilityChecks[i]).MatchString; if (matchedString.Equals(".*")) { continue; } GenerateRegexWorkerIfNecessary(cmm, ref regexWorkerGenerated); //GEN: regexWorker.ProcessRegex((string)dictionary["xxxCapability"], "xxxRegexString"); CodeMethodInvokeExpression cmie = new CodeMethodInvokeExpression(_regexWorkerRefExpr, _processRegexMethod); cmie.Parameters.Add( new CodeCastExpression(typeof(string), new CodeIndexerExpression( _dictionaryRefExpr, new CodeExpression[] { new CodePrimitiveExpression(((CheckPair)bd.CaptureCapabilityChecks[i]).Header) } ) ) ); cmie.Parameters.Add(new CodePrimitiveExpression(matchedString)); cmm.Statements.Add(cmie); } } } //generate part of the xxxProcess method for assigning capability values private void GenerateSetCapabilitiesCode(BrowserDefinition bd, CodeMemberMethod cmm, ref bool regexWorkerGenerated) { //GEN: browserCaps[aaa] = "bbb"; //GEN: browserCaps[xxx] = "yyy"; NameValueCollection nvc = bd.Capabilities; CodeAssignStatement assign; AddComment("Capabilities: set capabilities", cmm); foreach (string s in nvc.Keys) { string capsString = nvc[s]; //GEN: dictionary["xxx"] = regexWorker["xxx"]; assign = new CodeAssignStatement(); assign.Left = new CodeIndexerExpression( _dictionaryRefExpr, new CodeExpression[] { new CodePrimitiveExpression(s) } ); CodePrimitiveExpression capabilityExpr = new CodePrimitiveExpression(capsString); if (RegexWorker.RefPat.Match(capsString).Success) { GenerateRegexWorkerIfNecessary(cmm, ref regexWorkerGenerated); assign.Right = new CodeIndexerExpression( _regexWorkerRefExpr, new CodeExpression[] {capabilityExpr}); } else { assign.Right = capabilityExpr; } cmm.Statements.Add(assign); } } //generate part of the xxxProcess method for setting specific adapters for this browser internal void GenerateSetAdaptersCode(BrowserDefinition bd, CodeMemberMethod cmm) { //GEN: browserCaps.Adapters[xxxControl] = yyyAdapter; foreach (DictionaryEntry entry in bd.Adapters) { string controlString = (string)entry.Key; string adapterString = (string)entry.Value; CodePropertyReferenceExpression cpre = new CodePropertyReferenceExpression(_browserCapsRefExpr, "Adapters"); CodeIndexerExpression indexerExpression = new CodeIndexerExpression( cpre, new CodeExpression[] { new CodePrimitiveExpression(controlString) } ); CodeAssignStatement assignAdapter = new CodeAssignStatement(); assignAdapter.Left = indexerExpression; assignAdapter.Right = new CodePrimitiveExpression(adapterString); cmm.Statements.Add(assignAdapter); } //GEN: browser.HtmlTextWriter = xxxHtmlTextWriter; if(bd.HtmlTextWriterString != null) { CodeAssignStatement assignHtmlTextWriter = new CodeAssignStatement(); assignHtmlTextWriter.Left = new CodePropertyReferenceExpression(_browserCapsRefExpr, "HtmlTextWriter"); assignHtmlTextWriter.Right = new CodePrimitiveExpression(bd.HtmlTextWriterString); cmm.Statements.Add(assignHtmlTextWriter); } return; } internal void AddComment(string comment, CodeMemberMethod cmm) { cmm.Statements.Add(new CodeCommentStatement(comment)); } internal CodeStatementCollection GenerateTrackedSingleProcessCall(CodeStatementCollection stmts, BrowserDefinition bd, CodeMemberMethod cmm) { return GenerateTrackedSingleProcessCall(stmts, bd, cmm, String.Empty); } internal CodeStatementCollection GenerateTrackedSingleProcessCall(CodeStatementCollection stmts, BrowserDefinition bd, CodeMemberMethod cmm, string prefix) { //GEN: if (xProcess(headers, browserCaps)) { // } // else { // ... // } CodeMethodInvokeExpression xProcess = new CodeMethodInvokeExpression(new CodeThisReferenceExpression(), prefix + bd.Name + "Process"); xProcess.Parameters.Add(new CodeVariableReferenceExpression(_headersRefName)); xProcess.Parameters.Add(new CodeVariableReferenceExpression(browserCapsVariable)); CodeConditionStatement conditionStmt = new CodeConditionStatement(); conditionStmt.Condition = xProcess; stmts.Add(conditionStmt); return conditionStmt.FalseStatements; } internal void GenerateSingleProcessCall(BrowserDefinition bd, CodeMemberMethod cmm) { GenerateSingleProcessCall(bd, cmm, String.Empty); } //generate code to call the xxxProcess for a given browser //and store the result in a local variable internal void GenerateSingleProcessCall(BrowserDefinition bd, CodeMemberMethod cmm, string prefix) { //GEN: xProcess(headers, browserCaps); CodeMethodInvokeExpression xProcess = new CodeMethodInvokeExpression(new CodeThisReferenceExpression(), prefix + bd.Name + "Process"); xProcess.Parameters.Add(new CodeVariableReferenceExpression(_headersRefName)); xProcess.Parameters.Add(new CodeVariableReferenceExpression(browserCapsVariable)); cmm.Statements.Add(xProcess); } } }