将程序集加载到WindowsAzure WorkerRole中单独的AppDomain中

本文关键字:单独 AppDomain WorkerRole WindowsAzure 程序集 加载 | 更新日期: 2023-09-27 18:09:24

我正在开发一个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; }
}

将程序集加载到WindowsAzure WorkerRole中单独的AppDomain中

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

AppDomain currentDomain = AppDomain.CurrentDomain;
AppDomain BrokerJobDomain = AppDomain.CreateDomain("WorkerPluginDomain" + Guid.NewGuid().ToString(), null,  AppDomain.CurrentDomain.SetupInformation );