由于某种原因,如果我的XML文件只包含一个计算机节点,则以下Read-TestControllerXml powershell函数返回System.Xml.XmlElement类型的对象,但它返回一个列表(这是我一直期望的)如果有的话是多个计算机节点。
XML文件的示例如下:
<test_controller>
<defaults>
<nunit path='C:\Program Files (x86)\NUnit 2.6.2\bin\nunit-console.exe' />
<nunit_results local_path='C:\Test_Results' remote_path='C:\Test_Results' />
<powershell path='C:\windows\system32\WindowsPowerShell\v1.0\powershell.exe' />
<psexec path='C:\PSTools\PsExec.exe' />
</defaults>
<platform os='windows' version='2012' cpu='x64'>
<computer hostname='VM-2012R2-03' alias='standalone-01' configuration='standalone' image='VM-2012R2-03:Clean' />
</platform>
</test_controller>
这是powershell函数:
####################################################################
# Parses the TestController.xml file and returns a list of Computer XML objects.
#
# Syntax:
# Read-TestControllerXml [-TestControllerXml] <path to TestController.xml> [<CommonParameters>]
#
# Returns: An array of [Xml] nodes for each of the <computer> tags in TestController.xml.
####################################################################
function Read-TestControllerXml
{
[CmdletBinding()]
Param
(
[parameter(Mandatory=$true)]
[string] $TestControllerXml
)
$ErrorActionPreference = "Stop"
# Read the TestController XML configuration file
$TestController = [xml] (Get-Content $TestControllerXml)
Write-Verbose "Selecting computer nodes..."
$computerNodes = $TestController.SelectNodes("//test_controller/platform/computer")
Write-Verbose "Selecting default nodes..."
$defaultNodes = $TestController.SelectSingleNode("//test_controller/defaults")
$computers = @()
foreach ($computerNode in $computerNodes)
{
Write-Verbose "***** Before adding default attributes *****"
Write-Verbose "$($computerNode.OuterXml)"
# Add default attributes to the node
foreach ($dnode in $defaultNodes.ChildNodes)
{
if ($dnode.NodeType -eq "Comment") { continue } # Skip comments.
# Search for the node items in the defaults section
$cnode = $computerNode
if ($dnode.Name -ne $computerNode.Name)
{
Write-Verbose "Selecting '$($dnode.Name)' nodes..."
$cnode = $computerNode.SelectSingleNode($dnode.Name)
}
# Append the default nodes that do not exist in the computer specific sections
if ($cnode -eq $null)
{
Write-Verbose "*** cnode != null cnode.OuterXml is: '$($cnode.OuterXml)'"
Write-Verbose "Appending default '$($dnode.Name)' node that don't exist in the computer nodes..."
$cnode=$computerNode.AppendChild($TestController.ImportNode($dnode, $true))
}
# Append the default attributes that do not exist in the computer specific sections
else
{
Write-Verbose "Appending default attributes that don't exist in computer nodes..."
foreach ($attribute in $dnode.Attributes)
{
if (-not $cnode.HasAttribute($attribute.Name))
{
Write-Verbose "Adding attribute '$($attribute.Name)=$($attribute.Value)' to the computer node."
$cnode.SetAttribute($attribute.Name, $attribute.Value)
}
}
}
}
Write-Verbose "***** After adding default attributes *****"
Write-Verbose "$($computerNode.OuterXml)"
Write-Verbose "************************************************************"
$computers += $computerNode
}
return $computers
}
答案 0 :(得分:1)
我拿了你的例子函数和样本数据。运行该函数我得到你描述的输出。
PS C:\Users\Matt> (Read-TestControllerXml C:\temp\xml.txt).GetType().FullName
System.Xml.XmlElement
然后我将一个计算机节点添加到示例文件并再次运行该函数。
PS C:\Users\Matt> (Read-TestControllerXml C:\temp\xml.txt).GetType().FullName
System.Object[]
这是正确的,因为你的函数返回和对象数组。如果您分解项目并检查其类型,它应该是有道理的。
PS C:\Users\Matt> Read-TestControllerXml C:\temp\xml.txt | ForEach-Object{$_.GetType().FullName}
System.Xml.XmlElement
System.Xml.XmlElement
我有两个项目并将它们放入ForEach-Object循环并单独检查其类型。由于您的输出是一个数组,期望有多个对象。总之,只要有多个[System.Xml.XmlElement]
数组,您的函数就会返回System.Object[]
。
出于好奇的缘故
我试图强制$ computers成为像这个$computers = [System.Xml.XmlElement[]]@()
的xml元素数组。它没有改变输出。多个项目仍作为System.Object[]
答案 1 :(得分:0)
我刚刚发现了这个类似的问题,其解决方案有效: Function not returning expected object
即。变化:
return $computers
为:
return ,$computers