require "active_support/core_ext/class/attribute" require "active_support/core_ext/module/anonymous" require "set" module ActiveModel # Active Model Serializer # # Provides a basic serializer implementation that allows you to easily # control how a given object is going to be serialized. On initialization, # it expects two objects as arguments, a resource and options. For example, # one may do in a controller: # # PostSerializer.new(@post, :scope => current_user).to_json # # The object to be serialized is the +@post+ and the current user is passed # in for authorization purposes. # # We use the scope to check if a given attribute should be serialized or not. # For example, some attributes may only be returned if +current_user+ is the # author of the post: # # class PostSerializer < ActiveModel::Serializer # attributes :title, :body # has_many :comments # # private # # def attributes # hash = super # hash.merge!(:email => post.email) if author? # hash # end # # def author? # post.author == scope # end # end # class Serializer INCLUDE_METHODS = {} INSTRUMENT = { :serialize => :"serialize.serializer", :associations => :"associations.serializer" } class IncludeError < StandardError attr_reader :source, :association def initialize(source, association) @source, @association = source, association end def to_s "Cannot serialize #{association} when #{source} does not have a root!" end end module Associations #:nodoc: class Config #:nodoc: class_attribute :options def self.refine(name, class_options) current_class = self Class.new(self) do singleton_class.class_eval do define_method(:to_s) do "(subclass of #{current_class.name})" end alias inspect to_s end self.options = class_options end end self.options = {} def initialize(name, source, options={}) @name = name @source = source @options = options end def option(key, default=nil) if @options.key?(key) @options[key] elsif self.class.options.key?(key) self.class.options[key] else default end end def target_serializer option(:serializer) end def source_serializer @source end def key option(:key) || @name end def root option(:root) || plural_key end def name option(:name) || @name end def associated_object option(:value) || source_serializer.send(name) end def embed_ids? option(:embed, source_serializer._embed) == :ids end def embed_objects? option(:embed, source_serializer._embed) == :objects end def embed_in_root? option(:include, source_serializer._root_embed) end def embeddable? !associated_object.nil? end protected def find_serializable(object) if target_serializer target_serializer.new(object, source_serializer.options) elsif object.respond_to?(:active_model_serializer) && (ams = object.active_model_serializer) ams.new(object, source_serializer.options) else object end end end class HasMany < Config #:nodoc: alias plural_key key def serialize associated_object.map do |item| find_serializable(item).serializable_hash end end alias serialize_many serialize def serialize_ids # Use pluck or select_columns if available # return collection.ids if collection.respond_to?(:ids) associated_object.map do |item| item.read_attribute_for_serialization(:id) end end end class HasOne < Config #:nodoc: def embeddable? if polymorphic? && associated_object.nil? false else true end end def polymorphic? option :polymorphic end def polymorphic_key associated_object.class.to_s.demodulize.underscore.to_sym end def plural_key if polymorphic? associated_object.class.to_s.pluralize.demodulize.underscore.to_sym else key.to_s.pluralize.to_sym end end def serialize object = associated_object if object && polymorphic? { :type => polymorphic_key, polymorphic_key => find_serializable(object).serializable_hash } elsif object find_serializable(object).serializable_hash end end def serialize_many object = associated_object value = object && find_serializable(object).serializable_hash value ? [value] : [] end def serialize_ids object = associated_object if object && polymorphic? { :type => polymorphic_key, :id => object.read_attribute_for_serialization(:id) } elsif object object.read_attribute_for_serialization(:id) else nil end end end end class_attribute :_attributes self._attributes = {} class_attribute :_associations self._associations = {} class_attribute :_root class_attribute :_embed self._embed = :objects class_attribute :_root_embed class << self # Define attributes to be used in the serialization. def attributes(*attrs) self._attributes = _attributes.dup attrs.each do |attr| attribute attr end end def attribute(attr, options={}) self._attributes = _attributes.merge(attr => options[:key] || attr.to_s.gsub(/\?$/, '').to_sym) unless method_defined?(attr) define_method attr do object.read_attribute_for_serialization(attr.to_sym) end end define_include_method attr end def associate(klass, attrs) #:nodoc: options = attrs.extract_options! self._associations = _associations.dup attrs.each do |attr| unless method_defined?(attr) define_method attr do object.send attr end end define_include_method attr self._associations[attr] = klass.refine(attr, options) end end def define_include_method(name) method = "include_#{name}?".to_sym INCLUDE_METHODS[name] = method unless method_defined?(method) define_method method do true end end end # Defines an association in the object should be rendered. # # The serializer object should implement the association name # as a method which should return an array when invoked. If a method # with the association name does not exist, the association name is # dispatched to the serialized object. def has_many(*attrs) associate(Associations::HasMany, attrs) end # Defines an association in the object should be rendered. # # The serializer object should implement the association name # as a method which should return an object when invoked. If a method # with the association name does not exist, the association name is # dispatched to the serialized object. def has_one(*attrs) associate(Associations::HasOne, attrs) end # Return a schema hash for the current serializer. This information # can be used to generate clients for the serialized output. # # The schema hash has two keys: +attributes+ and +associations+. # # The +attributes+ hash looks like this: # # { :name => :string, :age => :integer } # # The +associations+ hash looks like this: # { :posts => { :has_many => :posts } } # # If :key is used: # # class PostsSerializer < ActiveModel::Serializer # has_many :posts, :key => :my_posts # end # # the hash looks like this: # # { :my_posts => { :has_many => :posts } # # This information is extracted from the serializer's model class, # which is provided by +SerializerClass.model_class+. # # The schema method uses the +columns_hash+ and +reflect_on_association+ # methods, provided by default by ActiveRecord. You can implement these # methods on your custom models if you want the serializer's schema method # to work. # # TODO: This is currently coupled to Active Record. We need to # figure out a way to decouple those two. def schema klass = model_class columns = klass.columns_hash attrs = _attributes.inject({}) do |hash, (name,key)| column = columns[name.to_s] hash.merge key => column.type end associations = _associations.inject({}) do |hash, (attr,association_class)| association = association_class.new(attr, self) model_association = klass.reflect_on_association(association.name) hash.merge association.key => { model_association.macro => model_association.name } end { :attributes => attrs, :associations => associations } end # The model class associated with this serializer. def model_class name.sub(/Serializer$/, '').constantize end # Define how associations should be embedded. # # embed :objects # Embed associations as full objects # embed :ids # Embed only the association ids # embed :ids, :include => true # Embed the association ids and include objects in the root # def embed(type, options={}) self._embed = type self._root_embed = true if options[:include] end # Defines the root used on serialization. If false, disables the root. def root(name) self._root = name end def inherited(klass) #:nodoc: return if klass.anonymous? name = klass.name.demodulize.underscore.sub(/_serializer$/, '') klass.class_eval do alias_method name.to_sym, :object root name.to_sym unless self._root == false end end end attr_reader :object, :options def initialize(object, options={}) @object, @options = object, options end def url_options @options[:url_options] || {} end # Returns a json representation of the serializable # object including the root. def as_json(options={}) if root = options.fetch(:root, @options.fetch(:root, _root)) @options[:hash] = hash = {} @options[:unique_values] = {} hash.merge!(root => serializable_hash) hash else serializable_hash end end # Returns a hash representation of the serializable # object without the root. def serializable_hash instrument(:serialize, :serializer => self.class.name) do @node = attributes instrument :associations do include_associations! if _embed end @node end end def include_associations! _associations.each_key do |name| include!(name) if include?(name) end end def include?(name) send INCLUDE_METHODS[name] end def include!(name, options={}) # Make sure that if a special options[:hash] was passed in, we generate # a new unique values hash and don't clobber the original. If the hash # passed in is the same as the current options hash, use the current # unique values. # # TODO: Should passing in a Hash even be public API here? unique_values = if hash = options[:hash] if @options[:hash] == hash @options[:unique_values] ||= {} else {} end else hash = @options[:hash] @options[:unique_values] ||= {} end node = options[:node] ||= @node value = options[:value] if options[:include] == nil if @options.key?(:include) options[:include] = @options[:include].include?(name) elsif @options.include?(:exclude) options[:include] = !@options[:exclude].include?(name) end end association_class = if klass = _associations[name] klass elsif value.respond_to?(:to_ary) Associations::HasMany else Associations::HasOne end association = association_class.new(name, self, options) if association.embed_ids? node[association.key] = association.serialize_ids if association.embed_in_root? && hash.nil? raise IncludeError.new(self.class, association.name) elsif association.embed_in_root? && association.embeddable? merge_association hash, association.root, association.serialize_many, unique_values end elsif association.embed_objects? node[association.key] = association.serialize end end # In some cases, an Array of associations is built by merging the associated # content for all of the children. For instance, if a Post has_many comments, # which has_many tags, the top-level :tags key will contain the merged list # of all tags for all comments of the post. # # In order to make this efficient, we store a :unique_values hash containing # a unique list of all of the objects that are already in the Array. This # avoids the need to scan through the Array looking for entries every time # we want to merge a new list of values. def merge_association(hash, key, value, unique_values) if current_value = unique_values[key] current_value.merge! value hash[key] = current_value.to_a elsif value hash[key] = value unique_values[key] = OrderedSet.new(value) end end # Returns a hash representation of the serializable # object attributes. def attributes hash = {} _attributes.each do |name,key| hash[key] = read_attribute_for_serialization(name) if include?(name) end hash end # Returns options[:scope] def scope @options[:scope] end alias :read_attribute_for_serialization :send # Use ActiveSupport::Notifications to send events to external systems. # The event name is: name.class_name.serializer def instrument(name, payload = {}, &block) event_name = INSTRUMENT[name] ActiveSupport::Notifications.instrument(event_name, payload, &block) end end end