lib/dry/types/constrained/coercible.rb



# frozen_string_literal: true

module Dry
  module Types
    class Constrained
      # Common coercion-related API for constrained types
      #
      # @api public
      class Coercible < Constrained
        # @return [Object]
        #
        # @api private
        def call_unsafe(input)
          coerced = type.call_unsafe(input)
          result = rule.(coerced)

          if result.success?
            coerced
          else
            raise ConstraintError.new(result, input)
          end
        end

        # @return [Object]
        #
        # @api private
        def call_safe(input)
          coerced = type.call_safe(input) { return yield }

          if rule[coerced]
            coerced
          else
            yield(coerced)
          end
        end

        # @see Dry::Types::Constrained#try
        #
        # @api public
        def try(input, &)
          result = type.try(input)

          if result.success?
            validation = rule.(result.input)

            if validation.success?
              result
            else
              failure = failure(result.input, ConstraintError.new(validation, input))
              block_given? ? yield(failure) : failure
            end
          else
            block_given? ? yield(result) : result
          end
        end
      end
    end
  end
end