mirror of
https://github.com/ditkrg/active_model_serializers.git
synced 2026-01-23 06:16:50 +00:00
Merge pull request #730 from rails-api/fixes-nested-has-many-links
Fixes nested has_many links in JSONAPI
This commit is contained in:
commit
a933d44673
@ -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)
|
||||||
|
|||||||
@ -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]
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user