XMLからHashへの変換とその逆変換を考えます。ここでは2つの方法を取り上げますが、結論としてはXmlSimpleを使うほうがよさそうです。

サンプルXML

XMLのサンプルとしてはてなブログのAPIで返ってくるXMLを使います。入っている値は実際とは異なっていますがフォーマットは同じです。

1. ActiveSupportを使う

Railsに一緒に入っているActiveSupportを使ってXML文字列をRubyのHashに変換できるようになります。

require "active_support/core_ext/hash/conversions"により

  • Hash.from_xml
  • Hash#to_xml

これらのメソッドが使えるようになります。

 1require "active_support/core_ext/hash/conversions"
 2xml = %Q{
 3<?xml version="1.0" encoding="utf-8"?>
 4<feed xmlns="http://www.w3.org/2005/Atom"
 5      xmlns:app="http://www.w3.org/2007/app">
 6  ...
 7</feed>
 8}
 9
10hash = Hash.from_xml(xml)
11xml2 = hash.to_xml(hash)

上のように使い方は簡単ですが結果に癖があります。

上のhash変数を見てみるとわかるのですが、XMLのattributeが潰れることもあれば残っている場合もあり得ます。

 1{"feed"=>
 2  {"xmlns"=>"http://www.w3.org/2005/Atom",
 3   "xmlns:app"=>"http://www.w3.org/2007/app",
 4   "link"=>
 5    [{"rel"=>"first",
 6      "href"=>
 7       "https://blog.hatena.ne.jp/hatena_id/blog_id.hatenablog.com/atom/entry"},
 8     {"rel"=>"next",
 9      "href"=>
10       "https://blog.hatena.ne.jp/hatena_id/blog_id.hatenablog.com/atom/entry?page=2"},
11     {"rel"=>"alternate", "href"=>"http://blog_id.hatenablog.com/"}],
12   "title"=>"ブログタイトル",
13   "updated"=>"2013-08-27T15:17:06+09:00",
14   "author"=>{"name"=>"hatena_id"},
15   "generator"=>"Hatena::Blog",
16   "id"=>"hatenablog://blog/2000000000000",
17   "entry"=>
18    {"id"=>
19      "tag:blog.hatena.ne.jp,2013:blog-hatena_id-20000000000000-3000000000000000",
20     "link"=>
21      [{"rel"=>"edit",
22        "href"=>
23         "https://blog.hatena.ne.jp/hatena_id/blog_id.hatenablog.com/atom/edit/2500000000"},
24       {"rel"=>"alternate",
25        "type"=>"text/html",
26        "href"=>"http://blog_id.hatenablog.com/entry/2013/09/02/112823"}],
27     "author"=>{"name"=>"hatena_id"},
28     "title"=>"記事タイトル",
29     "updated"=>"2013-09-02T11:28:23+09:00",
30     "published"=>"2013-09-02T11:28:23+09:00",
31     "edited"=>"2013-09-02T11:28:23+09:00",
32     "summary"=>"hoge ",
33     "content"=>"\n      hoge\n    ",
34     "formatted_content"=>"\n      hoge\n    ",
35     "control"=>{"draft"=>"yes"}}}}

ケース1

この部分(タイトル部分を除く)が

1<link rel="first" href="https://blog.hatena.ne.jp/hatena_id/blog_id.hatenablog.com/atom/entry" />
2<link rel="next" href="https://blog.hatena.ne.jp/hatena_id/blog_id.hatenablog.com/atom/entry?page=2" />
3<title>ブログタイトル</title>
4<link rel="alternate" href="http://blog_id.hatenablog.com/"/>

こうなっています。

1"link"=>
2    [{"rel"=>"first",
3      "href"=>
4       "https://blog.hatena.ne.jp/hatena_id/blog_id.hatenablog.com/atom/entry"},
5     {"rel"=>"next",
6      "href"=>
7       "https://blog.hatena.ne.jp/hatena_id/blog_id.hatenablog.com/atom/entry?page=2"},
8     {"rel"=>"alternate", "href"=>"http://blog_id.hatenablog.com/"}]

このケースではrelhrefが残っています。

ケース2

それに対してこの部分では、、、

1<content type="text/x-hatena-syntax">
2  hoge
3</content>

次のようになってしまいます。typeattributeが残っていません。同じように<generator>タグでもattributeが消えてしまっていることがわかります。

1"content"=>"\n      hoge\n    "

ActiveSupportを使う場合には注意が必要と言えそうです。

2. XmlSimpleを使う

xmlsimpleというgemで似たようなことができます。attributeもちゃんと残せます。

1$ gem install xml-simple
1require "xmlsimple"
2
3hash = XmlSimple.xml_in(xml, options) # XML文字列をHashに変換
4xml = XmlSimple.xml_out(hash, options) # HashをXMLに変換

オプションが豊富で全てを説明しきれないので詳細はこちらで→XmlSimple - XML made easy

1xml = # さっきと一緒の文字列
2hash = XmlSimple.xml_in(xml, ContentKey:"__content__")
3xml2 = XmlSimple.xml_out(hash, ContentKey:"__content__", RootName:"feed")

ここで使っているオプションは次の通り。

  • ContentKey: 値を表すキーでデフォルト値はcontentです。元々のXMLにあるcontentタグと重複するのでそれとは異なるキーを指定します。
  • RootName: HashをXMLに変換する際のRootノードの名前はデフォルトでは<opt>となっています。元のXMLに合わせるためにfeedを指定しています。

xml_inで変換したHashはこうなりました。長くてすいません。ActiveSupportのときに潰れていたような<content type=""><generator uri="">などがちゃんと残っています。

また、要素が1つであろうが複数であろうが配列になっているので、単一の値なのか配列なのかを区別してプログラムを書く必要がありません。

ActiveSupportに依存する必要もなくなるのでこちらがオススメかと。

 1{"xmlns"=>"http://www.w3.org/2005/Atom",
 2 "xmlns:app"=>"http://www.w3.org/2007/app",
 3 "link"=>
 4  [{"rel"=>"first",
 5    "href"=>
 6     "https://blog.hatena.ne.jp/hatena_id/blog_id.hatenablog.com/atom/entry"},
 7   {"rel"=>"next",
 8    "href"=>
 9     "https://blog.hatena.ne.jp/hatena_id/blog_id.hatenablog.com/atom/entry?page=2"},
10   {"rel"=>"alternate", "href"=>"http://blog_id.hatenablog.com/"}],
11 "title"=>["ブログタイトル"],
12 "updated"=>["2013-08-27T15:17:06+09:00"],
13 "author"=>[{"name"=>["hatena_id"]}],
14 "generator"=>
15  [{"uri"=>"http://blog.hatena.ne.jp/",
16    "version"=>"100000000",
17    "__content__"=>"Hatena::Blog"}],
18 "id"=>["hatenablog://blog/2000000000000"],
19 "entry"=>
20  [{"id"=>
21     ["tag:blog.hatena.ne.jp,2013:blog-hatena_id-20000000000000-3000000000000000"],
22    "link"=>
23     [{"rel"=>"edit",
24       "href"=>
25        "https://blog.hatena.ne.jp/hatena_id/blog_id.hatenablog.com/atom/edit/2500000000"},
26      {"rel"=>"alternate",
27       "type"=>"text/html",
28       "href"=>"http://blog_id.hatenablog.com/entry/2013/09/02/112823"}],
29    "author"=>[{"name"=>["hatena_id"]}],
30    "title"=>["記事タイトル"],
31    "updated"=>["2013-09-02T11:28:23+09:00"],
32    "published"=>["2013-09-02T11:28:23+09:00"],
33    "edited"=>["2013-09-02T11:28:23+09:00"],
34    "summary"=>[{"type"=>"text", "__content__"=>"hoge "}],
35    "content"=>
36     [{"type"=>"text/x-hatena-syntax", "__content__"=>"\n      hoge\n    "}],
37    "formatted-content"=>
38     [{"type"=>"text/html",
39       "xmlns:hatena"=>"http://www.hatena.ne.jp/info/xmlns#",
40       "__content__"=>"\n      hoge\n    "}],
41    "control"=>[{"draft"=>["yes"]}]}]}