859 lines
34 KiB
C#
Raw Normal View History

//------------------------------------------------------------------------------
// <copyright file="TraceContext.cs" company="Microsoft">
// Copyright (c) Microsoft Corporation. All rights reserved.
// </copyright>
//------------------------------------------------------------------------------
/*
* The context for outputting trace information in the page.
*
* Copyright (c) 1999 Microsoft Corporation
*/
namespace System.Web {
using System;
//
using System.Web.UI;
using System.Web.Handlers;
using System.Web.Util;
using System.Web.SessionState;
using System.Text;
using System.Data;
using System.Globalization;
using System.Collections;
using System.Collections.Specialized;
using System.Diagnostics;
using System.Security.Permissions;
using System.Web.Configuration;
using System.Threading;
using System.ComponentModel;
/// <devdoc>
/// <para>Captures and presents execution details about a Web request.</para>
/// <para>Use the TraceContext
/// class by appending trace messages to specific trace categories. For example, a
/// calendar class might append the message ?Calendar1-&gt;Starting
/// To Render? within the Render category, and append the message ?Calendar1-&gt;Firing OnChange Event? within
/// the Events category. Tracing is enabled by setting the <see topic='cpdirSystem.Web.UI.PageDirectives'/>
/// Trace attribute or the System.Web.UI.TraceContext.IsEnabled
/// property.</para>
/// <para>When tracing is enabled, In addition to showing
/// user-provided trace content, the <see cref='System.Web.UI.Page'/> class not only shows user-provided trace content, it automatically includes
/// performance data, tree-structure information, and state management content.</para>
/// </devdoc>
public sealed
class TraceContext {
private static DataSet _masterRequest;
private static bool _writeToDiagnosticsTrace = false;
private static readonly object EventTraceFinished = new object();
private EventHandlerList _events = new EventHandlerList();
private TraceMode _traceMode;
private TraceEnable _isEnabled;
private HttpContext _context;
private DataSet _requestData;
private long _firstTime;
private long _lastTime;
private int _uniqueIdCounter = 0;
private const string PAGEKEYNAME = "__PAGE";
private const string NULLSTRING = "<null>"; // this will get HtmlEncoded later...
private const string NULLIDPREFIX = "__UnassignedID";
private ArrayList _traceRecords;
private bool _endDataCollected;
private bool _writing = false;
/// <devdoc>
/// <para>Initializes a new instance of the <see cref='System.Web.TraceContext'/> class.</para>
/// </devdoc>
public TraceContext(HttpContext context) {
_traceMode = TraceMode.Default;
// Always disable trace in retail deployment mode (DevDiv 36396)
_isEnabled = DeploymentSection.RetailInternal ? TraceEnable.Disable : TraceEnable.Default;
_context = context;
_firstTime = -1;
_lastTime = -1;
_endDataCollected = false;
_traceRecords = new ArrayList();
}
/// <devdoc>
/// <para>Indicates the order in which trace messages should be
/// output to a requesting browser. Trace messages can be sorted in the order they
/// were processed, or alphabetically by user-defined category.</para>
/// </devdoc>
public TraceMode TraceMode {
get {
if (_traceMode == TraceMode.Default)
return HttpRuntime.Profile.OutputMode;
return _traceMode;
}
set {
if (value < TraceMode.SortByTime || value > TraceMode.Default) {
throw new ArgumentOutOfRangeException("value");
}
_traceMode = value;
if (IsEnabled)
ApplyTraceMode();
}
}
/// <devdoc>
/// <para>Indicates whether tracing is enabled for the current Web request.
/// Use this flag to check whether you should output tracing information before
/// writing anything to the trace log.</para>
/// </devdoc>
public bool IsEnabled {
get {
if (_isEnabled == TraceEnable.Default)
return HttpRuntime.Profile.IsEnabled;
else {
return (_isEnabled == TraceEnable.Enable) ? true : false;
}
}
set {
// Always disable trace in retail deployment mode (DevDiv 36396)
if (DeploymentSection.RetailInternal) {
System.Web.Util.Debug.Assert(_isEnabled == TraceEnable.Disable);
return;
}
if (value)
_isEnabled = TraceEnable.Enable;
else
_isEnabled = TraceEnable.Disable;
}
}
/// <devdoc>
/// <para>Indicates whether trace information should be output to a Web Forms
/// page. This property is used only in application-level tracing situations. You
/// can set this property in your application's config.web configuration file which
/// resides in the root directory of the application. </para>
/// </devdoc>
internal bool PageOutput {
get {
if (_isEnabled == TraceEnable.Default)
return HttpRuntime.Profile.PageOutput;
else {
return (_isEnabled == TraceEnable.Enable) ? true : false;
}
}
}
// this is used only for error pages, called from Page.HandleError.
internal int StatusCode {
set {
VerifyStart();
DataRow row = _requestData.Tables[SR.Trace_Request].Rows[0];
row[SR.Trace_Status_Code] = value;
}
}
/// <devdoc>
/// Raised after the trace has been finished
/// </devdoc>
public event TraceContextEventHandler TraceFinished {
add {
_events.AddHandler(EventTraceFinished, value);
}
remove {
_events.RemoveHandler(EventTraceFinished, value);
}
}
private void ApplyTraceMode() {
VerifyStart();
if (TraceMode == TraceMode.SortByCategory)
_requestData.Tables[SR.Trace_Trace_Information].DefaultView.Sort = SR.Trace_Category;
else
_requestData.Tables[SR.Trace_Trace_Information].DefaultView.Sort = SR.Trace_From_First;
}
internal void CopySettingsTo(TraceContext tc) {
tc._traceMode = this._traceMode;
tc._isEnabled = this._isEnabled;
}
internal void OnTraceFinished(TraceContextEventArgs e) {
TraceContextEventHandler handler = (TraceContextEventHandler)_events[EventTraceFinished];
if (handler != null) {
handler(this, e);
}
}
internal static void SetWriteToDiagnosticsTrace(bool value) {
_writeToDiagnosticsTrace = value;
}
/// <devdoc>
/// <para>Writes trace information to the trace log including any
/// user defined categories and trace messages.</para>
/// </devdoc>
public void Write(string message) {
Write(String.Empty, message, null, false, _writeToDiagnosticsTrace);
}
/// <devdoc>
/// <para>Writes trace information to the trace log including any
/// user defined categories and trace messages.</para>
/// </devdoc>
public void Write(string category, string message) {
Write(category, message, null, false, _writeToDiagnosticsTrace);
}
/// <devdoc>
/// <para>Writes trace information to the trace log including any user defined
/// categories,trace messages and error information.</para>
/// </devdoc>
public void Write(string category, string message, Exception errorInfo) {
Write(category, message, errorInfo, false, _writeToDiagnosticsTrace);
}
internal void WriteInternal(string message, bool writeToDiagnostics) {
Write(String.Empty, message, null, false, writeToDiagnostics);
}
internal void WriteInternal(string category, string message, bool writeToDiagnostics) {
Write(category, message, null, false, writeToDiagnostics);
}
/// <devdoc>
/// <para>Writes trace information to the trace log including any
/// user defined categories and trace messages. All warnings appear as red text.</para>
/// </devdoc>
public void Warn(string message) {
Write(String.Empty, message, null, true, _writeToDiagnosticsTrace);
}
/// <devdoc>
/// <para>Writes trace information to the trace log including any
/// user defined categories and trace messages. All warnings appear as red text.</para>
/// </devdoc>
public void Warn(string category, string message) {
Write(category, message, null, true, _writeToDiagnosticsTrace);
}
/// <devdoc>
/// <para>Writes trace information to the trace log including any user defined categories,trace messages and error information. All
/// warnings appear as red text. </para>
/// </devdoc>
public void Warn(string category, string message, Exception errorInfo) {
Write(category, message, errorInfo, true, _writeToDiagnosticsTrace);
}
internal void WarnInternal(string category, string message, bool writeToDiagnostics) {
Write(category, message, null, true, writeToDiagnostics);
}
void Write(string category, string message, Exception errorInfo, bool isWarning, bool writeToDiagnostics) {
// Guard against
lock(this) {
// Bail if trace isn't enabled or this is from our call to System.Diagonostics.Trace below
if (!IsEnabled || _writing || _endDataCollected)
return;
VerifyStart();
if (category == null)
category = String.Empty;
if (message == null)
message = String.Empty;
long messagetime = Counter.Value;
DataRow row = NewRow(_requestData, SR.Trace_Trace_Information);
row[SR.Trace_Category] = category;
row[SR.Trace_Message] = message;
row[SR.Trace_Warning] = isWarning ? "yes" : "no";
if (errorInfo != null) {
row["ErrorInfoMessage"] = errorInfo.Message;
row["ErrorInfoStack"] = errorInfo.StackTrace;
}
if (_firstTime != -1) {
row[SR.Trace_From_First] = (((double)(messagetime - _firstTime)) / Counter.Frequency);
}
else
_firstTime = messagetime;
if (_lastTime != -1) {
row[SR.Trace_From_Last] = (((double)(messagetime - _lastTime)) / Counter.Frequency);
}
_lastTime = messagetime;
AddRow(_requestData, SR.Trace_Trace_Information, row);
string msg = message;
if (errorInfo != null) {
string eMsg = errorInfo.Message;
if (eMsg == null) eMsg = String.Empty;
string eTrace = errorInfo.StackTrace;
if (eTrace == null) eTrace = String.Empty;
StringBuilder str = new StringBuilder(message.Length + eMsg.Length + eTrace.Length);
str.Append(message);
str.Append(" -- ");
str.Append(eMsg);
str.Append(": ");
str.Append(eTrace);
msg = str.ToString();
}
if (writeToDiagnostics) {
_writing = true;
System.Diagnostics.Trace.WriteLine(msg, category);
_writing = false;
}
// Send to IIS tracing
if (_context != null && _context.WorkerRequest != null) {
_context.WorkerRequest.RaiseTraceEvent(isWarning ? IntegratedTraceType.TraceWarn : IntegratedTraceType.TraceWrite, msg);
}
}
// Add the trace record to our collection
_traceRecords.Add(new TraceContextRecord(category, message, isWarning, errorInfo));
}
internal void AddNewControl(string id, string parentId, string type, int viewStateSize, int controlStateSize) {
VerifyStart();
DataRow row = NewRow(_requestData, SR.Trace_Control_Tree);
if (id == null)
id = NULLIDPREFIX+(_uniqueIdCounter++);
row[SR.Trace_Control_Id] = id;
if (parentId == null)
parentId = PAGEKEYNAME;
row[SR.Trace_Parent_Id] = parentId;
row[SR.Trace_Type] = type;
row[SR.Trace_Viewstate_Size] = viewStateSize;
row[SR.Trace_Controlstate_Size] = controlStateSize;
row[SR.Trace_Render_Size] = 0;
try {
AddRow(_requestData, SR.Trace_Control_Tree, row);
}
catch (ConstraintException) {
throw new HttpException(SR.GetString(SR.Duplicate_id_used, id, "Trace"));
}
}
/*
* Add the render size to the control
*/
internal void AddControlSize(String controlId, int renderSize) {
VerifyStart();
DataTable dt = _requestData.Tables[SR.Trace_Control_Tree];
// find the row for this control
if (controlId == null)
controlId = PAGEKEYNAME;
DataRow row = dt.Rows.Find((object) controlId);
// if the row is null, we couldn't find it, so we'll just skip this control
if (row != null)
row[SR.Trace_Render_Size] = renderSize;
}
internal void AddControlStateSize(String controlId, int viewstateSize, int controlstateSize) {
VerifyStart();
DataTable dt = _requestData.Tables[SR.Trace_Control_Tree];
// find the row for this control
if (controlId == null)
controlId = PAGEKEYNAME;
DataRow row = dt.Rows.Find((object) controlId);
// if the row is null, we couldn't find it, so we'll just skip this control
if (row != null) {
row[SR.Trace_Viewstate_Size] = viewstateSize;
row[SR.Trace_Controlstate_Size] = controlstateSize;
}
}
internal void Render(HtmlTextWriter output) {
if (PageOutput && _requestData != null) {
TraceEnable oldEnabled = _isEnabled;
_isEnabled = TraceEnable.Disable;
Control table;
output.Write("<div id=\"__asptrace\">\r\n");
output.Write(TraceHandler.StyleSheet);
output.Write("<span class=\"tracecontent\">\r\n");
table = TraceHandler.CreateDetailsTable(_requestData.Tables[SR.Trace_Request]);
if (table != null)
table.RenderControl(output);
table = TraceHandler.CreateTraceTable(_requestData.Tables[SR.Trace_Trace_Information]);
if (table != null)
table.RenderControl(output);
table = TraceHandler.CreateControlTable(_requestData.Tables[SR.Trace_Control_Tree]);
if (table != null)
table.RenderControl(output);
table = TraceHandler.CreateTable(_requestData.Tables[SR.Trace_Session_State], true /*encodeSpaces*/);
if (table != null)
table.RenderControl(output);
table = TraceHandler.CreateTable(_requestData.Tables[SR.Trace_Application_State], true /*encodeSpaces*/);
if (table != null)
table.RenderControl(output);
table = TraceHandler.CreateTable(_requestData.Tables[SR.Trace_Request_Cookies_Collection], true /*encodeSpaces*/);
if (table != null)
table.RenderControl(output);
table = TraceHandler.CreateTable(_requestData.Tables[SR.Trace_Response_Cookies_Collection], true /*encodeSpaces*/);
if (table != null)
table.RenderControl(output);
table = TraceHandler.CreateTable(_requestData.Tables[SR.Trace_Headers_Collection], true /*encodeSpaces*/);
if (table != null)
table.RenderControl(output);
table = TraceHandler.CreateTable(_requestData.Tables[SR.Trace_Response_Headers_Collection], true /*encodeSpaces*/);
if (table != null)
table.RenderControl(output);
table = TraceHandler.CreateTable(_requestData.Tables[SR.Trace_Form_Collection]);
if (table != null)
table.RenderControl(output);
table = TraceHandler.CreateTable(_requestData.Tables[SR.Trace_Querystring_Collection]);
if (table != null)
table.RenderControl(output);
table = TraceHandler.CreateTable(_requestData.Tables[SR.Trace_Server_Variables], true /*encodeSpaces*/);
if (table != null)
table.RenderControl(output);
output.Write("<hr width=100% size=1 color=silver>\r\n\r\n");
output.Write(SR.GetString(SR.Error_Formatter_CLR_Build) + VersionInfo.ClrVersion +
SR.GetString(SR.Error_Formatter_ASPNET_Build) + VersionInfo.EngineVersion + "\r\n\r\n");
output.Write("</font>\r\n\r\n");
output.Write("</span>\r\n</div>\r\n");
_isEnabled = oldEnabled;
}
}
internal DataSet GetData() {
return _requestData;
}
internal void VerifyStart() {
// if we have already started, we can skip the lock
if (_masterRequest == null) {
// otherwise we need to make sure two
// requests don't try to call InitMaster at the same time
lock(this) {
if (_masterRequest == null)
InitMaster();
}
}
if (_requestData == null) {
InitRequest();
}
}
internal void StopTracing() {
_endDataCollected = true;
}
/*
* Finalize the request
*/
internal void EndRequest() {
VerifyStart();
if (_endDataCollected)
return;
// add some more information about the reponse
DataRow row = _requestData.Tables[SR.Trace_Request].Rows[0];
row[SR.Trace_Status_Code] = _context.Response.StatusCode;
row[SR.Trace_Response_Encoding] = _context.Response.ContentEncoding.EncodingName;
IEnumerator en;
string temp;
object obj;
int i;
// Application State info
_context.Application.Lock();
try {
en = _context.Application.GetEnumerator();
while (en.MoveNext()) {
row = NewRow(_requestData, SR.Trace_Application_State);
temp = (string) en.Current;
//the key might be null
row[SR.Trace_Application_Key] = (temp != null) ? temp : NULLSTRING;
obj = _context.Application[temp];
// the value could also be null
if (obj != null) {
row[SR.Trace_Type] = obj.GetType();
row[SR.Trace_Value] = obj.ToString();
}
else {
row[SR.Trace_Type] = NULLSTRING;
row[SR.Trace_Value] = NULLSTRING;
}
AddRow(_requestData, SR.Trace_Application_State, row);
}
}
finally {
_context.Application.UnLock();
}
// request cookie info
HttpCookieCollection cookieCollection = new HttpCookieCollection();
_context.Request.FillInCookiesCollection(cookieCollection, false /*includeResponse */);
HttpCookie[] cookies = new HttpCookie[cookieCollection.Count];
cookieCollection.CopyTo(cookies, 0);
for (i = 0; i<cookies.Length; i++) {
row = NewRow(_requestData, SR.Trace_Request_Cookies_Collection);
row[SR.Trace_Name] = cookies[i].Name;
if (cookies[i].Values.HasKeys()) {
NameValueCollection subvalues = cookies[i].Values;
StringBuilder sb = new StringBuilder();
en = subvalues.GetEnumerator();
while (en.MoveNext()) {
temp = (string) en.Current;
sb.Append("(");
sb.Append(temp + "=");
sb.Append(cookies[i][temp] + ") ");
}
row[SR.Trace_Value] = sb.ToString();
}
else
row[SR.Trace_Value] = cookies[i].Value;
int size = (cookies[i].Name == null) ? 0 : cookies[i].Name.Length;
size += (cookies[i].Value == null) ? 0 : cookies[i].Value.Length;
row[SR.Trace_Size] = size + 1; // plus 1 for =
AddRow(_requestData, SR.Trace_Request_Cookies_Collection, row);
}
// response cookie info
cookies = new HttpCookie[_context.Response.Cookies.Count];
_context.Response.Cookies.CopyTo(cookies, 0);
for (i = 0; i<cookies.Length; i++) {
row = NewRow(_requestData, SR.Trace_Response_Cookies_Collection);
row[SR.Trace_Name] = cookies[i].Name;
if (cookies[i].Values.HasKeys()) {
NameValueCollection subvalues = cookies[i].Values;
StringBuilder sb = new StringBuilder();
en = subvalues.GetEnumerator();
while (en.MoveNext()) {
temp = (string) en.Current;
sb.Append("(");
sb.Append(temp + "=");
sb.Append(cookies[i][temp] + ") ");
}
row[SR.Trace_Value] = sb.ToString();
}
else
row[SR.Trace_Value] = cookies[i].Value;
int size = (cookies[i].Name == null) ? 0 : cookies[i].Name.Length;
size += (cookies[i].Value == null) ? 0 : cookies[i].Value.Length;
row[SR.Trace_Size] = size + 1; // plus 1 for =
AddRow(_requestData, SR.Trace_Response_Cookies_Collection, row);
}
HttpSessionState session = _context.Session;
// session state info
if (session != null) {
row = _requestData.Tables[SR.Trace_Request].Rows[0];
try {
row[SR.Trace_Session_Id] = HttpUtility.UrlEncode(session.SessionID);
}
catch {
// VSWhidbey 527536
// Accessing SessionID can cause creation of the session id, this will throw in the
// cross page post scenario, as the response has already been flushed when we try
// to add the session cookie, since this is just trace output, we can just not set a session ID.
//
}
en = session.GetEnumerator();
while (en.MoveNext()) {
row = NewRow(_requestData, SR.Trace_Session_State);
temp = (string) en.Current;
// the key could be null
row[SR.Trace_Session_Key] = (temp != null) ? temp : NULLSTRING;
obj = session[temp];
// the value could also be null
if (obj != null) {
row[SR.Trace_Type] = obj.GetType();
row[SR.Trace_Value] = obj.ToString();
}
else {
row[SR.Trace_Type] = NULLSTRING;
row[SR.Trace_Value] = NULLSTRING;
}
AddRow(_requestData, SR.Trace_Session_State, row);
}
}
ApplyTraceMode();
OnTraceFinished(new TraceContextEventArgs(_traceRecords));
}
/* InitMaster
* Initialize the _masterRequest dataset with the schema we use
* to store profiling information
*/
private void InitMaster() {
DataSet tempset = new DataSet();
tempset.Locale = CultureInfo.InvariantCulture;
// populate the _masterRequest's schema
DataTable tab;
Type strtype = typeof(string);
Type inttype = typeof(int);
Type doubletype = typeof(double);
// request table
tab = tempset.Tables.Add(SR.Trace_Request);
tab.Columns.Add(SR.Trace_No, inttype);
tab.Columns.Add(SR.Trace_Time_of_Request, strtype);
tab.Columns.Add(SR.Trace_Url, strtype);
tab.Columns.Add(SR.Trace_Request_Type, strtype);
tab.Columns.Add(SR.Trace_Status_Code, inttype);
tab.Columns.Add(SR.Trace_Session_Id, strtype);
tab.Columns.Add(SR.Trace_Request_Encoding, strtype);
tab.Columns.Add(SR.Trace_Response_Encoding, strtype);
// control heirarchy table
tab = tempset.Tables.Add(SR.Trace_Control_Tree);
tab.Columns.Add(SR.Trace_Parent_Id, strtype);
DataColumn[] col = new DataColumn[1];
col[0] = new DataColumn(SR.Trace_Control_Id, strtype);
tab.Columns.Add(col[0]);
tab.PrimaryKey = col; // set the control id to be the primary key
tab.Columns.Add(SR.Trace_Type, strtype);
tab.Columns.Add(SR.Trace_Render_Size, inttype);
tab.Columns.Add(SR.Trace_Viewstate_Size, inttype);
tab.Columns.Add(SR.Trace_Controlstate_Size, inttype);
// session state table
tab = tempset.Tables.Add(SR.Trace_Session_State);
tab.Columns.Add(SR.Trace_Session_Key, strtype);
tab.Columns.Add(SR.Trace_Type, strtype);
tab.Columns.Add(SR.Trace_Value, strtype);
// application state table
tab = tempset.Tables.Add(SR.Trace_Application_State);
tab.Columns.Add(SR.Trace_Application_Key, strtype);
tab.Columns.Add(SR.Trace_Type, strtype);
tab.Columns.Add(SR.Trace_Value, strtype);
// request cookies table
tab = tempset.Tables.Add(SR.Trace_Request_Cookies_Collection);
tab.Columns.Add(SR.Trace_Name, strtype);
tab.Columns.Add(SR.Trace_Value, strtype);
tab.Columns.Add(SR.Trace_Size, inttype);
// resposne cookies table
tab = tempset.Tables.Add(SR.Trace_Response_Cookies_Collection);
tab.Columns.Add(SR.Trace_Name, strtype);
tab.Columns.Add(SR.Trace_Value, strtype);
tab.Columns.Add(SR.Trace_Size, inttype);
// header table
tab = tempset.Tables.Add(SR.Trace_Headers_Collection);
tab.Columns.Add(SR.Trace_Name, strtype);
tab.Columns.Add(SR.Trace_Value, strtype);
// response header table
tab = tempset.Tables.Add(SR.Trace_Response_Headers_Collection);
tab.Columns.Add(SR.Trace_Name, strtype);
tab.Columns.Add(SR.Trace_Value, strtype);
// form variables table
tab = tempset.Tables.Add(SR.Trace_Form_Collection);
tab.Columns.Add(SR.Trace_Name, strtype);
tab.Columns.Add(SR.Trace_Value, strtype);
// querystring table
tab = tempset.Tables.Add(SR.Trace_Querystring_Collection);
tab.Columns.Add(SR.Trace_Name, strtype);
tab.Columns.Add(SR.Trace_Value, strtype);
// trace info
tab = tempset.Tables.Add(SR.Trace_Trace_Information);
tab.Columns.Add(SR.Trace_Category, strtype);
tab.Columns.Add(SR.Trace_Warning, strtype);
tab.Columns.Add(SR.Trace_Message, strtype);
tab.Columns.Add(SR.Trace_From_First, doubletype);
tab.Columns.Add(SR.Trace_From_Last, doubletype);
tab.Columns.Add("ErrorInfoMessage", strtype);
tab.Columns.Add("ErrorInfoStack", strtype);
// server variables
tab = tempset.Tables.Add(SR.Trace_Server_Variables);
tab.Columns.Add(SR.Trace_Name, strtype);
tab.Columns.Add(SR.Trace_Value, strtype);
_masterRequest = tempset;
}
private DataRow NewRow(DataSet ds, string table) {
return ds.Tables[table].NewRow();
}
private void AddRow(DataSet ds, string table, DataRow row) {
ds.Tables[table].Rows.Add(row);
}
/* InitRequest
* Initialize the given dataset with basic
* request information
*/
private void InitRequest() {
// Master request is assumed to be initialized first
System.Web.Util.Debug.Assert(_masterRequest != null);
DataSet requestData = _masterRequest.Clone();
// request info
DataRow row = NewRow(requestData, SR.Trace_Request);
row[SR.Trace_Time_of_Request] = _context.Timestamp.ToString("G");
string url = _context.Request.RawUrl;
int loc = url.IndexOf("?", StringComparison.Ordinal);
if (loc != -1)
url = url.Substring(0, loc);
row[SR.Trace_Url] = url;
row[SR.Trace_Request_Type] = _context.Request.HttpMethod;
try {
row[SR.Trace_Request_Encoding] = _context.Request.ContentEncoding.EncodingName;
}
catch {
// if we get an exception getting the ContentEncoding, most likely
// there's an error in the config file. Just ignore it so we can finish InitRequest.
}
if (TraceMode == TraceMode.SortByCategory)
requestData.Tables[SR.Trace_Trace_Information].DefaultView.Sort = SR.Trace_Category;
AddRow(requestData, SR.Trace_Request, row);
// header info
try {
// Bug 867196: Use Request.Unvalidated to ensure request validation will not
// be triggered when the entries of the collection are accessed.
AddCollectionToRequestData(requestData, SR.Trace_Headers_Collection, _context.Request.Unvalidated.Headers);
}
catch {
// ---- exceptions when we fail to get the unvalidated collection
}
// response header info
ArrayList headers = _context.Response.GenerateResponseHeaders(false);
int n = (headers != null) ? headers.Count : 0;
for (int i = 0; i < n; i++) {
HttpResponseHeader h = (HttpResponseHeader)headers[i];
row = NewRow(requestData, SR.Trace_Response_Headers_Collection);
row[SR.Trace_Name] = h.Name;
row[SR.Trace_Value] = h.Value;
AddRow(requestData, SR.Trace_Response_Headers_Collection, row);
}
//form info
try {
AddCollectionToRequestData(requestData, SR.Trace_Form_Collection, _context.Request.Unvalidated.Form);
}
catch {
// ---- exceptions when we fail to get the unvalidated collection
}
//QueryString info
try {
AddCollectionToRequestData(requestData, SR.Trace_Querystring_Collection, _context.Request.Unvalidated.QueryString);
}
catch {
// ---- exceptions when we fail to get the unvalidated collection
}
//Server Variable info
if (HttpRuntime.HasAppPathDiscoveryPermission()) {
AddCollectionToRequestData(requestData, SR.Trace_Server_Variables, _context.Request.ServerVariables);
}
_requestData = requestData;
if (HttpRuntime.UseIntegratedPipeline) {
// Dev10 914119: When trace is enabled, the request entity is read and no longer
// available to IIS. In integrated mode, we have an API that allows us to reinsert
// the entity. Although it is expensive, performance is not a concern when
// trace is enalbed, so we will reinsert just in case a native handler needs
// to access the entity. I decided not to check the current handler, since
// that can be changed if someone sets the IIS script map.
_context.Request.InsertEntityBody();
}
}
private void AddCollectionToRequestData(DataSet requestData, string traceCollectionTitle, NameValueCollection collection) {
if (null != collection) {
var keys = collection.AllKeys;
for (int i = 0; i < keys.Length; i++) {
var row = NewRow(requestData, traceCollectionTitle);
row[SR.Trace_Name] = keys[i];
row[SR.Trace_Value] = collection[keys[i]];
AddRow(requestData, traceCollectionTitle, row);
}
}
}
}
}