c#-将程序集加载到WindowsAzure WorkerRole中的单独的AppDomain中
我正在开发一个Azure工作者角色,该角色需要根据从WCF服务提取的工作来定期执行操作.由于Azure部署的性质,因此决定在Azure中实现插件框架,以允许快速更新我们的产品,同时通过消除对完全蔚蓝的部署的需求以及支持各种程序集的多个版本来最大程度地减少停机时间.
插件是从Cloud存储中下载的,并加载到一个类中,该类包含加载程序集所需的所有相关信息. (请参阅下面的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; }
}
解决方法:
该问题似乎与未在子域上设置RelativeSearchPath有关.确保子域具有相同的安装凭据似乎可以解决此问题.
AppDomain currentDomain = AppDomain.CurrentDomain;
AppDomain BrokerJobDomain = AppDomain.CreateDomain("WorkerPluginDomain" + Guid.NewGuid().ToString(), null, AppDomain.CurrentDomain.SetupInformation );