How to add a new node to XML
Asked Answered
A

2

8

I have a simple XML file, items.xml:

 <?xml version="1.0" encoding="UTF-8" ?>

<items>
  <item>
    <name>mouse</name>
    <manufacturer>Logicteh</manufacturer>
  </item>
  <item>
    <name>keyboard</name>
    <manufacturer>Logitech - Inc.</manufacturer>
  </item>
  <item>
    <name>webcam</name>
    <manufacturer>Logistech</manufacturer>
  </item>
</items>

I am trying to insert a new node with the following code:

require 'rubygems'
require 'nokogiri'

f = File.open('items.xml')
@items = Nokogiri::XML(f)
f.close

price = Nokogiri::XML::Node.new "price", @items
price.content = "10"

@items.xpath('//items/item/manufacturer').each do |node|
  node.add_next_sibling(price)
end

file = File.open("items_fixed.xml",'w')
file.puts @items.to_xml
file.close

However this code adds a new node only after the last <manufacturer> node, items_fixed.xml:

<?xml version="1.0" encoding="UTF-8"?>
<items>
  <item>
    <name>mouse</name>
    <manufacturer>Logitech</manufacturer>
  </item>
  <item>
    <name>keyboard</name>
    <manufacturer>Logitech</manufacturer>
  </item>
  <item>
    <name>webcam</name>
    <manufacturer>Logitech</manufacturer><price>10</price>
  </item>
</items>

Why?

Absorption answered 15/3, 2011 at 18:11 Comment(1)
You make it way too difficult. Nokogiri makes it easy to add nodes using a string, which it then parses into a new node. See my answer below.Hawking
S
13

It would be helpful to distinguish between a Node (a particular piece of structured XML data at a particular place in a tree), and a "node template" which is the structure of the data.

Nokogiri (and most other XML libraries) only allow you to specify Nodes, not node templates. So when you created price = Nokogiri::XML::Node.new "price", @items, you had a particular piece of data that belongs in a particular place, but hadn't defined the place yet.

When you added it to the first <item>, you defined its place. When you added it to the second <item>, you uprooted it from its place and put it in a new place. At that point this Node appeared only in the second <item>. This continues when you add the same Node to each item, until you reach the last <item>, which is where the node stays.

Nokogiri doesn't have any way to specify a node template. What you need to do is:

@items.xpath('//items/item/manufacturer').each do |node|
  price = Nokogiri::XML::Node.new "price", @items
  price.content = "10"
  node.add_next_sibling(price)
end
Scottie answered 15/3, 2011 at 18:22 Comment(4)
I saw the exact behavour you explained in debugger but couldn't understand it. Thank you for your explanation and your answer!Absorption
Alternatively, you may want to simply use Node#dup to clone the node before inserting it in a new location.Metamathematics
Nice explanation.. I was having the same issue since last 3-4 days back. Now I found out why it was happening. Thanks for the OP and you.Robinetta
It's easier than this. See my example.Hawking
H
2

I'd start with this:

require 'nokogiri'

doc = Nokogiri::XML(<<EOT)
<?xml version="1.0" encoding="UTF-8"?>
<items>
  <item>
    <name>mouse</name>
    <manufacturer>Logitech</manufacturer>
  </item>
  <item>
    <name>keyboard</name>
    <manufacturer>Logitech - Inc.</manufacturer>
  </item>
</items>
EOT

doc.search('manufacturer').each { |n| n.after('<price>10</price>') }

Which results in:

puts doc.to_xml
# >> <?xml version="1.0" encoding="UTF-8"?>
# >> <items>
# >>   <item>
# >>     <name>mouse</name>
# >>     <manufacturer>Logitech</manufacturer><price>10</price>
# >>   </item>
# >>   <item>
# >>     <name>keyboard</name>
# >>     <manufacturer>Logitech - Inc.</manufacturer><price>10</price>
# >>   </item>
# >> </items>

It's easy to build upon this to insert different values for the price.

Hawking answered 19/8, 2016 at 21:18 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.