我想拦截插件和主机程序集之间的通信。
我尝试将ThirdPartyPlugin.dll
加载到AppDomain沙箱(下面的代码)中,以拦截其加载HostLibrary.dll
程序集的尝试(我正在使用技巧处理{ {3}}事件)。而不是原来的HostLibrary.dll
,我试图注入一个假的,具有一些不同的功能。
不幸的是,原来的HostLibrary.dll
是一个强名称大会,我认为因为我在我的程序中得到了如下所示的异常:
未处理的异常:System.IO.FileLoadException:无法加载文件或程序集'HostLibrary,Version = 7.0.0.0,Culture = neutral,PublicKeyToken = 1a2b3c4d5e6f7890'或其依赖项之一。定位的程序集的清单定义与程序集引用不匹配。 (HRESULT异常:0x80131040)---> System.IO.FileLoadException:定位程序集的清单定义与程序集引用不匹配。 (HRESULT异常:0x80131040)
我创建的虚假程序集与原始程序集的名称("HostLibrary"
)和版本("7.0.0.0"
)匹配;有一段时间我认为我甚至使它显然与PublicKeyToken
匹配(当我打印AssemblyResolve时,它是相同的),但异常仍然发生。
您对如何解决此问题有任何想法吗?有可能解决吗?是否因为数字签名或其他原因而发生?如果签名,那么可以禁用检查吗?实际上,从阅读Assembly.FullName开始,我认为我不应该真的遇到这个问题......那么什么是错的?
代码是:
static void Main(string[] args)
{
AppDomainSetup setup = new AppDomainSetup();
setup.ApplicationName = "WrapperBuddy";
setup.ApplicationBase = System.Environment.CurrentDirectory;
AppDomain sandbox = AppDomain.CreateDomain("SandboxBuddy", null, setup);
sandbox.AssemblyResolve += delegate(object sender, ResolveEventArgs args)
{
String name = new AssemblyName(args.Name).Name;
return BuildFakeAssembly(name);
};
sandbox.DoCallBack(delegate()
{
Assembly plugin = LoadAssembly(PLUGIN_DIR, "ThirdPartyPlugin");
System.Console.WriteLine(plugin.GetExportedTypes()); // (1)
});
}
将辅助函数定义为:
static Assembly LoadAssembly(String dir, String name)
{
string path = Path.Combine(dir, name) + ".dll";
return Assembly.Load(File.ReadAllBytes(path));
}
static Assembly BuildFakeAssembly(String name)
{
AssemblyName aName = new AssemblyName(name);
// see: http://msdn.microsoft.com/en-us/library/w58ww7se%28v=vs.85%29.aspx
// http://msdn.microsoft.com/en-us/library/w58ww7se%28v=vs.100%29.aspx
aName.KeyPair = new StrongNameKeyPair(new FileStream(@"MyKey.snk", FileMode.Open));
// MyKey.snk is a random key generated using sn.exe
aName.Version = new Version("7.0.0.0");
//aName.SetPublicKeyToken(new byte[] { 0x11, 0x22, ... }); // didn't help
AssemblyBuilder ab = AppDomain.CurrentDomain.DefineDynamicAssembly(
aName, AssemblyBuilderAccess.RunAndSave);
ModuleBuilder mb = ab.DefineDynamicModule(aName.Name, aName.Name + ".dll");
return ab;
}
我会感激任何帮助。