require 'test_helper'
class InlineTest < MiniTest::Spec
let(:song) { Song.new("Alive") }
let(:request) { representer.prepare(OpenStruct.new(:song => song)) }
{
:hash => [Representable::Hash, {"song"=>{"name"=>"Alive"}}, {"song"=>{"name"=>"You've Taken Everything"}}],
:json => [Representable::JSON, "{\"song\":{\"name\":\"Alive\"}}", "{\"song\":{\"name\":\"You've Taken Everything\"}}"],
:xml => [Representable::XML, "\n \n Alive\n \n", "You've Taken Everything/open_struct>"],
:yaml => [Representable::YAML, "---\nsong:\n name: Alive\n", "---\nsong:\n name: You've Taken Everything\n"],
}.each do |format, cfg|
mod, output, input = cfg
describe "[#{format}] with :class" do
representer!(:module => mod) do
property :song, :class => Song do
property :name
end
end
let(:format) { format }
it { render(request).must_equal_document output }
it { _(parse(request, input).song.name).must_equal "You've Taken Everything"}
end
end
{
:hash => [Representable::Hash, {"songs"=>[{"name"=>"Alive"}]}, {"songs"=>[{"name"=>"You've Taken Everything"}]}],
:json => [Representable::JSON, "{\"songs\":[{\"name\":\"Alive\"}]}", "{\"songs\":[{\"name\":\"You've Taken Everything\"}]}"],
:xml => [Representable::XML, "\n \n Alive\n \n", "You've Taken Everything", { :as => :song }],
:yaml => [Representable::YAML, "---\nsongs:\n- name: Alive\n", "---\nsongs:\n- name: You've Taken Everything\n"],
}.each do |format, cfg|
mod, output, input, collection_options = cfg
collection_options ||= {}
describe "[#{format}] collection with :class" do
let(:request) { representer.prepare(OpenStruct.new(:songs => [song])) }
representer!(:module => mod) do
collection :songs, collection_options.merge(:class => Song) do
property :name
end
end
let(:format) { format } # FIXME: why do we have to define this?
it { render(request).must_equal_document output }
it { _(parse(request, input).songs.first.name).must_equal "You've Taken Everything"}
end
end
describe "without :class" do
representer! do
property :song do
property :name
end
end
it { _(request.to_hash).must_equal({"song"=>{"name"=>"Alive"}}) }
end
for_formats(
:hash => [Representable::Hash, {}],
# :xml => [Representable::XML, "\n \n Alive\n \n", "You've Taken Everything/open_struct>"],
# :yaml => [Representable::YAML, "---\nsong:\n name: Alive\n", "---\nsong:\n name: You've Taken Everything\n"],
) do |format, mod, input|
describe "parsing [#{format}] where nested property missing" do
representer!(:module => mod) do
property :song do
property :name
end
end
it "doesn't change represented object" do
_(request.send("from_#{format}", input).song).must_equal song
end
end
end
describe "inheriting from outer representer" do
let(:request) { Struct.new(:song, :requester).new(song, "Josephine") }
[false, true].each do |is_decorator| # test for module and decorator.
representer!(:decorator => is_decorator) do
property :requester
property :song, :class => Song do
property :name
end
end
let(:decorator) { representer.prepare(request) }
it { _(decorator.to_hash).must_equal({"requester"=>"Josephine", "song"=>{"name"=>"Alive"}}) }
it { _(decorator.from_hash({"song"=>{"name"=>"You've Taken Everything"}}).song.name).must_equal "You've Taken Everything"}
end
end
describe "object pollution" do
representer!(:decorator => true) do
property :song do
property :name
end
end
it "uses an inline decorator and doesn't alter represented" do
representer.prepare(Struct.new(:song).new(song)).to_hash
_(song).wont_be_kind_of Representable
end
end
# TODO: should be in extend:/decorator: test.
# FIXME: this tests :getter{represented}+:extend - represented gets extended twice and the inline decorator overrides original config.
# for_formats(
# :hash => [Representable::Hash, {"album" => {"artist" => {"label"=>"Epitaph"}}}],
# # :xml => [Representable::XML, ""],
# #:yaml => [Representable::YAML, "---\nlabel:\n label: Epitaph\n owner: Brett Gurewitz\n"]
# ) do |format, mod, output, input|
# module ArtistRepresenter
# include Representable::JSON
# property :label
# end
# describe ":getter with inline representer" do
# let(:format) { format }
# representer!(:module => mod) do
# self.representation_wrap = :album
# property :artist, :getter => lambda { |args| represented }, :extend => ArtistRepresenter
# end
# let(:album) { OpenStruct.new(:label => "Epitaph").extend(representer) }
# it "renders nested Album-properties in separate section" do
# render(album).must_equal_document output
# end
# end
# end
for_formats({
:hash => [Representable::Hash, {"album" => {"artist" => {"label"=>"Epitaph"}}}],
# :xml => [Representable::XML, ""],
#:yaml => [Representable::YAML, "---\nlabel:\n label: Epitaph\n owner: Brett Gurewitz\n"]
}) do |format, mod, output, input|
class ArtistDecorator < Representable::Decorator
include Representable::JSON
property :label
end
describe ":getter with :decorator" do
let(:format) { format }
representer!(:module => mod) do
self.representation_wrap = "album"
property :artist, :getter => lambda { |args| represented }, :decorator => ArtistDecorator
end
let(:album) { OpenStruct.new(:label => "Epitaph").extend(representer) }
it "renders nested Album-properties in separate section" do
render(album).must_equal_document output
end
end
end
# test helper methods within inline representer
for_formats({
:hash => [Representable::Hash, {"song"=>{"name"=>"ALIVE"}}],
:xml => [Representable::XML, "\n \n ALIVE\n \n"],
:yaml => [Representable::YAML, "---\nsong:\n name: ALIVE\n"],
}) do |format, mod, output|
describe "helper method within inline representer [#{format}]" do
let(:format) { format }
representer!(:module => mod, :decorator => true) do
self.representation_wrap = :request if format == :xml
property :requester
property :song do
property :name, :exec_context => :decorator
define_method :name do
represented.name.upcase
end
self.representation_wrap = :song if format == :xml
end
end
let(:request) { representer.prepare(OpenStruct.new(:song => Song.new("Alive"))) }
it do
render(request).must_equal_document output
end
end
end
describe "include module in inline representers" do
representer! do
extension = Module.new do
include Representable::Hash
property :title
end
property :song do
include extension
property :artist
end
end
it do _(OpenStruct.new(:song => OpenStruct.new(:title => "The Fever And The Sound", :artist => "Strung Out")).extend(representer).
to_hash).
must_equal({"song"=>{"artist"=>"Strung Out", "title"=>"The Fever And The Sound"}})
end
end
# define method in inline representer
describe "define method in inline representer" do
Mod = Module.new do
include Representable::Hash
def song
"Object.new"
end
property :song do
property :duration
def duration
"6:53"
end
end
end
it { _(Object.new.extend(Mod).to_hash).must_equal("song"=>{"duration"=>"6:53"}) }
end
# define method inline with Decorator
describe "define method inline with Decorator" do
dec = Class.new(Representable::Decorator) do
include Representable::Hash
def song
"Object.new"
end
property :song, :exec_context => :decorator do
property :duration, :exec_context => :decorator
def duration
"6:53"
end
end
end
it { _(dec.new(Object.new).to_hash).must_equal("song"=>{"duration"=>"6:53"}) }
end
end