<PackageReference Include="Castle.Core" Version="4.4.1" />

ProxyUtil

public static class ProxyUtil
using Castle.Core.Internal; using Castle.DynamicProxy.Generators; using Castle.DynamicProxy.Internal; using System; using System.Linq; using System.Reflection; using System.Runtime.CompilerServices; namespace Castle.DynamicProxy { public static class ProxyUtil { private static readonly SynchronizedDictionary<Assembly, bool> internalsVisibleToDynamicProxy = new SynchronizedDictionary<Assembly, bool>(); public static TDelegate CreateDelegateToMixin<TDelegate>(object proxy) { return (TDelegate)(object)CreateDelegateToMixin(proxy, typeof(TDelegate)); } public static Delegate CreateDelegateToMixin(object proxy, Type delegateType) { if (proxy == null) throw new ArgumentNullException("proxy"); if ((object)delegateType == null) throw new ArgumentNullException("delegateType"); if (!delegateType.IsDelegateType()) throw new ArgumentException("Type is not a delegate type.", "delegateType"); MethodInfo invokeMethod = System.Reflection.TypeExtensions.GetMethod(delegateType, "Invoke"); MethodInfo methodInfo = System.Reflection.TypeExtensions.GetMember(proxy.GetType(), "Invoke", BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic).Cast<MethodInfo>().FirstOrDefault((MethodInfo m) => MethodSignatureComparer.Instance.EqualParameters(m, invokeMethod)); if ((object)methodInfo == null) throw new MissingMethodException("The proxy does not have an Invoke method that is compatible with the requested delegate type."); return methodInfo.CreateDelegate(delegateType, proxy); } public static object GetUnproxiedInstance(object instance) { IProxyTargetAccessor proxyTargetAccessor = instance as IProxyTargetAccessor; if (proxyTargetAccessor != null) instance = proxyTargetAccessor.DynProxyGetTarget(); return instance; } public static Type GetUnproxiedType(object instance) { IProxyTargetAccessor proxyTargetAccessor = instance as IProxyTargetAccessor; if (proxyTargetAccessor != null) { object obj = proxyTargetAccessor.DynProxyGetTarget(); if (obj != null) { if (obj == instance) return instance.GetType().GetTypeInfo().get_BaseType(); instance = obj; } } return instance.GetType(); } public static bool IsProxy(object instance) { return instance is IProxyTargetAccessor; } public static bool IsProxyType(Type type) { return typeof(IProxyTargetAccessor).IsAssignableFrom(type); } public static bool IsAccessible(MethodBase method) { if (IsAccessibleMethod(method)) return IsAccessibleType(method.DeclaringType); return false; } public static bool IsAccessible(MethodBase method, out string message) { if (IsAccessible(method)) { message = null; return true; } message = CreateMessageForInaccessibleMethod(method); return false; } public static bool IsAccessible(Type type) { return IsAccessibleType(type); } internal static bool AreInternalsVisibleToDynamicProxy(Assembly asm) { return internalsVisibleToDynamicProxy.GetOrAdd(asm, (Assembly a) => asm.GetCustomAttributes<InternalsVisibleToAttribute>().Any((InternalsVisibleToAttribute attr) => attr.AssemblyName.Contains(ModuleScope.DEFAULT_ASSEMBLY_NAME))); } internal static bool IsAccessibleType(Type target) { TypeInfo typeInfo = target.GetTypeInfo(); if (typeInfo.get_IsPublic() || typeInfo.get_IsNestedPublic()) return true; bool isNested = target.IsNested; bool flag = isNested && (typeInfo.get_IsNestedAssembly() || typeInfo.get_IsNestedFamORAssem()); if (((!typeInfo.get_IsVisible() && !isNested) | flag) && AreInternalsVisibleToDynamicProxy(typeInfo.get_Assembly())) return true; return false; } internal static bool IsAccessibleMethod(MethodBase method) { if (method.IsPublic || method.IsFamily || method.IsFamilyOrAssembly) return true; if (method.IsAssembly || method.IsFamilyAndAssembly) return AreInternalsVisibleToDynamicProxy(method.DeclaringType.GetTypeInfo().get_Assembly()); return false; } internal static bool IsInternal(MethodBase method) { if (!method.IsAssembly) { if (method.IsFamilyAndAssembly) return !method.IsFamilyOrAssembly; return false; } return true; } private static string CreateMessageForInaccessibleMethod(MethodBase inaccessibleMethod) { Assembly assembly = inaccessibleMethod.DeclaringType.GetTypeInfo().get_Assembly(); string str = $"""{inaccessibleMethod}"""; string str2 = ExceptionMessageBuilder.CreateInstructionsToMakeVisible(assembly); return str + str2; } } }