Как получить дополнительные типы PowerShell для использования дополнительных типов - PullRequest
4 голосов
/ 19 апреля 2010

Я работаю над проектом PoSh, который генерирует код CSharp, а затем Add-Type сохраняет его в памяти.

Новые типы используют существующие типы в дисковой DLL, которая загружается через Add-Type.

Все хорошо, пока я не попытаюсь вызвать методы для новых типов. Вот пример того, что я делаю:

$PWD = "."
rm -Force $PWD\TestClassOne*
$code = "
namespace TEST{
public class TestClassOne
{
    public int DoNothing()
    {
        return 1;
    }
}
}"
$code | Out-File tcone.cs
Add-Type -OutputAssembly $PWD\TestClassOne.dll -OutputType Library -Path $PWD\tcone.cs
Add-Type -Path $PWD\TestClassOne.dll
$a = New-Object TEST.TestClassOne
"Using TestClassOne"
$a.DoNothing()


"Compiling TestClassTwo"
Add-Type -Language CSharpVersion3 -TypeDefinition "
namespace TEST{
public class TestClassTwo
{
    public int CallTestClassOne()
    {
        var a = new TEST.TestClassOne();
        return a.DoNothing();
    }
}
}" -ReferencedAssemblies $PWD\TestClassOne.dll
"OK"
$b = New-Object TEST.TestClassTwo
"Using TestClassTwo"
$b.CallTestClassOne()

Запуск вышеуказанного скрипта приводит к следующей ошибке в последней строке:

Исключение, вызывающее "CallTestClassOne" с аргументом (ами) "0": "Не удалось загрузить файл или сборку" TestClassOne, ... " или одна из его зависимостей. Система не может найти указанный файл." На AddTypeTest.ps1: 39 символов: 20 + $ b.CallTestClassOne <<<< () + CategoryInfo: NotSpecified: (:) [], MethodInvocationException + FullyQualifiedErrorId: DotNetMethodException </p>

Что я делаю не так?

Ответы [ 2 ]

7 голосов
/ 19 апреля 2010

Это происходит потому, что любые сборки ищутся загрузчиком CLR в базовом каталоге приложения (PowerShell). Конечно, там нет вашей сборки. Лучший способ решить эту проблему - перехватить событие AssemblyResolve, как об этом говорится, но использовать его, чтобы сообщить CLR, где находится сборка. Вы не можете сделать это с помощью PowerShell 2.0 Register-ObjectEvent, потому что он не работает с событиями, которые требуют возвращаемого значения (т.е. сборка). В этом случае давайте использовать C # через Add-Type, чтобы сделать эту работу за нас. Этот фрагмент кода работает:

ri .\TestClassOne.dll -for -ea 0

$resolver = @'
using System;
using System.Collections.Generic;
using System.IO;
using System.Reflection;
namespace Utils
{
    public static class AssemblyResolver
    {
        private static Dictionary<string, string> _assemblies;

        static AssemblyResolver()
        {
            var comparer = StringComparer.CurrentCultureIgnoreCase;
            _assemblies = new Dictionary<string,string>(comparer);
            AppDomain.CurrentDomain.AssemblyResolve += ResolveHandler;
        }

        public static void AddAssemblyLocation(string path)
        {
            // This should be made threadsafe for production use
            string name = Path.GetFileNameWithoutExtension(path);
            _assemblies.Add(name, path);
        }

        private static Assembly ResolveHandler(object sender, 
                                               ResolveEventArgs args) 
        {
            var assemblyName = new AssemblyName(args.Name);
            if (_assemblies.ContainsKey(assemblyName.Name))
            {
                return Assembly.LoadFrom(_assemblies[assemblyName.Name]);
            }
            return null;
        }
    }
}
'@

Add-Type -TypeDefinition $resolver -Language CSharpVersion3

$code = @'
namespace TEST {
    public class TestClassOne {
        public int DoNothing() {
            return 1;
        }
    }
}
'@
$code | Out-File tcone.cs
Add-Type -OutputAssembly TestClassOne.dll -OutputType Library -Path tcone.cs

# This is the key, register this assembly's location with our resolver utility
[Utils.AssemblyResolver]::AddAssemblyLocation("$pwd\TestClassOne.dll")

Add-Type -Language CSharpVersion3 `
         -ReferencedAssemblies "$pwd\TestClassOne.dll" `
         -TypeDefinition @'
namespace TEST {
    public class TestClassTwo {
        public int CallTestClassOne() {
            var a = new TEST.TestClassOne();
            return a.DoNothing();
        }
    }
}
'@ 

$b = new-object Test.TestClassTwo
$b.CallTestClassOne()
4 голосов
/ 19 апреля 2010

Когда вы выводите TestClassTwo в dll (в том же каталоге, что и TestClassOne) и Add-Type, это работает. Или, по крайней мере, на моей машине;) Так что это уродливый обходной путь.

При вызове $b.CallTestClassOne() PowerShell пытается (по какой-то причине я не знаю) найти сборку TestClassOne.dll в следующих местах:

LOG: Pokus o stažení nové adresy URL file:///C:/Windows/SysWOW64/WindowsPowerShell/v1.0/TestClassOne.DLL
LOG: Pokus o stažení nové adresy URL file:///C:/Windows/SysWOW64/WindowsPowerShell/v1.0/TestClassOne/TestClassOne.DLL
LOG: Pokus o stažení nové adresy URL file:///C:/Windows/SysWOW64/WindowsPowerShell/v1.0/TestClassOne.EXE
LOG: Pokus o stažení nové adresy URL file:///C:/Windows/SysWOW64/WindowsPowerShell/v1.0/TestClassOne/TestClassOne.EXE

Это вывод инструмента fuslogvw. Это может быть полезно для вас. Этот же список путей можно увидеть в реальном времени с помощью ProcessMonitor.

Вы также можете попробовать это (перед вызовом CallTestClassOne()

[appdomain]::CurrentDomain.add_assemblyResolve({
    $global:x = $args
})
$b.CallTestClassOne()
$x | fl

Это покажет вам, какая сборка не удалась, и немного больше информации.

Я согласен, что это должно работать так, как вы ожидаете. Вот почему это выглядит несколько глючно.

...