//------------------------------------------------------------------------------
// 
//     Copyright (c) Microsoft Corporation.  All rights reserved.
// 
//------------------------------------------------------------------------------
/*********************************
Class hierarchy
ErrorFormatter (abstract)
    UnhandledErrorFormatter
        SecurityErrorFormatter
        UseLastUnhandledErrorFormatter
        TemplatedMailRuntimeErrorFormatter
    PageNotFoundErrorFormatter
    PageForbiddenErrorFormatter
    GenericApplicationErrorFormatter
    FormatterWithFileInfo (abstract)
        ParseErrorFormatter
        ConfigErrorFormatter
    DynamicCompileErrorFormatter
        TemplatedMailCompileErrorFormatter    
    UrlAuthFailedErrorFormatter
    TraceHandlerErrorFormatter
    TemplatedMailErrorFormatterGenerator
    AuthFailedErrorFormatter
    FileAccessFailedErrorFormatter
    PassportAuthFailedErrorFormatter
**********************************/
/*
 * Object used to put together ASP.NET HTML error messages
 *
 * Copyright (c) 1999 Microsoft Corporation
 */
namespace System.Web {
    using System.Runtime.Serialization.Formatters;
    using System.Text;
    using System.Diagnostics;
    using System.Drawing;
    using System.Reflection;
    using System.Configuration.Assemblies;
    using System.Runtime.InteropServices;
    using System.Runtime.Serialization;
    using System.IO;
    using System.Globalization;
    using System.Web.Hosting;
    using System.Web.UI;
    using System.Web.UI.HtmlControls;
    using System.Web.UI.WebControls;
    using System.Web.Util;
    using System.Web.Compilation;
    using System.Collections;
    using System.Collections.Specialized;
    using System.Text.RegularExpressions;
    using System.CodeDom.Compiler;
    using System.ComponentModel;
    using Debug=System.Web.Util.Debug;
    using System.Web.Management;
    using System.Configuration;
    using System.Security;
    using System.Security.Permissions;
    /*
     * This is an abstract base class from which we derive other formatters.
     */
    internal abstract class ErrorFormatter {
        private StringCollection _adaptiveMiscContent;
        private StringCollection _adaptiveStackTrace;
        protected bool           _dontShowVersion = false;
        private const string startExpandableBlock =
            "
" +
            "{1}" +
            ":
\r\n" +
            "\r\n" +
            "            
\r\n" +
            "               \r\n" +
            "                  | \r\n" +
            " \r\n" +
            "\r\n\r\n" +
            "\r\n\r\n";
        private const string endExpandableBlock =
            "                       | 
\r\n" +
            "            
\r\n\r\n" +
            "            \r\n\r\n" +
            "
";
        protected const string EndLeftToRightTag = "
";
        internal static bool RequiresAdaptiveErrorReporting(HttpContext context)
        {
            // If HostingInit failed, don't try to continue, as we are not sufficiently
            // initialized to execute this code (VSWhidbey 210495)
            if (HttpRuntime.HostingInitFailed)
                return false;
            HttpRequest request = (context != null) ? context.Request : null;
            if (context != null && context.WorkerRequest is System.Web.SessionState.StateHttpWorkerRequest)
                return false;
            // Request.Browser might throw if the configuration file has some
            // bad format.
            HttpBrowserCapabilities browser = null;
            try {
                browser = (request != null) ? request.Browser : null;
            }
            catch {
                return false;
            }
            if (browser != null &&
                browser["requiresAdaptiveErrorReporting"] == "true") {
                return true;
            }
            return false;
        }
        private Literal CreateBreakLiteral() {
            Literal breakControl = new Literal();
            breakControl.Text = "
";
            return breakControl;
        }
        private Label CreateLabelFromText(String text) {
            Label label = new Label();
            label.Text = text;
            return label;
        }
        // Return error message in markup using adaptive rendering of web
        // controls.  This would also set the corresponding headers of the
        // response accordingly so content can be shown properly on devices.
        // This method has been added with the same signature of
        // GetHtmlErrorMessage for consistency.
        internal virtual string GetAdaptiveErrorMessage(HttpContext context, bool dontShowSensitiveInfo) {
            // This call will compute and set all the necessary properties of
            // this instance of ErrorFormatter.  Then the controls below can
            // collect info from the properties.  The returned html is safely
            // ignored.
            GetHtmlErrorMessage(dontShowSensitiveInfo);
            // We need to inform the Response object that adaptive error is used
            // so it can adjust the status code right before headers are written out.
            // It is because some mobile devices/browsers can display a page
            // content only if it is a normal response instead of response that
            // has error status code.
            context.Response.UseAdaptiveError = true;
            try {
                Page page = new ErrorFormatterPage();
                page.EnableViewState = false;
                HtmlForm form = new HtmlForm();
                page.Controls.Add(form);
                IParserAccessor formAdd = (IParserAccessor) form;
                // Display a server error text with the application name
                Label label = CreateLabelFromText(SR.GetString(SR.Error_Formatter_ASPNET_Error, HttpRuntime.AppDomainAppVirtualPath));
                label.ForeColor = Color.Red;
                label.Font.Bold = true;
                label.Font.Size = FontUnit.Large;
                formAdd.AddParsedSubObject(label);
                formAdd.AddParsedSubObject(CreateBreakLiteral());
                // Title
                label = CreateLabelFromText(ErrorTitle);
                label.ForeColor = Color.Maroon;
                label.Font.Bold = true;
                label.Font.Italic = true;
                formAdd.AddParsedSubObject(label);
                formAdd.AddParsedSubObject(CreateBreakLiteral());
                // Description
                formAdd.AddParsedSubObject(CreateLabelFromText(SR.GetString(SR.Error_Formatter_Description) + " " + Description));
                formAdd.AddParsedSubObject(CreateBreakLiteral());
                // Misc Title
                String miscTitle = MiscSectionTitle;
                if (!String.IsNullOrEmpty(miscTitle)) {
                    formAdd.AddParsedSubObject(CreateLabelFromText(miscTitle));
                    formAdd.AddParsedSubObject(CreateBreakLiteral());
                }
                // Misc Info
                StringCollection miscContent = AdaptiveMiscContent;
                if (miscContent != null && miscContent.Count > 0) {
                    foreach (String contentLine in miscContent) {
                        formAdd.AddParsedSubObject(CreateLabelFromText(contentLine));
                        formAdd.AddParsedSubObject(CreateBreakLiteral());
                    }
                }
                // File & line# info
                String sourceFilePath = GetDisplayPath();
                if (!String.IsNullOrEmpty(sourceFilePath)) {
                    String text = SR.GetString(SR.Error_Formatter_Source_File) + " " + sourceFilePath;
                    formAdd.AddParsedSubObject(CreateLabelFromText(text));
                    formAdd.AddParsedSubObject(CreateBreakLiteral());
                    text = SR.GetString(SR.Error_Formatter_Line) + " " + SourceFileLineNumber;
                    formAdd.AddParsedSubObject(CreateLabelFromText(text));
                    formAdd.AddParsedSubObject(CreateBreakLiteral());
                }
                // Stack trace info
                StringCollection stackTrace = AdaptiveStackTrace;
                if (stackTrace != null && stackTrace.Count > 0) {
                    foreach (String stack in stackTrace) {
                        formAdd.AddParsedSubObject(CreateLabelFromText(stack));
                        formAdd.AddParsedSubObject(CreateBreakLiteral());
                    }
                }
                // Temporarily use a string writer to capture the output and
                // return it accordingly.
                StringWriter stringWriter = new StringWriter(CultureInfo.CurrentCulture);
                TextWriter textWriter = context.Response.SwitchWriter(stringWriter);
                page.ProcessRequest(context);
                context.Response.SwitchWriter(textWriter);
                return stringWriter.ToString();
            }
            catch {
                return GetStaticErrorMessage(context);
            }
        }
        private string GetPreferredRenderingType(HttpContext context) {
            HttpRequest request = (context != null) ? context.Request : null;
            // Request.Browser might throw if the configuration file has some
            // bad format.
            HttpBrowserCapabilities browser = null;
            try {
                browser = (request != null) ? request.Browser : null;
            }
            catch {
                return String.Empty;
            }
            return ((browser != null) ? browser["preferredRenderingType"] : String.Empty);
        }
        private string GetStaticErrorMessage(HttpContext context) {
            string preferredRenderingType = GetPreferredRenderingType(context);
            Debug.Assert(preferredRenderingType != null);
            string errorMessage;
            if (StringUtil.StringStartsWithIgnoreCase(preferredRenderingType, "xhtml")) {
                errorMessage = FormatStaticErrorMessage(StaticErrorFormatterHelper.XhtmlErrorBeginTemplate,
                                                        StaticErrorFormatterHelper.XhtmlErrorEndTemplate);
            }
            else if (StringUtil.StringStartsWithIgnoreCase(preferredRenderingType, "wml")) {
                errorMessage = FormatStaticErrorMessage(StaticErrorFormatterHelper.WmlErrorBeginTemplate,
                                                        StaticErrorFormatterHelper.WmlErrorEndTemplate);
                // VSWhidbey 161754: In the case that headers have been written,
                // we should try to set the content type only if needed.
                const string wmlContentType = "text/vnd.wap.wml";
                if (String.Compare(context.Response.ContentType, 0,
                                   wmlContentType, 0, wmlContentType.Length,
                                   StringComparison.OrdinalIgnoreCase) != 0) {
                    context.Response.ContentType = wmlContentType;
                }
            }
            else {
                errorMessage = FormatStaticErrorMessage(StaticErrorFormatterHelper.ChtmlErrorBeginTemplate,
                                                        StaticErrorFormatterHelper.ChtmlErrorEndTemplate);
            }
            return errorMessage;
        }
        private string FormatStaticErrorMessage(string errorBeginTemplate,
                                                string errorEndTemplate) {
            StringBuilder errorContent = new StringBuilder();
            // Server error text with the application name and Title
            string errorHeader = SR.GetString(SR.Error_Formatter_ASPNET_Error, HttpRuntime.AppDomainAppVirtualPath);
            errorContent.Append(String.Format(CultureInfo.CurrentCulture, errorBeginTemplate, errorHeader, ErrorTitle));
            // Description
            errorContent.Append(SR.GetString(SR.Error_Formatter_Description) + " " + Description);
            errorContent.Append(StaticErrorFormatterHelper.Break);
            // Misc Title
            String miscTitle = MiscSectionTitle;
            if (miscTitle != null && miscTitle.Length > 0) {
                errorContent.Append(miscTitle);
                errorContent.Append(StaticErrorFormatterHelper.Break);
            }
            // Misc Info
            StringCollection miscContent = AdaptiveMiscContent;
            if (miscContent != null && miscContent.Count > 0) {
                foreach (String contentLine in miscContent) {
                    errorContent.Append(contentLine);
                    errorContent.Append(StaticErrorFormatterHelper.Break);
                }
            }
            // File & line# info
            String sourceFilePath = GetDisplayPath();
            if (!String.IsNullOrEmpty(sourceFilePath)) {
                String text = SR.GetString(SR.Error_Formatter_Source_File) + " " + sourceFilePath;
                errorContent.Append(text);
                errorContent.Append(StaticErrorFormatterHelper.Break);
                text = SR.GetString(SR.Error_Formatter_Line) + " " + SourceFileLineNumber;
                errorContent.Append(text);
                errorContent.Append(StaticErrorFormatterHelper.Break);
            }
            // Stack trace info
            StringCollection stackTrace = AdaptiveStackTrace;
            if (stackTrace != null && stackTrace.Count > 0) {
                foreach (String stack in stackTrace) {
                    errorContent.Append(stack);
                    errorContent.Append(StaticErrorFormatterHelper.Break);
                }
            }
            errorContent.Append(errorEndTemplate);
            return errorContent.ToString();
        }
        internal string GetErrorMessage() {
            return GetErrorMessage(HttpContext.Current, true);
        }
        // Return error message by checking if adaptive error formatting
        // should be used.
        internal virtual string GetErrorMessage(HttpContext context, bool dontShowSensitiveInfo) {
            if (RequiresAdaptiveErrorReporting(context)) {
                return GetAdaptiveErrorMessage(context, dontShowSensitiveInfo);
            }
            return GetHtmlErrorMessage(dontShowSensitiveInfo);
        }
        internal /*public*/ string GetHtmlErrorMessage() {
            return GetHtmlErrorMessage(true);
        }
        internal /*public*/ string GetHtmlErrorMessage(bool dontShowSensitiveInfo) {
            // Give the formatter a chance to prepare its state
            PrepareFormatter();
            StringBuilder sb = new StringBuilder();
            // 
            sb.Append("\r\n");
            sb.Append("\r\n");
            sb.Append("    \r\n");
            sb.Append("        " + ErrorTitle + "\r\n");
            sb.Append("        \r\n");
            sb.Append("        \r\n");
            sb.Append("    \r\n\r\n");
            sb.Append("    \r\n\r\n");
            sb.Append("            " + SR.GetString(SR.Error_Formatter_ASPNET_Error, HttpRuntime.AppDomainAppVirtualPath) + "
\r\n\r\n");
            sb.Append("             " + ErrorTitle + " 
\r\n\r\n");
            sb.Append("            \r\n\r\n");
            sb.Append("             " + SR.GetString(SR.Error_Formatter_Description) +  " " + Description + "\r\n");
            sb.Append("            
\r\n\r\n");
            if (MiscSectionTitle != null) {
                sb.Append("             " + MiscSectionTitle + ": " + MiscSectionContent + "
\r\n\r\n");
            }
            WriteColoredSquare(sb, ColoredSquareTitle, ColoredSquareDescription, ColoredSquareContent, WrapColoredSquareContentLines);
            if (ShowSourceFileInfo) {
                string displayPath = GetDisplayPath();
                if (displayPath == null)
                    displayPath = SR.GetString(SR.Error_Formatter_No_Source_File);
                sb.Append("             " + SR.GetString(SR.Error_Formatter_Source_File) + "  " + displayPath + "    " + SR.GetString(SR.Error_Formatter_Line) + "  " + SourceFileLineNumber + "\r\n");
                sb.Append("            
\r\n\r\n");
            }
            ConfigurationErrorsException configErrors = Exception as ConfigurationErrorsException;
            if (configErrors != null && configErrors.Errors.Count > 1) {
                sb.Append(String.Format(CultureInfo.InvariantCulture, startExpandableBlock, "additionalConfigurationErrors",
                    SR.GetString(SR.TmplConfigurationAdditionalError)));
                //
                // Get the configuration message as though there were user code on the stack,
                // so that the full path to the configuration file is not shown if the app
                // does not have PathDiscoveryPermission.
                // 
                bool revertPermitOnly = false;
                try {
                    PermissionSet ps = HttpRuntime.NamedPermissionSet;
                    if (ps != null) {
                        ps.PermitOnly();
                        revertPermitOnly = true;
                    }
                    
                    int errorNumber = 0;
                    foreach(ConfigurationException configurationError in configErrors.Errors) {
                        if (errorNumber > 0) {
                            sb.Append(configurationError.Message);
                            sb.Append("
\r\n");
                        }
                        errorNumber++;
                    }
                }
                finally {
                    if (revertPermitOnly) {
                        CodeAccessPermission.RevertPermitOnly();
                    }
                }
                sb.Append(endExpandableBlock);
                sb.Append(toggleScript);
            }
            // If it's a FileNotFoundException/FileLoadException/BadImageFormatException with a FusionLog,
            // write it out (ASURT 83587)
            if (!dontShowSensitiveInfo && Exception != null) {
                // (Only display the fusion log in medium or higher (ASURT 126827)
                if (HttpRuntime.HasAspNetHostingPermission(AspNetHostingPermissionLevel.Medium)) {
                    WriteFusionLogWithAssert(sb);
                }
            }
            WriteColoredSquare(sb, ColoredSquare2Title, ColoredSquare2Description, ColoredSquare2Content, false);
            if (!(dontShowSensitiveInfo || _dontShowVersion)) {  // don't show version for security reasons
                sb.Append("            
\r\n\r\n");
                sb.Append("            " + SR.GetString(SR.Error_Formatter_Version) + " " +
                                       SR.GetString(SR.Error_Formatter_CLR_Build) + VersionInfo.ClrVersion +
                                       SR.GetString(SR.Error_Formatter_ASPNET_Build) + VersionInfo.EngineVersion + "\r\n\r\n");
                sb.Append("            \r\n\r\n");
            }
            sb.Append("    \r\n");
            sb.Append("\r\n");
            sb.Append(PostMessage);
            return sb.ToString();
        }
        [PermissionSet(SecurityAction.Assert, Unrestricted=true)]
        private void WriteFusionLogWithAssert(StringBuilder sb) {
            for (Exception e = Exception; e != null; e = e.InnerException) {
                string fusionLog = null;
                string filename = null;
                FileNotFoundException fnfException = e as FileNotFoundException;
                if (fnfException != null) {
                    fusionLog = fnfException.FusionLog;
                    filename = fnfException.FileName;
                }
                FileLoadException flException = e as FileLoadException;
                if (flException != null) {
                    fusionLog = flException.FusionLog;
                    filename = flException.FileName;
                }
                BadImageFormatException bifException = e as BadImageFormatException;
                if (bifException != null) {
                    fusionLog = bifException.FusionLog;
                    filename = bifException.FileName;
                }
                if (!String.IsNullOrEmpty(fusionLog)) {
                    WriteColoredSquare(sb,
                                       SR.GetString(SR.Error_Formatter_FusionLog),
                                       SR.GetString(SR.Error_Formatter_FusionLogDesc, filename),
                                       HttpUtility.HtmlEncode(fusionLog),
                                       false /*WrapColoredSquareContentLines*/);
                    break;
                }
            }
        }
        private void WriteColoredSquare(StringBuilder sb, string title, string description,
            string content, bool wrapContentLines) {
            if (title != null) {
                sb.Append("            " + title + ": " + description + "
\r\n\r\n");
                sb.Append("            \r\n");
                sb.Append("               \r\n");
                sb.Append("                  | \r\n");
                sb.Append(" \r\n");
                sb.Append("");
                if (!wrapContentLines)
                    sb.Append("\r\n\r\n");
                sb.Append("");
                sb.Append("\r\n\r\n");
                sb.Append(content);
                if (!wrapContentLines)
                    sb.Append("");
                sb.Append(" | 
\r\n");
                sb.Append("            
\r\n\r\n");
                sb.Append("            
\r\n\r\n");
            }
        }
        internal /*public*/ virtual void PrepareFormatter() {
            // VSWhidbey 139210: ErrorFormatter object might be reused and
            // the properties would be gone through again.  So we need to
            // clear the adaptive error content to avoid duplicate content.
            if (_adaptiveMiscContent != null) {
                _adaptiveMiscContent.Clear();
            }
            if (_adaptiveStackTrace != null) {
                _adaptiveStackTrace.Clear();
            }
        }
        /*
         * Return the associated exception object (if any)
         */
        protected virtual Exception Exception {
            get { return null; }
        }
        /*
         * Return the type of error.  e.g. "Compilation Error."
         */
        protected abstract string ErrorTitle {
            get;
        }
        /*
         * Return a description of the error
         * e.g. "An error occurred during the compilation of a resource required to service"
         */
        protected abstract string Description {
            get;
        }
        /*
         * A section used differently by different types of errors (title)
         * e.g. "Compiler Error Message"
         * e.g. "Exception Details"
         */
        protected abstract string MiscSectionTitle {
            get;
        }
        /*
         * A section used differently by different types of errors (content)
         * e.g. "BC30198: Expected: )"
         * e.g. "System.NullReferenceException"
         */
        protected abstract string MiscSectionContent {
            get;
        }
        /*
         * e.g. "Source Error"
         */
        protected virtual string ColoredSquareTitle {
            get { return null;}
        }
        /*
         * Optional text between color square title and the color square itself
         */
        protected virtual string ColoredSquareDescription {
            get { return null;}
        }
        /*
         * e.g. a piece of source code with the error context
         */
        protected virtual string ColoredSquareContent {
            get { return null;}
        }
        /*
         * If false, use a  tag around it
         */
        protected virtual bool WrapColoredSquareContentLines {
            get { return false;}
        }
        /*
         * e.g. "Source Error"
         */
        protected virtual string ColoredSquare2Title {
            get { return null;}
        }
        /*
         * Optional text between color square title and the color square itself
         */
        protected virtual string ColoredSquare2Description {
            get { return null;}
        }
        /*
         * e.g. a piece of source code with the error context
         */
        protected virtual string ColoredSquare2Content {
            get { return null;}
        }
        /*
         * Misc content which will be shown to mobile devices
         * e.g. compile error code
         */
        protected virtual StringCollection AdaptiveMiscContent {
            get {
                if (_adaptiveMiscContent == null) {
                    _adaptiveMiscContent = new StringCollection();
                }
                return _adaptiveMiscContent;
            }
        }
        /*
         * Exception stack trace which will be shown to mobile devices
         * e.g. stack trace of a runtime error
         */
        protected virtual StringCollection AdaptiveStackTrace {
            get {
                if (_adaptiveStackTrace == null) {
                    _adaptiveStackTrace = new StringCollection();
                }
                return _adaptiveStackTrace;
            }
        }
        /*
         * Determines whether SourceFileName and SourceFileLineNumber will be used
         */
        protected abstract bool ShowSourceFileInfo {
            get;
        }
        /*
         * e.g. d:\samples\designpreview\test.aspx
         */
        protected virtual string PhysicalPath {
            get { return null;}
        }
        /*
         * e.g. /myapp/test.aspx
         */
        protected virtual string VirtualPath {
            get { return null;}
        }
        /*
         * The line number in the source file
         */
        protected virtual int SourceFileLineNumber {
            get { return 0;}
        }
        protected virtual String PostMessage {
            get { return null; }
        }
        /*
         * Does this error have only information that we want to
         * show over the web to random users?
         */
        internal virtual bool CanBeShownToAllUsers {
            get { return false;}
        }
        // VSWhidbey 477678: Respect current language text format that is right
        // to left.  To be used by subclasses who need to adjust text format for
        // code area accordingly.
        protected static bool IsTextRightToLeft {
            get {
                return CultureInfo.CurrentUICulture.TextInfo.IsRightToLeft;
            }
        }
        protected string WrapWithLeftToRightTextFormatIfNeeded(string content) {
            if (IsTextRightToLeft) {
                content = BeginLeftToRightTag + content + EndLeftToRightTag;
            }
            return content;
        }
        // Make an HTTP line pragma from a virtual path
        internal static string MakeHttpLinePragma(string virtualPath) {
            string server = "http://server";
            // We should only append a "/" if the virtual path does not
            // already start with "/". Otherwise, we end up with double
            // slashes, eg http://server//vpp/foo.aspx , and this breaks
            // the VirtualPathProvider. (DevDiv 157238)
            if (virtualPath != null && !virtualPath.StartsWith("/", StringComparison.Ordinal)) {
                server += "/";
            }
            return (new Uri(server + virtualPath)).ToString();
        }
        internal static string GetSafePath(string linePragma) {
            // First, check if it's an http line pragma
            string virtualPath = GetVirtualPathFromHttpLinePragma(linePragma);
            // If so, just return the virtual path
            if (virtualPath != null)
                return virtualPath;
            // If not, it must be a physical path, which we need to make safe
            return HttpRuntime.GetSafePath(linePragma);
        }
        internal static string GetVirtualPathFromHttpLinePragma(string linePragma) {
            if (String.IsNullOrEmpty(linePragma))
                return null;
            try {
                Uri uri = new Uri(linePragma);
                if (uri.Scheme == Uri.UriSchemeHttp || uri.Scheme == Uri.UriSchemeHttps)
                    return uri.LocalPath;
            }
            catch {}
            return null;
        }
        internal static string ResolveHttpFileName(string linePragma) {
            // When running under VS debugger, we use URL's instead of paths in our #line pragmas.
            // When we detect this situation, we need to do a MapPath to get back to the file name (ASURT 76211/114867)
            string virtualPath = GetVirtualPathFromHttpLinePragma(linePragma);
            // If we didn't detect a virtual path, just return the input
            if (virtualPath == null)
                return linePragma;
            return HostingEnvironment.MapPathInternal(virtualPath);
        }
        /*
         * This can be either a virtual or physical path, depending on what's available
         */
        private string GetDisplayPath() {
            if (VirtualPath != null)
                return VirtualPath;
            // It used to be an Assert on the following check but since
            // adaptive error rendering uses this method where both
            // VirtualPath and PhysicalPath might not set, it is changed to
            // an if statement.
            if (PhysicalPath != null)
                return HttpRuntime.GetSafePath(PhysicalPath);
            return null;
        }
    }
    /*
     * This formatter is used for runtime exceptions that don't fall into a
     * specific category.
     */
    internal class UnhandledErrorFormatter : ErrorFormatter {
        protected Exception _e;
        protected Exception _initialException;
        protected ArrayList _exStack = new ArrayList();
        protected string _physicalPath;
        protected int _line;
        private string _coloredSquare2Content;
        private bool _fGeneratedCodeOnStack;
        protected String _message;
        protected String _postMessage;
        internal UnhandledErrorFormatter(Exception e) : this(e, null, null){
        }
        internal UnhandledErrorFormatter(Exception e, String message, String postMessage) {
            _message = message;
            _postMessage = postMessage;
            _e = e;
        }
        internal /*public*/ override void PrepareFormatter() {
            // Build a stack of exceptions
            for (Exception e = _e; e != null; e = e.InnerException) {
                _exStack.Add(e);
                // Keep track of the initial exception (first one thrown)
                _initialException = e;
            }
            // Get the Square2Content first so the line number gets calculated
            _coloredSquare2Content = ColoredSquare2Content;
        }
        protected override Exception Exception {
            get { return _e; }
        }
        protected override string ErrorTitle {
            get {
                // Use the exception's message if there is one
                string msg = _initialException.Message;
                if (!String.IsNullOrEmpty(msg))
                    return HttpUtility.FormatPlainTextAsHtml(msg);
                // Otherwise, use some default string
                return SR.GetString(SR.Unhandled_Err_Error);
            }
        }
        protected override string Description {
            get {
                if (_message != null) {
                    return _message;
                }
                else {
                    return SR.GetString(SR.Unhandled_Err_Desc);
                }
            }
        }
        protected override string MiscSectionTitle {
            get { return SR.GetString(SR.Unhandled_Err_Exception_Details);}
        }
        protected override string MiscSectionContent {
            get {
                string exceptionName = _initialException.GetType().FullName;
                StringBuilder msg = new StringBuilder(exceptionName);
                string adaptiveMiscLine = exceptionName;
                if (_initialException.Message != null) {
                    string errorMessage = HttpUtility.FormatPlainTextAsHtml(_initialException.Message);
                    msg.Append(": ");
                    msg.Append(errorMessage);
                    adaptiveMiscLine += ": " + errorMessage;
                }
                AdaptiveMiscContent.Add(adaptiveMiscLine);
                if (_initialException is UnauthorizedAccessException) {
                    msg.Append("\r\n
");
                    String errDesc = SR.GetString(SR.Unauthorized_Err_Desc1);
                    errDesc = HttpUtility.HtmlEncode(errDesc);
                    msg.Append(errDesc);
                    AdaptiveMiscContent.Add(errDesc);
                    msg.Append("\r\n
");
                    errDesc = SR.GetString(SR.Unauthorized_Err_Desc2);
                    errDesc = HttpUtility.HtmlEncode(errDesc);
                    msg.Append(errDesc);
                    AdaptiveMiscContent.Add(errDesc);
                }
                else if (_initialException is HostingEnvironmentException) {
                    String details = ((HostingEnvironmentException)_initialException).Details;
                    if (!String.IsNullOrEmpty(details)) {
                        msg.Append("\r\n
");
                        msg.Append(details);
                        msg.Append("");
                        AdaptiveMiscContent.Add(details);
                    }
                }
                return msg.ToString();
            }
        }
        protected override string ColoredSquareTitle {
            get { return SR.GetString(SR.TmplCompilerSourceSecTitle);}
        }
        protected override string ColoredSquareContent {
            get {
                // If we couldn't get line info for the error, display a standard message
                if (_physicalPath == null) {
                    const string BeginLeftToRightMarker = "BeginMarker";
                    const string EndLeftToRightMarker = "EndMarker";
                    bool setLeftToRightMarker = false;
                    // The error text depends on whether .aspx code was found on the stack
                    // Also, if trust is less than medium, never display the message that
                    // explains how to turn on debugging, since it's not allowed (Whidbey 9176)
                    string msg;
                    if (!_fGeneratedCodeOnStack ||
                        !HttpRuntime.HasAspNetHostingPermission(AspNetHostingPermissionLevel.Medium)) {
                        msg = SR.GetString(SR.Src_not_available_nodebug);
                    }
                    else {
                        if (IsTextRightToLeft) {
                            setLeftToRightMarker = true;
                        }
                        // Because the resource string has both normal language text and config/code samples,
                        // left-to-right markup tags need to be wrapped around the config/code samples if
                        // right to left language format is being used.
                        //
                        // Note that the retrieved resource string will be passed to the call
                        // HttpUtility.FormatPlainTextAsHtml(), which does HtmlEncode.  In order to preserve
                        // the left-to-right markup tags, the resource string has been added with markers
                        // that identify the beginnings and ends of config/code samples.  After
                        // FormatPlainTextAsHtml() is called, and the markers will be replaced with
                        // left-to-right markup tags below.
                        msg = SR.GetString(SR.Src_not_available,
                                           ((setLeftToRightMarker) ? BeginLeftToRightMarker : string.Empty),
                                           ((setLeftToRightMarker) ? EndLeftToRightMarker : string.Empty),
                                           ((setLeftToRightMarker) ? BeginLeftToRightMarker : string.Empty),
                                           ((setLeftToRightMarker) ? EndLeftToRightMarker : string.Empty));
                    }
                    msg = HttpUtility.FormatPlainTextAsHtml(msg);
                    if (setLeftToRightMarker) {
                        // If only  was used to wrap around the left-to-right code text,
                        // the font rendering on Firefox was not good.  We use 
 in addition to
                        // the  tag to workaround the problem.
                        const string BeginLeftToRightTags = "" + BeginLeftToRightTag + "
";
                        const string EndLeftToRightTags = "" + EndLeftToRightTag + "
";
                        msg = msg.Replace(BeginLeftToRightMarker, BeginLeftToRightTags);
                        msg = msg.Replace(EndLeftToRightMarker, EndLeftToRightTags);
                    }
                    return msg;
                }
                return FormatterWithFileInfo.GetSourceFileLines(_physicalPath, Encoding.Default, null, _line);
            }
        }
        protected override bool WrapColoredSquareContentLines {
            // Only wrap the text if we're displaying the standard message
            get { return (_physicalPath == null);}
        }
        protected override string ColoredSquare2Title {
            get { return SR.GetString(SR.Unhandled_Err_Stack_Trace);}
        }
        protected override string ColoredSquare2Content {
            get {
                if (_coloredSquare2Content != null)
                    return _coloredSquare2Content;
                StringBuilder sb = new StringBuilder();
                bool addAdaptiveStackTrace = true;
                int sbBeginIndex = 0;
                for (int i = _exStack.Count - 1; i >=0; i--) {
                    if (i < _exStack.Count - 1)
                        sb.Append("\r\n");
                    Exception e = (Exception)_exStack[i];
                    sb.Append("[" + _exStack[i].GetType().Name);
                    // Display the error code if there is one
                    if ((e is ExternalException) && ((ExternalException) e).ErrorCode != 0)
                        sb.Append(" (0x" + (((ExternalException)e).ErrorCode).ToString("x", CultureInfo.CurrentCulture) + ")");
                    // Display the message if there is one
                    if (e.Message != null && e.Message.Length > 0)
                        sb.Append(": " + e.Message);
                    sb.Append("]\r\n");
                    // Display the stack trace
                    StackTrace st = new StackTrace(e, true /*fNeedFileInfo*/);
                    for (int j = 0; j < st.FrameCount; j++) {
                        if (addAdaptiveStackTrace) {
                            sbBeginIndex = sb.Length;
                        }
                        StackFrame sf = st.GetFrame(j);
                        MethodBase mb = sf.GetMethod();
                        Type declaringType = mb.DeclaringType;
                        string ns = String.Empty;
                        if (declaringType != null) {
                            // Check if this stack item is for ASP generated code (ASURT 51063).
                            // To do this, we check if the assembly lives in the codegen dir.
                            // But if the native offset is 0, it is likely that the method simply
                            // failed to JIT, in which case don't treat it as an ASP.NET stack,
                            // since no line number can ever be shown for it (VSWhidbey 87014).
                            string assemblyDir = null;
                            try {
                                // This could throw if the assembly is dynamic
                                assemblyDir = System.Web.UI.Util.GetAssemblyCodeBase(declaringType.Assembly);
                            }
                            catch {}
                            if (assemblyDir != null) {
                                assemblyDir = Path.GetDirectoryName(assemblyDir);
                                if (string.Compare(assemblyDir, HttpRuntime.CodegenDirInternal,
                                    StringComparison.OrdinalIgnoreCase) == 0 && sf.GetNativeOffset() > 0) {
                                    _fGeneratedCodeOnStack = true;
                                }
                            }
                            ns = declaringType.Namespace;
                        }
                        if (ns != null)
                            ns = ns + ".";
                        if (declaringType == null) {
                            sb.Append("   " + mb.Name + "(");
                        }
                        else {
                            sb.Append("   " + ns + declaringType.Name + "." +
                                mb.Name + "(");
                        }
                        ParameterInfo[] arrParams = mb.GetParameters();
                        for (int k = 0; k < arrParams.Length; k++) {
                            sb.Append((k != 0 ? ", " : String.Empty) + arrParams[k].ParameterType.Name + " " +
                                arrParams[k].Name);
                        }
                        sb.Append(")");
                        string fileName = GetFileName(sf);
                        if (fileName != null) {
                            // ASURT 114867: if it's an http path, turn it into a local path
                            fileName = ResolveHttpFileName(fileName);
                            if (fileName != null) {
                                // Remember the file/line number of the top level stack
                                // item for which we have symbols
                                if (_physicalPath == null && FileUtil.FileExists(fileName)) {
                                    _physicalPath = fileName;
                                    _line = sf.GetFileLineNumber();
                                }
                                sb.Append(" in " + HttpRuntime.GetSafePath(fileName) +
                                    ":" + sf.GetFileLineNumber());
                            }
                        }
                        else {
                            sb.Append(" +" + sf.GetNativeOffset());
                        }
                        if (addAdaptiveStackTrace) {
                            string stackTraceText = sb.ToString(sbBeginIndex,
                                                                sb.Length - sbBeginIndex);
                            AdaptiveStackTrace.Add(HttpUtility.HtmlEncode(stackTraceText));
                        }
                        sb.Append("\r\n");
                    }
                    // Due to size limitation, we only want to add the top
                    // stack trace for mobile devices.
                    addAdaptiveStackTrace = false;
                }
                _coloredSquare2Content = HttpUtility.HtmlEncode(sb.ToString());
                _coloredSquare2Content = WrapWithLeftToRightTextFormatIfNeeded(_coloredSquare2Content);
                return _coloredSquare2Content;
            }
        }
        // Dev10 786146: partial trust apps may not have PathDiscovery, so just pretend
        // the path is unknown.
        private string GetFileName(StackFrame sf) {
            string fileName = null;
            try {
                fileName = sf.GetFileName();
            }
            catch (SecurityException) {
            }
            return fileName;
        }
        protected override String PostMessage {
            get { return _postMessage; }
        }
        protected override bool ShowSourceFileInfo {
            get { return _physicalPath != null; }
        }
        protected override string PhysicalPath {
            get { return _physicalPath; }
        }
        protected override int SourceFileLineNumber {
            get { return _line; }
        }
    }
    /*
     * This formatter is used for security exceptions.
     */
    internal class SecurityErrorFormatter : UnhandledErrorFormatter {
        internal SecurityErrorFormatter(Exception e) : base(e) {}
        protected override string ErrorTitle {
            get {
                return SR.GetString(SR.Security_Err_Error);
            }
        }
        protected override string Description {
            get {
                // VSWhidbey 493720: Do Html encode to preserve space characters
                return HttpUtility.FormatPlainTextAsHtml(SR.GetString(SR.Security_Err_Desc));
            }
        }
    }
    /*
     * This formatter is used for 404: page not found errors
     */
    internal class PageNotFoundErrorFormatter : ErrorFormatter {
        protected string _htmlEncodedUrl;
        private StringCollection _adaptiveMiscContent = new StringCollection();
        internal PageNotFoundErrorFormatter(string url) {
            _htmlEncodedUrl = HttpUtility.HtmlEncode(url);
            _adaptiveMiscContent.Add(_htmlEncodedUrl);
        }
        protected override string ErrorTitle {
            get { return SR.GetString(SR.NotFound_Resource_Not_Found);}
        }
        protected override string Description {
            get { return HttpUtility.FormatPlainTextAsHtml(SR.GetString(SR.NotFound_Http_404));}
        }
        protected override string MiscSectionTitle {
            get { return SR.GetString(SR.NotFound_Requested_Url);}
        }
        protected override string MiscSectionContent {
            get { return _htmlEncodedUrl;}
        }
        protected override StringCollection AdaptiveMiscContent {
            get { return _adaptiveMiscContent;}
        }
        protected override bool ShowSourceFileInfo {
            get { return false;}
        }
        internal override bool CanBeShownToAllUsers {
            get { return true;}
        }
    }
    /*
     * This formatter is used for 403: forbidden
     */
    internal class PageForbiddenErrorFormatter : ErrorFormatter {
        protected string _htmlEncodedUrl;
        private StringCollection _adaptiveMiscContent = new StringCollection();
        private string _description;
        internal PageForbiddenErrorFormatter(string url): this(url, null) {
        }
        internal PageForbiddenErrorFormatter(string url, string description) {
            _htmlEncodedUrl = HttpUtility.HtmlEncode(url);
            _adaptiveMiscContent.Add(_htmlEncodedUrl);
            _description = description;
        }
        protected override string ErrorTitle {
            get { return SR.GetString(SR.Forbidden_Type_Not_Served);}
        }
        protected override string Description {
            get {
                if (_description != null) {
                    return _description;
                }
                Match m = Regex.Match(_htmlEncodedUrl, @"\.\w+$");
                String extMessage = String.Empty;
                if (m.Success)
                    extMessage = SR.GetString(SR.Forbidden_Extension_Incorrect, m.ToString());
                return HttpUtility.FormatPlainTextAsHtml(SR.GetString(SR.Forbidden_Extension_Desc, extMessage));
            }
        }
        protected override string MiscSectionTitle {
            get { return SR.GetString(SR.NotFound_Requested_Url);}
        }
        protected override string MiscSectionContent {
            get { return _htmlEncodedUrl;}
        }
        protected override StringCollection AdaptiveMiscContent {
            get { return _adaptiveMiscContent;}
        }
        protected override bool ShowSourceFileInfo {
            get { return false;}
        }
        internal override bool CanBeShownToAllUsers {
            get { return true;}
        }
    }
    /*
     * This formatter is used for generic errors that hide sensitive information
     * error text is sometimes different for remote vs. local machines
     */
    internal class GenericApplicationErrorFormatter : ErrorFormatter {
        private bool _local;
        internal GenericApplicationErrorFormatter(bool local) {
            _local = local;
        }
        protected override string ErrorTitle {
            get {
                return SR.GetString(SR.Generic_Err_Title);
            }
        }
        protected override string Description {
            get {
                return SR.GetString(
                                    _local ? SR.Generic_Err_Local_Desc
                                           : SR.Generic_Err_Remote_Desc);
            }
        }
        protected override string MiscSectionTitle {
            get {
                return null;
            }
        }
        protected override string MiscSectionContent {
            get {
                return null;
            }
        }
        protected override string ColoredSquareTitle {
            get {
                String detailsTitle = SR.GetString(SR.Generic_Err_Details_Title);
                AdaptiveMiscContent.Add(detailsTitle);
                return detailsTitle;
            }
        }
        protected override string ColoredSquareDescription {
            get {
                String detailsDesc = SR.GetString(
                                    _local ? SR.Generic_Err_Local_Details_Desc
                                           : SR.Generic_Err_Remote_Details_Desc);
                detailsDesc = HttpUtility.HtmlEncode(detailsDesc);
                AdaptiveMiscContent.Add(detailsDesc);
                return detailsDesc;
            }
        }
        protected override string ColoredSquareContent {
            get {
                string content = HttpUtility.HtmlEncode(SR.GetString(
                                    _local ? SR.Generic_Err_Local_Details_Sample
                                           : SR.Generic_Err_Remote_Details_Sample));
                return (WrapWithLeftToRightTextFormatIfNeeded(content));
            }
        }
        protected override string ColoredSquare2Title {
            get {
                String noteTitle = SR.GetString(SR.Generic_Err_Notes_Title);
                AdaptiveMiscContent.Add(noteTitle);
                return noteTitle;
            }
        }
        protected override string ColoredSquare2Description {
            get {
                String notesDesc = SR.GetString(SR.Generic_Err_Notes_Desc);
                notesDesc = HttpUtility.HtmlEncode(notesDesc);
                AdaptiveMiscContent.Add(notesDesc);
                return notesDesc;
            }
        }
        protected override string ColoredSquare2Content {
            get {
                string content = HttpUtility.HtmlEncode(SR.GetString(
                                    _local ? SR.Generic_Err_Local_Notes_Sample
                                           : SR.Generic_Err_Remote_Notes_Sample));
                return (WrapWithLeftToRightTextFormatIfNeeded(content));
            }
        }
        protected override bool ShowSourceFileInfo {
            get { return false;}
        }
        internal override bool CanBeShownToAllUsers {
            get { return true;}
        }
    }
    /*
    * This formatter is used when we couldn't run the normal custom error page (due to it also failing)
    */
    internal class CustomErrorFailedErrorFormatter : ErrorFormatter {
        internal CustomErrorFailedErrorFormatter() {
        }
        protected override string ErrorTitle {
            get { return SR.GetString(SR.Generic_Err_Title); }
        }
        protected override string Description {
            get { return HttpUtility.FormatPlainTextAsHtml(SR.GetString(SR.CustomErrorFailed_Err_Desc)); }
        }
        protected override string MiscSectionTitle {
            get { return null; }
        }
        protected override string MiscSectionContent {
            get { return null; }
        }
        protected override bool ShowSourceFileInfo {
            get { return false; }
        }
        internal override bool CanBeShownToAllUsers {
            get { return true; }
        }
    }
    /*
     * This is the base class for formatter that handle errors that have an
     * associated file / line number.
     */
    internal abstract class FormatterWithFileInfo : ErrorFormatter {
        protected string _virtualPath;
        protected string _physicalPath;
        protected string _sourceCode;
        protected int _line;
        // Number of lines before and after the error lines included in the report
        private const int errorRange = 2;
        /*
         * Return the text of the error line in the source file, with a few
         * lines around it.  It is returned in HTML format.
         */
        internal static string GetSourceFileLines(string fileName, Encoding encoding, string sourceCode, int lineNumber) {
            // Don't show any source file if the user doesn't have access to it (ASURT 122430)
            if (fileName != null && !HttpRuntime.HasFilePermission(fileName))
                return SR.GetString(SR.WithFile_No_Relevant_Line);
            // 
            StringBuilder sb = new StringBuilder();
            if (lineNumber <= 0) {
                return SR.GetString(SR.WithFile_No_Relevant_Line);
            }
            TextReader reader = null;
            // Check if it's an http line pragma, from which we can get a VirtualPath
            string virtualPath = GetVirtualPathFromHttpLinePragma(fileName);
            // If we got a virtual path, open a TextReader from it
            if (virtualPath != null) {
                Stream stream = VirtualPathProvider.OpenFile(virtualPath);
                if (stream != null)
                    reader = System.Web.UI.Util.ReaderFromStream(stream, System.Web.VirtualPath.Create(virtualPath));
            }
            try {
                // Otherwise, open the physical file
                if (reader == null && fileName != null)
                    reader = new StreamReader(fileName, encoding, true, 4096);
            }
            catch { }
            if (reader == null) {
                if (sourceCode == null)
                    return SR.GetString(SR.WithFile_No_Relevant_Line);
                // Can't open the file?  Use the dynamically generated content...
                reader = new StringReader(sourceCode);
            }
            try {
                bool fFoundLine = false;
                if (IsTextRightToLeft) {
                    sb.Append(BeginLeftToRightTag);
                }
                for (int i=1; ; i++) {
                    // Get the current line from the source file
                    string sourceLine = reader.ReadLine();
                    if (sourceLine == null)
                        break;
                    // If it's the error line, make it red
                    if (i == lineNumber)
                        sb.Append("");
                    // Is it in the range we want to display
                    if (i >= lineNumber-errorRange && i <= lineNumber+errorRange) {
                        fFoundLine = true;
                        String linestr = i.ToString("G", CultureInfo.CurrentCulture);
                        sb.Append(SR.GetString(SR.WithFile_Line_Num, linestr));
                        if (linestr.Length < 3)
                            sb.Append(' ', 3 - linestr.Length);
                        sb.Append(HttpUtility.HtmlEncode(sourceLine));
                        if (i != lineNumber+errorRange)
                            sb.Append("\r\n");
                    }
                    if (i == lineNumber)
                        sb.Append("");
                    if (i>lineNumber+errorRange)
                        break;
                }
                if (IsTextRightToLeft) {
                    sb.Append(EndLeftToRightTag);
                }
                if (!fFoundLine)
                    return SR.GetString(SR.WithFile_No_Relevant_Line);
            }
            finally {
                // Make sure we always close the reader
                reader.Close();
            }
            return sb.ToString();
        }
        private string GetSourceFileLines() {
            return GetSourceFileLines(_physicalPath, SourceFileEncoding, _sourceCode, _line);
        }
        internal FormatterWithFileInfo(string virtualPath, string physicalPath,
            string sourceCode, int line) {
            _virtualPath = virtualPath;
            _physicalPath = physicalPath;
            if (sourceCode == null && _physicalPath == null && _virtualPath != null) {
                // Make sure _virtualPath is really a virtual path.  Sometimes,
                // it can actually be a physical path, in which case we keep
                // it as is.
                if (UrlPath.IsValidVirtualPathWithoutProtocol(_virtualPath))
                    _physicalPath = HostingEnvironment.MapPath(_virtualPath);
                else
                    _physicalPath = _virtualPath;
            }
            _sourceCode = sourceCode;
            _line = line;
        }
        protected virtual Encoding SourceFileEncoding {
            get { return Encoding.Default; }
        }
        protected override string ColoredSquareContent {
            get { return GetSourceFileLines();}
        }
        protected override bool ShowSourceFileInfo {
            get { return true;}
        }
        protected override string PhysicalPath {
            get { return _physicalPath;}
        }
        protected override string VirtualPath {
            get { return _virtualPath;}
        }
        protected override int SourceFileLineNumber {
            get { return _line;}
        }
    }
    /*
     * Formatter used for compilation errors
     */
    internal class DynamicCompileErrorFormatter : ErrorFormatter {
        private const string startExpandableBlock =
            "
" +
            "{1}" +
            ":
\r\n" +
            "\r\n" +
            "            
\r\n" +
            "               \r\n" +
            "                  | \r\n" +
            " \r\n" +
            "\r\n\r\n" +
            "\r\n\r\n";
        private const string endExpandableBlock =
            " | 
\r\n" +
            "            
\r\n\r\n" +
            "            \r\n\r\n" +
            "
\r\n");
                }
                if (results.Errors.HasErrors) {
                    CompilerError e = _excep.FirstCompileError;
                    if (e != null) {
                        string htmlEncodedText = HttpUtility.HtmlEncode(e.ErrorNumber);
                        string adaptiveContentLine = htmlEncodedText;
                        sb.Append(htmlEncodedText);
                        // Don't show the error message in low trust (VSWhidbey 87012)
                        if (HttpRuntime.HasAspNetHostingPermission(AspNetHostingPermissionLevel.Medium)) {
                            htmlEncodedText = HttpUtility.HtmlEncode(e.ErrorText);
                            sb.Append(": ");
                            sb.Append(htmlEncodedText);
                            adaptiveContentLine += ": " + htmlEncodedText;
                        }
                        AdaptiveMiscContent.Add(adaptiveContentLine);
                        sb.Append("
\r\n");
                        sb.Append("");
                        sb.Append(SR.GetString(SR.TmplCompilerSourceSecTitle));
                        sb.Append(":
\r\n");
                        sb.Append("            \r\n");
                        sb.Append("               | \r\n");
                        sb.Append("               ");
                        sb.Append(" | 
\r\n");
                        sb.Append("               \r\n");
                        sb.Append("                  | \r\n");
                        sb.Append(" \r\n");
                        sb.Append("\r\n\r\n");
                        sb.Append("\r\n\r\n");
                        sb.Append(FormatterWithFileInfo.GetSourceFileLines(e.FileName, Encoding.Default, _excep.SourceCodeWithoutDemand, e.Line));
                        sb.Append(" | 
\r\n");
                        sb.Append("            
\r\n\r\n");
                        sb.Append("            
\r\n\r\n");
                        // display file
                        sb.Append("            ");
                        sb.Append(SR.GetString(SR.TmplCompilerSourceFileTitle));
                        sb.Append(": ");
                        _sourceFilePath = GetSafePath(e.FileName);
                        sb.Append(HttpUtility.HtmlEncode(_sourceFilePath));
                        sb.Append("\r\n");
                        // display number
                        TypeConverter itc = new Int32Converter();
                        sb.Append("               ");
                        sb.Append(SR.GetString(SR.TmplCompilerSourceFileLine));
                        sb.Append(":  ");
                        _sourceFileLineNumber = e.Line;
                        sb.Append(HttpUtility.HtmlEncode(itc.ConvertToString(_sourceFileLineNumber)));
                        sb.Append("\r\n");
                        sb.Append("            
\r\n");
                    }
                }
                if (results.Errors.HasWarnings) {
                    sb.Append("
");
                    sb.Append(SR.GetString(SR.TmplCompilerWarningBanner));
                    sb.Append(":
\r\n");
                    sb.Append("\r\n");
                    foreach (CompilerError e in results.Errors) {
                        if (e.IsWarning) {
                            sb.Append("
");
                            sb.Append(SR.GetString(SR.TmplCompilerWarningSecTitle));
                            sb.Append(": ");
                            sb.Append(HttpUtility.HtmlEncode(e.ErrorNumber));
                            // Don't show the error message in low trust (VSWhidbey 87012)
                            if (HttpRuntime.HasAspNetHostingPermission(AspNetHostingPermissionLevel.Medium)) {
                                sb.Append(": ");
                                sb.Append(HttpUtility.HtmlEncode(e.ErrorText));
                            }
                            sb.Append("
\r\n");
                            sb.Append("
");
                            sb.Append(SR.GetString(SR.TmplCompilerSourceSecTitle));
                            sb.Append(":\r\n");
                            sb.Append("            
\r\n");
                            sb.Append("               | \r\n");
                            sb.Append("               ");
                            sb.Append(HttpUtility.HtmlEncode(HttpRuntime.GetSafePath(e.FileName)));
                            sb.Append("\r\n");
                            sb.Append(" | 
\r\n");
                            sb.Append("               \r\n");
                            sb.Append("                  | \r\n");
                            sb.Append(" \r\n");
                            sb.Append("\r\n\r\n");
                            sb.Append("\r\n\r\n");
                            sb.Append(FormatterWithFileInfo.GetSourceFileLines(e.FileName, Encoding.Default, _excep.SourceCodeWithoutDemand, e.Line));
                            sb.Append(" | 
\r\n");
                            sb.Append("            
\r\n\r\n");
                            sb.Append("            
\r\n\r\n");
                        }
                    }
                    sb.Append("
{0}
{1}
";
        internal const string WmlErrorEndTemplate = @"
";
        internal const string XhtmlErrorBeginTemplate = @"
";
        internal const string Break = "
\r\n";
    }
}