faraday 0.9.1 → 0.16.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (98) hide show
  1. checksums.yaml +5 -5
  2. data/LICENSE.md +1 -1
  3. data/README.md +30 -195
  4. data/lib/faraday/adapter/em_http.rb +148 -99
  5. data/lib/faraday/adapter/em_http_ssl_patch.rb +24 -18
  6. data/lib/faraday/adapter/em_synchrony/parallel_manager.rb +18 -15
  7. data/lib/faraday/adapter/em_synchrony.rb +107 -49
  8. data/lib/faraday/adapter/excon.rb +102 -55
  9. data/lib/faraday/adapter/httpclient.rb +80 -36
  10. data/lib/faraday/adapter/net_http.rb +119 -44
  11. data/lib/faraday/adapter/net_http_persistent.rb +68 -27
  12. data/lib/faraday/adapter/patron.rb +76 -34
  13. data/lib/faraday/adapter/rack.rb +28 -12
  14. data/lib/faraday/adapter/test.rb +136 -52
  15. data/lib/faraday/adapter/typhoeus.rb +7 -115
  16. data/lib/faraday/adapter.rb +43 -20
  17. data/lib/faraday/adapter_registry.rb +28 -0
  18. data/lib/faraday/autoload.rb +47 -36
  19. data/lib/faraday/connection.rb +359 -165
  20. data/lib/faraday/dependency_loader.rb +37 -0
  21. data/lib/faraday/encoders/flat_params_encoder.rb +94 -0
  22. data/lib/faraday/encoders/nested_params_encoder.rb +171 -0
  23. data/lib/faraday/error.rb +71 -24
  24. data/lib/faraday/file_part.rb +128 -0
  25. data/lib/faraday/logging/formatter.rb +92 -0
  26. data/lib/faraday/middleware.rb +4 -28
  27. data/lib/faraday/middleware_registry.rb +129 -0
  28. data/lib/faraday/options/connection_options.rb +22 -0
  29. data/lib/faraday/options/env.rb +181 -0
  30. data/lib/faraday/options/proxy_options.rb +28 -0
  31. data/lib/faraday/options/request_options.rb +21 -0
  32. data/lib/faraday/options/ssl_options.rb +59 -0
  33. data/lib/faraday/options.rb +57 -185
  34. data/lib/faraday/param_part.rb +53 -0
  35. data/lib/faraday/parameters.rb +4 -180
  36. data/lib/faraday/rack_builder.rb +74 -38
  37. data/lib/faraday/request/authorization.rb +42 -31
  38. data/lib/faraday/request/basic_authentication.rb +14 -7
  39. data/lib/faraday/request/instrumentation.rb +45 -27
  40. data/lib/faraday/request/multipart.rb +81 -45
  41. data/lib/faraday/request/retry.rb +212 -121
  42. data/lib/faraday/request/token_authentication.rb +15 -10
  43. data/lib/faraday/request/url_encoded.rb +41 -23
  44. data/lib/faraday/request.rb +84 -30
  45. data/lib/faraday/response/logger.rb +22 -48
  46. data/lib/faraday/response/raise_error.rb +36 -14
  47. data/lib/faraday/response.rb +29 -18
  48. data/lib/faraday/utils/headers.rb +139 -0
  49. data/lib/faraday/utils/params_hash.rb +61 -0
  50. data/lib/faraday/utils.rb +28 -216
  51. data/lib/faraday.rb +102 -204
  52. data/spec/external_adapters/faraday_specs_setup.rb +14 -0
  53. metadata +24 -94
  54. data/.document +0 -6
  55. data/CHANGELOG.md +0 -20
  56. data/CONTRIBUTING.md +0 -36
  57. data/Gemfile +0 -25
  58. data/Rakefile +0 -71
  59. data/faraday.gemspec +0 -34
  60. data/lib/faraday/upload_io.rb +0 -67
  61. data/script/cached-bundle +0 -46
  62. data/script/console +0 -7
  63. data/script/generate_certs +0 -42
  64. data/script/package +0 -7
  65. data/script/proxy-server +0 -42
  66. data/script/release +0 -17
  67. data/script/s3-put +0 -71
  68. data/script/server +0 -36
  69. data/script/test +0 -172
  70. data/test/adapters/default_test.rb +0 -14
  71. data/test/adapters/em_http_test.rb +0 -20
  72. data/test/adapters/em_synchrony_test.rb +0 -20
  73. data/test/adapters/excon_test.rb +0 -20
  74. data/test/adapters/httpclient_test.rb +0 -21
  75. data/test/adapters/integration.rb +0 -254
  76. data/test/adapters/logger_test.rb +0 -82
  77. data/test/adapters/net_http_persistent_test.rb +0 -20
  78. data/test/adapters/net_http_test.rb +0 -14
  79. data/test/adapters/patron_test.rb +0 -20
  80. data/test/adapters/rack_test.rb +0 -31
  81. data/test/adapters/test_middleware_test.rb +0 -114
  82. data/test/adapters/typhoeus_test.rb +0 -28
  83. data/test/authentication_middleware_test.rb +0 -65
  84. data/test/composite_read_io_test.rb +0 -111
  85. data/test/connection_test.rb +0 -522
  86. data/test/env_test.rb +0 -218
  87. data/test/helper.rb +0 -81
  88. data/test/live_server.rb +0 -67
  89. data/test/middleware/instrumentation_test.rb +0 -88
  90. data/test/middleware/retry_test.rb +0 -177
  91. data/test/middleware_stack_test.rb +0 -173
  92. data/test/multibyte.txt +0 -1
  93. data/test/options_test.rb +0 -252
  94. data/test/parameters_test.rb +0 -64
  95. data/test/request_middleware_test.rb +0 -142
  96. data/test/response_middleware_test.rb +0 -72
  97. data/test/strawberry.rb +0 -2
  98. data/test/utils_test.rb +0 -58
@@ -1,12 +1,20 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'faraday/adapter_registry'
4
+
1
5
  module Faraday
2
6
  # A Builder that processes requests into responses by passing through an inner
3
7
  # middleware stack (heavily inspired by Rack).
4
8
  #
5
- # Faraday::Connection.new(:url => 'http://sushi.com') do |builder|
9
+ # @example
10
+ # Faraday::Connection.new(url: 'http://sushi.com') do |builder|
6
11
  # builder.request :url_encoded # Faraday::Request::UrlEncoded
7
12
  # builder.adapter :net_http # Faraday::Adapter::NetHttp
8
13
  # end
9
14
  class RackBuilder
15
+ # Used to detect missing arguments
16
+ NO_ARGUMENT = Object.new
17
+
10
18
  attr_accessor :handlers
11
19
 
12
20
  # Error raised when trying to modify the stack after calling `lock!`
@@ -15,28 +23,28 @@ module Faraday
15
23
  # borrowed from ActiveSupport::Dependencies::Reference &
16
24
  # ActionDispatch::MiddlewareStack::Middleware
17
25
  class Handler
18
- @@constants_mutex = Mutex.new
19
- @@constants = Hash.new { |h, k|
20
- value = k.respond_to?(:constantize) ? k.constantize : Object.const_get(k)
21
- @@constants_mutex.synchronize { h[k] = value }
22
- }
26
+ REGISTRY = Faraday::AdapterRegistry.new
23
27
 
24
28
  attr_reader :name
25
29
 
26
30
  def initialize(klass, *args, &block)
27
31
  @name = klass.to_s
28
- if klass.respond_to?(:name)
29
- @@constants_mutex.synchronize { @@constants[@name] = klass }
30
- end
31
- @args, @block = args, block
32
+ REGISTRY.set(klass) if klass.respond_to?(:name)
33
+ @args = args
34
+ @block = block
35
+ end
36
+
37
+ def klass
38
+ REGISTRY.get(@name)
32
39
  end
33
40
 
34
- def klass() @@constants[@name] end
35
- def inspect() @name end
41
+ def inspect
42
+ @name
43
+ end
36
44
 
37
45
  def ==(other)
38
46
  if other.is_a? Handler
39
- self.name == other.name
47
+ name == other.name
40
48
  elsif other.respond_to? :name
41
49
  klass == other
42
50
  else
@@ -44,18 +52,19 @@ module Faraday
44
52
  end
45
53
  end
46
54
 
47
- def build(app)
55
+ def build(app = nil)
48
56
  klass.new(app, *@args, &@block)
49
57
  end
50
58
  end
51
59
 
52
- def initialize(handlers = [])
60
+ def initialize(handlers = [], adapter = nil, &block)
61
+ @adapter = adapter
53
62
  @handlers = handlers
54
63
  if block_given?
55
- build(&Proc.new)
64
+ build(&block)
56
65
  elsif @handlers.empty?
57
66
  # default stack, if nothing else is configured
58
- self.request :url_encoded
67
+ request :url_encoded
59
68
  self.adapter Faraday.default_adapter
60
69
  end
61
70
  end
@@ -64,13 +73,14 @@ module Faraday
64
73
  raise_if_locked
65
74
  @handlers.clear unless options[:keep]
66
75
  yield(self) if block_given?
76
+ adapter(Faraday.default_adapter) unless @adapter
67
77
  end
68
78
 
69
79
  def [](idx)
70
80
  @handlers[idx]
71
81
  end
72
82
 
73
- # Locks the middleware stack to ensure no further modifications are possible.
83
+ # Locks the middleware stack to ensure no further modifications are made.
74
84
  def lock!
75
85
  @handlers.freeze
76
86
  end
@@ -84,6 +94,7 @@ module Faraday
84
94
  use_symbol(Faraday::Middleware, klass, *args, &block)
85
95
  else
86
96
  raise_if_locked
97
+ raise_if_adapter(klass)
87
98
  @handlers << self.class::Handler.new(klass, *args, &block)
88
99
  end
89
100
  end
@@ -96,8 +107,11 @@ module Faraday
96
107
  use_symbol(Faraday::Response, key, *args, &block)
97
108
  end
98
109
 
99
- def adapter(key, *args, &block)
100
- use_symbol(Faraday::Adapter, key, *args, &block)
110
+ def adapter(klass = NO_ARGUMENT, *args, &block)
111
+ return @adapter if klass == NO_ARGUMENT
112
+
113
+ klass = Faraday::Adapter.lookup_middleware(klass) if klass.is_a?(Symbol)
114
+ @adapter = self.class::Handler.new(klass, *args, &block)
101
115
  end
102
116
 
103
117
  ## methods to push onto the various positions in the stack:
@@ -109,7 +123,7 @@ module Faraday
109
123
  @handlers.insert(index, handler)
110
124
  end
111
125
 
112
- alias_method :insert_before, :insert
126
+ alias insert_before insert
113
127
 
114
128
  def insert_after(index, *args, &block)
115
129
  index = assert_index(index)
@@ -131,10 +145,10 @@ module Faraday
131
145
  # Processes a Request into a Response by passing it through this Builder's
132
146
  # middleware stack.
133
147
  #
134
- # connection - Faraday::Connection
135
- # request - Faraday::Request
148
+ # @param connection [Faraday::Connection]
149
+ # @param request [Faraday::Request]
136
150
  #
137
- # Returns a Faraday::Response.
151
+ # @return [Faraday::Response]
138
152
  def build_response(connection, request)
139
153
  app.call(build_env(connection, request))
140
154
  end
@@ -149,25 +163,26 @@ module Faraday
149
163
  def app
150
164
  @app ||= begin
151
165
  lock!
152
- to_app(lambda { |env|
153
- response = Response.new
154
- response.finish(env) unless env.parallel?
155
- env.response = response
156
- })
166
+ to_app
157
167
  end
158
168
  end
159
169
 
160
- def to_app(inner_app)
170
+ def to_app
161
171
  # last added handler is the deepest and thus closest to the inner app
162
- @handlers.reverse.inject(inner_app) { |app, handler| handler.build(app) }
172
+ # adapter is always the last one
173
+ @handlers.reverse.inject(@adapter.build) do |app, handler|
174
+ handler.build(app)
175
+ end
163
176
  end
164
177
 
165
178
  def ==(other)
166
- other.is_a?(self.class) && @handlers == other.handlers
179
+ other.is_a?(self.class) &&
180
+ @handlers == other.handlers &&
181
+ @adapter == other.adapter
167
182
  end
168
183
 
169
184
  def dup
170
- self.class.new(@handlers.dup)
185
+ self.class.new(@handlers.dup, @adapter.dup)
171
186
  end
172
187
 
173
188
  # ENV Keys
@@ -187,16 +202,36 @@ module Faraday
187
202
  # :password - Proxy server password
188
203
  # :ssl - Hash of options for configuring SSL requests.
189
204
  def build_env(connection, request)
190
- Env.new(request.method, request.body,
191
- connection.build_exclusive_url(request.path, request.params),
192
- request.options, request.headers, connection.ssl,
193
- connection.parallel_manager)
205
+ exclusive_url = connection.build_exclusive_url(
206
+ request.path, request.params,
207
+ request.options.params_encoder
208
+ )
209
+
210
+ Env.new(request.method, request.body, exclusive_url,
211
+ request.options, request.headers, connection.ssl,
212
+ connection.parallel_manager)
194
213
  end
195
214
 
196
215
  private
197
216
 
217
+ LOCK_ERR = "can't modify middleware stack after making a request"
218
+
198
219
  def raise_if_locked
199
- raise StackLocked, "can't modify middleware stack after making a request" if locked?
220
+ raise StackLocked, LOCK_ERR if locked?
221
+ end
222
+
223
+ def raise_if_adapter(klass)
224
+ return unless is_adapter?(klass)
225
+
226
+ raise 'Adapter should be set using the `adapter` method, not `use`'
227
+ end
228
+
229
+ def adapter_set?
230
+ !@adapter.nil?
231
+ end
232
+
233
+ def is_adapter?(klass) # rubocop:disable Naming/PredicateName
234
+ klass.ancestors.include?(Faraday::Adapter)
200
235
  end
201
236
 
202
237
  def use_symbol(mod, key, *args, &block)
@@ -206,6 +241,7 @@ module Faraday
206
241
  def assert_index(index)
207
242
  idx = index.is_a?(Integer) ? index : @handlers.index(index)
208
243
  raise "No such handler: #{index.inspect}" unless idx
244
+
209
245
  idx
210
246
  end
211
247
  end
@@ -1,42 +1,53 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module Faraday
2
- class Request::Authorization < Faraday::Middleware
3
- KEY = "Authorization".freeze unless defined? KEY
4
+ class Request
5
+ # Request middleware for the Authorization HTTP header
6
+ class Authorization < Faraday::Middleware
7
+ KEY = 'Authorization' unless defined? KEY
4
8
 
5
- # Public
6
- def self.header(type, token)
7
- case token
8
- when String, Symbol
9
- "#{type} #{token}"
10
- when Hash
11
- build_hash(type.to_s, token)
12
- else
13
- raise ArgumentError, "Can't build an Authorization #{type} header from #{token.inspect}"
9
+ # @param type [String, Symbol]
10
+ # @param token [String, Symbol, Hash]
11
+ # @return [String] a header value
12
+ def self.header(type, token)
13
+ case token
14
+ when String, Symbol
15
+ "#{type} #{token}"
16
+ when Hash
17
+ build_hash(type.to_s, token)
18
+ else
19
+ raise ArgumentError,
20
+ "Can't build an Authorization #{type}" \
21
+ "header from #{token.inspect}"
22
+ end
14
23
  end
15
- end
16
24
 
17
- # Internal
18
- def self.build_hash(type, hash)
19
- offset = KEY.size + type.size + 3
20
- comma = ",\n#{' ' * offset}"
21
- values = []
22
- hash.each do |key, value|
23
- values << "#{key}=#{value.to_s.inspect}"
25
+ # @param type [String]
26
+ # @param hash [Hash]
27
+ # @return [String] type followed by comma-separated key=value pairs
28
+ # @api private
29
+ def self.build_hash(type, hash)
30
+ comma = ', '
31
+ values = []
32
+ hash.each do |key, value|
33
+ values << "#{key}=#{value.to_s.inspect}"
34
+ end
35
+ "#{type} #{values * comma}"
24
36
  end
25
- "#{type} #{values * comma}"
26
- end
27
37
 
28
- def initialize(app, type, token)
29
- @header_value = self.class.header(type, token)
30
- super(app)
31
- end
38
+ # @param app [#call]
39
+ # @param type [String, Symbol] Type of Authorization
40
+ # @param token [String, Symbol, Hash] Token value for the Authorization
41
+ def initialize(app, type, token)
42
+ @header_value = self.class.header(type, token)
43
+ super(app)
44
+ end
32
45
 
33
- # Public
34
- def call(env)
35
- unless env.request_headers[KEY]
36
- env.request_headers[KEY] = @header_value
46
+ # @param env [Faraday::Env]
47
+ def call(env)
48
+ env.request_headers[KEY] = @header_value unless env.request_headers[KEY]
49
+ @app.call(env)
37
50
  end
38
- @app.call(env)
39
51
  end
40
52
  end
41
53
  end
42
-
@@ -1,13 +1,20 @@
1
+ # frozen_string_literal: true
2
+
1
3
  require 'base64'
2
4
 
3
5
  module Faraday
4
- class Request::BasicAuthentication < Request.load_middleware(:authorization)
5
- # Public
6
- def self.header(login, pass)
7
- value = Base64.encode64([login, pass].join(':'))
8
- value.gsub!("\n", '')
9
- super(:Basic, value)
6
+ class Request
7
+ # Authorization middleware for Basic Authentication.
8
+ class BasicAuthentication < load_middleware(:authorization)
9
+ # @param login [String]
10
+ # @param pass [String]
11
+ #
12
+ # @return [String] a Basic Authentication header line
13
+ def self.header(login, pass)
14
+ value = Base64.encode64([login, pass].join(':'))
15
+ value.delete!("\n")
16
+ super(:Basic, value)
17
+ end
10
18
  end
11
19
  end
12
20
  end
13
-
@@ -1,35 +1,53 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module Faraday
2
- class Request::Instrumentation < Faraday::Middleware
3
- class Options < Faraday::Options.new(:name, :instrumenter)
4
- def name
5
- self[:name] ||= 'request.faraday'
6
- end
4
+ class Request
5
+ # Middleware for instrumenting Requests.
6
+ class Instrumentation < Faraday::Middleware
7
+ # Options class used in Request::Instrumentation class.
8
+ class Options < Faraday::Options.new(:name, :instrumenter)
9
+ # @return [String]
10
+ def name
11
+ self[:name] ||= 'request.faraday'
12
+ end
7
13
 
8
- def instrumenter
9
- self[:instrumenter] ||= ActiveSupport::Notifications
14
+ # @return [Class]
15
+ def instrumenter
16
+ self[:instrumenter] ||= ActiveSupport::Notifications
17
+ end
10
18
  end
11
- end
12
19
 
13
- # Public: Instruments requests using Active Support.
14
- #
15
- # Measures time spent only for synchronous requests.
16
- #
17
- # Examples
18
- #
19
- # ActiveSupport::Notifications.subscribe('request.faraday') do |name, starts, ends, _, env|
20
- # url = env[:url]
21
- # http_method = env[:method].to_s.upcase
22
- # duration = ends - starts
23
- # $stderr.puts '[%s] %s %s (%.3f s)' % [url.host, http_method, url.request_uri, duration]
24
- # end
25
- def initialize(app, options = nil)
26
- super(app)
27
- @name, @instrumenter = Options.from(options).values_at(:name, :instrumenter)
28
- end
20
+ # Instruments requests using Active Support.
21
+ #
22
+ # Measures time spent only for synchronous requests.
23
+ #
24
+ # @example Using ActiveSupport::Notifications to measure time spent
25
+ # for Faraday requests.
26
+ # ActiveSupport::Notifications
27
+ # .subscribe('request.faraday') do |name, starts, ends, _, env|
28
+ # url = env[:url]
29
+ # http_method = env[:method].to_s.upcase
30
+ # duration = ends - starts
31
+ # $stderr.puts '[%s] %s %s (%.3f s)' %
32
+ # [url.host, http_method, url.request_uri, duration]
33
+ # end
34
+ # @param app [#call]
35
+ # @param options [nil, Hash] Options hash
36
+ # @option options [String] :name ('request.faraday')
37
+ # Name of the instrumenter
38
+ # @option options [Class] :instrumenter (ActiveSupport::Notifications)
39
+ # Active Support instrumenter class.
40
+ def initialize(app, options = nil)
41
+ super(app)
42
+ @name, @instrumenter = Options.from(options)
43
+ .values_at(:name, :instrumenter)
44
+ end
29
45
 
30
- def call(env)
31
- @instrumenter.instrument(@name, env) do
32
- @app.call(env)
46
+ # @param env [Faraday::Env]
47
+ def call(env)
48
+ @instrumenter.instrument(@name, env) do
49
+ @app.call(env)
50
+ end
33
51
  end
34
52
  end
35
53
  end
@@ -1,61 +1,97 @@
1
- require File.expand_path("../url_encoded", __FILE__)
1
+ # frozen_string_literal: true
2
+
3
+ require File.expand_path('url_encoded', __dir__)
4
+ require 'securerandom'
2
5
 
3
6
  module Faraday
4
- class Request::Multipart < Request::UrlEncoded
5
- self.mime_type = 'multipart/form-data'.freeze
6
- DEFAULT_BOUNDARY = "-----------RubyMultipartPost".freeze unless defined? DEFAULT_BOUNDARY
7
+ class Request
8
+ # Middleware for supporting multi-part requests.
9
+ class Multipart < UrlEncoded
10
+ self.mime_type = 'multipart/form-data'
11
+ unless defined? DEFAULT_BOUNDARY_PREFIX
12
+ DEFAULT_BOUNDARY_PREFIX = '-----------RubyMultipartPost'
13
+ end
7
14
 
8
- def call(env)
9
- match_content_type(env) do |params|
10
- env.request.boundary ||= DEFAULT_BOUNDARY
11
- env.request_headers[CONTENT_TYPE] += "; boundary=#{env.request.boundary}"
12
- env.body = create_multipart(env, params)
15
+ # Checks for files in the payload, otherwise leaves everything untouched.
16
+ #
17
+ # @param env [Faraday::Env]
18
+ def call(env)
19
+ match_content_type(env) do |params|
20
+ env.request.boundary ||= unique_boundary
21
+ env.request_headers[CONTENT_TYPE] +=
22
+ "; boundary=#{env.request.boundary}"
23
+ env.body = create_multipart(env, params)
24
+ end
25
+ @app.call env
13
26
  end
14
- @app.call env
15
- end
16
27
 
17
- def process_request?(env)
18
- type = request_type(env)
19
- env.body.respond_to?(:each_key) and !env.body.empty? and (
20
- (type.empty? and has_multipart?(env.body)) or
21
- type == self.class.mime_type
22
- )
23
- end
28
+ # @param env [Faraday::Env]
29
+ def process_request?(env)
30
+ type = request_type(env)
31
+ env.body.respond_to?(:each_key) && !env.body.empty? && (
32
+ (type.empty? && has_multipart?(env.body)) ||
33
+ (type == self.class.mime_type)
34
+ )
35
+ end
24
36
 
25
- def has_multipart?(obj)
26
- # string is an enum in 1.8, returning list of itself
27
- if obj.respond_to?(:each) && !obj.is_a?(String)
28
- (obj.respond_to?(:values) ? obj.values : obj).each do |val|
29
- return true if (val.respond_to?(:content_type) || has_multipart?(val))
37
+ # Returns true if obj is an enumerable with values that are multipart.
38
+ #
39
+ # @param obj [Object]
40
+ # @return [Boolean]
41
+ def has_multipart?(obj) # rubocop:disable Naming/PredicateName
42
+ if obj.respond_to?(:each)
43
+ (obj.respond_to?(:values) ? obj.values : obj).each do |val|
44
+ return true if val.respond_to?(:content_type) || has_multipart?(val)
45
+ end
30
46
  end
47
+ false
31
48
  end
32
- false
33
- end
34
49
 
35
- def create_multipart(env, params)
36
- boundary = env.request.boundary
37
- parts = process_params(params) do |key, value|
38
- Faraday::Parts::Part.new(boundary, key, value)
50
+ # @param env [Faraday::Env]
51
+ # @param params [Hash]
52
+ def create_multipart(env, params)
53
+ boundary = env.request.boundary
54
+ parts = process_params(params) do |key, value|
55
+ part(boundary, key, value)
56
+ end
57
+ parts << Faraday::Parts::EpiloguePart.new(boundary)
58
+
59
+ body = Faraday::CompositeReadIO.new(parts)
60
+ env.request_headers[Faraday::Env::ContentLength] = body.length.to_s
61
+ body
39
62
  end
40
- parts << Faraday::Parts::EpiloguePart.new(boundary)
41
63
 
42
- body = Faraday::CompositeReadIO.new(parts)
43
- env.request_headers[Faraday::Env::ContentLength] = body.length.to_s
44
- return body
45
- end
64
+ def part(boundary, key, value)
65
+ if value.respond_to?(:to_part)
66
+ value.to_part(boundary, key)
67
+ else
68
+ Faraday::Parts::Part.new(boundary, key, value)
69
+ end
70
+ end
71
+
72
+ # @return [String]
73
+ def unique_boundary
74
+ "#{DEFAULT_BOUNDARY_PREFIX}-#{SecureRandom.hex}"
75
+ end
46
76
 
47
- def process_params(params, prefix = nil, pieces = nil, &block)
48
- params.inject(pieces || []) do |all, (key, value)|
49
- key = "#{prefix}[#{key}]" if prefix
77
+ # @param params [Hash]
78
+ # @param prefix [String]
79
+ # @param pieces [Array]
80
+ def process_params(params, prefix = nil, pieces = nil, &block)
81
+ params.inject(pieces || []) do |all, (key, value)|
82
+ key = "#{prefix}[#{key}]" if prefix
50
83
 
51
- case value
52
- when Array
53
- values = value.inject([]) { |a,v| a << [nil, v] }
54
- process_params(values, key, all, &block)
55
- when Hash
56
- process_params(value, key, all, &block)
57
- else
58
- all << block.call(key, value)
84
+ case value
85
+ when Array
86
+ values = value.inject([]) { |a, v| a << [nil, v] }
87
+ process_params(values, key, all, &block)
88
+ when Hash
89
+ process_params(value, key, all, &block)
90
+ else
91
+ # rubocop:disable Performance/RedundantBlockCall
92
+ all << block.call(key, value)
93
+ # rubocop:enable Performance/RedundantBlockCall
94
+ end
59
95
  end
60
96
  end
61
97
  end