Browse Source

Added transactional changes support to GeneratedStore

pull/46/head
Anairkoen Schno 4 years ago
parent
commit
0d60158c7b
4 changed files with 181 additions and 98 deletions
  1. +2
    -2
      IPA.Loader/Config/SelfConfig.cs
  2. +169
    -92
      IPA.Loader/Config/Stores/GeneratedStore.cs
  3. +7
    -3
      IPA.Loader/Loader/DisabledConfig.cs
  4. +3
    -1
      IPA.Loader/Loader/PluginManager.cs

+ 2
- 2
IPA.Loader/Config/SelfConfig.cs View File

@ -29,7 +29,7 @@ namespace IPA.Config
protected internal virtual void OnReload() protected internal virtual void OnReload()
{ {
if (Regenerate) if (Regenerate)
CopyFrom(new SelfConfig());
CopyFrom(new SelfConfig { Regenerate = false });
StandardLogger.Configure(); StandardLogger.Configure();
} }
@ -75,7 +75,7 @@ namespace IPA.Config
// END: section ignore // END: section ignore
public virtual bool Regenerate { get; set; } = false;
public virtual bool Regenerate { get; set; } = true;
public class Updates_ public class Updates_
{ {


+ 169
- 92
IPA.Loader/Config/Stores/GeneratedStore.cs View File

@ -77,7 +77,12 @@ namespace IPA.Config.Stores
/// another object of its type easily, and more importantly, as only one change. Its body will be executed after the values have been copied. /// another object of its type easily, and more importantly, as only one change. Its body will be executed after the values have been copied.
/// </para> /// </para>
/// <para> /// <para>
/// TODO: describe details of generated stores
/// Similarly, <typeparamref name="T"/> can declare a public or protected, <see langword="virtual"/>
/// method <c>ChangeTransaction()</c> returning <see cref="IDisposable"/>, which may be called to get an object representing a transactional
/// change. This may be used to change a lot of properties at once without triggering a save multiple times. Ideally, this is used in a
/// <see langword="using"/> block or declaration. The <see cref="IDisposable"/> returned from your implementation will have its
/// <see cref="IDisposable.Dispose"/> called <i>after</i> <c>Changed()</c> is called, but <i>before</i> the write lock is released.
/// Unless you have a very good reason to use the nested <see cref="IDisposable"/>, avoid it.
/// </para> /// </para>
/// </remarks> /// </remarks>
/// <typeparam name="T">the type to wrap</typeparam> /// <typeparam name="T">the type to wrap</typeparam>
@ -107,6 +112,7 @@ namespace IPA.Config.Stores
void OnReload(); void OnReload();
void Changed(); void Changed();
IDisposable ChangeTransaction();
Value Serialize(); Value Serialize();
void Deserialize(Value val); void Deserialize(Value val);
@ -119,6 +125,7 @@ namespace IPA.Config.Stores
internal class Impl : IConfigStore internal class Impl : IConfigStore
{ {
private readonly IGeneratedStore generated; private readonly IGeneratedStore generated;
private bool inChangeTransaction = false;
internal static ConstructorInfo Ctor = typeof(Impl).GetConstructor(new[] { typeof(IGeneratedStore) }); internal static ConstructorInfo Ctor = typeof(Impl).GetConstructor(new[] { typeof(IGeneratedStore) });
public Impl(IGeneratedStore store) => generated = store; public Impl(IGeneratedStore store) => generated = store;
@ -142,11 +149,19 @@ namespace IPA.Config.Stores
internal static MethodInfo ImplTakeReadMethod = typeof(Impl).GetMethod(nameof(ImplTakeRead)); internal static MethodInfo ImplTakeReadMethod = typeof(Impl).GetMethod(nameof(ImplTakeRead));
public static void ImplTakeRead(IGeneratedStore s) => FindImpl(s).TakeRead(); public static void ImplTakeRead(IGeneratedStore s) => FindImpl(s).TakeRead();
public void TakeRead() => WriteSyncObject.EnterReadLock();
public void TakeRead()
{
if (!WriteSyncObject.IsWriteLockHeld)
WriteSyncObject.EnterReadLock();
}
internal static MethodInfo ImplReleaseReadMethod = typeof(Impl).GetMethod(nameof(ImplReleaseRead)); internal static MethodInfo ImplReleaseReadMethod = typeof(Impl).GetMethod(nameof(ImplReleaseRead));
public static void ImplReleaseRead(IGeneratedStore s) => FindImpl(s).ReleaseRead(); public static void ImplReleaseRead(IGeneratedStore s) => FindImpl(s).ReleaseRead();
public void ReleaseRead() => WriteSyncObject.ExitReadLock();
public void ReleaseRead()
{
if (!WriteSyncObject.IsWriteLockHeld)
WriteSyncObject.ExitReadLock();
}
internal static MethodInfo ImplTakeWriteMethod = typeof(Impl).GetMethod(nameof(ImplTakeWrite)); internal static MethodInfo ImplTakeWriteMethod = typeof(Impl).GetMethod(nameof(ImplTakeWrite));
public static void ImplTakeWrite(IGeneratedStore s) => FindImpl(s).TakeWrite(); public static void ImplTakeWrite(IGeneratedStore s) => FindImpl(s).TakeWrite();
@ -156,6 +171,44 @@ namespace IPA.Config.Stores
public static void ImplReleaseWrite(IGeneratedStore s) => FindImpl(s).ReleaseWrite(); public static void ImplReleaseWrite(IGeneratedStore s) => FindImpl(s).ReleaseWrite();
public void ReleaseWrite() => WriteSyncObject.ExitWriteLock(); public void ReleaseWrite() => WriteSyncObject.ExitWriteLock();
internal static MethodInfo ImplChangeTransactionMethod = typeof(Impl).GetMethod(nameof(ImplChangeTransaction));
public static IDisposable ImplChangeTransaction(IGeneratedStore s, IDisposable nest) => FindImpl(s).ChangeTransaction(nest);
// TODO: use some fixed pool of these, because their lifetimes are hella short
public IDisposable ChangeTransaction(IDisposable nest, bool takeWrite = true)
=> new ChangeTransactionObj(this, !inChangeTransaction, nest, takeWrite && !WriteSyncObject.IsWriteLockHeld);
private sealed class ChangeTransactionObj : IDisposable
{
private readonly Impl impl;
private readonly bool owns;
private readonly bool ownsWrite;
private readonly IDisposable nested;
public ChangeTransactionObj(Impl impl, bool owning, IDisposable nest, bool takeWrite)
{
this.impl = impl;
nested = nest;
if (owns = owning)
impl.inChangeTransaction = true;
if (ownsWrite = takeWrite)
impl.TakeWrite();
}
public void Dispose()
{
if (owns)
{
impl.inChangeTransaction = false;
impl.InvokeChanged();
}
nested?.Dispose();
if (ownsWrite)
impl.ReleaseWrite();
GC.SuppressFinalize(this);
}
~ChangeTransactionObj() => Dispose();
}
public static Impl FindImpl(IGeneratedStore store) public static Impl FindImpl(IGeneratedStore store)
{ {
@ -173,9 +226,8 @@ namespace IPA.Config.Stores
Logger.config.Debug($"Read {values}"); Logger.config.Debug($"Read {values}");
generated.Deserialize(values); generated.Deserialize(values);
ReleaseWrite();
using var transaction = generated.ChangeTransaction();
generated.OnReload(); generated.OnReload();
TakeWrite(); // must take again for runtime to be happy (which is unfortunate)
} }
internal static MethodInfo ImplWriteToMethod = typeof(Impl).GetMethod(nameof(ImplWriteTo)); internal static MethodInfo ImplWriteToMethod = typeof(Impl).GetMethod(nameof(ImplWriteTo));
@ -290,6 +342,7 @@ namespace IPA.Config.Stores
internal delegate IConfigStore GeneratedStoreCreator(IGeneratedStore parent); internal delegate IConfigStore GeneratedStoreCreator(IGeneratedStore parent);
private static bool IsMethodInvalid(MethodInfo m, Type ret) => !m.IsVirtual || m.ReturnType != ret;
private static (GeneratedStoreCreator ctor, Type type) MakeCreator(Type type) private static (GeneratedStoreCreator ctor, Type type) MakeCreator(Type type)
{ // note that this does not and should not use converters by default for everything { // note that this does not and should not use converters by default for everything
if (!type.IsClass) throw new ArgumentException("Config type is not a class"); if (!type.IsClass) throw new ArgumentException("Config type is not a class");
@ -299,12 +352,18 @@ namespace IPA.Config.Stores
throw new ArgumentException("Config type does not have a public parameterless constructor"); throw new ArgumentException("Config type does not have a public parameterless constructor");
#region Parse base object structure #region Parse base object structure
var baseChanged = type.GetMethod("Changed", BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance, null, Type.EmptyTypes, Array.Empty<ParameterModifier>());
if (baseChanged != null && !baseChanged.IsVirtual) baseChanged = null;
var baseOnReload = type.GetMethod("OnReload", BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance, null, Type.EmptyTypes, Array.Empty<ParameterModifier>());
if (baseOnReload != null && !baseOnReload.IsVirtual) baseOnReload = null;
var baseCopyFrom = type.GetMethod("CopyFrom", BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance, null, new[] { type }, Array.Empty<ParameterModifier>());
if (baseCopyFrom != null && !baseCopyFrom.IsVirtual) baseCopyFrom = null;
const BindingFlags overrideMemberFlags = BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance;
var baseChanged = type.GetMethod("Changed", overrideMemberFlags, null, Type.EmptyTypes, Array.Empty<ParameterModifier>());
if (baseChanged != null && IsMethodInvalid(baseChanged, typeof(void))) baseChanged = null;
var baseOnReload = type.GetMethod("OnReload", overrideMemberFlags, null, Type.EmptyTypes, Array.Empty<ParameterModifier>());
if (baseOnReload != null && IsMethodInvalid(baseOnReload, typeof(void))) baseOnReload = null;
var baseCopyFrom = type.GetMethod("CopyFrom", overrideMemberFlags, null, new[] { type }, Array.Empty<ParameterModifier>());
if (baseCopyFrom != null && IsMethodInvalid(baseCopyFrom, typeof(void))) baseCopyFrom = null;
var baseChangeTransaction = type.GetMethod("ChangeTransaction", overrideMemberFlags, null, Type.EmptyTypes, Array.Empty<ParameterModifier>());
if (baseChangeTransaction != null && IsMethodInvalid(baseChangeTransaction, typeof(IDisposable))) baseChangeTransaction = null;
var structure = new List<SerializedMemberInfo>(); var structure = new List<SerializedMemberInfo>();
@ -529,6 +588,7 @@ namespace IPA.Config.Stores
var IGeneratedStore_Deserialize = IGeneratedStore_t.GetMethod(nameof(IGeneratedStore.Deserialize)); var IGeneratedStore_Deserialize = IGeneratedStore_t.GetMethod(nameof(IGeneratedStore.Deserialize));
var IGeneratedStore_OnReload = IGeneratedStore_t.GetMethod(nameof(IGeneratedStore.OnReload)); var IGeneratedStore_OnReload = IGeneratedStore_t.GetMethod(nameof(IGeneratedStore.OnReload));
var IGeneratedStore_Changed = IGeneratedStore_t.GetMethod(nameof(IGeneratedStore.Changed)); var IGeneratedStore_Changed = IGeneratedStore_t.GetMethod(nameof(IGeneratedStore.Changed));
var IGeneratedStore_ChangeTransaction = IGeneratedStore_t.GetMethod(nameof(IGeneratedStore.ChangeTransaction));
#region IGeneratedStore.OnReload #region IGeneratedStore.OnReload
var onReload = typeBuilder.DefineMethod($"<>{nameof(IGeneratedStore.OnReload)}", virtualMemberMethod, null, Type.EmptyTypes); var onReload = typeBuilder.DefineMethod($"<>{nameof(IGeneratedStore.OnReload)}", virtualMemberMethod, null, Type.EmptyTypes);
@ -691,56 +751,6 @@ namespace IPA.Config.Stores
#endregion #endregion
#endregion #endregion
#region IGeneratedStore<T>
var IGeneratedStore_T_t = typeof(IGeneratedStore<>).MakeGenericType(type);
typeBuilder.AddInterfaceImplementation(IGeneratedStore_T_t);
var IGeneratedStore_T_CopyFrom = IGeneratedStore_T_t.GetMethod(nameof(IGeneratedStore<Config>.CopyFrom));
#region IGeneratedStore<T>.CopyFrom
var copyFrom = typeBuilder.DefineMethod($"<>{nameof(IGeneratedStore<Config>.CopyFrom)}", virtualMemberMethod, null, new[] { type, typeof(bool) });
typeBuilder.DefineMethodOverride(copyFrom, IGeneratedStore_T_CopyFrom);
{
var il = copyFrom.GetILGenerator();
var startLock = il.DefineLabel();
il.Emit(OpCodes.Ldarg_2);
il.Emit(OpCodes.Brfalse, startLock);
il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Call, Impl.ImplTakeWriteMethod); // take the write lock
il.MarkLabel(startLock);
var GetLocal = MakeGetLocal(il);
foreach (var member in structure)
{
il.BeginExceptionBlock();
EmitStore(il, member, il =>
{
EmitLoad(il, member, il => il.Emit(OpCodes.Ldarg_1));
EmitCorrectMember(il, member, false, false, GetLocal);
});
il.BeginCatchBlock(typeof(Exception));
EmitWarnException(il, $"Error while copying from member {member.Name}");
il.EndExceptionBlock();
}
var endLock = il.DefineLabel();
il.Emit(OpCodes.Ldarg_2);
il.Emit(OpCodes.Brfalse, endLock);
il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Call, Impl.ImplReleaseWriteMethod); // release write lock
il.MarkLabel(endLock);
il.Emit(OpCodes.Ret);
}
#endregion
#endregion
#region IConfigStore #region IConfigStore
typeBuilder.AddInterfaceImplementation(typeof(IConfigStore)); typeBuilder.AddInterfaceImplementation(typeof(IConfigStore));
@ -815,39 +825,104 @@ namespace IPA.Config.Stores
"<>Changed", "<>Changed",
virtualMemberMethod, virtualMemberMethod,
null, Type.EmptyTypes); null, Type.EmptyTypes);
typeBuilder.DefineMethodOverride(coreChanged, IGeneratedStore_Changed);
if (baseChanged != null)
typeBuilder.DefineMethodOverride(coreChanged, baseChanged);
{ {
var il = coreChanged.GetILGenerator(); var il = coreChanged.GetILGenerator();
il.Emit(OpCodes.Ldarg_0);
if (baseChanged != null)
{
il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Call, baseChanged); // call base
}
il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Tailcall);
il.Emit(OpCodes.Call, Impl.ImplSignalChangedMethod); il.Emit(OpCodes.Call, Impl.ImplSignalChangedMethod);
il.Emit(OpCodes.Ret); // simply call our impl's SignalChanged method and return il.Emit(OpCodes.Ret); // simply call our impl's SignalChanged method and return
}
}
#endregion
#region ChangeTransaction
var coreChangeTransaction = typeBuilder.DefineMethod(
"<>ChangeTransaction",
virtualMemberMethod,
typeof(IDisposable), Type.EmptyTypes);
typeBuilder.DefineMethodOverride(coreChangeTransaction, IGeneratedStore_ChangeTransaction);
if (baseChangeTransaction != null)
typeBuilder.DefineMethodOverride(coreChangeTransaction, baseChangeTransaction);
{
var il = coreChangeTransaction.GetILGenerator();
il.Emit(OpCodes.Ldarg_0);
if (baseChangeTransaction != null)
{
il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Call, baseChangeTransaction);
}
else
il.Emit(OpCodes.Ldnull);
il.Emit(OpCodes.Tailcall);
il.Emit(OpCodes.Call, Impl.ImplChangeTransactionMethod);
il.Emit(OpCodes.Ret);
}
#endregion
#region IGeneratedStore<T>
var IGeneratedStore_T_t = typeof(IGeneratedStore<>).MakeGenericType(type);
typeBuilder.AddInterfaceImplementation(IGeneratedStore_T_t);
var IGeneratedStore_T_CopyFrom = IGeneratedStore_T_t.GetMethod(nameof(IGeneratedStore<Config>.CopyFrom));
#region IGeneratedStore<T>.CopyFrom
var copyFrom = typeBuilder.DefineMethod($"<>{nameof(IGeneratedStore<Config>.CopyFrom)}", virtualMemberMethod, null, new[] { type, typeof(bool) });
typeBuilder.DefineMethodOverride(copyFrom, IGeneratedStore_T_CopyFrom);
{
var il = copyFrom.GetILGenerator();
var transactionLocal = il.DeclareLocal(IDisposable_t);
var startLock = il.DefineLabel();
il.Emit(OpCodes.Ldarg_2);
il.Emit(OpCodes.Brfalse, startLock);
il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Call, coreChangeTransaction); // take the write lock
il.Emit(OpCodes.Stloc, transactionLocal);
il.MarkLabel(startLock);
if (baseChanged != null) {
var changedMethod = typeBuilder.DefineMethod( // copy to override baseChanged
baseChanged.Name,
virtualMemberMethod,
null, Type.EmptyTypes);
typeBuilder.DefineMethodOverride(changedMethod, baseChanged);
var GetLocal = MakeGetLocal(il);
foreach (var member in structure)
{ {
var il = changedMethod.GetILGenerator();
il.BeginExceptionBlock();
il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Call, baseChanged); // call base
EmitStore(il, member, il =>
{
EmitLoad(il, member, il => il.Emit(OpCodes.Ldarg_1));
EmitCorrectMember(il, member, false, false, GetLocal);
});
il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Tailcall);
il.Emit(OpCodes.Call, coreChanged); // call back to the core change method
il.BeginCatchBlock(typeof(Exception));
il.Emit(OpCodes.Ret);
}
EmitWarnException(il, $"Error while copying from member {member.Name}");
coreChanged = changedMethod; // switch to calling this version instead of just the default
il.EndExceptionBlock();
}
var endLock = il.DefineLabel();
il.Emit(OpCodes.Ldarg_2);
il.Emit(OpCodes.Brfalse, endLock);
il.Emit(OpCodes.Ldloc, transactionLocal);
il.Emit(OpCodes.Callvirt, IDisposable_Dispose);
il.MarkLabel(endLock);
il.Emit(OpCodes.Ret);
} }
typeBuilder.DefineMethodOverride(coreChanged, IGeneratedStore_Changed);
#endregion
#endregion #endregion
#region base.CopyFrom #region base.CopyFrom
@ -862,20 +937,20 @@ namespace IPA.Config.Stores
{ {
var il = pubCopyFrom.GetILGenerator(); var il = pubCopyFrom.GetILGenerator();
// TODO: use transactional changes
il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Call, coreChangeTransaction);
il.Emit(OpCodes.Ldarg_0); il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Ldarg_1); il.Emit(OpCodes.Ldarg_1);
il.Emit(OpCodes.Ldc_I4_1);
il.Emit(OpCodes.Ldc_I4_0);
il.Emit(OpCodes.Call, copyFrom); // call internal il.Emit(OpCodes.Call, copyFrom); // call internal
il.Emit(OpCodes.Ldarg_0); il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Ldarg_1); il.Emit(OpCodes.Ldarg_1);
il.Emit(OpCodes.Call, baseCopyFrom); // call base il.Emit(OpCodes.Call, baseCopyFrom); // call base
il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Tailcall); il.Emit(OpCodes.Tailcall);
il.Emit(OpCodes.Call, coreChanged); // call changed
il.Emit(OpCodes.Callvirt, IDisposable_Dispose); // dispose transaction (which calls changed)
il.Emit(OpCodes.Ret); il.Emit(OpCodes.Ret);
} }
} }
@ -925,10 +1000,12 @@ namespace IPA.Config.Stores
{ // TODO: decide if i want to correct the value before or after i take the write lock { // TODO: decide if i want to correct the value before or after i take the write lock
var il = propSet.GetILGenerator(); var il = propSet.GetILGenerator();
var transactionLocal = il.DeclareLocal(IDisposable_t);
var GetLocal = MakeGetLocal(il); var GetLocal = MakeGetLocal(il);
il.Emit(OpCodes.Ldarg_0); il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Call, Impl.ImplTakeWriteMethod); // take the write lock
il.Emit(OpCodes.Call, coreChangeTransaction); // take the write lock
il.Emit(OpCodes.Stloc, transactionLocal);
il.BeginExceptionBlock(); il.BeginExceptionBlock();
@ -937,16 +1014,13 @@ namespace IPA.Config.Stores
EmitCorrectMember(il, member, false, false, GetLocal); EmitCorrectMember(il, member, false, false, GetLocal);
il.Emit(OpCodes.Call, set); il.Emit(OpCodes.Call, set);
il.BeginFinallyBlock();
il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Call, Impl.ImplReleaseWriteMethod); // release the write lock
il.BeginFinallyBlock();
il.Emit(OpCodes.Ldloc, transactionLocal);
il.Emit(OpCodes.Callvirt, IDisposable_Dispose);
il.EndExceptionBlock(); il.EndExceptionBlock();
il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Call, Impl.ImplInvokeChangedMethod);
il.Emit(OpCodes.Ret); il.Emit(OpCodes.Ret);
} }
@ -1125,6 +1199,9 @@ namespace IPA.Config.Stores
il.Emit(OpCodes.Call, Type_GetTypeFromHandle); il.Emit(OpCodes.Call, Type_GetTypeFromHandle);
} }
private static Type IDisposable_t = typeof(IDisposable);
private static MethodInfo IDisposable_Dispose = IDisposable_t.GetMethod(nameof(IDisposable.Dispose));
private static Type Decimal_t = typeof(decimal); private static Type Decimal_t = typeof(decimal);
private static ConstructorInfo Decimal_FromFloat = Decimal_t.GetConstructor(new[] { typeof(float) }); private static ConstructorInfo Decimal_FromFloat = Decimal_t.GetConstructor(new[] { typeof(float) });
private static ConstructorInfo Decimal_FromDouble = Decimal_t.GetConstructor(new[] { typeof(double) }); private static ConstructorInfo Decimal_FromDouble = Decimal_t.GetConstructor(new[] { typeof(double) });


+ 7
- 3
IPA.Loader/Loader/DisabledConfig.cs View File

@ -23,18 +23,22 @@ namespace IPA.Loader
Instance = Disabled.Generated<DisabledConfig>(); Instance = Disabled.Generated<DisabledConfig>();
} }
public virtual bool Reset { get; set; } = false;
public virtual bool Reset { get; set; } = true;
[NonNullable] [NonNullable]
[UseConverter(typeof(CollectionConverter<string, HashSet<string>>))] [UseConverter(typeof(CollectionConverter<string, HashSet<string>>))]
public virtual HashSet<string> DisabledModIds { get; set; } = new HashSet<string>(); public virtual HashSet<string> DisabledModIds { get; set; } = new HashSet<string>();
protected internal virtual void Changed() { } protected internal virtual void Changed() { }
protected internal virtual IDisposable ChangeTransaction() => null;
protected virtual void OnReload() protected virtual void OnReload()
{ {
if (DisabledModIds == null || Reset)
DisabledModIds = new HashSet<string>();
if (DisabledModIds == null || Reset)
{
DisabledModIds = new HashSet<string>();
Reset = false;
}
} }
} }
} }

+ 3
- 1
IPA.Loader/Loader/PluginManager.cs View File

@ -98,6 +98,7 @@ namespace IPA.Loader
var toDisable = transaction.ToDisable; var toDisable = transaction.ToDisable;
transaction.Dispose(); transaction.Dispose();
using var disabledChangeTransaction = DisabledConfig.Instance.ChangeTransaction();
{ {
// first enable the mods that need to be // first enable the mods that need to be
void DeTree(List<PluginMetadata> into, IEnumerable<PluginMetadata> tree) void DeTree(List<PluginMetadata> into, IEnumerable<PluginMetadata> tree)
@ -198,7 +199,8 @@ namespace IPA.Loader
result = TaskEx.WhenAll(disableStructure.Select(d => Disable(d, disabled))); result = TaskEx.WhenAll(disableStructure.Select(d => Disable(d, disabled)));
} }
DisabledConfig.Instance.Changed();
//DisabledConfig.Instance.Changed();
// changed is handled by transaction
return result; return result;
} }
} }


Loading…
Cancel
Save