试图伪造强名称程序集,但“清单定义不匹配”

时间:2013-01-21 20:38:31

标签: c# .net appdomain .net-assembly interception

我想拦截插件和主机程序集之间的通信。

我尝试将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;
}

我会感激任何帮助。

0 个答案:

没有答案