Мне нужно выполнить метод в сборке, загруженной во время выполнения. Теперь я хочу выгрузить эти загруженные сборки после вызова метода. Я знаю, что мне нужен новый AppDomain, поэтому я могу выгрузить библиотеки. Но здесь проблема возникает.
Сборки, которые загружаются, являются плагинами в моей плагиновой структуре. У них нет точки входа. Все, что я знаю, это то, что они содержат некоторые типы, которые реализуют данный интерфейс. Старый, не-AppDomain-код выглядит следующим образом (немного сокращен):
try
{
string path = Path.GetFullPath("C:\library.dll");
AppDomain.CurrentDomain.AssemblyResolve += CurrentDomain_AssemblyResolve;
Assembly asm = Assembly.LoadFrom(path);
Type[] types = asm.GetExportedTypes();
foreach (Type t in types)
{
if ((t.GetInterface("IStarter") != null) && !t.IsAbstract)
{
object tempObj = Activator.CreateInstance(t);
MethodInfo info = t.GetMethod("GetParameters");
if (info != null)
{
return info.Invoke(tempObj, null) as string;
}
}
}
}
catch (Exception ex)
{
MessageBox.Show(String.Format("Damn '{0}'.", ex.Message), "Exception", MessageBoxButtons.OK, MessageBoxIcon.Error);
}
private Assembly CurrentDomain_AssemblyResolve(object sender, ResolveEventArgs args)
{
if (args.Name.StartsWith("MyProject.View,"))
{
string path = Path.GetFullPath("C:\view.dll"));
return Assembly.LoadFrom(path);
}
return null;
}
Теперь я хочу, чтобы они загружались в собственный AppDomain:
try
{
string path = Path.GetFullPath("C:\library.dll");
AppDomain domain = AppDomain.CreateDomain("TempDomain");
domain.AssemblyResolve += CurrentDomain_AssemblyResolve; // 1. Exception here!!
domain.ExecuteAssembly(path); // 2. Exception here!!
domain.CreateInstanceFrom(...); // 3. I have NO clue, how the type is named.
domain.Load(...); // 4. I have NO clue, how the assembly is named.
domain.DoCallBack(...); // 5. Exception here!!
// ...
}
catch (Exception ex)
{
MessageBox.Show(String.Format("Damn '{0}'.", ex.Message), "Exception", MessageBoxButtons.OK, MessageBoxIcon.Error);
}
Как вы можете видеть, я поставил 5 случаев.
-
Если я устанавливаю обработчик событий, я получаю исключение, что сборка (это консоль управления (mmc.exe) SnapIn. не может быть найдена/загружена.
-
ExecuteAssembly не находит точку входа (ну, ее нет).
-
Я не знаю, как называется этот тип. Как загрузить по интерфейсу?
-
Аналогично 3. Как получить имя сборки?
-
Те же ошибки, что и в 1.
Я думаю, что проблема может быть консолью управления каким-то образом, или я просто не знаю, что я делаю неправильно. Любая помощь приветствуется.
ОБНОВЛЕНИЕ 1
Теперь я попытался использовать размещенное прокси-решение.
AppDomain domain = AppDomain.CreateDomain("TempDomain");
InstanceProxy proxy = domain.CreateInstanceAndUnwrap(Assembly.GetAssembly(
typeof(InstanceProxy)).FullName, typeof(InstanceProxy).ToString()) as InstanceProxy;
if (proxy != null)
{
proxy.LoadAssembly(path);
}
AppDomain.Unload(domain);
public class InstanceProxy : MarshalByRefObject
{
public void LoadAssembly(string path)
{
AppDomain.CurrentDomain.AssemblyResolve += CurrentDomain_AssemblyResolve;
Assembly asm = Assembly.LoadFrom(path);
Type[] types = asm.GetExportedTypes();
// ...see above...
}
}
Это тоже не работает. При попытке создать прокси-объект я получаю исключение:
Не удалось загрузить файл или сборку "MyProject.SnapIn, Version = 1.0.0.0, Culture = neutral, PublicKeyToken = null" или одну из его зависимостей. Система не может найти указанный файл.
Файл в сообщении об ошибке - это значение, загруженное в mmc (SnapIn). Любая идея, как исправить эту ошибку? AppDomain.AssemblyResolve не вызывается (ни в старом, ни в новом домене).
ОБНОВЛЕНИЕ 2
Теперь я попробовал решение с AppDomainSetup. Теперь исключение изменилось на:
Не удалось загрузить файл или сборку:///C:/Development/MyProject/bin/SnapIn/MyProject.SnapIn.DLL 'или одну из его зависимостей. Данное имя сборки или кодовая база недействительны. (Исключение из HRESULT: 0x80131047)
Любая идея?