Files
SPT-Server-Build/Libraries/SPTarkov.Reflection/Patching/AbstractPatch.cs
T
2025-08-03 02:15:16 -04:00

177 lines
4.8 KiB
C#

using System.Reflection;
using HarmonyLib;
namespace SPTarkov.Reflection.Patching;
/// <summary>
/// Harmony patch wrapper class. See mod example 6.1 for usage.
/// </summary>
public abstract class AbstractPatch
{
/// <summary>
/// Method this patch targets
/// </summary>
public MethodBase? TargetMethod { get; private set; }
/// <summary>
/// Is this patch active?
/// </summary>
public bool IsActive { get; private set; }
/// <summary>
/// The harmony Id assigned to this patch, usually the name of the patch class.
/// </summary>
public string HarmonyId
{
get { return _harmony.Id; }
}
private readonly Harmony _harmony;
private readonly List<HarmonyMethod> _prefixList;
private readonly List<HarmonyMethod> _postfixList;
private readonly List<HarmonyMethod> _transpilerList;
private readonly List<HarmonyMethod> _finalizerList;
private readonly List<HarmonyMethod> _ilManipulatorList;
/// <summary>
/// Constructor
/// </summary>
/// <param name="name">Name</param>
protected AbstractPatch(string? name = null)
{
_harmony = new Harmony(name ?? GetType().Name);
_prefixList = GetPatchMethods(typeof(PatchPrefixAttribute));
_postfixList = GetPatchMethods(typeof(PatchPostfixAttribute));
_transpilerList = GetPatchMethods(typeof(PatchTranspilerAttribute));
_finalizerList = GetPatchMethods(typeof(PatchFinalizerAttribute));
_ilManipulatorList = GetPatchMethods(typeof(PatchIlManipulatorAttribute));
if (
_prefixList.Count == 0
&& _postfixList.Count == 0
&& _transpilerList.Count == 0
&& _finalizerList.Count == 0
&& _ilManipulatorList.Count == 0
)
{
throw new Exception($"{_harmony.Id}: At least one of the patch methods must be specified");
}
}
/// <summary>
/// Get original method
/// </summary>
/// <returns>Method</returns>
protected abstract MethodBase GetTargetMethod();
/// <summary>
/// Get HarmonyMethod from string
/// </summary>
/// <param name="attributeType">Attribute type</param>
/// <returns>Method</returns>
private List<HarmonyMethod> GetPatchMethods(Type attributeType)
{
var T = GetType();
var methods = new List<HarmonyMethod>();
foreach (var method in T.GetMethods(BindingFlags.Static | BindingFlags.NonPublic | BindingFlags.Public | BindingFlags.DeclaredOnly))
{
if (method.GetCustomAttribute(attributeType) != null)
{
methods.Add(new HarmonyMethod(method));
}
}
return methods;
}
/// <summary>
/// Apply patch to target
/// </summary>
public void Enable()
{
// We never want to have duplicated patches, prevent it.
if (IsActive)
{
return;
}
TargetMethod = GetTargetMethod();
if (TargetMethod == null)
{
throw new InvalidOperationException($"{_harmony.Id}: TargetMethod is null");
}
try
{
foreach (var prefix in _prefixList)
{
_harmony.Patch(TargetMethod, prefix: prefix);
}
foreach (var postfix in _postfixList)
{
_harmony.Patch(TargetMethod, postfix: postfix);
}
foreach (var transpiler in _transpilerList)
{
_harmony.Patch(TargetMethod, transpiler: transpiler);
}
foreach (var finalizer in _finalizerList)
{
_harmony.Patch(TargetMethod, finalizer: finalizer);
}
foreach (var ilmanipulator in _ilManipulatorList)
{
_harmony.Patch(TargetMethod, ilmanipulator: ilmanipulator);
}
ModPatchCache.AddPatch(this);
IsActive = true;
}
catch (Exception ex)
{
throw new Exception($"{_harmony.Id}:", ex);
}
}
/// <summary>
/// Remove applied patch from target
/// </summary>
public void Disable()
{
// Nothing to disable
if (!IsActive)
{
return;
}
var target = GetTargetMethod();
if (target == null)
{
throw new InvalidOperationException($"{_harmony.Id}: TargetMethod is null");
}
try
{
_harmony.Unpatch(target, HarmonyPatchType.All, _harmony.Id);
}
catch (Exception ex)
{
throw new Exception($"{_harmony.Id}:", ex);
}
if (!ModPatchCache.RemovePatch(this))
{
throw new Exception($"{_harmony.Id}: Target patch not present in cache, a mod is likely externally altering it.");
}
IsActive = false;
}
}