101 lines
2.8 KiB
C#
Raw Normal View History

2021-12-31 17:10:41 -08:00
using Microsoft.VisualStudio.TestTools.UnitTesting;
using Mono.Cecil;
using Mono.Cecil.Cil;
using Mono.Cecil.Rocks;
2022-01-08 02:39:32 -08:00
using MonoMod.Cil;
2021-12-31 17:10:41 -08:00
using QSB.Messaging;
using QSB.Utility;
2022-01-02 18:10:28 -08:00
using System;
2021-12-31 17:10:41 -08:00
using System.Linq;
using System.Reflection;
namespace QSBTests
{
[TestClass]
public class MessageTests
{
[TestMethod]
public void TestMessages()
{
var module = ModuleDefinition.ReadModule("QSB.dll");
var messageTypes = typeof(QSBMessage).GetDerivedTypes();
var fromField = module.ImportReference(typeof(QSBMessage).GetField("From", Util.Flags));
var toField = module.ImportReference(typeof(QSBMessage).GetField("To", Util.Flags));
var objectIdField = module.ImportReference(typeof(QSBWorldObjectMessage<>).GetField("ObjectId", Util.Flags));
foreach (var type in messageTypes)
{
var fields = type.GetFields(Util.Flags)
.Select(x => module.ImportReference(x));
2022-01-14 22:10:41 -08:00
var constructor = module.ImportReference(type.GetConstructors(Util.Flags).Single()).Resolve();
2021-12-31 17:10:41 -08:00
var serialize = module.ImportReference(type.GetMethod("Serialize", Util.Flags)).Resolve();
var deserialize = module.ImportReference(type.GetMethod("Deserialize", Util.Flags)).Resolve();
foreach (var field in fields)
{
2022-01-08 02:39:32 -08:00
if (!field.GenericEq(fromField) && !field.GenericEq(toField) && !field.GenericEq(objectIdField))
2021-12-31 18:54:32 -08:00
{
2021-12-31 17:10:41 -08:00
constructor.CheckUses(field, Util.UseType.Store);
2021-12-31 18:54:32 -08:00
}
2022-01-01 22:19:10 +00:00
2022-01-14 02:29:42 -08:00
serialize.CheckUses(field, Util.UseType.Load);
deserialize.CheckUses(field, Util.UseType.Store);
2021-12-31 17:10:41 -08:00
}
}
}
}
2022-01-08 02:39:32 -08:00
public static partial class Util
2021-12-31 17:10:41 -08:00
{
public const BindingFlags Flags = BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic;
2022-01-08 02:39:32 -08:00
/// <summary>
/// ignores open vs closed generic type
/// </summary>
public static bool GenericEq(this MemberReference a, MemberReference b) =>
2021-12-31 17:10:41 -08:00
a.DeclaringType.Namespace == b.DeclaringType.Namespace &&
a.DeclaringType.Name == b.DeclaringType.Name &&
a.Name == b.Name;
2022-01-08 02:39:32 -08:00
public enum UseType { Store, Load }
2021-12-31 17:10:41 -08:00
public static void CheckUses(this MethodDefinition method, FieldReference field, UseType useType)
{
2022-01-08 02:39:32 -08:00
Func<Instruction, bool> matches = useType switch
2021-12-31 17:10:41 -08:00
{
2022-01-08 02:39:32 -08:00
UseType.Store => x => x.MatchStfld(out var f) && f.GenericEq(field),
2022-01-13 23:42:47 -08:00
UseType.Load => x => (x.MatchLdfld(out var f) || x.MatchLdflda(out f)) && f.GenericEq(field),
2022-01-02 18:10:28 -08:00
_ => throw new ArgumentOutOfRangeException(nameof(useType), useType, null)
2021-12-31 17:10:41 -08:00
};
while (true)
{
2021-12-31 18:54:32 -08:00
var il = method.Body.Instructions;
2022-01-08 02:39:32 -08:00
var uses = il.Any(matches);
2021-12-31 18:54:32 -08:00
if (uses)
{
return;
}
2021-12-31 17:10:41 -08:00
var baseMethod = method.GetBaseMethod();
2022-01-08 02:39:32 -08:00
if (baseMethod == method)
2021-12-31 18:54:32 -08:00
{
break;
}
2022-01-01 22:19:10 +00:00
2022-01-08 02:39:32 -08:00
var callsBase = il.Any(x => x.MatchCall(out var m) && m.GenericEq(baseMethod));
2022-01-02 18:10:28 -08:00
if (!callsBase)
2021-12-31 18:54:32 -08:00
{
break;
}
2022-01-01 22:19:10 +00:00
2021-12-31 17:10:41 -08:00
method = baseMethod;
}
Assert.Fail($"{method} does not {useType} {field}");
}
}
}