Merge pull request #696 from ggordon/explicit_serializer

Explicitly set serializer for associations
This commit is contained in:
Alexandre de Oliveira 2015-01-06 09:57:07 -02:00
commit 6eb75af96b
6 changed files with 119 additions and 13 deletions

View File

@ -221,6 +221,12 @@ The `has_many` and `belongs_to` declarations describe relationships between
resources. By default, when you serialize a `Post`, you will get its `Comment`s
as well.
You may also use the `:serializer` option to specify a custom serializer class, for example:
```ruby
has_many :comments, serializer: CommentPreviewSerializer
```
The `url` declaration describes which named routes to use while generating URLs
for your JSON. Not every adapter will require URLs.

View File

@ -23,7 +23,7 @@ module ActiveModel
attrs.each do |attr|
define_method attr do
object.read_attribute_for_serialization(attr)
object && object.read_attribute_for_serialization(attr)
end unless method_defined?(attr)
end
end
@ -67,7 +67,7 @@ module ActiveModel
end
end
self._associations[attr] = {type: type, options: options}
self._associations[attr] = {type: type, association_options: options}
end
end
@ -79,11 +79,13 @@ module ActiveModel
@_urls.concat attrs
end
def self.serializer_for(resource)
def self.serializer_for(resource, options = {})
if resource.respond_to?(:to_ary)
config.array_serializer
else
get_serializer_for(resource.class)
options
.fetch(:association_options, {})
.fetch(:serializer, get_serializer_for(resource.class))
end
end
@ -146,16 +148,27 @@ module ActiveModel
def each_association(&block)
self.class._associations.dup.each do |name, options|
next unless object
association = object.send(name)
serializer_class = ActiveModel::Serializer.serializer_for(association)
serializer = serializer_class.new(association) if serializer_class
serializer_class = ActiveModel::Serializer.serializer_for(association, options)
serializer = serializer_class.new(
association,
serializer_from_options(options)
) if serializer_class
if block_given?
block.call(name, serializer, options[:options])
block.call(name, serializer, options[:association_options])
end
end
end
def serializer_from_options(options)
opts = {}
serializer = options.fetch(:options, {}).fetch(:serializer, nil)
opts[:serializer] = serializer if serializer
opts
end
private
def self.get_serializer_for(klass)

View File

@ -24,7 +24,6 @@ module ActiveModel
end
else
@hash[@root] = attributes_for_serializer(serializer, @options)
add_resource_links(@hash[@root], serializer)
end
@ -52,7 +51,7 @@ module ActiveModel
resource[:links] ||= {}
resource[:links][name] = nil
if serializer
if serializer && serializer.object
type = serialized_object_type(serializer)
if name.to_s == type || !type
resource[:links][name] = serializer.id.to_s

View File

@ -0,0 +1,65 @@
require 'test_helper'
module ActiveModel
class Serializer
class Adapter
class JsonApi
# Test 'has_many :assocs, serializer: AssocXSerializer'
class HasManyExplicitSerializerTest < Minitest::Test
def setup
@post = Post.new(title: 'New Post', body: 'Body')
@author = Author.new(name: 'Jane Blogger')
@author.posts = [@post]
@post.author = @author
@first_comment = Comment.new(id: 1, body: 'ZOMG A COMMENT')
@second_comment = Comment.new(id: 2, body: 'ZOMG ANOTHER COMMENT')
@post.comments = [@first_comment, @second_comment]
@first_comment.post = @post
@first_comment.author = nil
@second_comment.post = @post
@second_comment.author = nil
@serializer = PostPreviewSerializer.new(@post)
@adapter = ActiveModel::Serializer::Adapter::JsonApi.new(
@serializer,
include: 'comments,author'
)
end
def test_includes_comment_ids
assert_equal(['1', '2'],
@adapter.serializable_hash[:posts][:links][:comments])
end
def test_includes_linked_comments
assert_equal([{ id: '1', body: "ZOMG A COMMENT", links: { post: @post.id.to_s, author: nil }},
{ id: '2', body: "ZOMG ANOTHER COMMENT", links: { post: @post.id.to_s, author: nil }}],
@adapter.serializable_hash[:linked][:comments])
end
def test_includes_author_id
assert_equal(@author.id.to_s,
@adapter.serializable_hash[:posts][:links][:author])
end
def test_includes_linked_authors
assert_equal([{ id: @author.id.to_s, links: { posts: [@post.id.to_s] } }],
@adapter.serializable_hash[:linked][:authors])
end
def test_explicit_serializer_with_null_resource
@post.author = nil
assert_equal(nil,
@adapter.serializable_hash[:posts][:links][:author])
end
def test_explicit_serializer_with_null_collection
@post.comments = []
assert_equal([],
@adapter.serializable_hash[:posts][:links][:comments])
end
end
end
end
end
end

23
test/fixtures/poro.rb vendored
View File

@ -100,3 +100,26 @@ AlternateBlogSerializer = Class.new(ActiveModel::Serializer) do
attribute :id
attribute :name, key: :title
end
CommentPreviewSerializer = Class.new(ActiveModel::Serializer) do
attributes :id
belongs_to :post
end
AuthorPreviewSerializer = Class.new(ActiveModel::Serializer) do
attributes :id
has_many :posts
end
PostPreviewSerializer = Class.new(ActiveModel::Serializer) do
def self.root_name
'posts'
end
attributes :title, :body, :id
has_many :comments, serializer: CommentPreviewSerializer
belongs_to :author, serializer: AuthorPreviewSerializer
end

View File

@ -42,9 +42,9 @@ module ActiveModel
def test_has_many
assert_equal(
{ posts: { type: :has_many, options: { embed: :ids } },
roles: { type: :has_many, options: { embed: :ids } },
bio: { type: :belongs_to, options: {} } },
{ posts: { type: :has_many, association_options: { embed: :ids } },
roles: { type: :has_many, association_options: { embed: :ids } },
bio: { type: :belongs_to, association_options: {} } },
@author_serializer.class._associations
)
@author_serializer.each_association do |name, serializer, options|
@ -64,7 +64,7 @@ module ActiveModel
end
def test_has_one
assert_equal({post: {type: :belongs_to, options: {}}, :author=>{:type=>:belongs_to, :options=>{}}}, @comment_serializer.class._associations)
assert_equal({post: {type: :belongs_to, association_options: {}}, :author=>{:type=>:belongs_to, :association_options=>{}}}, @comment_serializer.class._associations)
@comment_serializer.each_association do |name, serializer, options|
if name == :post
assert_equal({}, options)