1

我在非标准路径下有一个模块(MyModule),即不在$env:PSModulePath -split ";". 但是,我已将MyModule的“生产”路径添加到该环境变量中,同时继续处理“开发”副本。

在尝试调试某些东西时,我$VerbosePreference = "Continue"使用以下命令加载了模块(使用 ),并立即看到了两条看似矛盾的 Verbose 输出行:

[D:\Dev\UserA\]> Import-Module D:\Dev\UserA\libs\PowerShell\MyModule

VERBOSE: Loading module from path 'D:\Dev\UserA\libs\PowerShell\MyModule\MyModule.psd1'.
VERBOSE: Loading module from path 'D:\Dev\usera\MyModule2\MyModule.psm1'.

我想了解为什么 Import-Module 似乎两次加载模块,尤其是在第二条路径不正确的情况下。


更多详情:

该模块的文件夹结构为:

MyModule\MyModule.psd1
MyModule\MyModule.Test-Module.xml
MyModule\MyModule1\MyModule.psm1
MyModule\MyModule2\MyModule.psm1

注意 (1) 我在 MyModule1 子文件夹中保留了此模块的旧“版本 1”,并将更新后的“版本 2”文件放在 MyModule2 子文件夹中,并且 (2) .xml 文件由自定义使用列出测试用例的模块测试脚本。我很确定后者可以忽略不计。

我的模块清单 (.psd1) 文件包含以下内容,所有其他行均为空白或注释:

@{
  RootModule = '.\MyModule2\MyModule.psm1'
  ModuleVersion = '2.0.0.0'
  GUID = 'aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee'
  Author = 'Old Developer (v1.x) & New Developer (v2.x)'
  CompanyName = 'MyCompany'
  Copyright = '(c) 2013-2015 MyCompany. All rights reserved.'
  Description = 'Really useful functions'
  FileList = @(
    '.\MyModule.psd1'
    '.\MyModule.Test-Module.xml'
    '.\MyModule1\MyModule.psm1'
    '.\MyModule2\MyModule.psm1'
    '.\MyModule2\Examples\Archive-FilesWithCompression.ps1'
  )
}

显然我已经使用了文件的相对路径,尤其是。RootModule;这是必要的,因为我无法确定共享模块时将复制到哪里。

回到 Verbose 输出,我可以看到两行显示 (1) PSD1 文件的正确路径和 (2) PSM1 文件的无效路径。我确实注意到第二个路径的用户名是小写的,这就是我Set-Location在测试之前碰巧输入它的方式。因此,看起来第一个路径是通过将 MyModule.psd1 附加到给定Import-Modulecmdlet 的路径而采用的,第二个是 (Get-Location) 和 RootModule 路径的串联。

这似乎只发生在这个模块上。我在同一个“根”文件夹下有其他人没有表现出这种行为。

4

1 回答 1

0

嗯。好的。我可能已经解决了,至少部分...

我不小心在运行 Import-Module cmdlet 的位置下复制了MyModule2子文件夹。一旦我删除了那个文件夹,详细的输出就开始变得更有意义了:

[D:\Dev\UserA\]> Import-Module D:\Dev\UserA\libs\PowerShell\MyModule

VERBOSE: Loading module from path 'D:\Dev\UserA\libs\PowerShell\MyModule\MyModule.psd1'.
VERBOSE: Loading module from path 'D:\Dev\UserA\libs\PowerShell\MyModule\.\MyModule2\MyModule.psm1'.
VERBOSE: Exporting function '<function>'.
....

我猜这意味着当 PowerShell 解析清单文件中的RootModule时,它​​首先会在当前路径下查找,如果没有找到,则在主模块文件夹下查找。我发现这违反直觉,因为我希望清单中的任何相对路径总是相对于 PSD1 文件,而不是当前位置。

如果我尝试立即再次导入模块,我会得到:

[D:\Dev\UserA\]> Import-Module D:\Dev\UserA\libs\PowerShell\MyModule

VERBOSE: Loading module from path 'D:\Dev\UserA\libs\PowerShell\MyModule\MyModule.psd1'.
VERBOSE: Importing function '<function>'.
....

也就是说,只有一行“加载模块”,而不是两行详细输出。然后我尝试了-Force开关并得到以下信息:

[D:\Dev\UserA\]> Import-Module D:\Dev\UserA\libs\PowerShell\MyModule -Force

VERBOSE: Loading module from path 'D:\Dev\UserA\libs\PowerShell\MyModule\MyModule.psd1'.
VERBOSE: Removing the imported "<function>" function.
....
VERBOSE: Loading module from path 'D:\Dev\UserA\libs\PowerShell\MyModule\.\MyModule2\MyModule.psm1'.
VERBOSE: Exporting function '<function>'.
....
VERBOSE: Importing function '<function>'.
....

因此,似乎在模块首次导入或强制重新导入时出现了两行详细输出,但如果它已经是会话的一部分,那么您只会看到第一行。

此外,我终于发现详细输出在导出函数和导入函数之间存在差异。这些函数按它们在 PSM1 文件中定义的顺序导出,但按字母顺序导入。这表明使用一阶段或两阶段过程取决于是否从 PSM1 文件中重新读取函数定义,这与我们是否看到一两行详细输出有关。

或者,为了更直接地回答这个问题,当它需要读取 .psm1 文件时,似乎有两行说“正在加载模块...”,而当 PowerShell 已经知道模块及其内容时只有一行。

于 2016-01-21T12:25:44.403 回答