# frozen_string_literal: true require 'spec_helper' require 'uri' module Analytics class Property include HappyMapper tag 'property' namespace 'dxp' attribute :name, String attribute :value, String end class Goal include HappyMapper # Google Analytics does a dirtry trick where a user with no goals # returns a profile without any goals data or the declared namespace # which means Nokogiri does not pick up the namespace automatically. # To fix this, we manually register the namespace to avoid bad XPath # expression. Dirty, but works. register_namespace 'ga', 'http://schemas.google.com/ga/2009' namespace 'ga' tag 'goal' attribute :active, Boolean attribute :name, String attribute :number, Integer attribute :value, Float def clean_name name.gsub(/ga:/, '') end end class Profile include HappyMapper tag 'entry' element :title, String element :tableId, String, namespace: 'dxp' has_many :properties, Property has_many :goals, Goal end class Entry include HappyMapper tag 'entry' element :id, String element :updated, DateTime element :title, String element :table_id, String, namespace: 'dxp', tag: 'tableId' has_many :properties, Property end class Feed include HappyMapper tag 'feed' element :id, String element :updated, DateTime element :title, String has_many :entries, Entry end end module Atom class Feed include HappyMapper tag 'feed' attribute :xmlns, String, single: true element :id, String, single: true element :title, String, single: true element :updated, DateTime, single: true element :link, String, single: false, attributes: { rel: String, type: String, href: String } # has_many :entries, Entry # nothing interesting in the entries end end class Country include HappyMapper attribute :code, String content :name, String end class State include HappyMapper end class Address include HappyMapper attr_accessor :xml_value attr_accessor :xml_content tag 'address' element :street, String element :postcode, String element :housenumber, String element :city, String has_one :country, Country has_one :state, State end class Feature include HappyMapper element :name, String, xpath: './/text()' end class FeatureBullet include HappyMapper tag 'features_bullets' has_many :features, Feature element :bug, String end class Product include HappyMapper element :title, String has_one :feature_bullets, FeatureBullet has_one :address, Address end class Rate include HappyMapper end module FamilySearch class AlternateIds include HappyMapper tag 'alternateIds' has_many :ids, String, tag: 'id' end class Information include HappyMapper has_one :alternateIds, AlternateIds end class Person include HappyMapper attribute :version, String attribute :modified, Time attribute :id, String has_one :information, Information end class Persons include HappyMapper has_many :person, Person end class FamilyTree include HappyMapper tag 'familytree' attribute :version, String attribute :status_message, String, tag: 'statusMessage' attribute :status_code, String, tag: 'statusCode' has_one :persons, Persons end end module FedEx class Address include HappyMapper tag 'Address' namespace 'v2' element :city, String, tag: 'City' element :state, String, tag: 'StateOrProvinceCode' element :zip, String, tag: 'PostalCode' element :countrycode, String, tag: 'CountryCode' element :residential, Boolean, tag: 'Residential' end class Event include HappyMapper tag 'Events' namespace 'v2' element :timestamp, String, tag: 'Timestamp' element :eventtype, String, tag: 'EventType' element :eventdescription, String, tag: 'EventDescription' has_one :address, Address end class PackageWeight include HappyMapper tag 'PackageWeight' namespace 'v2' element :units, String, tag: 'Units' element :value, Integer, tag: 'Value' end class TrackDetails include HappyMapper tag 'TrackDetails' namespace 'v2' element :tracking_number, String, tag: 'TrackingNumber' element :status_code, String, tag: 'StatusCode' element :status_desc, String, tag: 'StatusDescription' element :carrier_code, String, tag: 'CarrierCode' element :service_info, String, tag: 'ServiceInfo' has_one :weight, PackageWeight, tag: 'PackageWeight' element :est_delivery, String, tag: 'EstimatedDeliveryTimestamp' has_many :events, Event end class Notification include HappyMapper tag 'Notifications' namespace 'v2' element :severity, String, tag: 'Severity' element :source, String, tag: 'Source' element :code, Integer, tag: 'Code' element :message, String, tag: 'Message' element :localized_message, String, tag: 'LocalizedMessage' end class TransactionDetail include HappyMapper tag 'TransactionDetail' namespace 'v2' element :cust_tran_id, String, tag: 'CustomerTransactionId' end class TrackReply include HappyMapper tag 'TrackReply' namespace 'v2' element :highest_severity, String, tag: 'HighestSeverity' element :more_data, Boolean, tag: 'MoreData' has_many :notifications, Notification, tag: 'Notifications' has_many :trackdetails, TrackDetails, tag: 'TrackDetails' has_one :tran_detail, TransactionDetail, tab: 'TransactionDetail' end end class Place include HappyMapper element :name, String end class Radar include HappyMapper has_many :places, Place, tag: :place end class Post include HappyMapper attribute :href, String attribute :hash, String attribute :description, String attribute :tag, String attribute :time, Time attribute :others, Integer attribute :extended, String end class User include HappyMapper element :id, Integer element :name, String element :screen_name, String element :location, String element :description, String element :profile_image_url, String element :url, String element :protected, Boolean element :followers_count, Integer end class Status include HappyMapper register_namespace 'fake', 'faka:namespace' element :id, Integer element :text, String element :created_at, Time element :source, String element :truncated, Boolean element :in_reply_to_status_id, Integer element :in_reply_to_user_id, Integer element :favorited, Boolean element :non_existent, String, tag: 'dummy', namespace: 'fake' has_one :user, User end class CurrentWeather include HappyMapper tag 'ob' namespace 'aws' element :temperature, Integer, tag: 'temp' element :feels_like, Integer, tag: 'feels-like' element :current_condition, String, tag: 'current-condition', attributes: { icon: String } end # for type coercion class ProductGroup < String; end module PITA class Item include HappyMapper tag 'Item' # if you put class in module you need tag element :asin, String, tag: 'ASIN' element :detail_page_url, URI, tag: 'DetailPageURL', parser: :parse element :manufacturer, String, tag: 'Manufacturer', deep: true element :point, String, tag: 'point', namespace: 'georss' element :product_group, ProductGroup, tag: 'ProductGroup', deep: true, parser: :new, raw: true end class Items include HappyMapper tag 'Items' # if you put class in module you need tag element :total_results, Integer, tag: 'TotalResults' element :total_pages, Integer, tag: 'TotalPages' has_many :items, Item end end module GitHub class Commit include HappyMapper tag 'commit' element :url, String element :tree, String element :message, String element :id, String element :'committed-date', Date end end module QuarterTest class Quarter include HappyMapper element :start, String end class Details include HappyMapper element :round, Integer element :quarter, Integer end class Game include HappyMapper # in an ideal world, the following elements would all be # called 'quarter' with an attribute indicating which quarter # it represented, but the refactoring that allows a single class # to be used for all these differently named elements is the next # best thing has_one :details, QuarterTest::Details has_one :q1, QuarterTest::Quarter, tag: 'q1' has_one :q2, QuarterTest::Quarter, tag: 'q2' has_one :q3, QuarterTest::Quarter, tag: 'q3' has_one :q4, QuarterTest::Quarter, tag: 'q4' end end # To check for multiple primitives class Artist include HappyMapper tag 'artist' element :images, String, tag: 'image', single: false element :name, String end class Location include HappyMapper tag 'point' namespace 'geo' element :latitude, String, tag: 'lat' end # Testing the XmlContent type module Dictionary class Variant include HappyMapper tag 'var' has_xml_content def to_html xml_content.gsub('', '').gsub('', '') end end class Definition include HappyMapper tag 'def' element :text, XmlContent, tag: 'dtext' end class Record include HappyMapper tag 'record' has_many :definitions, Definition has_many :variants, Variant, tag: 'var' end end module AmbigousItems class Item include HappyMapper tag 'item' element :name, String element :item, String end end class PublishOptions include HappyMapper tag 'publishOptions' element :author, String, tag: 'author' element :draft, Boolean, tag: 'draft' element :scheduled_day, String, tag: 'scheduledDay' element :scheduled_time, String, tag: 'scheduledTime' element :published_day, String, tag: 'publishDisplayDay' element :published_time, String, tag: 'publishDisplayTime' element :created_day, String, tag: 'publishDisplayDay' element :created_time, String, tag: 'publishDisplayTime' end class Article include HappyMapper tag 'Article' namespace 'article' attr_writer :xml_value element :title, String element :text, String has_many :photos, 'Photo', tag: 'Photo', namespace: 'photo', xpath: '/article:Article' has_many :galleries, 'Gallery', tag: 'Gallery', namespace: 'gallery' element :publish_options, PublishOptions, tag: 'publishOptions', namespace: 'article' end class PartiallyBadArticle include HappyMapper attr_writer :xml_value tag 'Article' namespace 'article' element :title, String element :text, String has_many :photos, 'Photo', tag: 'Photo', namespace: 'photo', xpath: '/article:Article' has_many :videos, 'Video', tag: 'Video', namespace: 'video' element :publish_options, PublishOptions, tag: 'publishOptions', namespace: 'article' end class Photo include HappyMapper tag 'Photo' namespace 'photo' attr_writer :xml_value element :title, String element :publish_options, PublishOptions, tag: 'publishOptions', namespace: 'photo' end class Gallery include HappyMapper tag 'Gallery' namespace 'gallery' attr_writer :xml_value element :title, String end class Video include HappyMapper tag 'Video' namespace 'video' attr_writer :xml_value element :title, String element :publish_options, PublishOptions, tag: 'publishOptions', namespace: 'video' end class OptionalAttribute include HappyMapper tag 'address' attribute :street, String end class DefaultNamespaceCombi include HappyMapper register_namespace 'bk', 'urn:loc.gov:books' register_namespace 'isbn', 'urn:ISBN:0-395-36341-6' register_namespace 'p', 'urn:loc.gov:people' namespace 'bk' tag 'book' element :title, String, namespace: 'bk', tag: 'title' element :number, String, namespace: 'isbn', tag: 'number' element :author, String, namespace: 'p', tag: 'author' end describe HappyMapper do describe 'being included into another class' do let(:klass) do Class.new do include HappyMapper def self.name 'Boo' end end end it 'sets attributes to an array' do expect(klass.attributes).to eq([]) end it 'sets @elements to a hash' do expect(klass.elements).to eq([]) end it 'allows adding an attribute' do expect do klass.attribute :name, String end.to change(klass, :attributes) end it 'allows adding an attribute containing a dash' do expect do klass.attribute :'bar-baz', String end.to change(klass, :attributes) end it 'is able to get all attributes in array' do klass.attribute :name, String expect(klass.attributes.size).to eq(1) end it 'allows adding an element' do expect do klass.element :name, String end.to change(klass, :elements) end it 'allows adding an element containing a dash' do expect do klass.element :'bar-baz', String end.to change(klass, :elements) end it 'is able to get all elements in array' do klass.element(:name, String) expect(klass.elements.size).to eq(1) end it 'allows has one association' do klass.has_one(:user, User) element = klass.elements.first expect(element.name).to eq('user') expect(element.type).to eq(User) expect(element.options[:single]).to eq(true) end it 'allows has many association' do klass.has_many(:users, User) element = klass.elements.first expect(element.name).to eq('users') expect(element.type).to eq(User) expect(element.options[:single]).to eq(false) end it 'defaults tag name to lowercase class' do expect(klass.tag_name).to eq('boo') end it 'generates no tag name for anonymous class' do @anon = Class.new { include HappyMapper } expect(@anon.tag_name).to be_nil end it 'defaults tag name of class in modules to the last constant lowercase' do module Bar; class Baz; include HappyMapper; end; end expect(Bar::Baz.tag_name).to eq('baz') end it 'allows setting tag name' do klass.tag('FooBar') expect(klass.tag_name).to eq('FooBar') end it 'allows setting a namespace' do klass.namespace(namespace = 'boo') expect(klass.namespace).to eq(namespace) end it 'provides #parse' do expect(klass).to respond_to(:parse) end end describe '#attributes' do it 'onlies return attributes for the current class' do expect(Post.attributes.size).to eq(7) expect(Status.attributes.size).to eq(0) end end describe '#elements' do it 'onlies return elements for the current class' do expect(Post.elements.size).to eq(0) expect(Status.elements.size).to eq(10) end end describe '#content' do it 'takes String as default argument for type' do State.content :name address = Address.parse(fixture_file('address.xml')) expect(address.state.name).to eq('Lower Saxony') address.state.name.class == String end it 'works when specific type is provided' do Rate.content :value, Float Product.has_one :rate, Rate product = Product.parse(fixture_file('product_default_namespace.xml'), single: true) expect(product.rate.value).to eq(120.25) product.rate.class == Float end end it 'parses xml attributes into ruby objects' do posts = Post.parse(fixture_file('posts.xml')) expect(posts.size).to eq(20) first = posts.first expect(first.href).to eq('http://roxml.rubyforge.org/') expect(first.hash).to eq('19bba2ab667be03a19f67fb67dc56917') expect(first.description).to eq('ROXML - Ruby Object to XML Mapping Library') expect(first.tag).to eq('ruby xml gems mapping') expect(first.time).to eq(Time.utc(2008, 8, 9, 5, 24, 20)) expect(first.others).to eq(56) expect(first.extended). to eq('ROXML is a Ruby library designed to make it easier for Ruby' \ ' developers to work with XML. Using simple annotations, it enables' \ ' Ruby classes to be custom-mapped to XML. ROXML takes care of the' \ ' marshalling and unmarshalling of mapped attributes so that developers can' \ ' focus on building first-class Ruby classes.') end it 'parses xml elements to ruby objcts' do statuses = Status.parse(fixture_file('statuses.xml')) expect(statuses.size).to eq(20) first = statuses.first expect(first.id).to eq(882_281_424) expect(first.created_at).to eq(Time.utc(2008, 8, 9, 5, 38, 12)) expect(first.source).to eq('web') expect(first.truncated).to be_falsey expect(first.in_reply_to_status_id).to eq(1234) expect(first.in_reply_to_user_id).to eq(12_345) expect(first.favorited).to be_falsey expect(first.user.id).to eq(4243) expect(first.user.name).to eq('John Nunemaker') expect(first.user.screen_name).to eq('jnunemaker') expect(first.user.location).to eq('Mishawaka, IN, US') expect(first.user.description).to eq('Loves his wife, ruby, notre dame football and iu basketball') expect(first.user.profile_image_url). to eq('http://s3.amazonaws.com/twitter_production/profile_images/53781608/Photo_75_normal.jpg') expect(first.user.url).to eq('http://addictedtonew.com') expect(first.user.protected).to be_falsey expect(first.user.followers_count).to eq(486) end it 'parses xml containing the desired element as root node' do address = Address.parse(fixture_file('address.xml'), single: true) expect(address.street).to eq('Milchstrasse') expect(address.postcode).to eq('26131') expect(address.housenumber).to eq('23') expect(address.city).to eq('Oldenburg') expect(address.country.class).to eq(Country) end it 'parses text node correctly' do address = Address.parse(fixture_file('address.xml'), single: true) expect(address.country.name).to eq('Germany') expect(address.country.code).to eq('de') end it 'treats Nokogiri::XML::Document as root' do doc = Nokogiri::XML(fixture_file('address.xml')) address = Address.parse(doc) expect(address.class).to eq(Address) end it 'parses xml with default namespace (amazon)' do file_contents = fixture_file('pita.xml') items = PITA::Items.parse(file_contents, single: true) expect(items.total_results).to eq(22) expect(items.total_pages).to eq(3) first = items.items[0] second = items.items[1] expect(first.asin).to eq('0321480791') expect(first.point).to eq('38.5351715088 -121.7948684692') expect(first.detail_page_url).to be_a_kind_of(URI) expect(first.detail_page_url.to_s).to eq('http://www.amazon.com/gp/redirect.html%3FASIN=0321480791%26tag=ws%26lcode=xm2%26cID=2025%26ccmID=165953%26location=/o/ASIN/0321480791%253FSubscriptionId=dontbeaswoosh') expect(first.manufacturer).to eq('Addison-Wesley Professional') expect(first.product_group).to eq('Book') expect(second.asin).to eq('047022388X') expect(second.manufacturer).to eq('Wrox') end it 'parses xml that has attributes of elements' do items = CurrentWeather.parse(fixture_file('current_weather.xml')) first = items[0] expect(first.temperature).to eq(51) expect(first.feels_like).to eq(51) expect(first.current_condition).to eq('Sunny') expect(first.current_condition.icon).to eq('http://deskwx.weatherbug.com/images/Forecast/icons/cond007.gif') end it "parses xml with attributes of elements that aren't :single => true" do feed = Atom::Feed.parse(fixture_file('atom.xml')) expect(feed.link.first.href).to eq('http://www.example.com') expect(feed.link.last.href).to eq('http://www.example.com/tv_shows.atom') end it 'parses xml with optional elements with embedded attributes' do expect { CurrentWeather.parse(fixture_file('current_weather_missing_elements.xml')) }.not_to raise_error end it 'returns nil rather than empty array for absent values when :single => true' do address = Address.parse('', single: true) expect(address).to be_nil end it 'returns same result for absent values when :single => true, regardless of :in_groups_of' do addr1 = Address.parse('', single: true) addr2 = Address.parse('', single: true, in_groups_of: 10) expect(addr1).to eq(addr2) end it 'parses xml with nested elements' do radars = Radar.parse(fixture_file('radar.xml')) first = radars[0] expect(first.places.size).to eq(1) expect(first.places[0].name).to eq('Store') second = radars[1] expect(second.places.size).to eq(0) third = radars[2] expect(third.places.size).to eq(2) expect(third.places[0].name).to eq('Work') expect(third.places[1].name).to eq('Home') end it 'parses xml with element name different to class name' do game = QuarterTest::Game.parse(fixture_file('quarters.xml')) expect(game.q1.start).to eq('4:40:15 PM') expect(game.q2.start).to eq('5:18:53 PM') end it 'parses xml that has elements with dashes' do commit = GitHub::Commit.parse(fixture_file('commit.xml')) expect(commit.message).to eq('move commands.rb and helpers.rb into commands/ dir') expect(commit.url).to eq('http://github.com/defunkt/github-gem/commit/c26d4ce9807ecf57d3f9eefe19ae64e75bcaaa8b') expect(commit.id).to eq('c26d4ce9807ecf57d3f9eefe19ae64e75bcaaa8b') expect(commit.committed_date).to eq(Date.parse('2008-03-02T16:45:41-08:00')) expect(commit.tree).to eq('28a1a1ca3e663d35ba8bf07d3f1781af71359b76') end it 'parses xml with no namespace' do product = Product.parse(fixture_file('product_no_namespace.xml'), single: true) expect(product.title).to eq('A Title') expect(product.feature_bullets.bug).to eq('This is a bug') expect(product.feature_bullets.features.size).to eq(2) expect(product.feature_bullets.features[0].name).to eq('This is feature text 1') expect(product.feature_bullets.features[1].name).to eq('This is feature text 2') end it 'parses xml with default namespace' do product = Product.parse(fixture_file('product_default_namespace.xml'), single: true) expect(product.title).to eq('A Title') expect(product.feature_bullets.bug).to eq('This is a bug') expect(product.feature_bullets.features.size).to eq(2) expect(product.feature_bullets.features[0].name).to eq('This is feature text 1') expect(product.feature_bullets.features[1].name).to eq('This is feature text 2') end it 'parses xml with single namespace' do product = Product.parse(fixture_file('product_single_namespace.xml'), single: true) expect(product.title).to eq('A Title') expect(product.feature_bullets.bug).to eq('This is a bug') expect(product.feature_bullets.features.size).to eq(2) expect(product.feature_bullets.features[0].name).to eq('This is feature text 1') expect(product.feature_bullets.features[1].name).to eq('This is feature text 2') end it 'parses xml with multiple namespaces' do track = FedEx::TrackReply.parse(fixture_file('multiple_namespaces.xml')) expect(track.highest_severity).to eq('SUCCESS') expect(track.more_data).to be_falsey notification = track.notifications.first expect(notification.code).to eq(0) expect(notification.localized_message).to eq('Request was successfully processed.') expect(notification.message).to eq('Request was successfully processed.') expect(notification.severity).to eq('SUCCESS') expect(notification.source).to eq('trck') detail = track.trackdetails.first expect(detail.carrier_code).to eq('FDXG') expect(detail.est_delivery).to eq('2009-01-02T00:00:00') expect(detail.service_info).to eq('Ground-Package Returns Program-Domestic') expect(detail.status_code).to eq('OD') expect(detail.status_desc).to eq('On FedEx vehicle for delivery') expect(detail.tracking_number).to eq('9611018034267800045212') expect(detail.weight.units).to eq('LB') expect(detail.weight.value).to eq(2) events = detail.events expect(events.size).to eq(10) first_event = events[0] expect(first_event.eventdescription).to eq('On FedEx vehicle for delivery') expect(first_event.eventtype).to eq('OD') expect(first_event.timestamp).to eq('2009-01-02T06:00:00') expect(first_event.address.city).to eq('WICHITA') expect(first_event.address.countrycode).to eq('US') expect(first_event.address.residential).to be_falsey expect(first_event.address.state).to eq('KS') expect(first_event.address.zip).to eq('67226') last_event = events[-1] expect(last_event.eventdescription).to eq('In FedEx possession') expect(last_event.eventtype).to eq('IP') expect(last_event.timestamp).to eq('2008-12-27T09:40:00') expect(last_event.address.city).to eq('LONGWOOD') expect(last_event.address.countrycode).to eq('US') expect(last_event.address.residential).to be_falsey expect(last_event.address.state).to eq('FL') expect(last_event.address.zip).to eq('327506398') expect(track.tran_detail.cust_tran_id).to eq('20090102-111321') end it 'is able to parse google analytics api xml' do data = Analytics::Feed.parse(fixture_file('analytics.xml')) expect(data.id).to eq('http://www.google.com/analytics/feeds/accounts/nunemaker@gmail.com') expect(data.entries.size).to eq(4) entry = data.entries[0] expect(entry.title).to eq('addictedtonew.com') expect(entry.properties.size).to eq(4) property = entry.properties[0] expect(property.name).to eq('ga:accountId') expect(property.value).to eq('85301') end it 'is able to parse google analytics profile xml with manually declared namespace' do data = Analytics::Profile.parse(fixture_file('analytics_profile.xml')) expect(data.entries.size).to eq(6) entry = data.entries[0] expect(entry.title).to eq('www.homedepot.com') expect(entry.properties.size).to eq(6) expect(entry.goals.size).to eq(0) end it 'allows instantiating with a string' do module StringFoo class Bar include HappyMapper has_many :things, 'StringFoo::Thing' end class Thing include HappyMapper end end end it 'parses family search xml' do tree = FamilySearch::FamilyTree.parse(fixture_file('family_tree.xml')) expect(tree.version).to eq('1.0.20071213.942') expect(tree.status_message).to eq('OK') expect(tree.status_code).to eq('200') expect(tree.persons.person.size).to eq(1) expect(tree.persons.person.first.version).to eq('1199378491000') expect(tree.persons.person.first.modified). to eq(Time.utc(2008, 1, 3, 16, 41, 31)) # 2008-01-03T09:41:31-07:00 expect(tree.persons.person.first.id).to eq('KWQS-BBQ') expect(tree.persons.person.first.information.alternateIds.ids).not_to be_kind_of(String) expect(tree.persons.person.first.information.alternateIds.ids.size).to eq(8) end it 'parses multiple images' do artist = Artist.parse(fixture_file('multiple_primitives.xml')) expect(artist.name).to eq('value') expect(artist.images.size).to eq(2) end it 'parses lastfm namespaces' do l = Location.parse(fixture_file('lastfm.xml')) expect(l.first.latitude).to eq('51.53469') end describe 'Parse optional attributes' do let(:parsed_result) { OptionalAttribute.parse(fixture_file('optional_attributes.xml')) } it 'parses an empty String as empty' do expect(parsed_result[0].street).to eq('') end it 'parses a String with value' do expect(parsed_result[1].street).to eq('Milchstrasse') end it 'parses an element with no value for the attribute' do expect(parsed_result[2].street).to be_nil end end describe 'Default namespace combi' do let(:file_contents) { fixture_file('default_namespace_combi.xml') } let(:book) { DefaultNamespaceCombi.parse(file_contents, single: true) } it 'parses author' do expect(book.author).to eq('Frank Gilbreth') end it 'parses title' do expect(book.title).to eq('Cheaper by the Dozen') end it 'parses number' do expect(book.number).to eq('1568491379') end end describe 'Xml Content' do let(:records) { Dictionary::Record.parse fixture_file('dictionary.xml') } it 'parses XmlContent' do expect(records.first.definitions.first.text). to eq('a large common parrot, Cacatua galerita, predominantly' \ ' white, with yellow on the undersides of wings and tail and a' \ ' forward curving yellow crest, found in Australia, New Guinea' \ ' and nearby islands.') end it "saves object's xml content" do expect(records.first.variants.first.xml_content).to eq( 'white cockatoo' ) expect(records.first.variants.last.to_html).to eq( 'white cockatoo' ) end end it 'parses ambigous items' do items = AmbigousItems::Item.parse(fixture_file('ambigous_items.xml'), xpath: '/ambigous/my-items') expect(items.map(&:name)).to eq(%w(first second third).map { |s| "My #{s} item" }) end context Article do let(:article) { Article.parse(fixture_file('subclass_namespace.xml')) } it 'parses the publish options for Article and Photo' do expect(article.title).not_to be_nil expect(article.text).not_to be_nil expect(article.photos).not_to be_nil expect(article.photos.first.title).not_to be_nil end it 'parses the publish options for Article' do expect(article.publish_options).not_to be_nil end it 'parses the publish options for Photo' do expect(article.photos.first.publish_options).not_to be_nil end it 'onlies find only items at the parent level' do expect(article.photos.length).to eq(1) end end describe 'Namespace is missing because an optional element that uses it is not present' do it 'parses successfully' do article = PartiallyBadArticle.parse(fixture_file('subclass_namespace.xml')) aggregate_failures do expect(article).not_to be_nil expect(article.title).not_to be_nil expect(article.text).not_to be_nil expect(article.photos).not_to be_nil expect(article.photos.first.title).not_to be_nil end end end describe 'with limit option' do it 'returns results with limited size: 6' do sizes = [] Post.parse(fixture_file('posts.xml'), in_groups_of: 6) do |a| sizes << a.size end expect(sizes).to eq([6, 6, 6, 2]) end it 'returns results with limited size: 10' do sizes = [] Post.parse(fixture_file('posts.xml'), in_groups_of: 10) do |a| sizes << a.size end expect(sizes).to eq([10, 10]) end end context 'when letting user set Nokogiri::XML::ParseOptions' do let(:default) do Class.new do include HappyMapper element :item, String end end let(:custom) do Class.new do include HappyMapper element :item, String with_nokogiri_config(&:default_xml) end end it 'initializes @nokogiri_config_callback to nil' do expect(default.nokogiri_config_callback).to be_nil end it 'defaults to Nokogiri::XML::ParseOptions::STRICT' do expect { default.parse(fixture_file('set_config_options.xml')) }. to raise_error(Nokogiri::XML::SyntaxError) end it 'accepts .on_config callback' do expect(custom.nokogiri_config_callback).not_to be_nil end it 'parses according to @nokogiri_config_callback' do expect { custom.parse(fixture_file('set_config_options.xml')) }.not_to raise_error end it 'can clear @nokogiri_config_callback' do custom.with_nokogiri_config {} expect { custom.parse(fixture_file('set_config_options.xml')) }. to raise_error(Nokogiri::XML::SyntaxError) end end describe '#xml_value' do it 'does not reformat the xml' do xml = fixture_file('unformatted_address.xml') address = Address.parse(xml, single: true) expect(address.xml_value). to eq %(
Milchstrasse23
) end end describe '#xml_content' do it 'does not reformat the xml' do xml = fixture_file('unformatted_address.xml') address = Address.parse(xml) expect(address.xml_content).to eq %(Milchstrasse23) end end describe '#to_xml' do let(:original) { 'baz' } let(:parsed) { described_class.parse original } it 'has UTF-8 encoding by default' do expect(original.encoding).to eq Encoding::UTF_8 expect(parsed.to_xml.encoding).to eq Encoding::UTF_8 end end end