3

我正在开发一个 Azure Worker Role,它需要根据从 WCF 服务中提取的工作定期执行操作。由于 Azure 部署的性质,我们决定在 Azure 中实施一个插件框架,以允许快速更新我们的产品,同时通过消除对完整 Azure 部署的需求以及支持各种程序集的多个版本来最大限度地减少停机时间。

插件从云存储下载并加载到包含加载程序集所需的所有相关信息的类中。(请参阅下面的 PluginAssemblyInfo)

在其他开发工作中,我开发了以下 ReflectionHelper 类。这个类在其他应用程序中工作,我已经验证它在控制台应用程序中工作。

但是,在 Azure 模拟器中运行时,出现以下错误:(我在引发异常的行上添加了注释。)

Type is not resolved for member 'MyCompanyName.ReflectionHelper,MyCompanyAssembly, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null'.

ReflectionHelper 类作为源文件添加到调用它的同一程序集中。(出于匿名原因,我确实编辑了程序集/命名空间的名称)

鉴于此类在其他情况下有效,我唯一的猜测是它可能是一个信任问题,但我没有在网上找到很多与此相关的信息。任何帮助将不胜感激。

这是 ReflectionHelper 类的代码。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System.Runtime.Remoting;
using System.Runtime.Serialization;
using System.Web;

namespace MyCompanyName
{
    [Serializable]
    public class ReflectionHelper
    {

        public AppDomain appDomain { get; set; }
        public byte[] PluginBytes { get; set; }
        public string PluginClassName { get; set; }
        public string PluginAssemblyName { get; set; }
        public Dictionary<string, byte[]> AssemblyArchive { get; set; }

        public object GetInstance(AppDomain NewDomain, byte[] PluginBytes, string PluginClassName, string PluginAssemblyName, Dictionary<string, byte[]> AssemblyArchive)
        {
            try
            {
                this.appDomain = NewDomain;
                this.PluginBytes = PluginBytes;
                this.PluginClassName = PluginClassName;
                this.AssemblyArchive = AssemblyArchive;
                this.PluginAssemblyName = PluginAssemblyName;

                //this is the line that throws the serializationexception
                appDomain.AssemblyResolve += new ResolveEventHandler((sender, args) =>
                {
                    AssemblyName x = new AssemblyName(args.Name);
                    string Name = x.Name;
                    if (System.IO.Path.GetExtension(x.Name) != ".dll")
                        Name += ".dll";
                    Assembly Ret = appDomain.Load(this.AssemblyArchive[Name]);

                    return Ret;
                });
                appDomain.DoCallBack(LoaderCallBack);
                return appDomain.GetData("Plugin");
            }
            catch (Exception ex)
            {
                throw ex;
            }
        }
        public void LoaderCallBack()
        {
            ObjectHandle PluginObject = appDomain.CreateInstance(string.Format(this.PluginAssemblyName), PluginClassName);
            appDomain.SetData("Plugin", PluginObject.Unwrap());
        }
    }


}

这是调用它的代码片段

AppDomain currentDomain = AppDomain.CurrentDomain;
AppDomainSetup domainSetup = new AppDomainSetup() { };
AppDomain BrokerJobDomain = AppDomain.CreateDomain("WorkerPluginDomain" + Guid.NewGuid().ToString());//, null, new AppDomainSetup { ApplicationBase = AppDomain.CurrentDomain.SetupInformation.ApplicationBase });

ReflectionHelper helper = new ReflectionHelper();
object Instance = helper.GetInstance(BrokerJobDomain, pluginAssembly.PluginBytes, pluginAssembly.ClassName, pluginAssembly.AssemblyName, pluginAssembly.AssemblyArchive);
IWorkPlugin executor = (IWorkPlugin)Instance;

这是从云存储下载插件时填充的程序集信息类

public class PluginAssemblyInfo
{
    public string ClassName { get; set; }
    public byte[] PluginBytes { get; set; }
    public Dictionary<string, byte[]> AssemblyArchive { get; set; }
    public string AssemblyName { get; set; }
    public DateTime LoadDate { get; set; }
}
4

1 回答 1

4

该问题似乎与未在子域上设置的 RelativeSearchPath 有关。确保子域具有相同的设置凭据似乎可以解决此问题。

AppDomain currentDomain = AppDomain.CurrentDomain;
AppDomain BrokerJobDomain = AppDomain.CreateDomain("WorkerPluginDomain" + Guid.NewGuid().ToString(), null,  AppDomain.CurrentDomain.SetupInformation );
于 2013-04-17T14:15:28.303 回答