# frozen_string_literal: true require 'securerandom' module Faraday module Multipart # Middleware for supporting multi-part requests. class Middleware < Faraday::Middleware CONTENT_TYPE = 'Content-Type' DEFAULT_BOUNDARY_PREFIX = '-----------RubyMultipartPost' def initialize(app = nil, options = {}) super(app) @options = options end # Checks for files in the payload, otherwise leaves everything untouched. # # @param env [Faraday::Env] def call(env) match_content_type(env) do |params| env.request.boundary ||= unique_boundary env.request_headers[CONTENT_TYPE] += "; boundary=#{env.request.boundary}" env.body = create_multipart(env, params) end @app.call env end private # @param env [Faraday::Env] # @yield [request_body] Body of the request def match_content_type(env) return unless process_request?(env) env.request_headers[CONTENT_TYPE] ||= mime_type return if env.body.respond_to?(:to_str) || env.body.respond_to?(:read) yield(env.body) end # @param env [Faraday::Env] def process_request?(env) type = request_type(env) env.body.respond_to?(:each_key) && !env.body.empty? && ( (type.empty? && has_multipart?(env.body)) || (type == mime_type) ) end # @param env [Faraday::Env] # # @return [String] def request_type(env) type = env.request_headers[CONTENT_TYPE].to_s type = type.split(';', 2).first if type.index(';') type end # Returns true if obj is an enumerable with values that are multipart. # # @param obj [Object] # @return [Boolean] def has_multipart?(obj) if obj.respond_to?(:each) (obj.respond_to?(:values) ? obj.values : obj).each do |val| return true if val.respond_to?(:content_type) || has_multipart?(val) end end false end # @param env [Faraday::Env] # @param params [Hash] def create_multipart(env, params) boundary = env.request.boundary parts = process_params(params) do |key, value| part(boundary, key, value) end parts << Faraday::Multipart::Parts::EpiloguePart.new(boundary) body = Faraday::Multipart::CompositeReadIO.new(parts) env.request_headers[Faraday::Env::ContentLength] = body.length.to_s body end def part(boundary, key, value) if value.respond_to?(:to_part) value.to_part(boundary, key) else Faraday::Multipart::Parts::Part.new(boundary, key, value) end end # @return [String] def unique_boundary "#{DEFAULT_BOUNDARY_PREFIX}-#{SecureRandom.hex}" end # @param params [Hash] # @param prefix [String] # @param pieces [Array] def process_params(params, prefix = nil, pieces = nil, &block) params.inject(pieces || []) do |all, (key, value)| if prefix key = @options[:flat_encode] ? prefix.to_s : "#{prefix}[#{key}]" end case value when Array values = value.inject([]) { |a, v| a << [nil, v] } process_params(values, key, all, &block) when Hash process_params(value, key, all, &block) else all << block.call(key, value) # rubocop:disable Performance/RedundantBlockCall end end end # Determines and provides the multipart mime type for the request. # # @return [String] the multipart mime type def mime_type @mime_type ||= if @options[:content_type].to_s.match?(%r{\Amultipart/.+}) @options[:content_type].to_s else 'multipart/form-data' end end end end end