//------------------------------------------------------------------------------
//
// 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";
private const string endExpandableBlock =
" \r\n\r\n" +
" | \r\n" +
"
\r\n" +
"
\r\n\r\n" +
" \r\n\r\n" +
"
\r\n";
private const string toggleScript = @"
";
protected const string BeginLeftToRightTag = "";
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(" ");
if (!wrapContentLines)
sb.Append("");
sb.Append("\r\n\r\n");
sb.Append(content);
if (!wrapContentLines)
sb.Append(" ");
sb.Append(" \r\n\r\n");
sb.Append(" | \r\n");
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";
private const string endExpandableBlock =
" \r\n\r\n" +
" | \r\n" +
"
\r\n" +
"
\r\n\r\n" +
" \r\n\r\n" +
"
\r\n";
// Number of lines before and after the error lines included in the report
private const int errorRange = 2;
HttpCompileException _excep;
private string _sourceFilePath = null;
private int _sourceFileLineNumber = 0;
protected bool _hideDetailedCompilerOutput = false;
internal DynamicCompileErrorFormatter(HttpCompileException excep) {
_excep = excep;
}
protected override Exception Exception {
get { return _excep; }
}
protected override bool ShowSourceFileInfo {
get {
return false;
}
}
protected override string ErrorTitle {
get {
return SR.GetString(SR.TmplCompilerErrorTitle);
}
}
protected override string Description {
get {
return SR.GetString(SR.TmplCompilerErrorDesc);
}
}
protected override string MiscSectionTitle {
get {
return SR.GetString(SR.TmplCompilerErrorSecTitle);
}
}
protected override string MiscSectionContent {
get {
StringBuilder sb = new StringBuilder(128);
CompilerResults results = _excep.ResultsWithoutDemand;
// Handle fatal errors where we couldn't find an error line
if (results.Errors.Count == 0 && results.NativeCompilerReturnValue != 0) {
string fatalError = SR.GetString(SR.TmplCompilerFatalError,
results.NativeCompilerReturnValue.ToString("G",
CultureInfo.CurrentCulture));
AdaptiveMiscContent.Add(fatalError);
sb.Append(fatalError);
sb.Append("
\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\r\n");
sb.Append(FormatterWithFileInfo.GetSourceFileLines(e.FileName, Encoding.Default, _excep.SourceCodeWithoutDemand, e.Line));
sb.Append(" \r\n\r\n");
sb.Append(" | \r\n");
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\r\n");
sb.Append(FormatterWithFileInfo.GetSourceFileLines(e.FileName, Encoding.Default, _excep.SourceCodeWithoutDemand, e.Line));
sb.Append(" \r\n\r\n");
sb.Append(" | \r\n");
sb.Append("
\r\n");
sb.Append("
\r\n\r\n");
sb.Append("
\r\n\r\n");
}
}
sb.Append("
\r\n");
}
if (!_hideDetailedCompilerOutput) {
if (results.Output.Count > 0) {
// (Only display the compiler output in medium or higher (ASURT 126827)
if (HttpRuntime.HasAspNetHostingPermission(AspNetHostingPermissionLevel.Medium)) {
sb.Append(String.Format(CultureInfo.CurrentCulture, startExpandableBlock, "compilerOutputDiv",
SR.GetString(SR.TmplCompilerCompleteOutput)));
foreach (string line in results.Output) {
sb.Append(HttpUtility.HtmlEncode(line));
sb.Append("\r\n");
}
sb.Append(endExpandableBlock);
}
}
// If we have the generated source code, display it
// (Only display the source in medium or higher (ASURT 128039)
if (_excep.SourceCodeWithoutDemand != null &&
HttpRuntime.HasAspNetHostingPermission(AspNetHostingPermissionLevel.Medium)) {
sb.Append(String.Format(CultureInfo.CurrentCulture, startExpandableBlock, "dynamicCodeDiv",
SR.GetString(SR.TmplCompilerGeneratedFile)));
string[] sourceLines = _excep.SourceCodeWithoutDemand.Split('\n');
int currentLine = 1;
foreach (string s in sourceLines) {
string number = currentLine.ToString("G", CultureInfo.CurrentCulture);
sb.Append(SR.GetString(SR.TmplCompilerLineHeader, number));
if (number.Length < 5) {
sb.Append(' ', 5 - number.Length);
}
currentLine++;
sb.Append(HttpUtility.HtmlEncode(s));
}
sb.Append(endExpandableBlock);
}
sb.Append(@"
");
}
return sb.ToString();
}
}
// This is calculated in MiscSectionContent
protected override string PhysicalPath {
get { return _sourceFilePath;}
}
protected override int SourceFileLineNumber {
get { return _sourceFileLineNumber;}
}
}
/*
* Formatter used for parse errors
*/
internal class ParseErrorFormatter : FormatterWithFileInfo {
protected string _message;
HttpParseException _excep;
private StringCollection _adaptiveMiscContent = new StringCollection();
internal ParseErrorFormatter(HttpParseException e, string virtualPath,
string sourceCode, int line, string message)
: base(virtualPath, null /*physicalPath*/, sourceCode, line) {
_excep = e;
_message = HttpUtility.FormatPlainTextAsHtml(message);
_adaptiveMiscContent.Add(_message);
}
protected override Exception Exception {
get { return _excep; }
}
protected override string ErrorTitle {
get { return SR.GetString(SR.Parser_Error);}
}
protected override string Description {
get { return SR.GetString(SR.Parser_Desc);}
}
protected override string MiscSectionTitle {
get { return SR.GetString(SR.Parser_Error_Message);}
}
protected override string MiscSectionContent {
get { return _message;}
}
protected override string ColoredSquareTitle {
get { return SR.GetString(SR.Parser_Source_Error);}
}
protected override StringCollection AdaptiveMiscContent {
get { return _adaptiveMiscContent;}
}
}
/*
* Formatter used for configuration errors
*/
internal class ConfigErrorFormatter : FormatterWithFileInfo {
protected string _message;
private Exception _e;
private StringCollection _adaptiveMiscContent = new StringCollection();
private bool _allowSourceCode;
internal ConfigErrorFormatter(System.Configuration.ConfigurationException e)
: base(null /*virtualPath*/, e.Filename, null, e.Line) {
_e = e;
PerfCounters.IncrementCounter(AppPerfCounter.ERRORS_PRE_PROCESSING);
PerfCounters.IncrementCounter(AppPerfCounter.ERRORS_TOTAL);
_message = HttpUtility.FormatPlainTextAsHtml(e.BareMessage);
_adaptiveMiscContent.Add(_message);
}
public bool AllowSourceCode {
get { return _allowSourceCode; }
set { _allowSourceCode = value; }
}
protected override Encoding SourceFileEncoding {
get { return Encoding.UTF8; }
}
protected override Exception Exception {
get { return _e; }
}
protected override string ErrorTitle {
get { return SR.GetString(SR.Config_Error);}
}
protected override string Description {
get { return SR.GetString(SR.Config_Desc);}
}
protected override string MiscSectionTitle {
get { return SR.GetString(SR.Parser_Error_Message);}
}
protected override string MiscSectionContent {
get { return _message;}
}
protected override string ColoredSquareTitle {
get { return SR.GetString(SR.Parser_Source_Error);}
}
protected override StringCollection AdaptiveMiscContent {
get { return _adaptiveMiscContent;}
}
protected override string ColoredSquareContent {
get {
if (!AllowSourceCode) {
return SR.GetString(SR.Generic_Err_Remote_Desc);
}
return base.ColoredSquareContent;
}
}
}
/*
* Formatter to allow user-specified description strings
* use if showing inner-most exception message is not appropriate
*/
internal class UseLastUnhandledErrorFormatter : UnhandledErrorFormatter {
internal UseLastUnhandledErrorFormatter(Exception e)
: base(e) {
}
internal /*public*/ override void PrepareFormatter() {
base.PrepareFormatter();
// use the outer-most exception instead of the inner-most in the misc section
_initialException = Exception;
}
}
internal class StaticErrorFormatterHelper {
internal const string ChtmlErrorBeginTemplate = @"
";
internal const string WmlErrorBeginTemplate = @"
{0}
{1}
";
internal const string WmlErrorEndTemplate = @"
";
internal const string XhtmlErrorBeginTemplate = @"
";
internal const string Break = "
\r\n";
}
}