faraday 2.13.1 → 2.14.2

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.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 7ab8d55c7b360596e25721ecc32dcf99c65cea19d04598bb468fc2a422e26146
4
- data.tar.gz: 624ddc291aec0a438fded658897b2127d62b8fe10e00590e3ad0a609307711d6
3
+ metadata.gz: 05f10cf15f3eb3f35b39edc67e14c374306d286f7b282ce86e131cab66ef9726
4
+ data.tar.gz: 17df5c4c9c054080f80038a1d8c6e1eaf88fd5d319c5ca1f04b10bc7bc0fe350
5
5
  SHA512:
6
- metadata.gz: e2394c6ccb0b3e12d16e2dcd739d8ecf053199b40ce5d8d4a1902cc141c12746190e21b928e4979405ffcb2660db80a4777d46ee839df10f75cb66b34039285e
7
- data.tar.gz: d91f5258bb49c40ff720aaba295ab4c86ea975cb7e0d4ad96647921e34924013501c1b4dddcf1c5497488fda0480ef39296f76b16188201f0162da004a490ba5
6
+ metadata.gz: 511bc21e7d0e4c4ecf866f84b434e6f38d6c4eada9d17148f25db9c6cf3194c4fe6fe600cc1b41d288e611e8dbfcbbbb5e2d711051a4340239741bb9dae54b73
7
+ data.tar.gz: e527f43b558124d58d4c4826dc46d0716a6689fec48dea4daf47a3943e1cae5c3612b2e490c956811d0d2ba501290548e85f2b0879bd924fd026566a2dc6056e
@@ -26,7 +26,7 @@ class Client
26
26
  end
27
27
 
28
28
  # Example API client test
29
- class ClientTest < Test::Unit::TestCase
29
+ class ClientTest < Test::Unit::TestCase # rubocop:disable Style/OneClassPerFile
30
30
  def test_httpbingo_name
31
31
  stubs = Faraday::Adapter::Test::Stubs.new
32
32
  stubs.get('/api') do |env|
@@ -8,6 +8,12 @@ module Faraday
8
8
 
9
9
  CONTENT_LENGTH = 'Content-Length'
10
10
 
11
+ TIMEOUT_KEYS = {
12
+ read: :read_timeout,
13
+ open: :open_timeout,
14
+ write: :write_timeout
15
+ }.freeze
16
+
11
17
  # This module marks an Adapter as supporting parallel requests.
12
18
  module Parallelism
13
19
  attr_writer :supports_parallel
@@ -23,6 +29,7 @@ module Faraday
23
29
  end
24
30
 
25
31
  extend Parallelism
32
+
26
33
  self.supports_parallel = false
27
34
 
28
35
  def initialize(_app = nil, opts = {}, &block)
@@ -89,12 +96,6 @@ module Faraday
89
96
  end
90
97
  options[key] || options[:timeout]
91
98
  end
92
-
93
- TIMEOUT_KEYS = {
94
- read: :read_timeout,
95
- open: :open_timeout,
96
- write: :write_timeout
97
- }.freeze
98
99
  end
99
100
  end
100
101
 
@@ -481,8 +481,11 @@ module Faraday
481
481
  if url && !base.path.end_with?('/')
482
482
  base.path = "#{base.path}/" # ensure trailing slash
483
483
  end
484
- # Ensure relative url will be parsed correctly (such as `service:search` )
485
- url = "./#{url}" if url.respond_to?(:start_with?) && !url.start_with?('http://', 'https://', '/', './', '../')
484
+ url = url.to_s if url.respond_to?(:host)
485
+ # Ensure relative url will be parsed correctly (such as `service:search` or `//evil.com`)
486
+ url = "./#{url}" if url.respond_to?(:start_with?) &&
487
+ (url.start_with?('//') ||
488
+ !url.start_with?('http://', 'https://', '/', './', '../'))
486
489
  uri = url ? base + url : base
487
490
  if params
488
491
  uri.query = params.to_query(params_encoder || options.params_encoder)
@@ -6,6 +6,7 @@ module Faraday
6
6
  module FlatParamsEncoder
7
7
  class << self
8
8
  extend Forwardable
9
+
9
10
  def_delegators :'Faraday::Utils', :escape, :unescape
10
11
  end
11
12
 
@@ -76,9 +77,9 @@ module Faraday
76
77
 
77
78
  empty_accumulator = {}
78
79
 
79
- split_query = (query.split('&').map do |pair|
80
+ split_query = query.split('&').filter_map do |pair|
80
81
  pair.split('=', 2) if pair && !pair.empty?
81
- end).compact
82
+ end
82
83
  split_query.each_with_object(empty_accumulator.dup) do |pair, accu|
83
84
  pair[0] = unescape(pair[0])
84
85
  pair[1] = true if pair[1].nil?
@@ -170,6 +170,7 @@ module Faraday
170
170
  attr_accessor :sort_params, :array_indices
171
171
 
172
172
  extend Forwardable
173
+
173
174
  def_delegators :'Faraday::Utils', :escape, :unescape
174
175
  end
175
176
 
data/lib/faraday/error.rb CHANGED
@@ -79,12 +79,46 @@ module Faraday
79
79
 
80
80
  # Pulls out potential parent exception and response hash.
81
81
  def exc_msg_and_response(exc, response = nil)
82
- return [exc, exc.message, response] if exc.respond_to?(:backtrace)
82
+ case exc
83
+ when Exception
84
+ [exc, exc.message, response]
85
+ when Hash
86
+ [nil, build_error_message_from_hash(exc), exc]
87
+ when Faraday::Env
88
+ [nil, build_error_message_from_env(exc), exc]
89
+ else
90
+ [nil, exc.to_s, response]
91
+ end
92
+ end
93
+
94
+ private
95
+
96
+ def build_error_message_from_hash(hash)
97
+ # Be defensive with external Hash objects - they might be missing keys
98
+ status = hash.fetch(:status, nil)
99
+ request = hash.fetch(:request, nil)
100
+
101
+ return fallback_error_message(status) if request.nil?
83
102
 
84
- return [nil, "the server responded with status #{exc[:status]}", exc] \
85
- if exc.respond_to?(:each_key)
103
+ method = request.fetch(:method, nil)
104
+ url = request.fetch(:url, nil)
105
+ build_status_error_message(status, method, url)
106
+ end
107
+
108
+ def build_error_message_from_env(env)
109
+ # Faraday::Env is internal - we can make reasonable assumptions about its structure
110
+ build_status_error_message(env.status, env.method, env.url)
111
+ end
86
112
 
87
- [nil, exc.to_s, response]
113
+ def build_status_error_message(status, method, url)
114
+ method_str = method ? method.to_s.upcase : ''
115
+ url_str = url ? url.to_s : ''
116
+ "the server responded with status #{status} for #{method_str} #{url_str}"
117
+ end
118
+
119
+ def fallback_error_message(status)
120
+ "the server responded with status #{status} - method and url are not available " \
121
+ 'due to include_request: false on Faraday::Response::RaiseError middleware'
88
122
  end
89
123
  end
90
124
 
@@ -121,9 +155,12 @@ module Faraday
121
155
  end
122
156
 
123
157
  # Raised by Faraday::Response::RaiseError in case of a 422 response.
124
- class UnprocessableEntityError < ClientError
158
+ class UnprocessableContentError < ClientError
125
159
  end
126
160
 
161
+ # Used to provide compatibility with legacy error name.
162
+ UnprocessableEntityError = UnprocessableContentError
163
+
127
164
  # Raised by Faraday::Response::RaiseError in case of a 429 response.
128
165
  class TooManyRequestsError < ClientError
129
166
  end
@@ -135,7 +172,7 @@ module Faraday
135
172
  # A unified client error for timeouts.
136
173
  class TimeoutError < ServerError
137
174
  def initialize(exc = 'timeout', response = nil)
138
- super(exc, response)
175
+ super
139
176
  end
140
177
  end
141
178
 
@@ -63,7 +63,7 @@ module Faraday
63
63
 
64
64
  def dump_body(body)
65
65
  if body.respond_to?(:to_str)
66
- body.to_str
66
+ body.to_str.encode(::Encoding::UTF_8, undef: :replace, invalid: :replace)
67
67
  else
68
68
  pretty_inspect(body)
69
69
  end
@@ -65,7 +65,7 @@ module Faraday
65
65
  if app.respond_to?(:close)
66
66
  app.close
67
67
  else
68
- warn "#{app} does not implement \#close!"
68
+ warn "#{app} does not implement #close!"
69
69
  end
70
70
  end
71
71
  end
@@ -78,7 +78,7 @@ module Faraday
78
78
  # @param value [Object] a value fitting Option.from(v).
79
79
  # @return [Env] from given value
80
80
  def self.from(value)
81
- env = super(value)
81
+ env = super
82
82
  if value.respond_to?(:custom_members)
83
83
  env.custom_members.update(value.custom_members)
84
84
  end
@@ -90,7 +90,7 @@ module Faraday
90
90
  return self[current_body] if key == :body
91
91
 
92
92
  if in_member_set?(key)
93
- super(key)
93
+ super
94
94
  else
95
95
  custom_members[key]
96
96
  end
@@ -105,7 +105,7 @@ module Faraday
105
105
  end
106
106
 
107
107
  if in_member_set?(key)
108
- super(key, value)
108
+ super
109
109
  else
110
110
  custom_members[key] = value
111
111
  end
@@ -7,6 +7,7 @@ module Faraday
7
7
  # class ProxyOptions < Options; end
8
8
  ProxyOptions = Options.new(:uri, :user, :password) do
9
9
  extend Forwardable
10
+
10
11
  def_delegators :uri, :scheme, :scheme=, :host, :host=, :port, :port=,
11
12
  :path, :path=
12
13
 
@@ -29,7 +30,7 @@ module Faraday
29
30
  end
30
31
  end
31
32
 
32
- super(value)
33
+ super
33
34
  end
34
35
 
35
36
  memoized(:user) { uri&.user && Utils.unescape(uri.user) }
@@ -12,7 +12,7 @@ module Faraday
12
12
  if key && key.to_sym == :proxy
13
13
  super(key, value ? ProxyOptions.from(value) : nil)
14
14
  else
15
- super(key, value)
15
+ super
16
16
  end
17
17
  end
18
18
 
@@ -186,7 +186,7 @@ module Faraday
186
186
  def [](key)
187
187
  key = key.to_sym
188
188
  if (method = self.class.memoized_attributes[key])
189
- super(key) || (self[key] = instance_eval(&method))
189
+ super || (self[key] = instance_eval(&method))
190
190
  else
191
191
  super
192
192
  end
@@ -15,6 +15,11 @@ module Faraday
15
15
  # Used to detect missing arguments
16
16
  NO_ARGUMENT = Object.new
17
17
 
18
+ LOCK_ERR = "can't modify middleware stack after making a request"
19
+ MISSING_ADAPTER_ERROR = "An attempt to run a request with a Faraday::Connection without adapter has been made.\n" \
20
+ "Please set Faraday.default_adapter or provide one when initializing the connection.\n" \
21
+ 'For more info, check https://lostisland.github.io/faraday/usage/.'
22
+
18
23
  attr_accessor :handlers
19
24
 
20
25
  # Error raised when trying to modify the stack after calling `lock!`
@@ -211,17 +216,12 @@ module Faraday
211
216
 
212
217
  private
213
218
 
214
- LOCK_ERR = "can't modify middleware stack after making a request"
215
- MISSING_ADAPTER_ERROR = "An attempt to run a request with a Faraday::Connection without adapter has been made.\n" \
216
- "Please set Faraday.default_adapter or provide one when initializing the connection.\n" \
217
- 'For more info, check https://lostisland.github.io/faraday/usage/.'
218
-
219
219
  def raise_if_locked
220
220
  raise StackLocked, LOCK_ERR if locked?
221
221
  end
222
222
 
223
223
  def raise_if_adapter(klass)
224
- return unless is_adapter?(klass)
224
+ return unless klass <= Faraday::Adapter
225
225
 
226
226
  raise 'Adapter should be set using the `adapter` method, not `use`'
227
227
  end
@@ -234,10 +234,6 @@ module Faraday
234
234
  !@adapter.nil?
235
235
  end
236
236
 
237
- def is_adapter?(klass) # rubocop:disable Naming/PredicateName
238
- klass <= Faraday::Adapter
239
- end
240
-
241
237
  def use_symbol(mod, key, ...)
242
238
  use(mod.lookup_middleware(key), ...)
243
239
  end
@@ -15,7 +15,7 @@ module Faraday
15
15
  404 => Faraday::ResourceNotFound,
16
16
  408 => Faraday::RequestTimeoutError,
17
17
  409 => Faraday::ConflictError,
18
- 422 => Faraday::UnprocessableEntityError,
18
+ 422 => Faraday::UnprocessableContentError,
19
19
  429 => Faraday::TooManyRequestsError
20
20
  }.freeze
21
21
  # rubocop:enable Naming/ConstantName
@@ -33,6 +33,10 @@ module Faraday
33
33
  finished? ? env.body : nil
34
34
  end
35
35
 
36
+ def url
37
+ finished? ? env.url : nil
38
+ end
39
+
36
40
  def finished?
37
41
  !!env
38
42
  end
@@ -60,9 +64,9 @@ module Faraday
60
64
 
61
65
  def to_hash
62
66
  {
63
- status: env.status, body: env.body,
64
- response_headers: env.response_headers,
65
- url: env.url
67
+ status: status, body: body,
68
+ response_headers: headers,
69
+ url: url
66
70
  }
67
71
  end
68
72
 
@@ -51,7 +51,7 @@ module Faraday
51
51
 
52
52
  def [](key)
53
53
  key = KeyMap[key]
54
- super(key) || super(@names[key.downcase])
54
+ super || super(@names[key.downcase])
55
55
  end
56
56
 
57
57
  def []=(key, val)
@@ -59,13 +59,13 @@ module Faraday
59
59
  key = (@names[key.downcase] ||= key)
60
60
  # join multiple values with a comma
61
61
  val = val.to_ary.join(', ') if val.respond_to?(:to_ary)
62
- super(key, val)
62
+ super
63
63
  end
64
64
 
65
65
  def fetch(key, ...)
66
66
  key = KeyMap[key]
67
67
  key = @names.fetch(key.downcase, key)
68
- super(key, ...)
68
+ super
69
69
  end
70
70
 
71
71
  def delete(key)
@@ -74,13 +74,13 @@ module Faraday
74
74
  return unless key
75
75
 
76
76
  @names.delete key.downcase
77
- super(key)
77
+ super
78
78
  end
79
79
 
80
80
  def dig(key, *rest)
81
81
  key = KeyMap[key]
82
82
  key = @names.fetch(key.downcase, key)
83
- super(key, *rest)
83
+ super
84
84
  end
85
85
 
86
86
  def include?(key)
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Faraday
4
- VERSION = '2.13.1'
4
+ VERSION = '2.14.2'
5
5
  end
data/lib/faraday.rb CHANGED
@@ -1,6 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'cgi'
3
+ require 'cgi/escape'
4
+ require 'cgi/util' if RUBY_VERSION < '3.5'
4
5
  require 'date'
5
6
  require 'set'
6
7
  require 'forwardable'
@@ -311,6 +311,46 @@ RSpec.describe Faraday::Connection do
311
311
  end
312
312
  end
313
313
 
314
+ context 'with protocol-relative URL (GHSA-33mh-2634-fwr2)' do
315
+ it 'does not allow host override with //evil.com/path' do
316
+ conn.url_prefix = 'http://httpbingo.org/api'
317
+ uri = conn.build_exclusive_url('//evil.com/path')
318
+ expect(uri.host).to eq('httpbingo.org')
319
+ end
320
+
321
+ it 'does not allow host override with URI("//evil.com/path")' do
322
+ conn.url_prefix = 'http://httpbingo.org/api'
323
+ uri = conn.build_exclusive_url(URI('//evil.com/path?token=1'))
324
+ expect(uri.host).to eq('httpbingo.org')
325
+ expect(uri.query).to eq('token=1')
326
+ end
327
+
328
+ it 'does not allow host override with //evil.com:8080/path' do
329
+ conn.url_prefix = 'http://httpbingo.org/api'
330
+ uri = conn.build_exclusive_url('//evil.com:8080/path')
331
+ expect(uri.host).to eq('httpbingo.org')
332
+ end
333
+
334
+ it 'does not allow host override with //user:pass@evil.com/path' do
335
+ conn.url_prefix = 'http://httpbingo.org/api'
336
+ uri = conn.build_exclusive_url('//user:pass@evil.com/path')
337
+ expect(uri.host).to eq('httpbingo.org')
338
+ end
339
+
340
+ it 'does not allow host override with ///evil.com' do
341
+ conn.url_prefix = 'http://httpbingo.org/api'
342
+ uri = conn.build_exclusive_url('///evil.com')
343
+ expect(uri.host).to eq('httpbingo.org')
344
+ end
345
+
346
+ it 'still allows single-slash absolute paths' do
347
+ conn.url_prefix = 'http://httpbingo.org/api'
348
+ uri = conn.build_exclusive_url('/safe/path')
349
+ expect(uri.host).to eq('httpbingo.org')
350
+ expect(uri.path).to eq('/safe/path')
351
+ end
352
+ end
353
+
314
354
  context 'with a custom `default_uri_parser`' do
315
355
  let(:url) { 'http://httpbingo.org' }
316
356
  let(:parser) { Addressable::URI }
@@ -23,7 +23,7 @@ RSpec.describe Faraday::Error do
23
23
 
24
24
  it { expect(subject.wrapped_exception).to be_nil }
25
25
  it { expect(subject.response).to eq(exception) }
26
- it { expect(subject.message).to eq('the server responded with status 400') }
26
+ it { expect(subject.message).to eq('the server responded with status 400 - method and url are not available due to include_request: false on Faraday::Response::RaiseError middleware') }
27
27
  if RUBY_VERSION >= '3.4'
28
28
  it { expect(subject.inspect).to eq('#<Faraday::Error response={status: 400}>') }
29
29
  else
@@ -89,5 +89,87 @@ RSpec.describe Faraday::Error do
89
89
  it { expect(subject.response_headers).to eq(headers) }
90
90
  it { expect(subject.response_body).to eq(body) }
91
91
  end
92
+
93
+ context 'with hash missing status key' do
94
+ let(:exception) { { body: 'error body' } }
95
+
96
+ it { expect(subject.wrapped_exception).to be_nil }
97
+ it { expect(subject.response).to eq(exception) }
98
+ it { expect(subject.message).to eq('the server responded with status - method and url are not available due to include_request: false on Faraday::Response::RaiseError middleware') }
99
+ end
100
+
101
+ context 'with hash with status but missing request data' do
102
+ let(:exception) { { status: 404, body: 'not found' } } # missing request key
103
+
104
+ it { expect(subject.wrapped_exception).to be_nil }
105
+ it { expect(subject.response).to eq(exception) }
106
+ it { expect(subject.message).to eq('the server responded with status 404 - method and url are not available due to include_request: false on Faraday::Response::RaiseError middleware') }
107
+ end
108
+
109
+ context 'with hash with status and request but missing method in request' do
110
+ let(:exception) { { status: 404, body: 'not found', request: { url: 'http://example.com/test' } } } # missing method
111
+
112
+ it { expect(subject.wrapped_exception).to be_nil }
113
+ it { expect(subject.response).to eq(exception) }
114
+ it { expect(subject.message).to eq('the server responded with status 404 for http://example.com/test') }
115
+ end
116
+
117
+ context 'with hash with status and request but missing url in request' do
118
+ let(:exception) { { status: 404, body: 'not found', request: { method: :get } } } # missing url
119
+
120
+ it { expect(subject.wrapped_exception).to be_nil }
121
+ it { expect(subject.response).to eq(exception) }
122
+ it { expect(subject.message).to eq('the server responded with status 404 for GET ') }
123
+ end
124
+
125
+ context 'with properly formed Faraday::Env' do
126
+ # This represents the normal case - a well-formed Faraday::Env object
127
+ # with all the standard properties populated as they would be during
128
+ # a typical HTTP request/response cycle
129
+ let(:exception) { Faraday::Env.new }
130
+
131
+ before do
132
+ exception.status = 500
133
+ exception.method = :post
134
+ exception.url = URI('https://api.example.com/users')
135
+ exception.request = Faraday::RequestOptions.new
136
+ exception.response_headers = { 'content-type' => 'application/json' }
137
+ exception.response_body = '{"error": "Internal server error"}'
138
+ exception.request_headers = { 'authorization' => 'Bearer token123' }
139
+ exception.request_body = '{"name": "John"}'
140
+ end
141
+
142
+ it { expect(subject.wrapped_exception).to be_nil }
143
+ it { expect(subject.response).to eq(exception) }
144
+ it { expect(subject.message).to eq('the server responded with status 500 for POST https://api.example.com/users') }
145
+ end
146
+
147
+ context 'with Faraday::Env missing status key' do
148
+ let(:exception) { Faraday::Env.new }
149
+
150
+ before do
151
+ exception[:body] = 'error body'
152
+ # Intentionally not setting status
153
+ end
154
+
155
+ it { expect(subject.wrapped_exception).to be_nil }
156
+ it { expect(subject.response).to eq(exception) }
157
+ it { expect(subject.message).to eq('the server responded with status for ') }
158
+ end
159
+
160
+ context 'with Faraday::Env with direct method and url properties' do
161
+ let(:exception) { Faraday::Env.new }
162
+
163
+ before do
164
+ exception.status = 404
165
+ exception.method = :get
166
+ exception.url = URI('http://example.com/test')
167
+ exception[:body] = 'not found'
168
+ end
169
+
170
+ it { expect(subject.wrapped_exception).to be_nil }
171
+ it { expect(subject.response).to eq(exception) }
172
+ it { expect(subject.message).to eq('the server responded with status 404 for GET http://example.com/test') }
173
+ end
92
174
  end
93
175
  end
@@ -52,14 +52,14 @@ RSpec.describe Faraday::Middleware do
52
52
  end
53
53
 
54
54
  describe '#close' do
55
- context "with app that doesn't support \#close" do
55
+ context "with app that doesn't support #close" do
56
56
  it 'should issue warning' do
57
57
  is_expected.to receive(:warn)
58
58
  subject.close
59
59
  end
60
60
  end
61
61
 
62
- context "with app that supports \#close" do
62
+ context 'with app that supports #close' do
63
63
  it 'should issue warning' do
64
64
  expect(app).to receive(:close)
65
65
  is_expected.to_not receive(:warn)
@@ -31,6 +31,15 @@ RSpec.describe Faraday::Request do
31
31
  it { expect(subject.to_env(conn).url.to_s).to eq('http://httpbingo.org/api/foo.json?a=1') }
32
32
  end
33
33
 
34
+ context 'when setting the url on setup with a protocol-relative URI' do
35
+ let(:block) { proc { |req| req.url URI.parse('//evil.com/path?token=1') } }
36
+ let(:url) { subject.to_env(conn).url }
37
+
38
+ it { expect(url.host).to eq('httpbingo.org') }
39
+ it { expect(url.path).to eq('/api///evil.com/path') }
40
+ it { expect(url.query).to eq('token=1') }
41
+ end
42
+
34
43
  context 'when setting the url on setup with a string path and params' do
35
44
  let(:block) { proc { |req| req.url 'foo.json', 'a' => 1 } }
36
45
 
@@ -106,7 +106,7 @@ RSpec.describe Faraday::Response::Json, type: :response do
106
106
  end
107
107
 
108
108
  it 'passes relevant options to JSON parse' do
109
- expect(::JSON).to receive(:parse)
109
+ expect(JSON).to receive(:parse)
110
110
  .with(body, options[:parser_options])
111
111
  .and_return(result)
112
112
 
@@ -21,6 +21,7 @@ RSpec.describe Faraday::Response::Logger do
21
21
  stubs.post('/ohai') { [200, { 'Content-Type' => 'text/html' }, 'fred'] }
22
22
  stubs.post('/ohyes') { [200, { 'Content-Type' => 'text/html' }, 'pebbles'] }
23
23
  stubs.get('/rubbles') { [200, { 'Content-Type' => 'application/json' }, rubbles] }
24
+ stubs.get('/8bit') { [200, { 'Content-Type' => 'text/html' }, (+'café!').force_encoding(Encoding::ASCII_8BIT)] }
24
25
  stubs.get('/filtered_body') { [200, { 'Content-Type' => 'text/html' }, 'soylent green is people'] }
25
26
  stubs.get('/filtered_headers') { [200, { 'Content-Type' => 'text/html' }, 'headers response'] }
26
27
  stubs.get('/filtered_params') { [200, { 'Content-Type' => 'text/html' }, 'params response'] }
@@ -217,7 +218,7 @@ RSpec.describe Faraday::Response::Logger do
217
218
 
218
219
  it 'logs response body object' do
219
220
  conn.get '/rubbles', nil, accept: 'text/html'
220
- expect(string_io.string).to match(%([\"Barney\", \"Betty\", \"Bam Bam\"]\n))
221
+ expect(string_io.string).to match(%(["Barney", "Betty", "Bam Bam"]\n))
221
222
  end
222
223
 
223
224
  it 'logs filter body' do
@@ -238,6 +239,11 @@ RSpec.describe Faraday::Response::Logger do
238
239
  expect(string_io.string).to match(%(fred))
239
240
  end
240
241
 
242
+ it 'converts to UTF-8' do
243
+ conn.get '/8bit'
244
+ expect(string_io.string).to match(%(caf��!))
245
+ end
246
+
241
247
  after do
242
248
  described_class.default_options = { bodies: false }
243
249
  end
@@ -13,7 +13,7 @@ RSpec.describe Faraday::Response::RaiseError do
13
13
  stub.get('proxy-error') { [407, { 'X-Reason' => 'because' }, 'keep looking'] }
14
14
  stub.get('request-timeout') { [408, { 'X-Reason' => 'because' }, 'keep looking'] }
15
15
  stub.get('conflict') { [409, { 'X-Reason' => 'because' }, 'keep looking'] }
16
- stub.get('unprocessable-entity') { [422, { 'X-Reason' => 'because' }, 'keep looking'] }
16
+ stub.get('unprocessable-content') { [422, { 'X-Reason' => 'because' }, 'keep looking'] }
17
17
  stub.get('too-many-requests') { [429, { 'X-Reason' => 'because' }, 'keep looking'] }
18
18
  stub.get('4xx') { [499, { 'X-Reason' => 'because' }, 'keep looking'] }
19
19
  stub.get('nil-status') { [nil, { 'X-Reason' => 'nil' }, 'fail'] }
@@ -28,7 +28,7 @@ RSpec.describe Faraday::Response::RaiseError do
28
28
 
29
29
  it 'raises Faraday::BadRequestError for 400 responses' do
30
30
  expect { conn.get('bad-request') }.to raise_error(Faraday::BadRequestError) do |ex|
31
- expect(ex.message).to eq('the server responded with status 400')
31
+ expect(ex.message).to eq('the server responded with status 400 for GET http:/bad-request')
32
32
  expect(ex.response[:headers]['X-Reason']).to eq('because')
33
33
  expect(ex.response[:status]).to eq(400)
34
34
  expect(ex.response_status).to eq(400)
@@ -39,7 +39,7 @@ RSpec.describe Faraday::Response::RaiseError do
39
39
 
40
40
  it 'raises Faraday::UnauthorizedError for 401 responses' do
41
41
  expect { conn.get('unauthorized') }.to raise_error(Faraday::UnauthorizedError) do |ex|
42
- expect(ex.message).to eq('the server responded with status 401')
42
+ expect(ex.message).to eq('the server responded with status 401 for GET http:/unauthorized')
43
43
  expect(ex.response[:headers]['X-Reason']).to eq('because')
44
44
  expect(ex.response[:status]).to eq(401)
45
45
  expect(ex.response_status).to eq(401)
@@ -50,7 +50,7 @@ RSpec.describe Faraday::Response::RaiseError do
50
50
 
51
51
  it 'raises Faraday::ForbiddenError for 403 responses' do
52
52
  expect { conn.get('forbidden') }.to raise_error(Faraday::ForbiddenError) do |ex|
53
- expect(ex.message).to eq('the server responded with status 403')
53
+ expect(ex.message).to eq('the server responded with status 403 for GET http:/forbidden')
54
54
  expect(ex.response[:headers]['X-Reason']).to eq('because')
55
55
  expect(ex.response[:status]).to eq(403)
56
56
  expect(ex.response_status).to eq(403)
@@ -61,7 +61,7 @@ RSpec.describe Faraday::Response::RaiseError do
61
61
 
62
62
  it 'raises Faraday::ResourceNotFound for 404 responses' do
63
63
  expect { conn.get('not-found') }.to raise_error(Faraday::ResourceNotFound) do |ex|
64
- expect(ex.message).to eq('the server responded with status 404')
64
+ expect(ex.message).to eq('the server responded with status 404 for GET http:/not-found')
65
65
  expect(ex.response[:headers]['X-Reason']).to eq('because')
66
66
  expect(ex.response[:status]).to eq(404)
67
67
  expect(ex.response_status).to eq(404)
@@ -83,7 +83,7 @@ RSpec.describe Faraday::Response::RaiseError do
83
83
 
84
84
  it 'raises Faraday::RequestTimeoutError for 408 responses' do
85
85
  expect { conn.get('request-timeout') }.to raise_error(Faraday::RequestTimeoutError) do |ex|
86
- expect(ex.message).to eq('the server responded with status 408')
86
+ expect(ex.message).to eq('the server responded with status 408 for GET http:/request-timeout')
87
87
  expect(ex.response[:headers]['X-Reason']).to eq('because')
88
88
  expect(ex.response[:status]).to eq(408)
89
89
  expect(ex.response_status).to eq(408)
@@ -94,7 +94,7 @@ RSpec.describe Faraday::Response::RaiseError do
94
94
 
95
95
  it 'raises Faraday::ConflictError for 409 responses' do
96
96
  expect { conn.get('conflict') }.to raise_error(Faraday::ConflictError) do |ex|
97
- expect(ex.message).to eq('the server responded with status 409')
97
+ expect(ex.message).to eq('the server responded with status 409 for GET http:/conflict')
98
98
  expect(ex.response[:headers]['X-Reason']).to eq('because')
99
99
  expect(ex.response[:status]).to eq(409)
100
100
  expect(ex.response_status).to eq(409)
@@ -103,9 +103,20 @@ RSpec.describe Faraday::Response::RaiseError do
103
103
  end
104
104
  end
105
105
 
106
- it 'raises Faraday::UnprocessableEntityError for 422 responses' do
107
- expect { conn.get('unprocessable-entity') }.to raise_error(Faraday::UnprocessableEntityError) do |ex|
108
- expect(ex.message).to eq('the server responded with status 422')
106
+ it 'raises legacy Faraday::UnprocessableEntityError for 422 responses' do
107
+ expect { conn.get('unprocessable-content') }.to raise_error(Faraday::UnprocessableEntityError) do |ex|
108
+ expect(ex.message).to eq('the server responded with status 422 for GET http:/unprocessable-content')
109
+ expect(ex.response[:headers]['X-Reason']).to eq('because')
110
+ expect(ex.response[:status]).to eq(422)
111
+ expect(ex.response_status).to eq(422)
112
+ expect(ex.response_body).to eq('keep looking')
113
+ expect(ex.response_headers['X-Reason']).to eq('because')
114
+ end
115
+ end
116
+
117
+ it 'raises Faraday::UnprocessableContentError for 422 responses' do
118
+ expect { conn.get('unprocessable-content') }.to raise_error(Faraday::UnprocessableContentError) do |ex|
119
+ expect(ex.message).to eq('the server responded with status 422 for GET http:/unprocessable-content')
109
120
  expect(ex.response[:headers]['X-Reason']).to eq('because')
110
121
  expect(ex.response[:status]).to eq(422)
111
122
  expect(ex.response_status).to eq(422)
@@ -116,7 +127,7 @@ RSpec.describe Faraday::Response::RaiseError do
116
127
 
117
128
  it 'raises Faraday::TooManyRequestsError for 429 responses' do
118
129
  expect { conn.get('too-many-requests') }.to raise_error(Faraday::TooManyRequestsError) do |ex|
119
- expect(ex.message).to eq('the server responded with status 429')
130
+ expect(ex.message).to eq('the server responded with status 429 for GET http:/too-many-requests')
120
131
  expect(ex.response[:headers]['X-Reason']).to eq('because')
121
132
  expect(ex.response[:status]).to eq(429)
122
133
  expect(ex.response_status).to eq(429)
@@ -138,7 +149,7 @@ RSpec.describe Faraday::Response::RaiseError do
138
149
 
139
150
  it 'raises Faraday::ClientError for other 4xx responses' do
140
151
  expect { conn.get('4xx') }.to raise_error(Faraday::ClientError) do |ex|
141
- expect(ex.message).to eq('the server responded with status 499')
152
+ expect(ex.message).to eq('the server responded with status 499 for GET http:/4xx')
142
153
  expect(ex.response[:headers]['X-Reason']).to eq('because')
143
154
  expect(ex.response[:status]).to eq(499)
144
155
  expect(ex.response_status).to eq(499)
@@ -149,7 +160,7 @@ RSpec.describe Faraday::Response::RaiseError do
149
160
 
150
161
  it 'raises Faraday::ServerError for 500 responses' do
151
162
  expect { conn.get('server-error') }.to raise_error(Faraday::ServerError) do |ex|
152
- expect(ex.message).to eq('the server responded with status 500')
163
+ expect(ex.message).to eq('the server responded with status 500 for GET http:/server-error')
153
164
  expect(ex.response[:headers]['X-Error']).to eq('bailout')
154
165
  expect(ex.response[:status]).to eq(500)
155
166
  expect(ex.response_status).to eq(500)
@@ -13,6 +13,7 @@ RSpec.describe Faraday::Response do
13
13
  it { expect(subject.success?).to be_falsey }
14
14
  it { expect(subject.status).to eq(404) }
15
15
  it { expect(subject.body).to eq('yikes') }
16
+ it { expect(subject.url).to eq(URI('https://lostisland.github.io/faraday')) }
16
17
  it { expect(subject.headers['Content-Type']).to eq('text/plain') }
17
18
  it { expect(subject['content-type']).to eq('text/plain') }
18
19
 
@@ -31,6 +32,12 @@ RSpec.describe Faraday::Response do
31
32
  it { expect(hash[:response_headers]).to eq(subject.headers) }
32
33
  it { expect(hash[:body]).to eq(subject.body) }
33
34
  it { expect(hash[:url]).to eq(subject.env.url) }
35
+
36
+ context 'when response is not finished' do
37
+ subject { Faraday::Response.new.to_hash }
38
+
39
+ it { is_expected.to eq({ status: nil, body: nil, response_headers: {}, url: nil }) }
40
+ end
34
41
  end
35
42
 
36
43
  describe 'marshal serialization support' do
@@ -33,7 +33,7 @@ RSpec.describe Faraday::Utils do
33
33
  end
34
34
 
35
35
  it 'parses with URI' do
36
- with_default_uri_parser(::URI) do
36
+ with_default_uri_parser(URI) do
37
37
  uri = normalize(url)
38
38
  expect(uri.host).to eq('example.com')
39
39
  end
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: faraday
3
3
  version: !ruby/object:Gem::Version
4
- version: 2.13.1
4
+ version: 2.14.2
5
5
  platform: ruby
6
6
  authors:
7
7
  - "@technoweenie"
@@ -144,7 +144,7 @@ licenses:
144
144
  - MIT
145
145
  metadata:
146
146
  homepage_uri: https://lostisland.github.io/faraday
147
- changelog_uri: https://github.com/lostisland/faraday/releases/tag/v2.13.1
147
+ changelog_uri: https://github.com/lostisland/faraday/releases/tag/v2.14.2
148
148
  source_code_uri: https://github.com/lostisland/faraday
149
149
  bug_tracker_uri: https://github.com/lostisland/faraday/issues
150
150
  rubygems_mfa_required: 'true'
@@ -163,7 +163,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
163
163
  - !ruby/object:Gem::Version
164
164
  version: '0'
165
165
  requirements: []
166
- rubygems_version: 3.6.7
166
+ rubygems_version: 3.6.9
167
167
  specification_version: 4
168
168
  summary: HTTP/REST API client library.
169
169
  test_files: []