将 XML 元素添加到 Nokogiri::XML::Builder 文档

Adding a XML Element to a Nokogiri::XML::Builder document

如何将 Nokogiri::XML::Element 添加到使用 Nokogiri::XML::Buider 创建的 XML 文档?

我目前的解决方案是序列化元素并使用 << 方法让 Builder 重新解释它。

orig_doc = Nokogiri::XML('<root xmlns="foobar"><a>test</a></root>')
node = orig_doc.at('/*/*[1]')

puts Nokogiri::XML::Builder.new do |doc|
    doc.another {
        # FIXME: this is the round-trip I would like to avoid
        xml_text = node.to_xml(:skip_instruct => true).to_s
        doc << xml_text

        doc.second("hi")
    }
end.to_xml

# The expected result is
#
# <another>
#    <a xmlns="foobar">test</a>
#    <second>hi</second>
# </another>

然而Nokogiri::XML::Element是一个相当大的节点(以千字节和数千个节点为单位)并且这段代码在热路径中。分析显示 serialization/parsing 往返非常昂贵。

如何指示 Nokogiri Builder 在 "current" 位置添加现有 XML 元素 node

查看 Nokogiri 源代码后,我发现了这个脆弱的解决方案:使用受保护的 #insert(node) 方法。

修改为使用该私有方法的代码如下所示:

doc.another {
    xml_text = node.to_xml(:skip_instruct => true).to_s
    doc.send('insert', xml_text) # <= use `#insert` instead of `<<`

    doc.second("hi")
}

在不使用私有方法的情况下,您可以使用 the parent method of the Builder 实例获取当前父元素的句柄。然后您可以向其附加一个元素(甚至来自另一个文档)。例如:

require 'nokogiri'
doc1 = Nokogiri.XML('<r><a>success!</a></r>')
a = doc1.at('a')

# note that `xml` is not a Nokogiri::XML::Document,
#  but rather a Nokogiri::XML::Builder instance.
doc2 = Nokogiri::XML::Builder.new do |xml|
  xml.some do
    xml.more do
      xml.parent << a
    end
  end
end.doc

puts doc2
#=> <?xml version="1.0"?>
#=> <some>
#=>   <more>
#=>     <a>success!</a>
#=>   </more>
#=> </some>