从[Reflection.Assembly] :: Load()加载的DLL中导出Powershell CMDlet。

时间:2019-04-18 13:09:43

标签: powershell dll .net-assembly gac cmdlet

我有一个无法将文件写入磁盘的环境,因此通常我会从GAC的内存中加载DLL。

虽然加载不是问题,但公开DLL Commandlets是一个问题。 程序集已加载:

[appdomain]::currentdomain.getassemblies() | sort -property fullname | format-table fullname

FullName
--------
AudioDeviceCmdlets, Version=3.0.0.4, Culture=neutral, PublicKeyToken=null

fl *

CodeBase            : file:///C:/WINDOWS/Microsoft.Net/assembly/GAC_64/mscorlib/v4.0_4.0.0.0__b77a5c561934e089/mscorlib.dll
FullName            : AudioDeviceCmdlets, Version=3.0.0.4, Culture=neutral, PublicKeyToken=null
EntryPoint          :
DefinedTypes        : {CoreAudioApi.AudioEndpointVolume, CoreAudioApi.AudioEndpointVolumeCallback, CoreAudioApi.AudioEndpointVolumeChannel, CoreAudioApi.AudioEndpointVolumeChannels...}
Evidence            : {<System.Security.Policy.Url version="1">
                      <Url>file:///C:/WINDOWS/Microsoft.Net/assembly/GAC_64/mscorlib/v4.0_4.0.0.0__b77a5c561934e089/mscorlib.dll</Url>
                      </System.Security.Policy.Url>
                      , <System.Security.Policy.Zone version="1">
                      <Zone>MyComputer</Zone>
                      </System.Security.Policy.Zone>
                      }
PermissionSet       : {}
SecurityRuleSet     : Level2
ManifestModule      : AudioDeviceCmdlets.dll
ReflectionOnly      : False
Location            :
ImageRuntimeVersion : v4.0.30319
GlobalAssemblyCache : False
HostContext         : 0
IsDynamic           : False
EscapedCodeBase     : file:///C:/WINDOWS/Microsoft.Net/assembly/GAC_64/mscorlib/v4.0_4.0.0.0__b77a5c561934e089/mscorlib.dll
ExportedTypes       : {CoreAudioApi.AudioEndpointVolume, CoreAudioApi.AudioEndpointVolumeChannel, CoreAudioApi.AudioEndpointVolumeChannels, CoreAudioApi.AudioEndpointVolumeNotificationDelegate...}
IsFullyTrusted      : True
CustomAttributes    : {[System.Runtime.CompilerServices.CompilationRelaxationsAttribute((Int32)8)], [System.Runtime.CompilerServices.RuntimeCompatibilityAttribute(WrapNonExceptionThrows = True)],
                      [System.Diagnostics.DebuggableAttribute((System.Diagnostics.DebuggableAttribute+DebuggingModes)2)], [System.Reflection.AssemblyTitleAttribute("AudioDeviceCmdlets")]...}
Modules             : {<unknown>}

..但是未导出CMDLets。.当然,我肯定缺少某些内容。

相同的DLL,以“常规”方式加载:

load-module AudioDeviceCmdlets.dll

正确导出几个CmdLets,例如 Get-AudioDevice Set-AudioDevice

关于在dll中公开Cmdlet缺少哪些步骤的任何提示?

2 个答案:

答案 0 :(得分:1)

使用Import-Module -Assembly

$assembly = [Reflection.Assembly]::Load($UncompressedFileBytes)    
Import-Module -Assembly $assembly

或者,创建一个模块清单来加载程序集,然后将其导入:

New-ModuleManifest .\AudioDeviceCmdlets.psd1 -RequiredAssemblies AudioDeviceCmdlets
Import-Module .\AudioDeviceCmdlets.psd1

答案 1 :(得分:1)

虽然看起来很不礼貌,但 (而且我仍然愿意寻求更好的答案!) 这有效:

1)选择可用程序集的第一个实例(因为可能有多个实例,或者在查找要加载的实例时更为具体):

$Assembly=([System.AppDomain]::CurrentDomain.GetAssemblies()|? FullName -Match "AudioDeviceCmdlets")[0]

2)使用Import-Module导入程序集,但不指定要加载的dll,而是直接引用程序集。 没有清单,它将默认导出其中的所有变量和所有函数

Import-Module -Assembly $Assembly

Voila,现在可以正确导出Commandlet了! :)