Backport SafeCallContext, DefaultDatabaseFactory fixes from 7.6
This commit is contained in:
94
src/Umbraco.Core/SafeCallContext.cs
Normal file
94
src/Umbraco.Core/SafeCallContext.cs
Normal file
@@ -0,0 +1,94 @@
|
||||
using System;
|
||||
using System.Linq;
|
||||
using System.Collections.Generic;
|
||||
|
||||
namespace Umbraco.Core
|
||||
{
|
||||
internal class SafeCallContext : IDisposable
|
||||
{
|
||||
private static readonly List<Func<object>> EnterFuncs = new List<Func<object>>();
|
||||
private static readonly List<Action<object>> ExitActions = new List<Action<object>>();
|
||||
private static int _count;
|
||||
private readonly List<object> _objects;
|
||||
private bool _disposed;
|
||||
|
||||
public static void Register(Func<object> enterFunc, Action<object> exitAction)
|
||||
{
|
||||
if (enterFunc == null) throw new ArgumentNullException("enterFunc");
|
||||
if (exitAction == null) throw new ArgumentNullException("exitAction");
|
||||
|
||||
lock (EnterFuncs)
|
||||
{
|
||||
if (_count > 0) throw new InvalidOperationException("Cannot register while some SafeCallContext instances exist.");
|
||||
EnterFuncs.Add(enterFunc);
|
||||
ExitActions.Add(exitAction);
|
||||
}
|
||||
}
|
||||
|
||||
// tried to make the UmbracoDatabase serializable but then it leaks to weird places
|
||||
// in ReSharper and so on, where Umbraco.Core is not available. Tried to serialize
|
||||
// as an object instead but then it comes *back* deserialized into the original context
|
||||
// as an object and of course it breaks everything. Cannot prevent this from flowing,
|
||||
// and ExecutionContext.SuppressFlow() works for threads but not domains. and we'll
|
||||
// have the same issue with anything that toys with logical call context...
|
||||
//
|
||||
// so this class lets anything that uses the logical call context register itself,
|
||||
// providing two methods:
|
||||
// - an enter func that removes and returns whatever is in the logical call context
|
||||
// - an exit action that restores the value into the logical call context
|
||||
// whenever a SafeCallContext instance is created, it uses these methods to capture
|
||||
// and clear the logical call context, and restore it when disposed.
|
||||
//
|
||||
// in addition, a static Clear method is provided - which uses the enter funcs to
|
||||
// remove everything from logical call context - not to be used when the app runs,
|
||||
// but can be useful during tests
|
||||
//
|
||||
// note
|
||||
// see System.Transactions
|
||||
// they are using a conditional weak table to store the data, and what they store in
|
||||
// LLC is the key - which is just an empty MarshalByRefObject that is created with
|
||||
// the transaction scope - that way, they can "clear current data" provided that
|
||||
// they have the key - but they need to hold onto a ref to the scope... not ok for us
|
||||
|
||||
public static void Clear()
|
||||
{
|
||||
lock (EnterFuncs)
|
||||
{
|
||||
foreach (var enter in EnterFuncs)
|
||||
enter();
|
||||
}
|
||||
}
|
||||
|
||||
public SafeCallContext()
|
||||
{
|
||||
lock (EnterFuncs)
|
||||
{
|
||||
_count++;
|
||||
_objects = EnterFuncs.Select(x => x()).ToList();
|
||||
}
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
if (_disposed) throw new ObjectDisposedException("this");
|
||||
_disposed = true;
|
||||
lock (EnterFuncs)
|
||||
{
|
||||
for (var i = 0; i < ExitActions.Count; i++)
|
||||
ExitActions[i](_objects[i]);
|
||||
_count--;
|
||||
}
|
||||
}
|
||||
|
||||
// for unit tests ONLY
|
||||
internal static void Reset()
|
||||
{
|
||||
lock (EnterFuncs)
|
||||
{
|
||||
if (_count > 0) throw new InvalidOperationException("Cannot reset while some SafeCallContext instances exist.");
|
||||
EnterFuncs.Clear();
|
||||
ExitActions.Clear();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user