Merge pull request #1426 from brigade/default-include

Add a default_include_tree config variable to ActiveModel::Serializer
This commit is contained in:
L. Preston Sego III 2016-05-26 13:31:53 -04:00
commit 7d7329bbcf
7 changed files with 132 additions and 38 deletions

View File

@ -5,6 +5,7 @@
Breaking changes:
Features:
- [#1426](https://github.com/rails-api/active_model_serializers/pull/1426) Add ActiveModelSerializers.config.default_includes (@empact)
Fixes:
- [#1710](https://github.com/rails-api/active_model_serializers/pull/1710) Prevent association loading when `include_data` option

View File

@ -52,10 +52,12 @@ Each adapter has a default key transform configured:
`config.key_transform` is a global override of the adapter default. Adapters
still prefer the render option `:key_transform` over this setting.
##### default_includes
What relationships to serialize by default. Default: `'*'`, which includes one level of related
objects. See [includes](adapters.md#included) for more info.
## JSON API
##### jsonapi_resource_type
Sets whether the [type](http://jsonapi.org/format/#document-resource-identifier-objects)

View File

@ -10,8 +10,6 @@ module ActiveModel
module Associations
extend ActiveSupport::Concern
DEFAULT_INCLUDE_TREE = ActiveModel::Serializer::IncludeTree.from_string('*')
included do
with_options instance_writer: false, instance_reader: true do |serializer|
serializer.class_attribute :_reflections
@ -80,10 +78,11 @@ module ActiveModel
end
end
# @param [IncludeTree] include_tree (defaults to all associations when not provided)
# @param [IncludeTree] include_tree (defaults to the
# default_includes config value when not provided)
# @return [Enumerator<Association>]
#
def associations(include_tree = DEFAULT_INCLUDE_TREE)
def associations(include_tree = ActiveModelSerializers.default_include_tree)
return unless object
Enumerator.new do |y|

View File

@ -19,6 +19,7 @@ module ActiveModel
collection_serializer
end
config.default_includes = '*'
config.adapter = :attributes
config.jsonapi_resource_type = :plural
config.jsonapi_version = '1.0'

View File

@ -31,6 +31,13 @@ module ActiveModelSerializers
[file, lineno]
end
# Memoized default include tree
# @return [ActiveModel::Serializer::IncludeTree]
def self.default_include_tree
@default_include_tree ||= ActiveModel::Serializer::IncludeTree
.from_include_args(config.default_includes)
end
require 'active_model/serializer/version'
require 'active_model/serializer'
require 'active_model/serializable_resource'

View File

@ -3,8 +3,13 @@ module ActiveModelSerializers
class Attributes < Base
def initialize(serializer, options = {})
super
@include_tree = ActiveModel::Serializer::IncludeTree.from_include_args(options[:include] || '*')
@cached_attributes = options[:cache_attributes] || {}
@include_tree =
if options[:include]
ActiveModel::Serializer::IncludeTree.from_include_args(options[:include])
else
ActiveModelSerializers.default_include_tree
end
end
def serializable_hash(options = nil)

View File

@ -4,6 +4,10 @@ module ActionController
module Serialization
class Json
class IncludeTest < ActionController::TestCase
INCLUDE_STRING = 'posts.comments'.freeze
INCLUDE_HASH = { posts: :comments }.freeze
DEEP_INCLUDE = 'posts.comments.author'.freeze
class IncludeTestController < ActionController::Base
def setup_data
ActionController::Base.cache_store.clear
@ -38,17 +42,28 @@ module ActionController
def render_resource_with_include_hash
setup_data
render json: @author, include: { posts: :comments }, adapter: :json
render json: @author, include: INCLUDE_HASH, adapter: :json
end
def render_resource_with_include_string
setup_data
render json: @author, include: 'posts.comments', adapter: :json
render json: @author, include: INCLUDE_STRING, adapter: :json
end
def render_resource_with_deep_include
setup_data
render json: @author, include: 'posts.comments.author', adapter: :json
render json: @author, include: DEEP_INCLUDE, adapter: :json
end
def render_without_recursive_relationships
# testing recursive includes ('**') can't have any cycles in the
# relationships, or we enter an infinite loop.
author = Author.new(id: 11, name: 'Jane Doe')
post = Post.new(id: 12, title: 'Hello World', body: 'My first post')
comment = Comment.new(id: 13, body: 'Commentary')
author.posts = [post]
post.comments = [comment]
render json: author
end
end
@ -77,34 +92,90 @@ module ActionController
def test_render_resource_with_include_hash
get :render_resource_with_include_hash
response = JSON.parse(@response.body)
expected = {
'author' => {
'id' => 1,
'name' => 'Steve K.',
'posts' => [
{
'id' => 42, 'title' => 'New Post', 'body' => 'Body',
'comments' => [
{
'id' => 1, 'body' => 'ZOMG A COMMENT'
},
{
'id' => 2, 'body' => 'ZOMG ANOTHER COMMENT'
}
]
}
]
}
}
assert_equal(expected, response)
assert_equal(expected_include_response, response)
end
def test_render_resource_with_include_string
get :render_resource_with_include_string
response = JSON.parse(@response.body)
assert_equal(expected_include_response, response)
end
def test_render_resource_with_deep_include
get :render_resource_with_deep_include
response = JSON.parse(@response.body)
assert_equal(expected_deep_include_response, response)
end
def test_render_with_empty_default_includes
with_default_includes '' do
get :render_without_include
response = JSON.parse(@response.body)
expected = {
'author' => {
'id' => 1,
'name' => 'Steve K.'
}
}
assert_equal(expected, response)
end
end
def test_render_with_recursive_default_includes
with_default_includes '**' do
get :render_without_recursive_relationships
response = JSON.parse(@response.body)
expected = {
'id' => 11,
'name' => 'Jane Doe',
'roles' => nil,
'bio' => nil,
'posts' => [
{
'id' => 12,
'title' => 'Hello World',
'body' => 'My first post',
'comments' => [
{
'id' => 13,
'body' => 'Commentary',
'post' => nil, # not set to avoid infinite recursion
'author' => nil, # not set to avoid infinite recursion
}
],
'blog' => {
'id' => 999,
'name' => 'Custom blog',
'writer' => nil,
'articles' => nil
},
'author' => nil # not set to avoid infinite recursion
}
]
}
assert_equal(expected, response)
end
end
def test_render_with_includes_overrides_default_includes
with_default_includes '' do
get :render_resource_with_include_hash
response = JSON.parse(@response.body)
assert_equal(expected_include_response, response)
end
end
private
def expected_include_response
{
'author' => {
'id' => 1,
'name' => 'Steve K.',
@ -123,15 +194,10 @@ module ActionController
]
}
}
assert_equal(expected, response)
end
def test_render_resource_with_deep_include
get :render_resource_with_deep_include
response = JSON.parse(@response.body)
expected = {
def expected_deep_include_response
{
'author' => {
'id' => 1,
'name' => 'Steve K.',
@ -158,8 +224,21 @@ module ActionController
]
}
}
end
assert_equal(expected, response)
def with_default_includes(include_tree)
original = ActiveModelSerializers.config.default_includes
ActiveModelSerializers.config.default_includes = include_tree
clear_include_tree_cache
yield
ensure
ActiveModelSerializers.config.default_includes = original
clear_include_tree_cache
end
def clear_include_tree_cache
ActiveModelSerializers
.instance_variable_set(:@default_include_tree, nil)
end
end
end