Manish Kumar
Manish Kumar

Reputation: 25

Need Help to Insert Child Node in XML FILE

Need help in PowerShell to correct the XML file. The Node I want to insert here is not the first child of the XML file. Structure of Current File - This is the current file and I need help on this

<?xml version="1.0" encoding="UTF-8" standalone="yes" ?>
<Shipment xmlns:xsi="http://www.w3.org/2001/xmlschema-instance" >
    <ShipmentNumber>0012546776</ShipmentNumber>
    <Container>Selleys/Yates</Container>
    <Extnl_id/>
    <AddressForwardingAgent>
        <PartnerNumber>0000400260</PartnerNumber>
        <LanguageKey>EN</LanguageKey>
        <Name>TEST</Name>
        <HouseandStreet>TEST1</HouseandStreet>
        </AddressForwardingAgent>
    <AddressSender>
        <TransportPlanningPoint>4001</TransportPlanningPoint>
        <LanguageKey>EN</LanguageKey>
        <Name>TEST2</Name>
        <HouseandStreet>TEST2</HouseandStreet>
    </AddressSender>
    <Consignment>
        <ConsignmentNumber>0001</ConsignmentNumber>
        <ConsignmentHeader>
            <DangerousGoodsIndicator>X</DangerousGoodsIndicator>
            <GrossWeight>2005.268</GrossWeight>
            <WeightUOM>KG</WeightUOM>
            <TotalVolume>3.856</TotalVolume>
            <VolumeUOM>M3</VolumeUOM>
        </ConsignmentHeader>
        <ConsignmentNumber>0002</ConsignmentNumber>
        <ConsignmentHeader>
            <DangerousGoodsIndicator>X</DangerousGoodsIndicator>
            <GrossWeight>21.12</GrossWeight>
            <WeightUOM>KG</WeightUOM>
            <TotalVolume>0.044</TotalVolume>
            <VolumeUOM>M3</VolumeUOM>
        </ConsignmentHeader>
        <ConsignmentNumber>0003</ConsignmentNumber>
        <ConsignmentHeader>
            <DangerousGoodsIndicator>Y</DangerousGoodsIndicator>
            <GrossWeight>12.45</GrossWeight>
            <WeightUOM>KG</WeightUOM>
            <TotalVolume>0.056</TotalVolume>
            <VolumeUOM>P3</VolumeUOM>
        </ConsignmentHeader>
    </Consignment>

Expected Output - This kind of structure I want in XML by using PowerShell.

<?xml version="1.0" encoding="UTF-8" standalone="yes" ?>
<Shipment xmlns:xsi="http://www.w3.org/2001/xmlschema-instance" >
    <ShipmentNumber>0012546776</ShipmentNumber>
    <Container>Selleys/Yates</Container>
    <Extnl_id/>
    <AddressForwardingAgent>
        <PartnerNumber>0000400260</PartnerNumber>
        <LanguageKey>EN</LanguageKey>
        <Name>TEST</Name>
        <HouseandStreet>TEST1</HouseandStreet>
        </AddressForwardingAgent>
    <AddressSender>
        <TransportPlanningPoint>4001</TransportPlanningPoint>
        <LanguageKey>EN</LanguageKey>
        <Name>TEST2</Name>
        <HouseandStreet>TEST2</HouseandStreet>
    </AddressSender>
    <Consignment>
    <ConsigmentLine>                 ## Need to insert this tag here
        <ConsignmentNumber>0001</ConsignmentNumber>
        <ConsignmentHeader>
            <DangerousGoodsIndicator>X</DangerousGoodsIndicator>
            <GrossWeight>2005.268</GrossWeight>
            <WeightUOM>KG</WeightUOM>
            <TotalVolume>3.856</TotalVolume>
            <VolumeUOM>M3</VolumeUOM>
        </ConsignmentHeader>
    </ConsigmentLine>
    <ConsigmentLine>    ## Need to insert this tag here
        <ConsignmentNumber>0002</ConsignmentNumber>
        <ConsignmentHeader>
            <DangerousGoodsIndicator>X</DangerousGoodsIndicator>
            <GrossWeight>21.12</GrossWeight>
            <WeightUOM>KG</WeightUOM>
            <TotalVolume>0.044</TotalVolume>
            <VolumeUOM>M3</VolumeUOM>
        </ConsignmentHeader>
    </ConsigmentLine>
    <ConsigmentLine>    ## Need to insert this tag here
        <ConsignmentNumber>0003</ConsignmentNumber>
        <ConsignmentHeader>
            <DangerousGoodsIndicator>Y</DangerousGoodsIndicator>
            <GrossWeight>12.45</GrossWeight>
            <WeightUOM>KG</WeightUOM>
            <TotalVolume>0.056</TotalVolume>
            <VolumeUOM>P3</VolumeUOM>
        </ConsignmentHeader>
    </ConsigmentLine>
    </Consignment>

Upvotes: 1

Views: 87

Answers (2)

RoadRunner
RoadRunner

Reputation: 26335

You can also try the following approach. I've added comments to explain the solution. Also added the missing </Shipment> closing tag at the end of your XML file.

# Function to zip two arrays together using .NET LINQ
function Select-Zip {
    [CmdletBinding()]
    Param(
        $First,
        $Second,
        $ResultSelector = { ,$args }
    )

    [System.Linq.Enumerable]::Zip($First, $Second, [Func[Object, Object, Object[]]]$ResultSelector)
}

# Create XML object to load data into
$xml = New-Object -TypeName System.Xml.XmlDocument

# Load in XML file
$xml.Load("test.xml")

# Get root node where we make the changes
$consignmentRootNode = $xml.Shipment.Consignment

# Get both number and header nodes
$numberNodes = $consignmentRootNode.ConsignmentNumber
$headerNodes = $consignmentRootNode.ConsignmentHeader

# Zip above nodes to form number & header pairs
$zippedNodes = Select-Zip -First $numberNodes -Second $headerNodes

# Remove consignment root node
$xml.Shipment.RemoveChild($consignmentRootNode)

# Rebuild new consignment node 
$newConsignmentRootNode = $xml.CreateElement("Consignment")

# Iterate each pair
foreach ($pair in $zippedNodes)
{
    # Create new line node
    $consignmentLineNode = $xml.CreateElement("ConsigmentLine")

    # Build and append number node
    # Need to create a new one because number node is stored as a string
    $consignmentNumberNode = $xml.CreateElement("ConsignmentNumber")
    $consignmentNumberNode.InnerText = $pair[0]
    $consignmentLineNode.AppendChild($consignmentNumberNode)

    # Append header node
    $consignmentLineNode.AppendChild($pair[1])
    $newConsignmentRootNode.AppendChild($consignmentLineNode)
}

# Add rebuilt consignment node to shipment node
$xml.Shipment.AppendChild($newConsignmentRootNode)

# Save output to new file
# Don't want to corrupt original
$xml.Save("output.xml")

Upvotes: 0

Theo
Theo

Reputation: 61218

You need to capture the 'ConsignmentNumber' and 'ConsignmentHeader' nodes first, remove the 'Consignment' node from the xml and rebuild that.

For demo, I'm using a Here-String, but you'd probably need to read it from file using [xml]$xml = Get-Content -Path 'D:\MyCurrentXml.xml'

[xml]$xml = @"
<?xml version="1.0" encoding="UTF-8" standalone="yes" ?>
<Shipment xmlns:xsi="http://www.w3.org/2001/xmlschema-instance" >
    <ShipmentNumber>0012546776</ShipmentNumber>
    <Container>Selleys/Yates</Container>
    <Extnl_id/>
    <AddressForwardingAgent>
        <PartnerNumber>0000400260</PartnerNumber>
        <LanguageKey>EN</LanguageKey>
        <Name>TEST</Name>
        <HouseandStreet>TEST1</HouseandStreet>
        </AddressForwardingAgent>
    <AddressSender>
        <TransportPlanningPoint>4001</TransportPlanningPoint>
        <LanguageKey>EN</LanguageKey>
        <Name>TEST2</Name>
        <HouseandStreet>TEST2</HouseandStreet>
    </AddressSender>
    <Consignment>
        <ConsignmentNumber>0001</ConsignmentNumber>
        <ConsignmentHeader>
            <DangerousGoodsIndicator>X</DangerousGoodsIndicator>
            <GrossWeight>2005.268</GrossWeight>
            <WeightUOM>KG</WeightUOM>
            <TotalVolume>3.856</TotalVolume>
            <VolumeUOM>M3</VolumeUOM>
        </ConsignmentHeader>
        <ConsignmentNumber>0002</ConsignmentNumber>
        <ConsignmentHeader>
            <DangerousGoodsIndicator>X</DangerousGoodsIndicator>
            <GrossWeight>21.12</GrossWeight>
            <WeightUOM>KG</WeightUOM>
            <TotalVolume>0.044</TotalVolume>
            <VolumeUOM>M3</VolumeUOM>
        </ConsignmentHeader>
        <ConsignmentNumber>0003</ConsignmentNumber>
        <ConsignmentHeader>
            <DangerousGoodsIndicator>Y</DangerousGoodsIndicator>
            <GrossWeight>12.45</GrossWeight>
            <WeightUOM>KG</WeightUOM>
            <TotalVolume>0.056</TotalVolume>
            <VolumeUOM>P3</VolumeUOM>
        </ConsignmentHeader>
    </Consignment>
</Shipment>
"@

To rebuild this the way you want, this should work:

# capture two arrays of nodes for 'ConsignmentNumber' and 'ConsignmentHeader'
$numbers = $xml.Shipment.Consignment.ChildNodes | Where-Object { $_.LocalName -eq 'ConsignmentNumber' }
$headers = $xml.Shipment.Consignment.ChildNodes | Where-Object { $_.LocalName -eq 'ConsignmentHeader' }

# remove the entire 'Consignment' node from the xml
$xml.Shipment.RemoveChild($xml.Shipment.Consignment)
# and create a new 'Consignment' node
$newConsignment = $xml.CreateElement('Consignment')

# loop through the arrays, create a new 'ConsigmentLine' node and add the captured nodes to it
for ($i = 0; $i -lt $numbers.Count; $i++) {
    $lineNode = $xml.CreateElement('ConsigmentLine')
    $lineNode.AppendChild($numbers[$i])
    $lineNode.AppendChild($headers[$i])
    # append this to the newly created 'Consignment' node
    $newConsignment.AppendChild($lineNode)
}
# finally, append the new 'Consignment' node to the 'Shipment' node
$xml.Shipment.AppendChild($newConsignment)

# and save the xml
$xml.Save('D:\MyCorrectedXml.xml')

Result:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<Shipment xmlns:xsi="http://www.w3.org/2001/xmlschema-instance">
  <ShipmentNumber>0012546776</ShipmentNumber>
  <Container>Selleys/Yates</Container>
  <Extnl_id />
  <AddressForwardingAgent>
    <PartnerNumber>0000400260</PartnerNumber>
    <LanguageKey>EN</LanguageKey>
    <Name>TEST</Name>
    <HouseandStreet>TEST1</HouseandStreet>
  </AddressForwardingAgent>
  <AddressSender>
    <TransportPlanningPoint>4001</TransportPlanningPoint>
    <LanguageKey>EN</LanguageKey>
    <Name>TEST2</Name>
    <HouseandStreet>TEST2</HouseandStreet>
  </AddressSender>
  <Consignment>
    <ConsigmentLine>
      <ConsignmentNumber>0001</ConsignmentNumber>
      <ConsignmentHeader>
        <DangerousGoodsIndicator>X</DangerousGoodsIndicator>
        <GrossWeight>2005.268</GrossWeight>
        <WeightUOM>KG</WeightUOM>
        <TotalVolume>3.856</TotalVolume>
        <VolumeUOM>M3</VolumeUOM>
      </ConsignmentHeader>
    </ConsigmentLine>
    <ConsigmentLine>
      <ConsignmentNumber>0002</ConsignmentNumber>
      <ConsignmentHeader>
        <DangerousGoodsIndicator>X</DangerousGoodsIndicator>
        <GrossWeight>21.12</GrossWeight>
        <WeightUOM>KG</WeightUOM>
        <TotalVolume>0.044</TotalVolume>
        <VolumeUOM>M3</VolumeUOM>
      </ConsignmentHeader>
    </ConsigmentLine>
    <ConsigmentLine>
      <ConsignmentNumber>0003</ConsignmentNumber>
      <ConsignmentHeader>
        <DangerousGoodsIndicator>Y</DangerousGoodsIndicator>
        <GrossWeight>12.45</GrossWeight>
        <WeightUOM>KG</WeightUOM>
        <TotalVolume>0.056</TotalVolume>
        <VolumeUOM>P3</VolumeUOM>
      </ConsignmentHeader>
    </ConsigmentLine>
  </Consignment>
</Shipment>

P.S. your example xml was missing the final closing </Shipment> tag

Upvotes: 2

Related Questions