okuramasafumi/alba

View on GitHub
lib/alba/resource.rb

Summary

Maintainability
B
5 hrs
Test Coverage
require_relative 'association'
require_relative 'conditional_attribute'
require_relative 'constants'
require_relative 'type'
require_relative 'typed_attribute'
require_relative 'nested_attribute'
require_relative 'deprecation'
require_relative 'layout'

module Alba
  # This module represents what should be serialized
  module Resource
    # @!parse include InstanceMethods
    # @!parse extend ClassMethods
    INTERNAL_VARIABLES = {_attributes: {}, _key: nil, _key_for_collection: nil, _meta: nil, _transform_type: :none, _transforming_root_key: false, _key_transformation_cascade: true, _on_error: nil, _on_nil: nil, _layout: nil, _collection_key: nil, _helper: nil, _resource_methods: []}.freeze # rubocop:disable Layout/LineLength
    private_constant :INTERNAL_VARIABLES

    WITHIN_DEFAULT = Object.new.freeze
    private_constant :WITHIN_DEFAULT

    # `setup` method is meta-programmatically defined here for performance.
    # @private
    def self.included(base) # rubocop:disable Metrics/MethodLength
      super
      setup_method_body = +'private def _setup;'
      base.class_eval do
        # Initialize
        INTERNAL_VARIABLES.each do |name, initial|
          instance_variable_set(:"@#{name}", initial.dup) unless instance_variable_defined?(:"@#{name}")
          setup_method_body << "@#{name} = self.class.#{name};"
        end
        base.define_method(:encode, Alba.encoder)
      end
      setup_method_body << 'end'
      base.class_eval(setup_method_body, __FILE__, __LINE__ + 1)
      base.include InstanceMethods
      base.extend ClassMethods
    end

    # Instance methods
    module InstanceMethods
      attr_reader :object, :params

      # @param object [Object] the object to be serialized
      # @param params [Hash] user-given Hash for arbitrary data
      # @param within [Object, nil, false, true] determines what associations to be serialized. If not set, it serializes all associations.
      def initialize(object, params: {}, within: WITHIN_DEFAULT)
        @object = object
        @params = params
        @within = within
        _setup
      end

      # Serialize object into JSON string
      #
      # @param root_key [Symbol, nil, true]
      # @param meta [Hash] metadata for this seialization
      # @return [String] serialized JSON string
      def serialize(root_key: nil, meta: {})
        serialize_with(as_json(root_key: root_key, meta: meta))
      end

      # For Rails compatibility
      # The first options is a dummy parameter
      #
      # @see #serialize
      # @see https://github.com/rails/rails/blob/7-0-stable/actionpack/lib/action_controller/metal/renderers.rb#L156
      def to_json(options = {}, root_key: nil, meta: {})
        confusing_options = options.keys.select { |k| k.to_sym == :only || k.to_sym == :except }
        unless confusing_options.empty?
          confusing_options.sort!
          confusing_options.map! { |s| "\"#{s}\"" }
          message = "You passed #{confusing_options.join(' and ')} options but ignored. Please refer to the document: https://github.com/okuramasafumi/alba/blob/main/docs/rails.md"
          Kernel.warn(message)
        end
        serialize(root_key: root_key, meta: meta)
      end

      # Returns a Hash correspondng {#serialize}
      #
      # @param _options [Hash] dummy parameter for Rails compatibility
      # @param root_key [Symbol, nil, true]
      # @param meta [Hash] metadata for this seialization
      # @return [Hash]
      def as_json(_options = {}, root_key: nil, meta: {})
        key = root_key.nil? ? fetch_key : root_key
        key = Alba.regularize_key(key)
        if key && !key.empty?
          h = {key => serializable_hash}
          hash_with_metadata(h, meta)
        else
          serializable_hash
        end
      end

      # A Hash for serialization
      #
      # @return [Hash]
      def serializable_hash
        collection? ? serializable_hash_for_collection : converter.call(@object)
      end
      alias to_h serializable_hash

      private

      def serialize_with(hash)
        serialized_json = encode(hash)
        return serialized_json unless @_layout

        @_layout.serialize(resource: self, serialized_json: serialized_json, binding: binding)
      end

      def hash_with_metadata(hash, meta)
        return hash if meta.empty? && @_meta&.last.nil?

        key, block = @_meta || :meta

        if key
          hash[key] = _metadata(block, meta)
        else
          _metadata(block, meta).each { |k, v| hash[k] = v }
        end
        hash
      end

      def _metadata(block, meta)
        block ? instance_eval(&block).merge(meta) : meta
      end

      def serializable_hash_for_collection
        if @_collection_key
          @object.to_h do |item|
            k = item.public_send(@_collection_key)
            key = Alba.regularize_key(k)
            [key, converter.call(item)]
          end
        else
          @object.each_with_object([], &collection_converter)
        end
      end

      # @return [String]
      def fetch_key
        k = collection? ? _key_for_collection : _key
        transforming_root_key? ? transform_key(k) : k
      end

      def _key_for_collection
        k = @_key_for_collection == true ? resource_name(pluralized: true) : @_key_for_collection
        Alba.regularize_key(k)
      end

      def _key
        k = @_key == true ? resource_name(pluralized: false) : @_key
        Alba.regularize_key(k)
      end

      def resource_name(pluralized: false)
        inflector = Alba.inflector
        raise Alba::Error, 'You must set inflector when setting root key as true.' unless inflector

        class_name = self.class.name
        suffix = class_name.end_with?('Resource') ? 'Resource' : 'Serializer'
        name = inflector.demodulize(class_name).delete_suffix(suffix)
        underscore_name = inflector.underscore(name)
        pluralized ? inflector.pluralize(underscore_name) : underscore_name
      end

      def transforming_root_key?
        @_transforming_root_key
      end

      def converter
        lambda do |obj|
          attributes_to_hash(obj, {})
        end
      end

      def collection_converter
        lambda do |obj, a|
          a << {}
          h = a.last
          attributes_to_hash(obj, h)
          a
        end
      end

      def attributes_to_hash(obj, hash)
        attributes.each do |key, attribute|
          set_key_and_attribute_body_from(obj, key, attribute, hash)
        rescue ::Alba::Error, FrozenError, TypeError
          raise
        rescue StandardError => e
          handle_error(e, obj, key, attribute, hash)
        end
        hash
      end

      # This is default behavior for getting attributes for serialization
      # Override this method to filter certain attributes
      def attributes
        @_attributes
      end

      # Default implementation for selecting attributes
      # Override this method to filter attributes based on key and value
      def select(_key, _value)
        true
      end

      def set_key_and_attribute_body_from(obj, key, attribute, hash)
        key = transform_key(key)
        value = fetch_attribute(obj, key, attribute)
        return unless select(key, value)

        hash[key] = value unless value == Alba::REMOVE_KEY
      end

      def handle_error(error, obj, key, attribute, hash)
        case (on_error = @_on_error || :raise)
        when :raise, nil then raise(error)
        when :nullify then hash[key] = nil
        when :ignore then nil
        when Proc
          key, value = on_error.call(error, obj, key, attribute, self.class)
          hash[key] = value
        else
          # :nocov:
          raise Alba::Error, 'Impossible path'
          # :nocov:
        end
      end

      def transform_key(key)
        return Alba.regularize_key(key) if @_transform_type == :none || key.nil? || key.empty? # We can skip transformation

        inflector = Alba.inflector
        raise Alba::Error, 'Inflector is nil. You must set inflector before transforming keys.' unless inflector

        Alba.regularize_key(_transform_key(inflector, key.to_s))
      end

      def _transform_key(inflector, key)
        case @_transform_type
        when :camel then inflector.camelize(key)
        when :lower_camel then inflector.camelize_lower(key)
        when :dash then inflector.dasherize(key)
        when :snake then inflector.underscore(key)
        else
          # :nocov:
          raise Alba::Error, "Unknown transform type: #{@_transform_type}"
          # :nocov:
        end
      end

      def fetch_attribute(obj, key, attribute) # rubocop:disable Metrics/CyclomaticComplexity
        value = case attribute
                when Symbol then fetch_attribute_from_object_and_resource(obj, attribute)
                when Proc then instance_exec(obj, &attribute)
                when Alba::Association then yield_if_within(attribute.name.to_sym) { |within| attribute.to_h(obj, params: params, within: within) }
                when TypedAttribute then attribute.value(obj)
                when NestedAttribute then attribute.value(object: obj, params: params, within: @within)
                when ConditionalAttribute then attribute.with_passing_condition(resource: self, object: obj) { |attr| fetch_attribute(obj, key, attr) }
                  # :nocov:
                else raise ::Alba::Error, "Unsupported type of attribute: #{attribute.class}"
                  # :nocov:
                end
        value.nil? && nil_handler ? instance_exec(obj, key, attribute, &nil_handler) : value
      end

      def fetch_attribute_from_object_and_resource(obj, attribute)
        _fetch_attribute_from_resource_first(obj, attribute)
      end

      def _fetch_attribute_from_object_first(obj, attribute)
        obj.__send__(attribute)
      rescue NoMethodError
        __send__(attribute, obj)
      end

      def _fetch_attribute_from_resource_first(obj, attribute)
        if @_resource_methods.include?(attribute)
          __send__(attribute, obj)
        else
          obj.__send__(attribute)
        end
      end

      def nil_handler
        @_on_nil
      end

      def yield_if_within(association_name)
        within = check_within(association_name)
        within ? yield(within) : Alba::REMOVE_KEY
      end

      def check_within(association_name)
        case @within
        when WITHIN_DEFAULT then WITHIN_DEFAULT # Default value, doesn't check within tree
        when Hash then @within.fetch(association_name, nil) # Traverse within tree
        when Array then @within.find { |item| item.to_sym == association_name }
        when Symbol then @within == association_name
        when nil, true, false then false # Stop here
        else
          raise Alba::Error, "Unknown type for within option: #{@within.class}"
        end
      end

      # Detect if object is a collection or not.
      # When object is a Struct, it's Enumerable but not a collection
      def collection?
        @object.is_a?(Enumerable) && !@object.is_a?(Struct)
      end
    end

    # Class methods
    module ClassMethods
      attr_reader(*INTERNAL_VARIABLES.keys)

      # This `method_added` is used for defining "resource methods"
      def method_added(method_name)
        _resource_methods << method_name.to_sym unless method_name.to_sym == :_setup
        super
      end

      # @private
      def inherited(subclass)
        super
        INTERNAL_VARIABLES.each_key { |name| subclass.instance_variable_set(:"@#{name}", instance_variable_get(:"@#{name}").clone) }
      end

      # Defining methods for DSLs and disable parameter number check since for users' benefits increasing params is fine

      # Set multiple attributes at once
      #
      # @param attrs [Array<String, Symbol>]
      # @param if [Proc] condition to decide if it should serialize these attributes
      # @param attrs_with_types [Hash<[Symbol, String], [Array<Symbol, Proc>, Symbol]>]
      #   attributes with name in its key and type and optional type converter in its value
      # @return [void]
      def attributes(*attrs, if: nil, **attrs_with_types)
        if_value = binding.local_variable_get(:if)
        assign_attributes(attrs, if_value)
        assign_attributes_with_types(attrs_with_types, if_value)
      end

      def assign_attributes(attrs, if_value)
        attrs.each do |attr_name|
          attr_name = attr_name.to_sym
          attr = if_value ? ConditionalAttribute.new(body: attr_name, condition: if_value) : attr_name
          @_attributes[attr_name] = attr
        end
      end
      private :assign_attributes

      def assign_attributes_with_types(attrs_with_types, if_value)
        attrs_with_types.each do |attr_name, type_and_converter|
          attr_name = attr_name.to_sym
          type, type_converter = type_and_converter
          typed_attr = TypedAttribute.new(name: attr_name, type: type, converter: type_converter)
          attr = if_value ? ConditionalAttribute.new(body: typed_attr, condition: if_value) : typed_attr
          @_attributes[attr_name] = attr
        end
      end
      private :assign_attributes_with_types

      # Set an attribute with the given block
      #
      # @param name [String, Symbol] key name
      # @param options [Hash<Symbol, Proc>]
      # @option options [Proc] if a condition to decide if this attribute should be serialized
      # @param block [Block] the block called during serialization
      # @raise [ArgumentError] if block is absent
      # @return [void]
      def attribute(name, **options, &block)
        raise ArgumentError, 'No block given in attribute method' unless block

        @_attributes[name.to_sym] = options[:if] ? ConditionalAttribute.new(body: block, condition: options[:if]) : block
      end

      # Set association
      #
      # @param name [String, Symbol] name of the association, used as key when `key` param doesn't exist
      # @param condition [Proc, nil] a Proc to modify the association
      # @param resource [Class<Alba::Resource>, String, Proc, nil] representing resource for this association
      # @param key [String, Symbol, nil] used as key when given
      # @param params [Hash] params override for the association
      # @param options [Hash<Symbol, Proc>]
      # @option options [Proc] if a condition to decide if this association should be serialized
      # @param block [Block]
      # @return [void]
      # @see Alba::Association#initialize
      def association(name, condition = nil, resource: nil, key: nil, params: {}, **options, &block)
        transformation = @_key_transformation_cascade ? @_transform_type : :none
        assoc = Association.new(
          name: name, condition: condition, resource: resource, params: params, nesting: nesting, key_transformation: transformation, helper: @_helper, &block
        )
        @_attributes[key&.to_sym || name.to_sym] = options[:if] ? ConditionalAttribute.new(body: assoc, condition: options[:if]) : assoc
      end
      alias one association
      alias many association
      alias has_one association
      alias has_many association

      def nesting
        if name.nil?
          nil
        else
          name.rpartition('::').first.then { |n| n.empty? ? nil : n }
        end
      end
      private :nesting

      # Set a nested attribute with the given block
      #
      # @param name [String, Symbol] key name
      # @param options [Hash<Symbol, Proc>]
      # @option options [Proc] if a condition to decide if this attribute should be serialized
      # @param block [Block] the block called during serialization
      # @raise [ArgumentError] if block is absent
      # @return [void]
      def nested_attribute(name, **options, &block)
        raise ArgumentError, 'No block given in attribute method' unless block

        key_transformation = @_key_transformation_cascade ? @_transform_type : :none
        attribute = NestedAttribute.new(key_transformation: key_transformation, &block)
        @_attributes[name.to_sym] = options[:if] ? ConditionalAttribute.new(body: attribute, condition: options[:if]) : attribute
      end
      alias nested nested_attribute

      # Set root key
      #
      # @param key [String, Symbol]
      # @param key_for_collection [String, Symbol]
      # @raise [NoMethodError] when key doesn't respond to `to_sym` method
      def root_key(key, key_for_collection = nil)
        @_key = key.to_sym
        @_key_for_collection = key_for_collection&.to_sym
      end

      # Set root key for collection
      #
      # @param key [String, Symbol]
      # @raise [NoMethodError] when key doesn't respond to `to_sym` method
      def root_key_for_collection(key)
        @_key = true
        @_key_for_collection = key.to_sym
      end

      # Set root key to true
      def root_key!
        @_key = true
        @_key_for_collection = true
      end

      # Set metadata
      def meta(key = :meta, &block)
        @_meta = [key, block]
      end

      # Set layout
      #
      # @param file [String] name of the layout file
      # @param inline [Proc] a proc returning JSON string or a Hash representing JSON
      def layout(file: nil, inline: nil)
        @_layout = Layout.new(file: file, inline: inline)
      end

      # Transform keys as specified type
      #
      # @param type [String, Symbol] one of `snake`, `:camel`, `:lower_camel`, `:dash` and `none`
      # @param root [Boolean] decides if root key also should be transformed
      # @param cascade [Boolean] decides if key transformation cascades into inline association
      #   Default is true but can be set false for old (v1) behavior
      # @raise [Alba::Error] when type is not supported
      def transform_keys(type, root: true, cascade: true)
        type = type.to_sym
        unless %i[none snake camel lower_camel dash].include?(type)
          # This should be `ArgumentError` but for backward compatibility it raises `Alba::Error`
          raise ::Alba::Error, "Unknown transform type: #{type}. Supported type are :camel, :lower_camel and :dash."
        end

        @_transform_type = type
        @_transforming_root_key = root
        @_key_transformation_cascade = cascade
      end

      # Sets key for collection serialization
      #
      # @param key [String, Symbol]
      def collection_key(key)
        @_collection_key = key.to_sym
      end

      # Set error handler
      # If this is set it's used as a error handler overriding global one
      #
      # @param handler [Symbol] `:raise`, `:ignore` or `:nullify`
      # @param block [Block]
      def on_error(handler = nil, &block)
        raise ArgumentError, 'You cannot specify error handler with both Symbol and block' if handler && block
        raise ArgumentError, 'You must specify error handler with either Symbol or block' unless handler || block

        @_on_error = block || validated_error_handler(handler)
      end

      def validated_error_handler(handler)
        unless %i[raise ignore nullify].include?(handler)
          # For backward compatibility
          # TODO: Change this to ArgumentError
          raise Alba::Error, "Unknown error handler: #{handler}. It must be one of `:raise`, `:ignore` or `:nullify`."
        end

        handler
      end
      private :validated_error_handler

      # Set nil handler
      #
      # @param block [Block]
      def on_nil(&block)
        @_on_nil = block
      end

      # Define helper methods
      #
      # @param mod [Module] a module to extend
      def helper(mod = @_helper || Module.new, &block)
        mod.module_eval(&block) if block
        extend mod
        @_helper = mod
      end

      # DSL for alias, purely for readability
      def prefer_resource_method!
        alias_method :fetch_attribute_from_object_and_resource, :_fetch_attribute_from_resource_first
      end

      # DSL for alias, purely for readability
      def prefer_object_method!
        alias_method :fetch_attribute_from_object_and_resource, :_fetch_attribute_from_object_first
      end
    end
  end

  Serializer = Resource
  public_constant :Serializer
end