class HTTParty::Request::Body

def boundary

def boundary
  @boundary ||= MultipartBoundary.generate
end

def call

def call
  if params.respond_to?(:to_hash)
    multipart? ? generate_multipart : normalize_query(params)
  else
    params
  end
end

def content_body(object)

def content_body(object)
  if file?(object)
    object = (file = object).read
    file.rewind if file.respond_to?(:rewind)
  end
  object.to_s
end

def content_type(object)

def content_type(object)
  return object.content_type if object.respond_to?(:content_type)
  mime = MiniMime.lookup_by_filename(object.path)
  mime ? mime.content_type : 'application/octet-stream'
end

def file?(object)

def file?(object)
  object.respond_to?(:path) && object.respond_to?(:read)
end

def file_name(object)

def file_name(object)
  object.respond_to?(:original_filename) ? object.original_filename : File.basename(object.path)
end

def generate_multipart

def generate_multipart
  normalized_params = params.flat_map { |key, value| HashConversions.normalize_keys(key, value) }
  multipart = normalized_params.inject(''.dup) do |memo, (key, value)|
    memo << "--#{boundary}#{NEWLINE}"
    memo << %(Content-Disposition: form-data; name="#{key}")
    # value.path is used to support ActionDispatch::Http::UploadedFile
    # https://github.com/jnunemaker/httparty/pull/585
    memo << %(; filename="#{file_name(value).gsub(/["\r\n]/, MULTIPART_FORM_DATA_REPLACEMENT_TABLE)}") if file?(value)
    memo << NEWLINE
    memo << "Content-Type: #{content_type(value)}#{NEWLINE}" if file?(value)
    memo << NEWLINE
    memo << content_body(value)
    memo << NEWLINE
  end
  multipart << "--#{boundary}--#{NEWLINE}"
end

def has_file?(value)

def has_file?(value)
  if value.respond_to?(:to_hash)
    value.to_hash.any? { |_, v| has_file?(v) }
  elsif value.respond_to?(:to_ary)
    value.to_ary.any? { |v| has_file?(v) }
  else
    file?(value)
  end
end

def initialize(params, query_string_normalizer: nil, force_multipart: false)

def initialize(params, query_string_normalizer: nil, force_multipart: false)
  @params = params
  @query_string_normalizer = query_string_normalizer
  @force_multipart = force_multipart
end

def multipart?

def multipart?
  params.respond_to?(:to_hash) && (force_multipart || has_file?(params))
end

def normalize_query(query)

def normalize_query(query)
  if query_string_normalizer
    query_string_normalizer.call(query)
  else
    HashConversions.to_params(query)
  end
end