Add key option to serializer associations

This commit is contained in:
Rodrigo Ra 2015-07-05 19:47:58 -03:00
parent ac1991fd6b
commit df63b59512
8 changed files with 109 additions and 29 deletions

View File

@ -264,6 +264,12 @@ You may also use the `:serializer` option to specify a custom serializer class,
has_many :comments, serializer: CommentPreviewSerializer has_many :comments, serializer: CommentPreviewSerializer
``` ```
And you can change the JSON key that the serializer should use for a particular association:
```ruby
has_many :comments, key: :reviews
```
The `url` declaration describes which named routes to use while generating URLs The `url` declaration describes which named routes to use while generating URLs
for your JSON. Not every adapter will require URLs. for your JSON. Not every adapter will require URLs.

View File

@ -220,8 +220,9 @@ module ActiveModel
association_options[:association_options][:virtual_value] = association_value association_options[:association_options][:virtual_value] = association_value
end end
association_key = association_options[:association_options][:key] || name
if block_given? if block_given?
block.call(name, serializer, association_options[:association_options]) block.call(association_key, serializer, association_options[:association_options])
end end
end end
end end

View File

@ -15,23 +15,23 @@ module ActiveModel
serializer.attributes(options) serializer.attributes(options)
end end
serializer.each_association do |name, association, opts| serializer.each_association do |key, association, opts|
if association.respond_to?(:each) if association.respond_to?(:each)
array_serializer = association array_serializer = association
@hash[name] = array_serializer.map do |item| @hash[key] = array_serializer.map do |item|
cache_check(item) do cache_check(item) do
item.attributes(opts) item.attributes(opts)
end end
end end
else else
if association && association.object if association && association.object
@hash[name] = cache_check(association) do @hash[key] = cache_check(association) do
association.attributes(options) association.attributes(options)
end end
elsif opts[:virtual_value] elsif opts[:virtual_value]
@hash[name] = opts[:virtual_value] @hash[key] = opts[:virtual_value]
else else
@hash[name] = nil @hash[key] = nil
end end
end end
end end

View File

@ -75,8 +75,8 @@ module ActiveModel
end end
serializers.each do |serializer| serializers.each do |serializer|
serializer.each_association do |name, association, opts| serializer.each_association do |key, association, opts|
add_included(name, association, resource_path) if association add_included(key, association, resource_path) if association
end if include_nested_assoc? resource_path end if include_nested_assoc? resource_path
end end
end end
@ -131,22 +131,22 @@ module ActiveModel
def add_resource_relationships(attrs, serializer, options = {}) def add_resource_relationships(attrs, serializer, options = {})
options[:add_included] = options.fetch(:add_included, true) options[:add_included] = options.fetch(:add_included, true)
serializer.each_association do |name, association, opts| serializer.each_association do |key, association, opts|
attrs[:relationships] ||= {} attrs[:relationships] ||= {}
if association.respond_to?(:each) if association.respond_to?(:each)
add_relationships(attrs, name, association) add_relationships(attrs, key, association)
else else
if opts[:virtual_value] if opts[:virtual_value]
add_relationship(attrs, name, nil, opts[:virtual_value]) add_relationship(attrs, key, nil, opts[:virtual_value])
else else
add_relationship(attrs, name, association) add_relationship(attrs, key, association)
end end
end end
if options[:add_included] if options[:add_included]
Array(association).each do |association| Array(association).each do |association|
add_included(name, association) add_included(key, association)
end end
end end
end end

View File

@ -0,0 +1,38 @@
require 'test_helper'
module ActiveModel
class Serializer
class Adapter
class JsonApiTest < Minitest::Test
def setup
ActionController::Base.cache_store.clear
@author = Author.new(id: 1, name: 'Steve K.')
@post = Post.new(id: 1, title: 'New Post', body: 'Body')
@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
@second_comment.post = @post
@post.author = @author
@blog = Blog.new(id: 1, name: "My Blog!!")
@post.blog = @blog
end
def test_custom_keys
serializer = PostWithCustomKeysSerializer.new(@post)
adapter = ActiveModel::Serializer::Adapter::JsonApi.new(serializer)
assert_equal({
reviews: { data: [
{type: "comments", id: "1"},
{type: "comments", id: "2"}
]},
writer: { data: {type: "authors", id: "1"} },
site: { data: {type: "blogs", id: "1" } }
}, adapter.serializable_hash[:data][:relationships])
end
end
end
end
end

View File

@ -7,7 +7,7 @@ module ActiveModel
def setup def setup
ActionController::Base.cache_store.clear ActionController::Base.cache_store.clear
@author = Author.new(id: 1, name: 'Steve K.') @author = Author.new(id: 1, name: 'Steve K.')
@post = Post.new(title: 'New Post', body: 'Body') @post = Post.new(id: 1, title: 'New Post', body: 'Body')
@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')
@post.comments = [@first_comment, @second_comment] @post.comments = [@first_comment, @second_comment]
@ -27,6 +27,20 @@ module ActiveModel
{id: 2, body: 'ZOMG ANOTHER COMMENT'} {id: 2, body: 'ZOMG ANOTHER COMMENT'}
], @adapter.serializable_hash[:post][:comments]) ], @adapter.serializable_hash[:post][:comments])
end end
def test_custom_keys
serializer = PostWithCustomKeysSerializer.new(@post)
adapter = ActiveModel::Serializer::Adapter::Json.new(serializer)
assert_equal({
id: 1,
reviews: [{id: 1, body: "ZOMG A COMMENT"},
{id: 2, body: "ZOMG ANOTHER COMMENT"}
],
writer: {id: 1, name: "Steve K."},
site: {id: 1, name: "My Blog!!"}
}, adapter.serializable_hash[:post_with_custom_keys])
end
end end
end end
end end

View File

@ -228,6 +228,14 @@ PostWithTagsSerializer = Class.new(ActiveModel::Serializer) do
has_many :tags has_many :tags
end end
PostWithCustomKeysSerializer = Class.new(ActiveModel::Serializer) do
attributes :id
has_many :comments, key: :reviews
belongs_to :author, key: :writer
has_one :blog, key: :site
end
VirtualValueSerializer = Class.new(ActiveModel::Serializer) do VirtualValueSerializer = Class.new(ActiveModel::Serializer) do
attributes :id attributes :id

View File

@ -51,33 +51,33 @@ module ActiveModel
bio: { type: :has_one, association_options: {} } }, bio: { type: :has_one, association_options: {} } },
@author_serializer.class._associations @author_serializer.class._associations
) )
@author_serializer.each_association do |name, serializer, options| @author_serializer.each_association do |key, serializer, options|
if name == :posts if key == :posts
assert_equal({embed: :ids}, options) assert_equal({embed: :ids}, options)
assert_kind_of(ActiveModel::Serializer.config.array_serializer, serializer) assert_kind_of(ActiveModel::Serializer.config.array_serializer, serializer)
elsif name == :bio elsif key == :bio
assert_equal({}, options) assert_equal({}, options)
assert_nil serializer assert_nil serializer
elsif name == :roles elsif key == :roles
assert_equal({embed: :ids}, options) assert_equal({embed: :ids}, options)
assert_kind_of(ActiveModel::Serializer.config.array_serializer, serializer) assert_kind_of(ActiveModel::Serializer.config.array_serializer, serializer)
else else
flunk "Unknown association: #{name}" flunk "Unknown association: #{key}"
end end
end end
end end
def test_has_many_with_no_serializer def test_has_many_with_no_serializer
PostWithTagsSerializer.new(@post).each_association do |name, serializer, options| PostWithTagsSerializer.new(@post).each_association do |key, serializer, options|
assert_equal name, :tags assert_equal key, :tags
assert_equal serializer, nil assert_equal serializer, nil
assert_equal [{ attributes: { name: "#hashtagged" }}].to_json, options[:virtual_value].to_json assert_equal [{ attributes: { name: "#hashtagged" }}].to_json, options[:virtual_value].to_json
end end
end end
def test_serializer_options_are_passed_into_associations_serializers def test_serializer_options_are_passed_into_associations_serializers
@post_serializer.each_association do |name, association| @post_serializer.each_association do |key, association|
if name == :comments if key == :comments
assert association.first.custom_options[:custom_options] assert association.first.custom_options[:custom_options]
end end
end end
@ -89,15 +89,15 @@ module ActiveModel
author: { type: :belongs_to, association_options: {} } }, author: { type: :belongs_to, association_options: {} } },
@comment_serializer.class._associations @comment_serializer.class._associations
) )
@comment_serializer.each_association do |name, serializer, options| @comment_serializer.each_association do |key, serializer, options|
if name == :post if key == :post
assert_equal({}, options) assert_equal({}, options)
assert_kind_of(PostSerializer, serializer) assert_kind_of(PostSerializer, serializer)
elsif name == :author elsif key == :author
assert_equal({}, options) assert_equal({}, options)
assert_nil serializer assert_nil serializer
else else
flunk "Unknown association: #{name}" flunk "Unknown association: #{key}"
end end
end end
end end
@ -105,8 +105,8 @@ module ActiveModel
def test_belongs_to_with_custom_method def test_belongs_to_with_custom_method
blog_is_present = false blog_is_present = false
@post_serializer.each_association do |name, serializer, options| @post_serializer.each_association do |key, serializer, options|
blog_is_present = true if name == :blog blog_is_present = true if key == :blog
end end
assert blog_is_present assert blog_is_present
@ -132,6 +132,19 @@ module ActiveModel
) )
assert_equal(inherited_klass._associations, expected_associations) assert_equal(inherited_klass._associations, expected_associations)
end end
def test_associations_custom_keys
serializer = PostWithCustomKeysSerializer.new(@post)
expected_association_keys = []
serializer.each_association do |key, serializer, options|
expected_association_keys << key
end
assert expected_association_keys.include? :reviews
assert expected_association_keys.include? :writer
assert expected_association_keys.include? :site
end
end end
end end
end end