This document gives you some exposure to the methods provided by the OM::XML::Document module and its related modules OM::XML::TermXPathGenerator & OM::XML::TermValueOperators
Note: In your code, don’t worry about including OM::XML::TermXPathGenerator and OM::XML::TermValueOperators into your classes. OM::XML::Document handles that for you.
These examples use the Document class defined in OM::Samples::ModsArticle
Download hydrangea_article1.xml into your working directory, then run this:
require "om/samples" sample_xml = File.new("hydrangea_article1.xml") doc = OM::Samples::ModsArticle.from_xml(sample_xml)
The OM Terminology declared by OM::Samples::ModsArticle handles generating xpath queries based on the structures you’ve defined. It will also run the queries for you in most cases. If you’re ever curious what the xpath queries are, or if you want to use them in some other way, they are a few keystrokes away.
Here are the xpaths for :name and two variants of :name that were created using the :ref argument in the Terminology builder.
OM::Samples::ModsArticle.terminology.xpath_for(:name) => "//oxns:name" OM::Samples::ModsArticle.terminology.xpath_for(:person) => "//oxns:name[@type=\"personal\"]"
To retrieve the values of xml nodes, use the term_values method
doc.term_values(:person, :first_name) doc.term_values(:person, :last_name)
If the xpath points to XML nodes that contain other nodes, the response to term_values will contain Nokogiri::XML::Node objects instead of text values.
doc.term_values(:name)
More examples of using term_values and find_by_terms:
doc.find_by_terms(:subject).to_xml doc.term_values(:subject, :topic)
You will often string together a series of term names to point to what you want
OM::Samples::ModsArticle.terminology.xpath_for(:journal, :issue, :pages, :start) => "//oxns:relatedItem[@type=\"host\"]/oxns:part/oxns:extent[@unit=\"pages\"]/oxns:start" doc.term_values(:journal, :issue, :pages, :start) => ["195"]
If you get one of the names wrong in the list, OM will tell you which one is causing problems. See what happens when you put :page instead of :pages in your argument to term_values.
doc.term_values(:journal, :issue, :page, :start) OM::XML::Terminology::BadPointerError: You attempted to retrieve a Term using this pointer: [:journal, :issue, :page] but no Term exists at that location. Everything is fine until ":page", which doesn't exist.
If you use a term often and you’re sick of typing all of those term names, you can define a proxy term. Here we have a proxy term called :start_page that saves you from having to remember the details of how MODS is structured.
OM::Samples::ModsArticle.terminology.xpath_for(:journal, :issue, :start_page) => "//oxns:relatedItem[@type=\"host\"]/oxns:part/oxns:extent[@unit=\"pages\"]/oxns:start"
In our MODS document, we have two types of title: 1) the title of the published article and 2) the title of the journal it was published in. They both use the same xml node. How can we deal with that?
doc.term_values(:title_info, :main_title) => ["ARTICLE TITLE HYDRANGEA ARTICLE 1", "Artikkelin otsikko Hydrangea artiklan 1", "TITLE OF HOST JOURNAL"] doc.term_values(:mods, :title_info, :main_title) => ["ARTICLE TITLE HYDRANGEA ARTICLE 1", "Artikkelin otsikko Hydrangea artiklan 1"] OM::Samples::ModsArticle.terminology.xpath_for(:title_info, :main_title) => "//oxns:titleInfo/oxns:title"
The solution: include the root node in your term pointer.
OM::Samples::ModsArticle.terminology.xpath_for(:mods, :title_info, :main_title) => "//oxns:mods/oxns:titleInfo/oxns:title" doc.term_values(:mods, :title_info, :main_title) => ["ARTICLE TITLE HYDRANGEA ARTICLE 1", "Artikkelin otsikko Hydrangea artiklan 1"]
We can still access the Journal title by its own pointers:
doc.term_values(:journal, :title_info, :main_title) => ["TITLE OF HOST JOURNAL"]
Sometimes all of these terms become tedious. That’s where proxy terms come in. You can use them to access frequently used Terms more easily. As you can see in OM::Samples::ModsArticle, we have defined a few proxy terms for convenience.
t.start_page(:proxy=>[:pages, :start]) t.end_page(:proxy=>[:pages, :end])
You can use them just like any other Term when querying the document.
OM::Samples::ModsArticle.terminology.xpath_for(:journal, :issue, :start_page) => "//oxns:relatedItem[@type=\"host\"]/oxns:part/oxns:extent[@unit=\"pages\"]/oxns:start" OM::Samples::ModsArticle.terminology.xpath_for(:journal, :issue, :end_page) => "//oxns:relatedItem[@type=\"host\"]/oxns:part/oxns:extent[@unit=\"pages\"]/oxns:end"