Merge pull request #730 from rails-api/fixes-nested-has-many-links

Fixes nested has_many links in JSONAPI
This commit is contained in:
Alexandre de Oliveira 2014-11-17 11:30:20 -02:00
commit a933d44673
2 changed files with 92 additions and 21 deletions

View File

@ -58,25 +58,30 @@ module ActiveModel
end end
end end
def add_linked(resource_name, serializer, parent = nil) def add_linked(resource_name, serializers, parent = nil)
serializers = Array(serializers) unless serializers.respond_to?(:each)
resource_path = [parent, resource_name].compact.join('.') resource_path = [parent, resource_name].compact.join('.')
if include_assoc?(resource_path) if include_assoc?(resource_path)
plural_name = serialized_object_type(serializer).pluralize.to_sym plural_name = serialized_object_type(serializers).pluralize.to_sym
attrs = [attributes_for_serializer(serializer, @options)].flatten
@top[:linked] ||= {} @top[:linked] ||= {}
@top[:linked][plural_name] ||= [] @top[:linked][plural_name] ||= []
attrs.each do |attrs| serializers.each do |serializer|
attrs = attributes_for_serializer(serializer, @options)
add_resource_links(attrs, serializer, add_linked: false) add_resource_links(attrs, serializer, add_linked: false)
@top[:linked][plural_name].push(attrs) unless @top[:linked][plural_name].include?(attrs) @top[:linked][plural_name].push(attrs) unless @top[:linked][plural_name].include?(attrs)
end end
end end
serializer.each_association do |name, association, opts| serializers.each do |serializer|
add_linked(name, association, resource_path) if association serializer.each_association do |name, association, opts|
end if include_nested_assoc? resource_path add_linked(name, association, resource_path) if association
end if include_nested_assoc? resource_path
end
end end
def attributes_for_serializer(serializer, options) def attributes_for_serializer(serializer, options)
@ -124,7 +129,7 @@ module ActiveModel
def add_resource_links(attrs, serializer, options = {}) def add_resource_links(attrs, serializer, options = {})
options[:add_linked] = options.fetch(:add_linked, true) options[:add_linked] = options.fetch(:add_linked, true)
Array(serializer).first.each_association do |name, association, opts| serializer.each_association do |name, association, opts|
attrs[:links] ||= {} attrs[:links] ||= {}
if association.respond_to?(:each) if association.respond_to?(:each)

View File

@ -6,24 +6,32 @@ module ActiveModel
class JsonApi class JsonApi
class LinkedTest < Minitest::Test class LinkedTest < Minitest::Test
def setup def setup
@author = Author.new(id: 1, name: 'Steve K.') @author1 = Author.new(id: 1, name: 'Steve K.')
@bio = Bio.new(id: 1, content: 'AMS Contributor') @author2 = Author.new(id: 2, name: 'Tenderlove')
@bio1 = Bio.new(id: 1, content: 'AMS Contributor')
@bio2 = Bio.new(id: 2, content: 'Rails Contributor')
@first_post = Post.new(id: 1, title: 'Hello!!', body: 'Hello, world!!') @first_post = Post.new(id: 1, title: 'Hello!!', body: 'Hello, world!!')
@second_post = Post.new(id: 2, title: 'New Post', body: 'Body') @second_post = Post.new(id: 2, title: 'New Post', body: 'Body')
@third_post = Post.new(id: 3, title: 'Yet Another Post', body: 'Body')
@first_post.comments = [] @first_post.comments = []
@second_post.comments = [] @second_post.comments = []
@first_post.author = @author @first_post.author = @author1
@second_post.author = @author @second_post.author = @author2
@author.posts = [@first_post, @second_post] @third_post.author = @author1
@author.bio = @bio @author1.posts = [@first_post, @third_post]
@author.roles = [] @author1.bio = @bio1
@bio.author = @author @author1.roles = []
@author2.posts = [@second_post]
@serializer = ArraySerializer.new([@first_post, @second_post]) @author2.bio = @bio2
@adapter = ActiveModel::Serializer::Adapter::JsonApi.new(@serializer, include: 'author,author.bio,comments') @author2.roles = []
@bio1.author = @author1
@bio2.author = @author2
end end
def test_include_multiple_posts_and_linked def test_include_multiple_posts_and_linked
@serializer = ArraySerializer.new([@first_post, @second_post])
@adapter = ActiveModel::Serializer::Adapter::JsonApi.new(@serializer, include: 'author,author.bio,comments')
@first_comment = Comment.new(id: 1, body: 'ZOMG A COMMENT') @first_comment = Comment.new(id: 1, body: 'ZOMG A COMMENT')
@second_comment = Comment.new(id: 2, body: 'ZOMG ANOTHER COMMENT') @second_comment = Comment.new(id: 2, body: 'ZOMG ANOTHER COMMENT')
@first_post.comments = [@first_comment, @second_comment] @first_post.comments = [@first_comment, @second_comment]
@ -33,7 +41,7 @@ module ActiveModel
@second_comment.author = nil @second_comment.author = nil
assert_equal([ assert_equal([
{ title: "Hello!!", body: "Hello, world!!", id: "1", links: { comments: ['1', '2'], author: "1" } }, { title: "Hello!!", body: "Hello, world!!", id: "1", links: { comments: ['1', '2'], author: "1" } },
{ title: "New Post", body: "Body", id: "2", links: { comments: [], :author => "1" } } { title: "New Post", body: "Body", id: "2", links: { comments: [], :author => "2" } }
], @adapter.serializable_hash[:posts]) ], @adapter.serializable_hash[:posts])
@ -57,10 +65,18 @@ module ActiveModel
id: "1", id: "1",
name: "Steve K.", name: "Steve K.",
links: { links: {
posts: ["1", "2"], posts: ["1"],
roles: [], roles: [],
bio: "1" bio: "1"
} }
}, {
id: "2",
name: "Tenderlove",
links: {
posts: ["2"],
roles: [],
bio: "2"
}
}], }],
bios: [{ bios: [{
id: "1", id: "1",
@ -68,6 +84,56 @@ module ActiveModel
links: { links: {
author: "1" author: "1"
} }
}, {
id: "2",
content: "Rails Contributor",
links: {
author: "2"
}
}]
}
assert_equal expected, @adapter.serializable_hash[:linked]
end
def test_include_multiple_posts_and_linked
@serializer = BioSerializer.new(@bio1)
@adapter = ActiveModel::Serializer::Adapter::JsonApi.new(@serializer, include: 'author,author.posts')
@first_comment = Comment.new(id: 1, body: 'ZOMG A COMMENT')
@second_comment = Comment.new(id: 2, body: 'ZOMG ANOTHER COMMENT')
@first_post.comments = [@first_comment, @second_comment]
@third_post.comments = []
@first_comment.post = @first_post
@first_comment.author = nil
@second_comment.post = @first_post
@second_comment.author = nil
expected = {
authors: [{
id: "1",
name: "Steve K.",
links: {
posts: ["1", "3"],
roles: [],
bio: "1"
}
}],
posts: [{
title: "Hello!!",
body: "Hello, world!!",
id: "1",
links: {
comments: ["1", "2"],
author: "1"
}
}, {
title: "Yet Another Post",
body: "Body",
id: "3",
links: {
comments: [],
author: "1"
}
}] }]
} }
assert_equal expected, @adapter.serializable_hash[:linked] assert_equal expected, @adapter.serializable_hash[:linked]