Calling AppDomain.DoCallback from Powershell

后端 未结 2 1439
暗喜
暗喜 2021-01-20 02:10

This is based on the Stack Overflow question: How to load an assembly as reflection-only in a new AppDomain?

I am attempting to determine the runtime version of an a

2条回答
  •  执念已碎
    2021-01-20 02:49

    First thought: don't muck around with AppDomains at all and use a completely separate process. Those are (relatively) easily launched from PowerShell, at least. The drawback is that it's potentially much slower if you're doing this for lots of files.

    $myAssemblyPath = "C:\..." 
    $getImageRuntimeVersion = {
        [Reflection.Assembly]::ReflectionOnlyLoadFrom($input).ImageRuntimeVersion
    }
    $encodedCommand = [Convert]::ToBase64String(
        [Text.Encoding]::Unicode.GetBytes($getImageRuntimeVersion)
    )
    $imageRuntimeVersion = $myAssemblyPath | powershell -EncodedCommand $encodedCommand
    

    So, is there no way at all to do this with AppDomains in PowerShell? Well, there is, but it's not pretty. You can't use AppDomain.DoCallBack because, as you've discovered, PowerShell can't remote delegates that way (because, under the covers, it produces dynamic methods).

    However, it's easy to host the PowerShell runtime, and all PowerShell objects know how to serialize (a requirement for cross-domain remoting), so invoking a PowerShell script in another AppDomain is fairly simple (but still ugly):

    $scriptInvokerAssembly = [System.IO.Path]::GetTempFileName() + ".dll"
    Add-Type -OutputAssembly $tempAssembly -TypeDefinition @"
      using System;
      using System.Reflection;
      using System.Collections.Generic;
      using System.Management.Automation;
    
      public class ScriptInvoker : MarshalByRefObject {
        public IEnumerable Invoke(ScriptBlock scriptBlock, PSObject[] parameters) {
          using (var powerShell = PowerShell.Create()) {
            powerShell.Commands.AddScript(scriptBlock.ToString());
            if (parameters != null) {
              powerShell.AddParameters(parameters);
            }
            return powerShell.Invoke();
          }
        }
      }
    "@
    [Reflection.Assembly]::LoadFile($scriptInvokerAssembly) | Out-Null
    
    Function Invoke-CommandInTemporaryAppDomain([ScriptBlock] $s, [object[]] $arguments) {
      $setup = New-Object System.AppDomainSetup
      $setup.ApplicationBase = Split-Path ([ScriptInvoker].Assembly.Location) -Parent
      $domain = [AppDomain]::CreateDomain([Guid]::NewGuid(), $null, $setup)
      $scriptInvoker = $domain.CreateInstanceAndUnwrap(
         [ScriptInvoker].Assembly.FullName, [ScriptInvoker]
      );
      $scriptInvoker.Invoke($s, $arguments)
      [AppDomain]::Unload($domain)
    }
    

    And now you can do

    Invoke-CommandInTemporaryAppDomain { 
      [Reflection.Assembly]::ReflectionOnlyLoadFrom($args[0]).ImageRuntimeVersion 
    } $myAssemblyPath
    

    Note that we have to generate a temporary assembly on disk and have AppDomain load it from there. This is ugly, but you can't have Add-Type produce an in-memory assembly, and even if you do end up with a byte[] getting that to load in another AppDomain is anything but trivial because you can't hook AppDomain.AssemblyResolve in PowerShell. If this command was packaged in a module, you'd compile the assembly containing the ScriptInvoker ahead of time, so I don't see working around this as a priority.

提交回复
热议问题