faraday 0.17.6 → 1.0.1
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 +4 -4
- data/CHANGELOG.md +52 -8
- data/LICENSE.md +1 -1
- data/README.md +18 -358
- data/Rakefile +1 -7
- data/examples/client_spec.rb +65 -0
- data/examples/client_test.rb +79 -0
- data/lib/faraday/adapter/em_http.rb +142 -99
- data/lib/faraday/adapter/em_http_ssl_patch.rb +24 -18
- data/lib/faraday/adapter/em_synchrony/parallel_manager.rb +18 -15
- data/lib/faraday/adapter/em_synchrony.rb +104 -60
- data/lib/faraday/adapter/excon.rb +98 -56
- data/lib/faraday/adapter/httpclient.rb +83 -59
- data/lib/faraday/adapter/net_http.rb +129 -63
- data/lib/faraday/adapter/net_http_persistent.rb +50 -27
- data/lib/faraday/adapter/patron.rb +80 -43
- data/lib/faraday/adapter/rack.rb +30 -13
- data/lib/faraday/adapter/test.rb +86 -53
- data/lib/faraday/adapter/typhoeus.rb +4 -1
- data/lib/faraday/adapter.rb +82 -22
- data/lib/faraday/adapter_registry.rb +30 -0
- data/lib/faraday/autoload.rb +47 -36
- data/lib/faraday/connection.rb +312 -182
- data/lib/faraday/dependency_loader.rb +37 -0
- data/lib/faraday/encoders/flat_params_encoder.rb +98 -0
- data/lib/faraday/encoders/nested_params_encoder.rb +171 -0
- data/lib/faraday/error.rb +9 -35
- data/lib/faraday/file_part.rb +128 -0
- data/lib/faraday/logging/formatter.rb +105 -0
- data/lib/faraday/middleware.rb +12 -28
- data/lib/faraday/middleware_registry.rb +129 -0
- data/lib/faraday/options/connection_options.rb +22 -0
- data/lib/faraday/options/env.rb +181 -0
- data/lib/faraday/options/proxy_options.rb +28 -0
- data/lib/faraday/options/request_options.rb +22 -0
- data/lib/faraday/options/ssl_options.rb +59 -0
- data/lib/faraday/options.rb +32 -183
- data/lib/faraday/param_part.rb +53 -0
- data/lib/faraday/parameters.rb +4 -197
- data/lib/faraday/rack_builder.rb +66 -55
- data/lib/faraday/request/authorization.rb +44 -30
- data/lib/faraday/request/basic_authentication.rb +14 -7
- data/lib/faraday/request/instrumentation.rb +45 -27
- data/lib/faraday/request/multipart.rb +79 -48
- data/lib/faraday/request/retry.rb +197 -171
- data/lib/faraday/request/token_authentication.rb +15 -10
- data/lib/faraday/request/url_encoded.rb +43 -23
- data/lib/faraday/request.rb +68 -38
- data/lib/faraday/response/logger.rb +22 -69
- data/lib/faraday/response/raise_error.rb +38 -18
- data/lib/faraday/response.rb +24 -14
- data/lib/faraday/utils/headers.rb +139 -0
- data/lib/faraday/utils/params_hash.rb +61 -0
- data/lib/faraday/utils.rb +36 -245
- data/lib/faraday.rb +94 -175
- data/spec/external_adapters/faraday_specs_setup.rb +14 -0
- data/spec/faraday/adapter/em_http_spec.rb +47 -0
- data/spec/faraday/adapter/em_synchrony_spec.rb +16 -0
- data/spec/faraday/adapter/excon_spec.rb +49 -0
- data/spec/faraday/adapter/httpclient_spec.rb +73 -0
- data/spec/faraday/adapter/net_http_persistent_spec.rb +57 -0
- data/spec/faraday/adapter/net_http_spec.rb +64 -0
- data/spec/faraday/adapter/patron_spec.rb +18 -0
- data/spec/faraday/adapter/rack_spec.rb +8 -0
- data/spec/faraday/adapter/typhoeus_spec.rb +7 -0
- data/spec/faraday/adapter_registry_spec.rb +28 -0
- data/spec/faraday/adapter_spec.rb +55 -0
- data/spec/faraday/composite_read_io_spec.rb +80 -0
- data/spec/faraday/connection_spec.rb +691 -0
- data/spec/faraday/error_spec.rb +0 -57
- data/spec/faraday/middleware_spec.rb +26 -0
- data/spec/faraday/options/env_spec.rb +70 -0
- data/spec/faraday/options/options_spec.rb +297 -0
- data/spec/faraday/options/proxy_options_spec.rb +37 -0
- data/spec/faraday/options/request_options_spec.rb +19 -0
- data/spec/faraday/params_encoders/flat_spec.rb +34 -0
- data/spec/faraday/params_encoders/nested_spec.rb +134 -0
- data/spec/faraday/rack_builder_spec.rb +196 -0
- data/spec/faraday/request/authorization_spec.rb +88 -0
- data/spec/faraday/request/instrumentation_spec.rb +76 -0
- data/spec/faraday/request/multipart_spec.rb +274 -0
- data/spec/faraday/request/retry_spec.rb +242 -0
- data/spec/faraday/request/url_encoded_spec.rb +83 -0
- data/spec/faraday/request_spec.rb +109 -0
- data/spec/faraday/response/logger_spec.rb +220 -0
- data/spec/faraday/response/middleware_spec.rb +68 -0
- data/spec/faraday/response/raise_error_spec.rb +15 -15
- data/spec/faraday/response_spec.rb +75 -0
- data/spec/faraday/utils/headers_spec.rb +82 -0
- data/spec/faraday/utils_spec.rb +56 -0
- data/spec/faraday_spec.rb +37 -0
- data/spec/spec_helper.rb +63 -36
- data/spec/support/disabling_stub.rb +14 -0
- data/spec/support/fake_safe_buffer.rb +15 -0
- data/spec/support/helper_methods.rb +133 -0
- data/spec/support/shared_examples/adapter.rb +104 -0
- data/spec/support/shared_examples/params_encoder.rb +18 -0
- data/spec/support/shared_examples/request_method.rb +234 -0
- data/spec/support/streaming_response_checker.rb +35 -0
- data/spec/support/webmock_rack_app.rb +68 -0
- metadata +66 -38
- data/lib/faraday/deprecate.rb +0 -109
- data/lib/faraday/upload_io.rb +0 -77
- data/spec/faraday/deprecate_spec.rb +0 -147
- data/test/adapters/default_test.rb +0 -14
- data/test/adapters/em_http_test.rb +0 -30
- data/test/adapters/em_synchrony_test.rb +0 -32
- data/test/adapters/excon_test.rb +0 -30
- data/test/adapters/httpclient_test.rb +0 -34
- data/test/adapters/integration.rb +0 -263
- data/test/adapters/logger_test.rb +0 -136
- data/test/adapters/net_http_persistent_test.rb +0 -114
- data/test/adapters/net_http_test.rb +0 -79
- data/test/adapters/patron_test.rb +0 -40
- data/test/adapters/rack_test.rb +0 -38
- data/test/adapters/test_middleware_test.rb +0 -157
- data/test/adapters/typhoeus_test.rb +0 -38
- data/test/authentication_middleware_test.rb +0 -65
- data/test/composite_read_io_test.rb +0 -109
- data/test/connection_test.rb +0 -738
- data/test/env_test.rb +0 -268
- data/test/helper.rb +0 -75
- data/test/live_server.rb +0 -67
- data/test/middleware/instrumentation_test.rb +0 -88
- data/test/middleware/retry_test.rb +0 -282
- data/test/middleware_stack_test.rb +0 -260
- data/test/multibyte.txt +0 -1
- data/test/options_test.rb +0 -333
- data/test/parameters_test.rb +0 -157
- data/test/request_middleware_test.rb +0 -126
- data/test/response_middleware_test.rb +0 -72
- data/test/strawberry.rb +0 -2
- data/test/utils_test.rb +0 -98
@@ -1,15 +1,19 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
begin
|
2
4
|
require 'net/https'
|
3
5
|
rescue LoadError
|
4
|
-
warn
|
6
|
+
warn 'Warning: no such file to load -- net/https. ' \
|
7
|
+
'Make sure openssl is installed if you want ssl support'
|
5
8
|
require 'net/http'
|
6
9
|
end
|
7
10
|
require 'zlib'
|
8
11
|
|
9
12
|
module Faraday
|
10
13
|
class Adapter
|
14
|
+
# Net::HTTP adapter.
|
11
15
|
class NetHttp < Faraday::Adapter
|
12
|
-
|
16
|
+
exceptions = [
|
13
17
|
IOError,
|
14
18
|
Errno::EADDRNOTAVAIL,
|
15
19
|
Errno::ECONNABORTED,
|
@@ -23,54 +27,78 @@ module Faraday
|
|
23
27
|
Net::HTTPHeaderSyntaxError,
|
24
28
|
Net::ProtocolError,
|
25
29
|
SocketError,
|
26
|
-
Zlib::GzipFile::Error
|
30
|
+
Zlib::GzipFile::Error
|
27
31
|
]
|
28
32
|
|
29
|
-
|
30
|
-
|
33
|
+
if defined?(::OpenSSL::SSL::SSLError)
|
34
|
+
exceptions << ::OpenSSL::SSL::SSLError
|
35
|
+
end
|
36
|
+
exceptions << ::Net::OpenTimeout if defined?(::Net::OpenTimeout)
|
37
|
+
|
38
|
+
NET_HTTP_EXCEPTIONS = exceptions.freeze
|
31
39
|
|
32
40
|
def initialize(app = nil, opts = {}, &block)
|
33
|
-
@
|
41
|
+
@ssl_cert_store = nil
|
34
42
|
super(app, opts, &block)
|
35
43
|
end
|
36
44
|
|
37
|
-
def
|
38
|
-
|
39
|
-
|
40
|
-
|
45
|
+
def build_connection(env)
|
46
|
+
net_http_connection(env).tap do |http|
|
47
|
+
if http.respond_to?(:use_ssl=)
|
48
|
+
http.use_ssl = env[:url].scheme == 'https'
|
49
|
+
end
|
50
|
+
configure_ssl(http, env[:ssl])
|
41
51
|
configure_request(http, env[:request])
|
52
|
+
end
|
53
|
+
end
|
42
54
|
|
55
|
+
def net_http_connection(env)
|
56
|
+
klass = if (proxy = env[:request][:proxy])
|
57
|
+
Net::HTTP::Proxy(proxy[:uri].hostname, proxy[:uri].port,
|
58
|
+
proxy[:user], proxy[:password])
|
59
|
+
else
|
60
|
+
Net::HTTP
|
61
|
+
end
|
62
|
+
port = env[:url].port || (env[:url].scheme == 'https' ? 443 : 80)
|
63
|
+
klass.new(env[:url].hostname, port)
|
64
|
+
end
|
65
|
+
|
66
|
+
def call(env)
|
67
|
+
super
|
68
|
+
http_response = connection(env) do |http|
|
43
69
|
begin
|
44
|
-
|
45
|
-
rescue *NET_HTTP_EXCEPTIONS =>
|
46
|
-
if defined?(OpenSSL) && OpenSSL::SSL::SSLError
|
47
|
-
raise Faraday::SSLError,
|
48
|
-
else
|
49
|
-
raise Faraday::ConnectionFailed, err
|
70
|
+
perform_request(http, env)
|
71
|
+
rescue *NET_HTTP_EXCEPTIONS => e
|
72
|
+
if defined?(OpenSSL) && e.is_a?(OpenSSL::SSL::SSLError)
|
73
|
+
raise Faraday::SSLError, e
|
50
74
|
end
|
75
|
+
|
76
|
+
raise Faraday::ConnectionFailed, e
|
51
77
|
end
|
78
|
+
end
|
52
79
|
|
53
|
-
|
54
|
-
|
55
|
-
|
56
|
-
|
80
|
+
save_response(env, http_response.code.to_i,
|
81
|
+
http_response.body || +'', nil,
|
82
|
+
http_response.message) do |response_headers|
|
83
|
+
http_response.each_header do |key, value|
|
84
|
+
response_headers[key] = value
|
57
85
|
end
|
58
86
|
end
|
59
87
|
|
60
88
|
@app.call env
|
61
|
-
rescue Timeout::Error, Errno::ETIMEDOUT =>
|
62
|
-
raise Faraday::TimeoutError,
|
89
|
+
rescue Timeout::Error, Errno::ETIMEDOUT => e
|
90
|
+
raise Faraday::TimeoutError, e
|
63
91
|
end
|
64
92
|
|
65
93
|
private
|
66
94
|
|
67
95
|
def create_request(env)
|
68
96
|
request = Net::HTTPGenericRequest.new \
|
69
|
-
env[:method].to_s.upcase,
|
70
|
-
!!env[:body],
|
71
|
-
:
|
72
|
-
env[:url].request_uri,
|
73
|
-
env[:request_headers]
|
97
|
+
env[:method].to_s.upcase, # request method
|
98
|
+
!!env[:body], # is there request body
|
99
|
+
env[:method] != :head, # is there response body
|
100
|
+
env[:url].request_uri, # request uri path
|
101
|
+
env[:request_headers] # request headers
|
74
102
|
|
75
103
|
if env[:body].respond_to?(:read)
|
76
104
|
request.body_stream = env[:body]
|
@@ -81,62 +109,100 @@ module Faraday
|
|
81
109
|
end
|
82
110
|
|
83
111
|
def perform_request(http, env)
|
84
|
-
if
|
112
|
+
if env[:request].stream_response?
|
113
|
+
size = 0
|
114
|
+
yielded = false
|
115
|
+
http_response = request_with_wrapped_block(http, env) do |chunk|
|
116
|
+
if chunk.bytesize.positive? || size.positive?
|
117
|
+
yielded = true
|
118
|
+
size += chunk.bytesize
|
119
|
+
env[:request].on_data.call(chunk, size)
|
120
|
+
end
|
121
|
+
end
|
122
|
+
env[:request].on_data.call(+'', 0) unless yielded
|
123
|
+
# Net::HTTP returns something,
|
124
|
+
# but it's not meaningful according to the docs.
|
125
|
+
http_response.body = nil
|
126
|
+
http_response
|
127
|
+
else
|
128
|
+
request_with_wrapped_block(http, env)
|
129
|
+
end
|
130
|
+
end
|
131
|
+
|
132
|
+
def request_with_wrapped_block(http, env, &block)
|
133
|
+
if (env[:method] == :get) && !env[:body]
|
85
134
|
# prefer `get` to `request` because the former handles gzip (ruby 1.9)
|
86
|
-
http
|
135
|
+
request_via_get_method(http, env, &block)
|
87
136
|
else
|
88
|
-
http
|
137
|
+
request_via_request_method(http, env, &block)
|
89
138
|
end
|
90
139
|
end
|
91
140
|
|
92
|
-
def
|
93
|
-
|
141
|
+
def request_via_get_method(http, env, &block)
|
142
|
+
# Must use Net::HTTP#start and pass it a block otherwise the server's
|
143
|
+
# TCP socket does not close correctly.
|
144
|
+
http.start do |opened_http|
|
145
|
+
opened_http.get env[:url].request_uri, env[:request_headers], &block
|
146
|
+
end
|
94
147
|
end
|
95
148
|
|
96
|
-
def
|
97
|
-
|
98
|
-
|
99
|
-
|
100
|
-
|
101
|
-
|
149
|
+
def request_via_request_method(http, env, &block)
|
150
|
+
# Must use Net::HTTP#start and pass it a block otherwise the server's
|
151
|
+
# TCP socket does not close correctly.
|
152
|
+
http.start do |opened_http|
|
153
|
+
if block_given?
|
154
|
+
opened_http.request create_request(env) do |response|
|
155
|
+
response.read_body(&block)
|
156
|
+
end
|
157
|
+
else
|
158
|
+
opened_http.request create_request(env)
|
159
|
+
end
|
160
|
+
end
|
102
161
|
end
|
103
162
|
|
104
163
|
def configure_ssl(http, ssl)
|
105
|
-
|
106
|
-
|
107
|
-
http.
|
108
|
-
|
109
|
-
|
110
|
-
http.
|
111
|
-
http.
|
112
|
-
http.
|
164
|
+
return unless ssl
|
165
|
+
|
166
|
+
http.verify_mode = ssl_verify_mode(ssl)
|
167
|
+
http.cert_store = ssl_cert_store(ssl)
|
168
|
+
|
169
|
+
http.cert = ssl[:client_cert] if ssl[:client_cert]
|
170
|
+
http.key = ssl[:client_key] if ssl[:client_key]
|
171
|
+
http.ca_file = ssl[:ca_file] if ssl[:ca_file]
|
172
|
+
http.ca_path = ssl[:ca_path] if ssl[:ca_path]
|
113
173
|
http.verify_depth = ssl[:verify_depth] if ssl[:verify_depth]
|
114
|
-
http.ssl_version
|
115
|
-
http.min_version
|
116
|
-
http.max_version
|
174
|
+
http.ssl_version = ssl[:version] if ssl[:version]
|
175
|
+
http.min_version = ssl[:min_version] if ssl[:min_version]
|
176
|
+
http.max_version = ssl[:max_version] if ssl[:max_version]
|
117
177
|
end
|
118
178
|
|
119
179
|
def configure_request(http, req)
|
120
|
-
if req
|
121
|
-
http.read_timeout
|
122
|
-
http.open_timeout = req[:timeout]
|
123
|
-
http.write_timeout = req[:timeout] if http.respond_to?(:write_timeout=)
|
180
|
+
if (sec = request_timeout(:read, req))
|
181
|
+
http.read_timeout = sec
|
124
182
|
end
|
125
|
-
http.open_timeout = req[:open_timeout] if req[:open_timeout]
|
126
|
-
http.write_timeout = req[:write_timeout] if req[:write_timeout] && http.respond_to?(:write_timeout=)
|
127
|
-
# Only set if Net::Http supports it, since Ruby 2.5.
|
128
|
-
http.max_retries = 0 if http.respond_to?(:max_retries=)
|
129
183
|
|
130
|
-
|
184
|
+
if (sec = http.respond_to?(:write_timeout=) &&
|
185
|
+
request_timeout(:write, req))
|
186
|
+
http.write_timeout = sec
|
187
|
+
end
|
188
|
+
|
189
|
+
if (sec = request_timeout(:open, req))
|
190
|
+
http.open_timeout = sec
|
191
|
+
end
|
192
|
+
|
193
|
+
# Only set if Net::Http supports it, since Ruby 2.5.
|
194
|
+
http.max_retries = 0 if http.respond_to?(:max_retries=)
|
195
|
+
|
196
|
+
@config_block&.call(http)
|
131
197
|
end
|
132
198
|
|
133
199
|
def ssl_cert_store(ssl)
|
134
200
|
return ssl[:cert_store] if ssl[:cert_store]
|
135
|
-
|
136
|
-
|
137
|
-
|
138
|
-
|
139
|
-
|
201
|
+
|
202
|
+
@ssl_cert_store ||= begin
|
203
|
+
# Use the default cert store by default, i.e. system ca certs
|
204
|
+
OpenSSL::X509::Store.new.tap(&:set_default_paths)
|
205
|
+
end
|
140
206
|
end
|
141
207
|
|
142
208
|
def ssl_verify_mode(ssl)
|
@@ -1,5 +1,8 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
module Faraday
|
2
4
|
class Adapter
|
5
|
+
# Net::HTTP::Persistent adapter.
|
3
6
|
class NetHttpPersistent < NetHttp
|
4
7
|
dependency 'net/http/persistent'
|
5
8
|
|
@@ -7,61 +10,81 @@ module Faraday
|
|
7
10
|
|
8
11
|
def net_http_connection(env)
|
9
12
|
@cached_connection ||=
|
10
|
-
if Net::HTTP::Persistent.instance_method(:initialize)
|
11
|
-
|
12
|
-
options
|
13
|
+
if Net::HTTP::Persistent.instance_method(:initialize)
|
14
|
+
.parameters.first == %i[key name]
|
15
|
+
options = { name: 'Faraday' }
|
16
|
+
if @connection_options.key?(:pool_size)
|
17
|
+
options[:pool_size] = @connection_options[:pool_size]
|
18
|
+
end
|
13
19
|
Net::HTTP::Persistent.new(**options)
|
14
20
|
else
|
15
21
|
Net::HTTP::Persistent.new('Faraday')
|
16
22
|
end
|
17
23
|
|
18
24
|
proxy_uri = proxy_uri(env)
|
19
|
-
|
25
|
+
if @cached_connection.proxy_uri != proxy_uri
|
26
|
+
@cached_connection.proxy = proxy_uri
|
27
|
+
end
|
20
28
|
@cached_connection
|
21
29
|
end
|
22
30
|
|
23
31
|
def proxy_uri(env)
|
24
32
|
proxy_uri = nil
|
25
33
|
if (proxy = env[:request][:proxy])
|
26
|
-
proxy_uri =
|
34
|
+
proxy_uri = if proxy[:uri].is_a?(::URI::HTTP)
|
35
|
+
proxy[:uri].dup
|
36
|
+
else
|
37
|
+
::URI.parse(proxy[:uri].to_s)
|
38
|
+
end
|
27
39
|
proxy_uri.user = proxy_uri.password = nil
|
28
|
-
# awful patch for net-http-persistent 2.8
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
40
|
+
# awful patch for net-http-persistent 2.8
|
41
|
+
# not unescaping user/password
|
42
|
+
if proxy[:user]
|
43
|
+
(class << proxy_uri; self; end).class_eval do
|
44
|
+
define_method(:user) { proxy[:user] }
|
45
|
+
define_method(:password) { proxy[:password] }
|
46
|
+
end
|
47
|
+
end
|
33
48
|
end
|
34
49
|
proxy_uri
|
35
50
|
end
|
36
51
|
|
37
52
|
def perform_request(http, env)
|
38
53
|
http.request env[:url], create_request(env)
|
39
|
-
rescue Errno::ETIMEDOUT =>
|
40
|
-
raise Faraday::TimeoutError,
|
41
|
-
rescue Net::HTTP::Persistent::Error =>
|
42
|
-
if
|
43
|
-
|
44
|
-
|
45
|
-
raise Faraday::ConnectionFailed,
|
46
|
-
else
|
47
|
-
raise
|
54
|
+
rescue Errno::ETIMEDOUT => e
|
55
|
+
raise Faraday::TimeoutError, e
|
56
|
+
rescue Net::HTTP::Persistent::Error => e
|
57
|
+
raise Faraday::TimeoutError, e if e.message.include? 'Timeout'
|
58
|
+
|
59
|
+
if e.message.include? 'connection refused'
|
60
|
+
raise Faraday::ConnectionFailed, e
|
48
61
|
end
|
62
|
+
|
63
|
+
raise
|
49
64
|
end
|
50
65
|
|
66
|
+
SSL_CONFIGURATIONS = {
|
67
|
+
certificate: :client_cert,
|
68
|
+
private_key: :client_key,
|
69
|
+
ca_file: :ca_file,
|
70
|
+
ssl_version: :version,
|
71
|
+
min_version: :min_version,
|
72
|
+
max_version: :max_version
|
73
|
+
}.freeze
|
74
|
+
|
51
75
|
def configure_ssl(http, ssl)
|
76
|
+
return unless ssl
|
77
|
+
|
52
78
|
http_set(http, :verify_mode, ssl_verify_mode(ssl))
|
53
|
-
http_set(http, :cert_store,
|
79
|
+
http_set(http, :cert_store, ssl_cert_store(ssl))
|
54
80
|
|
55
|
-
|
56
|
-
|
57
|
-
|
58
|
-
http_set(http, :ssl_version, ssl[:version]) if ssl[:version]
|
81
|
+
SSL_CONFIGURATIONS
|
82
|
+
.select { |_, key| ssl[key] }
|
83
|
+
.each { |target, key| http_set(http, target, ssl[key]) }
|
59
84
|
end
|
60
85
|
|
61
86
|
def http_set(http, attr, value)
|
62
|
-
if http.send(attr) != value
|
63
|
-
http.send("#{attr}=", value)
|
64
|
-
end
|
87
|
+
http.send("#{attr}=", value) if http.send(attr) != value
|
65
88
|
end
|
66
89
|
end
|
67
90
|
end
|
@@ -1,54 +1,66 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
module Faraday
|
2
4
|
class Adapter
|
5
|
+
# Patron adapter.
|
3
6
|
class Patron < Faraday::Adapter
|
4
7
|
dependency 'patron'
|
5
8
|
|
9
|
+
def build_connection(env)
|
10
|
+
session = ::Patron::Session.new
|
11
|
+
@config_block&.call(session)
|
12
|
+
if (env[:url].scheme == 'https') && env[:ssl]
|
13
|
+
configure_ssl(session, env[:ssl])
|
14
|
+
end
|
15
|
+
|
16
|
+
if (req = env[:request])
|
17
|
+
configure_timeouts(session, req)
|
18
|
+
configure_proxy(session, req[:proxy])
|
19
|
+
end
|
20
|
+
|
21
|
+
session
|
22
|
+
end
|
23
|
+
|
6
24
|
def call(env)
|
7
25
|
super
|
8
26
|
# TODO: support streaming requests
|
9
27
|
env[:body] = env[:body].read if env[:body].respond_to? :read
|
10
28
|
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
if proxy = req[:proxy]
|
20
|
-
proxy_uri = proxy[:uri].dup
|
21
|
-
proxy_uri.user = proxy[:user] && Utils.escape(proxy[:user]).gsub('+', '%20')
|
22
|
-
proxy_uri.password = proxy[:password] && Utils.escape(proxy[:password]).gsub('+', '%20')
|
23
|
-
session.proxy = proxy_uri.to_s
|
29
|
+
response = connection(env) do |session|
|
30
|
+
begin
|
31
|
+
data = env[:body] ? env[:body].to_s : nil
|
32
|
+
session.request(env[:method], env[:url].to_s,
|
33
|
+
env[:request_headers], data: data)
|
34
|
+
rescue Errno::ECONNREFUSED, ::Patron::ConnectionFailed
|
35
|
+
raise Faraday::ConnectionFailed, $ERROR_INFO
|
24
36
|
end
|
25
37
|
end
|
26
38
|
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
raise Faraday::ConnectionFailed, $!
|
39
|
+
if (req = env[:request]).stream_response?
|
40
|
+
warn "Streaming downloads for #{self.class.name} " \
|
41
|
+
'are not yet implemented.'
|
42
|
+
req.on_data.call(response.body, response.body.bytesize)
|
32
43
|
end
|
33
|
-
|
34
44
|
# Remove the "HTTP/1.1 200", leaving just the reason phrase
|
35
45
|
reason_phrase = response.status_line.gsub(/^.* \d{3} /, '')
|
36
46
|
|
37
|
-
save_response(env, response.status, response.body,
|
47
|
+
save_response(env, response.status, response.body,
|
48
|
+
response.headers, reason_phrase)
|
38
49
|
|
39
50
|
@app.call env
|
40
|
-
rescue ::Patron::TimeoutError =>
|
41
|
-
if connection_timed_out_message?(
|
42
|
-
raise Faraday::ConnectionFailed,
|
43
|
-
else
|
44
|
-
raise Faraday::TimeoutError, err
|
51
|
+
rescue ::Patron::TimeoutError => e
|
52
|
+
if connection_timed_out_message?(e.message)
|
53
|
+
raise Faraday::ConnectionFailed, e
|
45
54
|
end
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
raise Faraday::ConnectionFailed,
|
55
|
+
|
56
|
+
raise Faraday::TimeoutError, e
|
57
|
+
rescue ::Patron::Error => e
|
58
|
+
if e.message.include?('code 407')
|
59
|
+
raise Faraday::ConnectionFailed,
|
60
|
+
%(407 "Proxy Authentication Required ")
|
51
61
|
end
|
62
|
+
|
63
|
+
raise Faraday::ConnectionFailed, e
|
52
64
|
end
|
53
65
|
|
54
66
|
if loaded? && defined?(::Patron::Request::VALID_ACTIONS)
|
@@ -60,8 +72,8 @@ module Faraday
|
|
60
72
|
actions << :options unless actions.include? :options
|
61
73
|
else
|
62
74
|
# Patron 0.4.20 and up
|
63
|
-
actions <<
|
64
|
-
actions <<
|
75
|
+
actions << 'PATCH' unless actions.include? 'PATCH'
|
76
|
+
actions << 'OPTIONS' unless actions.include? 'OPTIONS'
|
65
77
|
end
|
66
78
|
end
|
67
79
|
end
|
@@ -74,22 +86,47 @@ module Faraday
|
|
74
86
|
end
|
75
87
|
end
|
76
88
|
|
89
|
+
def configure_timeouts(session, req)
|
90
|
+
return unless req
|
91
|
+
|
92
|
+
if (sec = request_timeout(:read, req))
|
93
|
+
session.timeout = sec
|
94
|
+
end
|
95
|
+
|
96
|
+
return unless (sec = request_timeout(:open, req))
|
97
|
+
|
98
|
+
session.connect_timeout = sec
|
99
|
+
end
|
100
|
+
|
101
|
+
def configure_proxy(session, proxy)
|
102
|
+
return unless proxy
|
103
|
+
|
104
|
+
proxy_uri = proxy[:uri].dup
|
105
|
+
proxy_uri.user = proxy[:user] &&
|
106
|
+
Utils.escape(proxy[:user]).gsub('+', '%20')
|
107
|
+
proxy_uri.password = proxy[:password] &&
|
108
|
+
Utils.escape(proxy[:password]).gsub('+', '%20')
|
109
|
+
session.proxy = proxy_uri.to_s
|
110
|
+
end
|
111
|
+
|
77
112
|
private
|
78
113
|
|
79
|
-
CURL_TIMEOUT_MESSAGES = [
|
80
|
-
|
81
|
-
|
82
|
-
|
83
|
-
|
84
|
-
|
85
|
-
|
86
|
-
|
87
|
-
|
114
|
+
CURL_TIMEOUT_MESSAGES = [
|
115
|
+
'Connection time-out',
|
116
|
+
'Connection timed out',
|
117
|
+
'Timed out before name resolve',
|
118
|
+
'server connect has timed out',
|
119
|
+
'Resolving timed out',
|
120
|
+
'name lookup timed out',
|
121
|
+
'timed out before SSL',
|
122
|
+
'connect() timed out'
|
123
|
+
].freeze
|
88
124
|
|
89
125
|
def connection_timed_out_message?(message)
|
90
|
-
CURL_TIMEOUT_MESSAGES.any?
|
126
|
+
CURL_TIMEOUT_MESSAGES.any? do |curl_message|
|
127
|
+
message.include?(curl_message)
|
128
|
+
end
|
91
129
|
end
|
92
|
-
|
93
130
|
end
|
94
131
|
end
|
95
132
|
end
|
data/lib/faraday/adapter/rack.rb
CHANGED
@@ -1,8 +1,10 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
module Faraday
|
2
4
|
class Adapter
|
3
5
|
# Sends requests to a Rack app.
|
4
6
|
#
|
5
|
-
#
|
7
|
+
# @example
|
6
8
|
#
|
7
9
|
# class MyRackApp
|
8
10
|
# def call(env)
|
@@ -17,7 +19,7 @@ module Faraday
|
|
17
19
|
dependency 'rack/test'
|
18
20
|
|
19
21
|
# not prefixed with "HTTP_"
|
20
|
-
SPECIAL_HEADERS = %w[
|
22
|
+
SPECIAL_HEADERS = %w[CONTENT_LENGTH CONTENT_TYPE].freeze
|
21
23
|
|
22
24
|
def initialize(faraday_app, rack_app)
|
23
25
|
super(faraday_app)
|
@@ -27,32 +29,47 @@ module Faraday
|
|
27
29
|
|
28
30
|
def call(env)
|
29
31
|
super
|
30
|
-
rack_env =
|
31
|
-
:method => env[:method],
|
32
|
-
:input => env[:body].respond_to?(:read) ? env[:body].read : env[:body],
|
33
|
-
'rack.url_scheme' => env[:url].scheme
|
34
|
-
}
|
32
|
+
rack_env = build_rack_env(env)
|
35
33
|
|
36
|
-
env[:request_headers]
|
34
|
+
env[:request_headers]&.each do |name, value|
|
37
35
|
name = name.upcase.tr('-', '_')
|
38
36
|
name = "HTTP_#{name}" unless SPECIAL_HEADERS.include? name
|
39
37
|
rack_env[name] = value
|
40
|
-
end
|
38
|
+
end
|
41
39
|
|
42
|
-
timeout
|
40
|
+
timeout = request_timeout(:open, env[:request])
|
41
|
+
timeout ||= request_timeout(:read, env[:request])
|
43
42
|
response = if timeout
|
44
|
-
|
45
|
-
|
46
|
-
|
43
|
+
Timer.timeout(timeout, Faraday::TimeoutError) do
|
44
|
+
execute_request(env, rack_env)
|
45
|
+
end
|
46
|
+
else
|
47
|
+
execute_request(env, rack_env)
|
48
|
+
end
|
49
|
+
|
50
|
+
if (req = env[:request]).stream_response?
|
51
|
+
warn "Streaming downloads for #{self.class.name} " \
|
52
|
+
'are not yet implemented.'
|
53
|
+
req.on_data.call(response.body, response.body.bytesize)
|
47
54
|
end
|
48
55
|
|
49
56
|
save_response(env, response.status, response.body, response.headers)
|
50
57
|
@app.call env
|
51
58
|
end
|
52
59
|
|
60
|
+
private
|
61
|
+
|
53
62
|
def execute_request(env, rack_env)
|
54
63
|
@session.request(env[:url].to_s, rack_env)
|
55
64
|
end
|
65
|
+
|
66
|
+
def build_rack_env(env)
|
67
|
+
{
|
68
|
+
method: env[:method],
|
69
|
+
input: env[:body].respond_to?(:read) ? env[:body].read : env[:body],
|
70
|
+
'rack.url_scheme' => env[:url].scheme
|
71
|
+
}
|
72
|
+
end
|
56
73
|
end
|
57
74
|
end
|
58
75
|
end
|