mirror of
https://github.com/ditkrg/active_model_serializers.git
synced 2026-01-23 06:16:50 +00:00
Extract Serializer Attributes into its own file
This commit is contained in:
parent
eceb2d5598
commit
036604b149
@ -3,6 +3,7 @@ require 'active_model/serializer/collection_serializer'
|
||||
require 'active_model/serializer/array_serializer'
|
||||
require 'active_model/serializer/include_tree'
|
||||
require 'active_model/serializer/associations'
|
||||
require 'active_model/serializer/attributes'
|
||||
require 'active_model/serializer/configuration'
|
||||
require 'active_model/serializer/fieldset'
|
||||
require 'active_model/serializer/lint'
|
||||
@ -13,35 +14,8 @@ module ActiveModel
|
||||
class Serializer
|
||||
include Configuration
|
||||
include Associations
|
||||
include Attributes
|
||||
require 'active_model/serializer/adapter'
|
||||
class Attribute
|
||||
delegate :call, to: :reader
|
||||
attr_reader :name, :reader
|
||||
def initialize(name)
|
||||
@name = name
|
||||
@reader = nil
|
||||
end
|
||||
|
||||
def self.build(name, block)
|
||||
if block
|
||||
AttributeBlock.new(name, block)
|
||||
else
|
||||
AttributeReader.new(name)
|
||||
end
|
||||
end
|
||||
end
|
||||
class AttributeReader < Attribute
|
||||
def initialize(name)
|
||||
super(name)
|
||||
@reader = ->(instance) { instance.read_attribute_for_serialization(name) }
|
||||
end
|
||||
end
|
||||
class AttributeBlock < Attribute
|
||||
def initialize(name, block)
|
||||
super(name)
|
||||
@reader = ->(instance) { instance.instance_eval(&block) }
|
||||
end
|
||||
end
|
||||
|
||||
# Matches
|
||||
# "c:/git/emberjs/ember-crm-backend/app/serializers/lead_serializer.rb:1:in `<top (required)>'"
|
||||
@ -73,12 +47,9 @@ module ActiveModel
|
||||
end
|
||||
|
||||
with_options instance_writer: false, instance_reader: false do |serializer|
|
||||
class_attribute :_type, instance_reader: true
|
||||
class_attribute :_attribute_mappings # @api private : maps attribute key names to names to names of implementing methods, @see Serializer#attribute
|
||||
self._attribute_mappings ||= {}
|
||||
class_attribute :_links # @api private : links definitions, @see Serializer#link
|
||||
serializer.class_attribute :_type, instance_reader: true
|
||||
serializer.class_attribute :_links # @api private : links definitions, @see Serializer#link
|
||||
self._links ||= {}
|
||||
|
||||
serializer.class_attribute :_cache # @api private : the cache object
|
||||
serializer.class_attribute :_fragmented # @api private : @see ::fragmented
|
||||
serializer.class_attribute :_cache_key # @api private : when present, is first item in cache_key
|
||||
@ -95,11 +66,10 @@ module ActiveModel
|
||||
serializer.class_attribute :_cache_digest # @api private : Generated
|
||||
end
|
||||
|
||||
# Serializers inherit serialized_attributes, _attributes_keys, and _reflections.
|
||||
# Serializers inherit _attribute_mappings, _reflections, and _links.
|
||||
# Generates a unique digest for each serializer at load.
|
||||
def self.inherited(base)
|
||||
caller_line = caller.first
|
||||
base._attribute_mappings = _attribute_mappings.dup
|
||||
base._links = _links.dup
|
||||
base._cache_digest = digest_caller_file(caller_line)
|
||||
super
|
||||
@ -116,54 +86,6 @@ module ActiveModel
|
||||
_links[name] = block || value
|
||||
end
|
||||
|
||||
# @example
|
||||
# class AdminAuthorSerializer < ActiveModel::Serializer
|
||||
# attributes :id, :name, :recent_edits
|
||||
def self.attributes(*attrs)
|
||||
attrs = attrs.first if attrs.first.class == Array
|
||||
|
||||
attrs.each do |attr|
|
||||
attribute(attr)
|
||||
end
|
||||
end
|
||||
|
||||
# TODO: remove the dynamic method definition
|
||||
# @example
|
||||
# class AdminAuthorSerializer < ActiveModel::Serializer
|
||||
# attributes :id, :recent_edits
|
||||
# attribute :name, key: :title
|
||||
#
|
||||
# attribute :full_name do
|
||||
# "#{object.first_name} #{object.last_name}"
|
||||
# end
|
||||
#
|
||||
# def recent_edits
|
||||
# object.edits.last(5)
|
||||
# end
|
||||
def self.attribute(attr, options = {}, &block)
|
||||
key = options.fetch(:key, attr)
|
||||
_attribute_mappings[key] = Attribute.build(attr, block)
|
||||
end
|
||||
|
||||
# @api private
|
||||
# names of attribute methods
|
||||
# @see Serializer::attribute
|
||||
def self._attributes
|
||||
_attribute_mappings.keys
|
||||
end
|
||||
|
||||
# @api private
|
||||
# maps attribute value to explict key name
|
||||
# @see Serializer::attribute
|
||||
# @see Adapter::FragmentCache#fragment_serializer
|
||||
def self._attributes_keys
|
||||
_attribute_mappings
|
||||
.each_with_object({}) do |(key, attribute_mapping), hash|
|
||||
next if key == attribute_mapping.name
|
||||
hash[attribute_mapping.name] = { key: key }
|
||||
end
|
||||
end
|
||||
|
||||
# @api private
|
||||
# Used by FragmentCache on the CachedSerializer
|
||||
# to call attribute methods on the fragmented cached serializer.
|
||||
@ -286,15 +208,6 @@ module ActiveModel
|
||||
root || object.class.model_name.to_s.underscore
|
||||
end
|
||||
|
||||
# Return the +attributes+ of +object+ as presented
|
||||
# by the serializer.
|
||||
def attributes(requested_attrs = nil)
|
||||
self.class._attribute_mappings.each_with_object({}) do |(key, attribute_mapping), hash|
|
||||
next unless requested_attrs.nil? || requested_attrs.include?(key)
|
||||
hash[key] = attribute_mapping.call(self)
|
||||
end
|
||||
end
|
||||
|
||||
def read_attribute_for_serialization(attr)
|
||||
if _serializer_method_defined?(attr)
|
||||
send(attr)
|
||||
|
||||
107
lib/active_model/serializer/attributes.rb
Normal file
107
lib/active_model/serializer/attributes.rb
Normal file
@ -0,0 +1,107 @@
|
||||
module ActiveModel
|
||||
class Serializer
|
||||
module Attributes
|
||||
class Attribute
|
||||
delegate :call, to: :reader
|
||||
attr_reader :name, :reader
|
||||
def initialize(name)
|
||||
@name = name
|
||||
@reader = nil
|
||||
end
|
||||
|
||||
def self.build(name, block)
|
||||
if block
|
||||
AttributeBlock.new(name, block)
|
||||
else
|
||||
AttributeReader.new(name)
|
||||
end
|
||||
end
|
||||
end
|
||||
class AttributeReader < Attribute
|
||||
def initialize(name)
|
||||
super(name)
|
||||
@reader = ->(instance) { instance.read_attribute_for_serialization(name) }
|
||||
end
|
||||
end
|
||||
class AttributeBlock < Attribute
|
||||
def initialize(name, block)
|
||||
super(name)
|
||||
@reader = ->(instance) { instance.instance_eval(&block) }
|
||||
end
|
||||
end
|
||||
|
||||
extend ActiveSupport::Concern
|
||||
|
||||
included do
|
||||
with_options instance_writer: false, instance_reader: false do |serializer|
|
||||
serializer.class_attribute :_attribute_mappings # @api private : maps attribute key names to names to names of implementing methods, @see #attribute
|
||||
self._attribute_mappings ||= {}
|
||||
end
|
||||
|
||||
# Return the +attributes+ of +object+ as presented
|
||||
# by the serializer.
|
||||
def attributes(requested_attrs = nil)
|
||||
self.class._attribute_mappings.each_with_object({}) do |(key, attribute_mapping), hash|
|
||||
next unless requested_attrs.nil? || requested_attrs.include?(key)
|
||||
hash[key] = attribute_mapping.call(self)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
module ClassMethods
|
||||
def inherited(base)
|
||||
super
|
||||
base._attribute_mappings = _attribute_mappings.dup
|
||||
end
|
||||
|
||||
# @example
|
||||
# class AdminAuthorSerializer < ActiveModel::Serializer
|
||||
# attributes :id, :name, :recent_edits
|
||||
def attributes(*attrs)
|
||||
attrs = attrs.first if attrs.first.class == Array
|
||||
|
||||
attrs.each do |attr|
|
||||
attribute(attr)
|
||||
end
|
||||
end
|
||||
|
||||
# TODO: remove the dynamic method definition
|
||||
# @example
|
||||
# class AdminAuthorSerializer < ActiveModel::Serializer
|
||||
# attributes :id, :recent_edits
|
||||
# attribute :name, key: :title
|
||||
#
|
||||
# attribute :full_name do
|
||||
# "#{object.first_name} #{object.last_name}"
|
||||
# end
|
||||
#
|
||||
# def recent_edits
|
||||
# object.edits.last(5)
|
||||
# end
|
||||
def attribute(attr, options = {}, &block)
|
||||
key = options.fetch(:key, attr)
|
||||
_attribute_mappings[key] = Attribute.build(attr, block)
|
||||
end
|
||||
|
||||
# @api private
|
||||
# names of attribute methods
|
||||
# @see Serializer::attribute
|
||||
def _attributes
|
||||
_attribute_mappings.keys
|
||||
end
|
||||
|
||||
# @api private
|
||||
# maps attribute value to explict key name
|
||||
# @see Serializer::attribute
|
||||
# @see Adapter::FragmentCache#fragment_serializer
|
||||
def _attributes_keys
|
||||
_attribute_mappings
|
||||
.each_with_object({}) do |(key, attribute_mapping), hash|
|
||||
next if key == attribute_mapping.name
|
||||
hash[attribute_mapping.name] = { key: key }
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
Loading…
Reference in New Issue
Block a user