active_model_serializers/lib/active_model/serializer/reflection.rb
2017-04-23 17:53:00 -05:00

280 lines
10 KiB
Ruby

require 'active_model/serializer/field'
require 'active_model/serializer/association'
module ActiveModel
class Serializer
# Holds all the meta-data about an association as it was specified in the
# ActiveModel::Serializer class.
#
# @example
# class PostSerializer < ActiveModel::Serializer
# has_one :author, serializer: AuthorSerializer
# has_many :comments
# has_many :comments, key: :last_comments do
# object.comments.last(1)
# end
# has_many :secret_meta_data, if: :is_admin?
#
# has_one :blog do |serializer|
# meta count: object.roles.count
# serializer.cached_blog
# end
#
# private
#
# def cached_blog
# cache_store.fetch("cached_blog:#{object.updated_at}") do
# Blog.find(object.blog_id)
# end
# end
#
# def is_admin?
# current_user.admin?
# end
# end
#
# Specifically, the association 'comments' is evaluated two different ways:
# 1) as 'comments' and named 'comments'.
# 2) as 'object.comments.last(1)' and named 'last_comments'.
#
# PostSerializer._reflections # =>
# # {
# # author: HasOneReflection.new(:author, serializer: AuthorSerializer),
# # comments: HasManyReflection.new(:comments)
# # last_comments: HasManyReflection.new(:comments, { key: :last_comments }, #<Block>)
# # secret_meta_data: HasManyReflection.new(:secret_meta_data, { if: :is_admin? })
# # }
#
# So you can inspect reflections in your Adapters.
class Reflection < Field
REFLECTION_OPTIONS = %i(key links polymorphic meta serializer virtual_value namespace).freeze
def initialize(*)
super
options[:links] = {}
options[:include_data_setting] = Serializer.config.include_data_default
options[:meta] = nil
end
# @api public
# @example
# has_one :blog do
# include_data false
# link :self, 'a link'
# link :related, 'another link'
# link :self, '//example.com/link_author/relationships/bio'
# id = object.profile.id
# link :related do
# "//example.com/profiles/#{id}" if id != 123
# end
# link :related do
# ids = object.likes.map(&:id).join(',')
# href "//example.com/likes/#{ids}"
# meta ids: ids
# end
# end
def link(name, value = nil)
options[:links][name] = block_given? ? Proc.new : value
:nil
end
# @api public
# @example
# has_one :blog do
# include_data false
# meta(id: object.blog.id)
# meta liked: object.likes.any?
# link :self do
# href object.blog.id.to_s
# meta(id: object.blog.id)
# end
def meta(value = nil)
options[:meta] = block_given? ? Proc.new : value
:nil
end
# @api public
# @example
# has_one :blog do
# include_data false
# link :self, 'a link'
# link :related, 'another link'
# end
#
# has_one :blog do
# include_data false
# link :self, 'a link'
# link :related, 'another link'
# end
#
# belongs_to :reviewer do
# meta name: 'Dan Brown'
# include_data true
# end
#
# has_many :tags, serializer: TagSerializer do
# link :self, '//example.com/link_author/relationships/tags'
# include_data :if_sideloaded
# end
def include_data(value = true)
options[:include_data_setting] = value
:nil
end
def collection?
false
end
# Build association. This method is used internally to
# build serializer's association by its reflection.
#
# @param [Serializer] parent_serializer for given association
# @param [Hash{Symbol => Object}] parent_serializer_options
#
# @example
# # Given the following serializer defined:
# class PostSerializer < ActiveModel::Serializer
# has_many :comments, serializer: CommentSummarySerializer
# end
#
# # Then you instantiate your serializer
# post_serializer = PostSerializer.new(post, foo: 'bar') #
# # to build association for comments you need to get reflection
# comments_reflection = PostSerializer._reflections.detect { |r| r.name == :comments }
# # and #build_association
# comments_reflection.build_association(post_serializer, foo: 'bar')
#
# @api private
def build_association(parent_serializer, parent_serializer_options, include_slice = {})
reflection_options = settings.merge(include_data: include_data?(include_slice)) unless block?
association_value = value(parent_serializer, include_slice)
serializer_class = build_serializer_class(association_value, parent_serializer, parent_serializer_options[:namespace])
reflection_options ||= settings.merge(include_data: include_data?(include_slice)) # Needs to be after association_value is evaluated unless reflection.block.nil?
if serializer_class
if (serializer = build_serializer!(association_value, serializer_class, parent_serializer, parent_serializer_options))
reflection_options[:serializer] = serializer
else
# BUG: per #2027, JSON API resource relationships are only id and type, and hence either
# *require* a serializer or we need to be a little clever about figuring out the id/type.
# In either case, returning the raw virtual value will almost always be incorrect.
#
# Should be reflection_options[:virtual_value] or adapter needs to figure out what to do
# with an object that is non-nil and has no defined serializer.
reflection_options[:virtual_value] = association_value.try(:as_json) || association_value
end
elsif !association_value.nil? && !association_value.instance_of?(Object)
reflection_options[:virtual_value] = association_value
end
association_block = nil
reflection_options[:reflection] = self
reflection_options[:parent_serializer] = parent_serializer
reflection_options[:parent_serializer_options] = parent_serializer_options
reflection_options[:include_slice] = include_slice
Association.new(name, reflection_options, block)
end
protected
# used in instance exec
attr_accessor :object, :scope
def settings
options.dup.reject { |k, _| !REFLECTION_OPTIONS.include?(k) }
end
# Evaluation of the reflection.block will mutate options.
# So, the settings cannot be used until the block is evaluated.
# This means that each time the block is evaluated, it may set a new
# value in the reflection instance. This is not thread-safe.
# @example
# has_many :likes do
# meta liked: object.likes.any?
# include_data: object.loaded?
# end
def block?
!block.nil?
end
def serializer?
options.key?(:serializer)
end
def serializer
options[:serializer]
end
def namespace
options[:namespace]
end
def include_data?(include_slice)
include_data_setting = options[:include_data_setting]
case include_data_setting
when :if_sideloaded then include_slice.key?(name)
when true then true
when false then false
else fail ArgumentError, "Unknown include_data_setting '#{include_data_setting.inspect}'"
end
end
# @param serializer [ActiveModel::Serializer]
# @yield [ActiveModel::Serializer]
# @return [:nil, associated resource or resource collection]
def value(serializer, include_slice)
@object = serializer.object
@scope = serializer.scope
block_value = instance_exec(serializer, &block) if block
return unless include_data?(include_slice)
if block && block_value != :nil
block_value
else
serializer.read_attribute_for_serialization(name)
end
end
def build_serializer!(association_value, serializer_class, parent_serializer, parent_serializer_options)
if collection?
build_association_collection_serializer(parent_serializer, parent_serializer_options, association_value, serializer_class)
else
build_association_serializer(parent_serializer, parent_serializer_options, association_value, serializer_class)
end
end
def build_serializer_class(association_value, parent_serializer, parent_serializer_namespace_option)
serializer_for_options = {
# Pass the parent's namespace onto the child serializer
namespace: namespace || parent_serializer_namespace_option
}
serializer_for_options[:serializer] = serializer if serializer?
parent_serializer.class.serializer_for(association_value, serializer_for_options)
end
# NOTE(BF): This serializer throw/catch should only happen when the serializer is a collection
# serializer.
#
# @return [ActiveModel::Serializer, nil]
def build_association_collection_serializer(parent_serializer, parent_serializer_options, association_value, serializer_class)
catch(:no_serializer) do
build_association_serializer(parent_serializer, parent_serializer_options, association_value, serializer_class)
end
end
# @return [ActiveModel::Serializer, nil]
def build_association_serializer(parent_serializer, parent_serializer_options, association_value, serializer_class)
# Make all the parent serializer instance options available to associations
# except ActiveModelSerializers-specific ones we don't want.
serializer_options = parent_serializer_options.except(:serializer)
serializer_options[:serializer_context_class] = parent_serializer.class
serializer_options[:serializer] = serializer if serializer
serializer_class.new(association_value, serializer_options)
end
end
end
end