2

我正在尝试使用 WMI 查询多个服务器,但我并不总是可以访问这些服务器。

代码如下。唉,它向控制台返回“访问被拒绝”,但我似乎无法摆脱它。那好吧。

但是,我正在捕获无法连接的服务器,以便我可以告诉其他人查看它们或请求访问。

但是当我运行代码时,它只返回第一个服务器列表;即使 $failed_servers 有值,也不会返回任何内容。如果我告诉两者都通过管道连接到 ogv,则会弹出两个窗口。

为什么两个“$variable|select”都不起作用?如果我删除 $failed_servers 上的选择,它就会显示出来,尽管它只是紧挨着成功的那些。这是好的,但不是很好。

$list = ("servera","serverb","serverc")
$failed_servers = @()
$final = foreach ($server_instance in $list)
{
$errors=@()
gwmi -query "select * from win32_service where name like '%SQLSERVER%'" -cn $server_instance -ErrorVariable +errors -ErrorAction SilentlyContinue
if ($errors.Count -gt 0) {$failed_servers += $server_instance
}
}

$final|select pscomputername, name, startmode, state |where {$_.pscomputername -ne $null}

$failed_servers |select @{N='Failed Servers'; E={$_}}
4

1 回答 1

3

您遇到的只是显示问题

  • 您的两个调用都会生成具有或更少属性Select-Object的输出对象,其类型没有与之关联的显式格式数据(如 报告)。4Get-FormatData

  • 这会导致 PowerShell 的显示输出格式化系统通过Format-Tablecmdlet 隐式呈现它们。

  • 使用的显示列根据接收到的第一个对象Format-Table属性锁定Format-Table

  • 因此,您的第二个 Select-Object调用(其输出对象与第一个输出的对象不共享任何属性)实际上不会产生可见的输出- 但是,这些对象发送到成功输出流并且用于编程处理

一个简单的演示:

& {
  # This locks in Month and Year as the display columns of the output table.
  Get-Date   | Select-Object Month, Year
  # This command's output will effectively be invisible,
  # because the property set Name, Attributes does not overlap with
  # Month, Year
  Get-Item \ | Select-Object Name, Attributes
}

输出看起来像这样 - 请注意第二条语句的输出实际上是如何不可见的(保存一个额外的空白行):

Month Year
----- ----
    9 2021


请注意,该问题甚至会影响输出不同类型对象的单个语句(其类型没有关联的格式化数据);例如:
(Get-Date | Select-Object Year), (Get-Item \ | Select-Object Name)


解决方法:

  • 应用| Format-List上面的命令会使所有对象都可见,但显然会改变显示格式。

  • 脚本内您可以通过管道将每个Select-Object管道传递Out-Host到以强制即时、特定于管道的格式化,但是 - 鉴于结果直接发送到主机而不是成功输出流 - 这种技术排除了进一步的编程处理


潜在的未来改进:

GitHub 问题 #7871建议如果输出对象实际上变得不可见,至少会发出警告。

于 2021-09-16T00:58:23.720 回答