How can I get PowerShell Added-Types to use Added Types

前端 未结 2 340
予麋鹿
予麋鹿 2021-01-03 05:35

I\'m working on a PoSh project that generates CSharp code, and then Add-Types it into memory.

The new types use existing types in an on disk DLL, which

相关标签:
2条回答
  • 2021-01-03 05:40

    When you output the TestClassTwo to a dll (in the same directory as TestClassOne) and Add-Type it, it works. Or at least at my machine ;) So that's the ugly workaround.

    When calling $b.CallTestClassOne() PowerShell tries (from some reason I don't know) to find assembly TestClassOne.dll at these locations:

    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
    

    This is output from fuslogvw tool. It might be useful for you. The same list of paths can bee seen live using ProcessMonitor.

    You might also try this (before calling CallTestClassOne()

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

    This will show you what assembly failed and some more info.

    I agree that it should work as you expect. So that's why this looks somewhat buggy.

    0 讨论(0)
  • 2021-01-03 05:48

    This happens because any assemblies are looked for by the CLR loader in the application's (PowerShell's) base directory. Of course, it doesn't find your assembly there. The best way to solve this is to hook the AssemblyResolve event as stej mentions but use it to tell the CLR where the assembly is. You can't do this with PowerShell 2.0's Register-ObjectEvent because it doesn't work with events that require a return value (ie the assembly). In this case, let's use more C# via Add-Type to do this work for us. This snippet of code works:

    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()
    
    0 讨论(0)
提交回复
热议问题