360 lines
16 KiB
C#
360 lines
16 KiB
C#
|
namespace System.Web {
|
|||
|
using System;
|
|||
|
using System.ComponentModel;
|
|||
|
using System.Globalization;
|
|||
|
using System.Runtime.Remoting.Messaging;
|
|||
|
using System.Security.Principal;
|
|||
|
using System.Text;
|
|||
|
using System.Threading;
|
|||
|
using System.Web.Configuration;
|
|||
|
using System.Web.UI;
|
|||
|
using System.Web.Util;
|
|||
|
|
|||
|
// Contains information about any modifications ASP.NET has made to the current
|
|||
|
// thread and how to undo them. See also the comments on
|
|||
|
// HttpApplication.OnThreadEnterPrivate.
|
|||
|
|
|||
|
internal sealed class ThreadContext : ISyncContextLock {
|
|||
|
|
|||
|
// This is a marker holding the current ThreadContext for the current
|
|||
|
// thread. Uses TLS so that it's not wiped away by ExecutionContext.Run.
|
|||
|
[ThreadStatic]
|
|||
|
private static ThreadContext _currentThreadContext;
|
|||
|
|
|||
|
private ImpersonationContext _newImpersonationContext;
|
|||
|
private HttpContext _originalHttpContext;
|
|||
|
private SynchronizationContext _originalSynchronizationContext;
|
|||
|
private ThreadContext _originalThreadContextCurrent;
|
|||
|
private CultureInfo _originalThreadCurrentCulture;
|
|||
|
private CultureInfo _originalThreadCurrentUICulture;
|
|||
|
private IPrincipal _originalThreadPrincipal;
|
|||
|
private bool _setCurrentThreadOnHttpContext;
|
|||
|
|
|||
|
internal ThreadContext(HttpContext httpContext) {
|
|||
|
HttpContext = httpContext;
|
|||
|
}
|
|||
|
|
|||
|
internal static ThreadContext Current {
|
|||
|
get { return _currentThreadContext; }
|
|||
|
private set { _currentThreadContext = value; }
|
|||
|
}
|
|||
|
|
|||
|
internal bool HasBeenDisassociatedFromThread {
|
|||
|
get;
|
|||
|
private set;
|
|||
|
}
|
|||
|
|
|||
|
internal HttpContext HttpContext {
|
|||
|
get;
|
|||
|
private set;
|
|||
|
}
|
|||
|
|
|||
|
// Associates this ThreadContext with the current thread. This restores certain
|
|||
|
// ambient values associated with the current HttpContext, such as the current
|
|||
|
// user and cultures. It also sets HttpContext.Current.
|
|||
|
internal void AssociateWithCurrentThread(bool setImpersonationContext) {
|
|||
|
Debug.Assert(HttpContext != null); // only to be used when context is available
|
|||
|
Debug.Assert(Current != this, "This ThreadContext is already associated with this thread.");
|
|||
|
Debug.Assert(!HasBeenDisassociatedFromThread, "This ThreadContext has already been disassociated from a thread.");
|
|||
|
|
|||
|
Debug.Trace("OnThread", GetTraceMessage("Enter1"));
|
|||
|
|
|||
|
/*
|
|||
|
* !! IMPORTANT !!
|
|||
|
* Keep this logic in [....] with DisassociateFromCurrentThread and EnterExecutionContext.
|
|||
|
*/
|
|||
|
|
|||
|
// attach http context to the call context
|
|||
|
_originalHttpContext = DisposableHttpContextWrapper.SwitchContext(HttpContext);
|
|||
|
|
|||
|
// set impersonation on the current thread
|
|||
|
if (setImpersonationContext) {
|
|||
|
SetImpersonationContext();
|
|||
|
}
|
|||
|
|
|||
|
// set synchronization context for the current thread to support the async pattern
|
|||
|
_originalSynchronizationContext = AsyncOperationManager.SynchronizationContext;
|
|||
|
AspNetSynchronizationContextBase aspNetSynchronizationContext = HttpContext.SyncContext;
|
|||
|
AsyncOperationManager.SynchronizationContext = aspNetSynchronizationContext;
|
|||
|
|
|||
|
// set ETW trace ID
|
|||
|
Guid g = HttpContext.WorkerRequest.RequestTraceIdentifier;
|
|||
|
if (g != Guid.Empty) {
|
|||
|
CallContext.LogicalSetData("E2ETrace.ActivityID", g);
|
|||
|
}
|
|||
|
|
|||
|
// set SqlDependecyCookie
|
|||
|
HttpContext.ResetSqlDependencyCookie();
|
|||
|
|
|||
|
// set principal on the current thread
|
|||
|
_originalThreadPrincipal = Thread.CurrentPrincipal;
|
|||
|
HttpApplication.SetCurrentPrincipalWithAssert(HttpContext.User);
|
|||
|
|
|||
|
// only set culture on the current thread if it is not initialized
|
|||
|
SetRequestLevelCulture();
|
|||
|
|
|||
|
// DevDivBugs 75042
|
|||
|
// set current thread in context if there is not there
|
|||
|
// the timeout manager uses this to abort the correct thread
|
|||
|
if (HttpContext.CurrentThread == null) {
|
|||
|
_setCurrentThreadOnHttpContext = true;
|
|||
|
HttpContext.CurrentThread = Thread.CurrentThread;
|
|||
|
}
|
|||
|
|
|||
|
// Store a reference to the original ThreadContext.Current. It is possible that a parent
|
|||
|
// ThreadContext might already be associated with the current thread, e.g. if the current
|
|||
|
// stack contains a call to MgdIndicateCompletion (via
|
|||
|
// PipelineRuntime.ProcessRequestNotificationHelper). If this is the case, the child
|
|||
|
// ThreadContext will temporarily take over.
|
|||
|
_originalThreadContextCurrent = Current;
|
|||
|
Current = this;
|
|||
|
|
|||
|
Debug.Trace("OnThread", GetTraceMessage("Enter2"));
|
|||
|
}
|
|||
|
|
|||
|
private ClientImpersonationContext CreateNewClientImpersonationContext() {
|
|||
|
// impersonation is set in the ClientImpersonationContext ctor
|
|||
|
return new ClientImpersonationContext(HttpContext);
|
|||
|
}
|
|||
|
|
|||
|
// Disassociates this ThreadContext from the current thread. Any ambient values (e.g., culture)
|
|||
|
// associated with the current request are stored in the HttpContext object so that they
|
|||
|
// can be restored the next time a ThreadContext associated with this HttpContext is active.
|
|||
|
// Impersonation and other similar modifications to the current thread are undone.
|
|||
|
internal void DisassociateFromCurrentThread() {
|
|||
|
Debug.Trace("OnThread", GetTraceMessage("Leave1"));
|
|||
|
Debug.Assert(Current == this, "This ThreadContext isn't associated with current thread.");
|
|||
|
Debug.Assert(!HasBeenDisassociatedFromThread, "This ThreadContext has already been disassociated from a thread.");
|
|||
|
|
|||
|
/*
|
|||
|
* !! IMPORTANT !!
|
|||
|
* Keep this logic in [....] with AssociateWithCurrentThread and EnterExecutionContext.
|
|||
|
*/
|
|||
|
|
|||
|
Current = _originalThreadContextCurrent;
|
|||
|
HasBeenDisassociatedFromThread = true;
|
|||
|
|
|||
|
// remove thread if set
|
|||
|
if (_setCurrentThreadOnHttpContext) {
|
|||
|
HttpContext.CurrentThread = null;
|
|||
|
}
|
|||
|
|
|||
|
// this thread should not be locking app state
|
|||
|
HttpApplicationFactory.ApplicationState.EnsureUnLock();
|
|||
|
|
|||
|
// stop impersonation
|
|||
|
UndoImpersonationContext();
|
|||
|
|
|||
|
// restore culture
|
|||
|
RestoreRequestLevelCulture();
|
|||
|
|
|||
|
// restrore synchronization context
|
|||
|
AsyncOperationManager.SynchronizationContext = _originalSynchronizationContext;
|
|||
|
|
|||
|
// restore thread principal
|
|||
|
HttpApplication.SetCurrentPrincipalWithAssert(_originalThreadPrincipal);
|
|||
|
|
|||
|
// Remove SqlCacheDependency cookie from call context if necessary
|
|||
|
HttpContext.RemoveSqlDependencyCookie();
|
|||
|
|
|||
|
// remove http context from the call context
|
|||
|
DisposableHttpContextWrapper.SwitchContext(_originalHttpContext);
|
|||
|
_originalHttpContext = null;
|
|||
|
|
|||
|
Debug.Trace("OnThread", GetTraceMessage("Leave2"));
|
|||
|
}
|
|||
|
|
|||
|
// Called by AspNetHostExecutionContextManager to signal that ExecutionContext.Run
|
|||
|
// is being called on a thread currently associated with our ThreadContext. Since
|
|||
|
// ExecutionContext.Run destroys some of our ambient state (HttpContext.Current, etc.),
|
|||
|
// we need to restore it. This method returns an Action which should be called when
|
|||
|
// the call to ExecutionContext.Run is concluding.
|
|||
|
internal Action EnterExecutionContext() {
|
|||
|
Debug.Trace("OnThread", GetTraceMessage("EnterExecutionContext1"));
|
|||
|
Debug.Assert(Current == this, "This ThreadContext isn't associated with current thread.");
|
|||
|
Debug.Assert(!HasBeenDisassociatedFromThread, "This ThreadContext has already been disassociated from a thread.");
|
|||
|
|
|||
|
/*
|
|||
|
* !! IMPORTANT !!
|
|||
|
* Keep this logic in [....] with AssociateWithCurrentThread and DisassociateFromCurrentThread.
|
|||
|
*/
|
|||
|
|
|||
|
// ExecutionContext.Run replaces the current impersonation token, so we need to impersonate
|
|||
|
// if AssociateWithCurrentThread also did so.
|
|||
|
|
|||
|
ClientImpersonationContext executionContextClientImpersonationContext = null;
|
|||
|
if (_newImpersonationContext != null) {
|
|||
|
executionContextClientImpersonationContext = CreateNewClientImpersonationContext();
|
|||
|
}
|
|||
|
|
|||
|
// ExecutionContext.Run resets the LogicalCallContext / IllogicalCallContext (which contains HttpContext.Current),
|
|||
|
// so we need to restore both of them.
|
|||
|
|
|||
|
DisposableHttpContextWrapper.SwitchContext(HttpContext);
|
|||
|
|
|||
|
Guid g = HttpContext.WorkerRequest.RequestTraceIdentifier;
|
|||
|
if (g != Guid.Empty) {
|
|||
|
CallContext.LogicalSetData("E2ETrace.ActivityID", g);
|
|||
|
}
|
|||
|
|
|||
|
HttpContext.ResetSqlDependencyCookie();
|
|||
|
|
|||
|
// ExecutionContext.Run resets the thread's CurrentPrincipal, so we need to restore it.
|
|||
|
|
|||
|
HttpApplication.SetCurrentPrincipalWithAssert(HttpContext.User);
|
|||
|
|
|||
|
// Other items like [ThreadStatic] fields, culture, etc. are untouched by ExecutionContext.Run,
|
|||
|
// so we don't need to worry about them.
|
|||
|
|
|||
|
Debug.Trace("OnThread", GetTraceMessage("EnterExecutionContext2"));
|
|||
|
|
|||
|
// This delegate is the cleanup routine.
|
|||
|
return () => {
|
|||
|
Debug.Trace("OnThread", GetTraceMessage("LeaveExecutionContext1"));
|
|||
|
|
|||
|
// Undo any impersonation that we performed.
|
|||
|
if (executionContextClientImpersonationContext != null) {
|
|||
|
executionContextClientImpersonationContext.Undo();
|
|||
|
}
|
|||
|
|
|||
|
// Other things, e.g. changes to the logical/illogical call contexts, changes
|
|||
|
// to CurrentPrincipal, etc., will automatically be reverted anyway when
|
|||
|
// the call to ExecutionContext.Run concludes, so we don't need to clean up
|
|||
|
// here.
|
|||
|
|
|||
|
Debug.Trace("OnThread", GetTraceMessage("LeaveExecutionContext2"));
|
|||
|
};
|
|||
|
}
|
|||
|
|
|||
|
private static string GetTraceMessage(string tag) {
|
|||
|
#if DBG
|
|||
|
StringBuilder sb = new StringBuilder(256);
|
|||
|
sb.Append(tag);
|
|||
|
sb.AppendFormat(" Thread={0}", SafeNativeMethods.GetCurrentThreadId().ToString(CultureInfo.InvariantCulture));
|
|||
|
sb.AppendFormat(" Context={0}", (HttpContext.Current != null) ? HttpContext.Current.GetHashCode().ToString(CultureInfo.InvariantCulture) : "NULL_CTX");
|
|||
|
sb.AppendFormat(" Principal={0}", (Thread.CurrentPrincipal != null) ? Thread.CurrentPrincipal.GetHashCode().ToString(CultureInfo.InvariantCulture) : "NULL_PRIN");
|
|||
|
sb.AppendFormat(" Culture={0}", Thread.CurrentThread.CurrentCulture.LCID.ToString(CultureInfo.InvariantCulture));
|
|||
|
sb.AppendFormat(" UICulture={0}", Thread.CurrentThread.CurrentUICulture.LCID.ToString(CultureInfo.InvariantCulture));
|
|||
|
sb.AppendFormat(" ActivityID={0}", CallContext.LogicalGetData("E2ETrace.ActivityID"));
|
|||
|
return sb.ToString();
|
|||
|
#else
|
|||
|
// This method should never be called in release mode.
|
|||
|
throw new NotImplementedException();
|
|||
|
#endif
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
// Restores the thread's CurrentCulture and CurrentUICulture back to what
|
|||
|
// they were before this ThreadContext was associated with the thread. If
|
|||
|
// any culture has changed from its original value, we squirrel the new
|
|||
|
// culture away in HttpContext so that we can restore it the next time any
|
|||
|
// ThreadContext associated with this HttpContext is active.
|
|||
|
private void RestoreRequestLevelCulture() {
|
|||
|
CultureInfo currentCulture = Thread.CurrentThread.CurrentCulture;
|
|||
|
CultureInfo currentUICulture = Thread.CurrentThread.CurrentUICulture;
|
|||
|
|
|||
|
if (_originalThreadCurrentCulture != null) {
|
|||
|
// Avoid the cost of the Demand when setting the culture by comparing the cultures first
|
|||
|
if (currentCulture != _originalThreadCurrentCulture) {
|
|||
|
HttpRuntime.SetCurrentThreadCultureWithAssert(_originalThreadCurrentCulture);
|
|||
|
if (HttpContext != null) {
|
|||
|
// remember changed culture for the rest of the request
|
|||
|
HttpContext.DynamicCulture = currentCulture;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
_originalThreadCurrentCulture = null;
|
|||
|
}
|
|||
|
|
|||
|
if (_originalThreadCurrentUICulture != null) {
|
|||
|
// Avoid the cost of the Demand when setting the culture by comparing the cultures first
|
|||
|
if (currentUICulture != _originalThreadCurrentUICulture) {
|
|||
|
Thread.CurrentThread.CurrentUICulture = _originalThreadCurrentUICulture;
|
|||
|
if (HttpContext != null) {
|
|||
|
// remember changed culture for the rest of the request
|
|||
|
HttpContext.DynamicUICulture = currentUICulture;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
_originalThreadCurrentUICulture = null;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// Sets impersonation on the current thread.
|
|||
|
internal void SetImpersonationContext() {
|
|||
|
if (_newImpersonationContext == null) {
|
|||
|
_newImpersonationContext = CreateNewClientImpersonationContext();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// Sets the thread's CurrentCulture and CurrentUICulture to those associated
|
|||
|
// with the current HttpContext. We do this since the culture of a request can
|
|||
|
// change over its lifetime and isn't necessarily the default for the AppDomain,
|
|||
|
// e.g. if the culture was read from the request headers.
|
|||
|
private void SetRequestLevelCulture() {
|
|||
|
CultureInfo culture = null;
|
|||
|
CultureInfo uiculture = null;
|
|||
|
|
|||
|
GlobalizationSection globConfig = RuntimeConfig.GetConfig(HttpContext).Globalization;
|
|||
|
if (!String.IsNullOrEmpty(globConfig.Culture))
|
|||
|
culture = HttpContext.CultureFromConfig(globConfig.Culture, true);
|
|||
|
|
|||
|
if (!String.IsNullOrEmpty(globConfig.UICulture))
|
|||
|
uiculture = HttpContext.CultureFromConfig(globConfig.UICulture, false);
|
|||
|
|
|||
|
if (HttpContext.DynamicCulture != null)
|
|||
|
culture = HttpContext.DynamicCulture;
|
|||
|
|
|||
|
if (HttpContext.DynamicUICulture != null)
|
|||
|
uiculture = HttpContext.DynamicUICulture;
|
|||
|
|
|||
|
// Page also could have its own culture settings
|
|||
|
Page page = HttpContext.CurrentHandler as Page;
|
|||
|
|
|||
|
if (page != null) {
|
|||
|
if (page.DynamicCulture != null)
|
|||
|
culture = page.DynamicCulture;
|
|||
|
|
|||
|
if (page.DynamicUICulture != null)
|
|||
|
uiculture = page.DynamicUICulture;
|
|||
|
}
|
|||
|
|
|||
|
_originalThreadCurrentCulture = Thread.CurrentThread.CurrentCulture;
|
|||
|
_originalThreadCurrentUICulture = Thread.CurrentThread.CurrentUICulture;
|
|||
|
|
|||
|
if (culture != null && culture != Thread.CurrentThread.CurrentCulture) {
|
|||
|
HttpRuntime.SetCurrentThreadCultureWithAssert(culture);
|
|||
|
}
|
|||
|
|
|||
|
if (uiculture != null && uiculture != Thread.CurrentThread.CurrentUICulture) {
|
|||
|
Thread.CurrentThread.CurrentUICulture = uiculture;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// Use of IndicateCompletion requires that we synchronize the cultures
|
|||
|
// with what may have been set by user code during execution of the
|
|||
|
// notification.
|
|||
|
internal void Synchronize() {
|
|||
|
HttpContext.DynamicCulture = Thread.CurrentThread.CurrentCulture;
|
|||
|
HttpContext.DynamicUICulture = Thread.CurrentThread.CurrentUICulture;
|
|||
|
}
|
|||
|
|
|||
|
// Undoes any impersonation that we did when associating this ThreadContext
|
|||
|
// with the current thread.
|
|||
|
internal void UndoImpersonationContext() {
|
|||
|
// remove impersonation on the current thread
|
|||
|
if (_newImpersonationContext != null) {
|
|||
|
_newImpersonationContext.Undo();
|
|||
|
_newImpersonationContext = null;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// Called by AspNetSynchronizationContext to signal that it is finished
|
|||
|
// processing on the current thread.
|
|||
|
void ISyncContextLock.Leave() {
|
|||
|
DisassociateFromCurrentThread();
|
|||
|
}
|
|||
|
|
|||
|
}
|
|||
|
}
|