faraday 1.4.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 +4 -4
- data/CHANGELOG.md +198 -4
- data/LICENSE.md +1 -1
- data/README.md +34 -20
- data/Rakefile +6 -1
- data/examples/client_spec.rb +67 -13
- data/examples/client_test.rb +81 -16
- data/lib/faraday/adapter/test.rb +117 -52
- data/lib/faraday/adapter.rb +12 -20
- data/lib/faraday/connection.rb +86 -134
- data/lib/faraday/encoders/flat_params_encoder.rb +3 -2
- data/lib/faraday/encoders/nested_params_encoder.rb +15 -7
- data/lib/faraday/error.rb +65 -15
- data/lib/faraday/logging/formatter.rb +30 -17
- data/lib/faraday/middleware.rb +44 -3
- data/lib/faraday/middleware_registry.rb +17 -63
- data/lib/faraday/options/connection_options.rb +7 -6
- data/lib/faraday/options/env.rb +88 -65
- data/lib/faraday/options/proxy_options.rb +17 -6
- data/lib/faraday/options/request_options.rb +8 -7
- data/lib/faraday/options/ssl_options.rb +62 -45
- data/lib/faraday/options.rb +8 -7
- data/lib/faraday/rack_builder.rb +46 -47
- data/lib/faraday/request/authorization.rb +37 -38
- data/lib/faraday/request/instrumentation.rb +5 -1
- data/lib/faraday/request/json.rb +70 -0
- data/lib/faraday/request/url_encoded.rb +5 -1
- data/lib/faraday/request.rb +20 -37
- data/lib/faraday/response/json.rb +74 -0
- data/lib/faraday/response/logger.rb +13 -7
- data/lib/faraday/response/raise_error.rb +45 -18
- data/lib/faraday/response.rb +15 -21
- data/lib/faraday/utils/headers.rb +18 -7
- data/lib/faraday/utils.rb +11 -7
- data/lib/faraday/version.rb +1 -1
- data/lib/faraday.rb +12 -32
- data/spec/faraday/adapter/test_spec.rb +182 -0
- data/spec/faraday/connection_spec.rb +219 -92
- data/spec/faraday/error_spec.rb +122 -7
- data/spec/faraday/middleware_registry_spec.rb +31 -0
- data/spec/faraday/middleware_spec.rb +163 -2
- data/spec/faraday/options/env_spec.rb +8 -2
- data/spec/faraday/options/options_spec.rb +1 -1
- data/spec/faraday/options/proxy_options_spec.rb +42 -0
- data/spec/faraday/params_encoders/nested_spec.rb +10 -1
- data/spec/faraday/rack_builder_spec.rb +26 -54
- data/spec/faraday/request/authorization_spec.rb +54 -24
- data/spec/faraday/request/instrumentation_spec.rb +5 -7
- data/spec/faraday/request/json_spec.rb +199 -0
- data/spec/faraday/request/url_encoded_spec.rb +12 -2
- data/spec/faraday/request_spec.rb +14 -15
- data/spec/faraday/response/json_spec.rb +206 -0
- data/spec/faraday/response/logger_spec.rb +84 -5
- data/spec/faraday/response/raise_error_spec.rb +133 -16
- data/spec/faraday/response_spec.rb +10 -1
- data/spec/faraday/utils/headers_spec.rb +31 -4
- data/spec/faraday/utils_spec.rb +66 -2
- data/spec/faraday_spec.rb +10 -4
- data/spec/spec_helper.rb +6 -5
- data/spec/support/fake_safe_buffer.rb +1 -1
- data/spec/support/faraday_middleware_subclasses.rb +18 -0
- data/spec/support/helper_methods.rb +0 -37
- data/spec/support/shared_examples/adapter.rb +2 -2
- data/spec/support/shared_examples/request_method.rb +22 -21
- metadata +27 -81
- data/lib/faraday/adapter/em_http.rb +0 -289
- data/lib/faraday/adapter/em_http_ssl_patch.rb +0 -62
- data/lib/faraday/adapter/em_synchrony/parallel_manager.rb +0 -69
- data/lib/faraday/adapter/em_synchrony.rb +0 -153
- data/lib/faraday/adapter/httpclient.rb +0 -152
- data/lib/faraday/adapter/patron.rb +0 -132
- data/lib/faraday/adapter/rack.rb +0 -75
- data/lib/faraday/adapter/typhoeus.rb +0 -15
- data/lib/faraday/autoload.rb +0 -92
- data/lib/faraday/dependency_loader.rb +0 -37
- data/lib/faraday/file_part.rb +0 -128
- data/lib/faraday/param_part.rb +0 -53
- data/lib/faraday/request/basic_authentication.rb +0 -20
- data/lib/faraday/request/multipart.rb +0 -106
- data/lib/faraday/request/retry.rb +0 -239
- data/lib/faraday/request/token_authentication.rb +0 -20
- data/spec/faraday/adapter/em_http_spec.rb +0 -47
- data/spec/faraday/adapter/em_synchrony_spec.rb +0 -16
- data/spec/faraday/adapter/excon_spec.rb +0 -49
- data/spec/faraday/adapter/httpclient_spec.rb +0 -73
- data/spec/faraday/adapter/net_http_spec.rb +0 -64
- data/spec/faraday/adapter/patron_spec.rb +0 -18
- data/spec/faraday/adapter/rack_spec.rb +0 -8
- data/spec/faraday/adapter/typhoeus_spec.rb +0 -7
- data/spec/faraday/composite_read_io_spec.rb +0 -80
- data/spec/faraday/request/multipart_spec.rb +0 -302
- data/spec/faraday/request/retry_spec.rb +0 -242
- data/spec/faraday/response/middleware_spec.rb +0 -68
- data/spec/support/webmock_rack_app.rb +0 -68
|
@@ -1,62 +0,0 @@
|
|
|
1
|
-
# frozen_string_literal: true
|
|
2
|
-
|
|
3
|
-
require 'openssl'
|
|
4
|
-
require 'em-http'
|
|
5
|
-
|
|
6
|
-
# EventMachine patch to make SSL work.
|
|
7
|
-
module EmHttpSslPatch
|
|
8
|
-
def ssl_verify_peer(cert_string)
|
|
9
|
-
begin
|
|
10
|
-
@last_seen_cert = OpenSSL::X509::Certificate.new(cert_string)
|
|
11
|
-
rescue OpenSSL::X509::CertificateError
|
|
12
|
-
return false
|
|
13
|
-
end
|
|
14
|
-
|
|
15
|
-
unless certificate_store.verify(@last_seen_cert)
|
|
16
|
-
raise OpenSSL::SSL::SSLError,
|
|
17
|
-
%(unable to verify the server certificate for "#{host}")
|
|
18
|
-
end
|
|
19
|
-
|
|
20
|
-
begin
|
|
21
|
-
certificate_store.add_cert(@last_seen_cert)
|
|
22
|
-
rescue OpenSSL::X509::StoreError => e
|
|
23
|
-
raise e unless e.message == 'cert already in hash table'
|
|
24
|
-
end
|
|
25
|
-
true
|
|
26
|
-
end
|
|
27
|
-
|
|
28
|
-
def ssl_handshake_completed
|
|
29
|
-
return true unless verify_peer?
|
|
30
|
-
|
|
31
|
-
unless verified_cert_identity?
|
|
32
|
-
raise OpenSSL::SSL::SSLError,
|
|
33
|
-
%(host "#{host}" does not match the server certificate)
|
|
34
|
-
end
|
|
35
|
-
|
|
36
|
-
true
|
|
37
|
-
end
|
|
38
|
-
|
|
39
|
-
def verify_peer?
|
|
40
|
-
parent.connopts.tls[:verify_peer]
|
|
41
|
-
end
|
|
42
|
-
|
|
43
|
-
def verified_cert_identity?
|
|
44
|
-
OpenSSL::SSL.verify_certificate_identity(@last_seen_cert, host)
|
|
45
|
-
end
|
|
46
|
-
|
|
47
|
-
def host
|
|
48
|
-
parent.uri.host
|
|
49
|
-
end
|
|
50
|
-
|
|
51
|
-
def certificate_store
|
|
52
|
-
@certificate_store ||= begin
|
|
53
|
-
store = OpenSSL::X509::Store.new
|
|
54
|
-
store.set_default_paths
|
|
55
|
-
ca_file = parent.connopts.tls[:cert_chain_file]
|
|
56
|
-
store.add_file(ca_file) if ca_file
|
|
57
|
-
store
|
|
58
|
-
end
|
|
59
|
-
end
|
|
60
|
-
end
|
|
61
|
-
|
|
62
|
-
EventMachine::HttpStubConnection.include(EmHttpSslPatch)
|
|
@@ -1,69 +0,0 @@
|
|
|
1
|
-
# frozen_string_literal: true
|
|
2
|
-
|
|
3
|
-
module Faraday
|
|
4
|
-
class Adapter
|
|
5
|
-
class EMSynchrony < Faraday::Adapter
|
|
6
|
-
# A parallel manager for EMSynchrony.
|
|
7
|
-
class ParallelManager
|
|
8
|
-
# Add requests to queue.
|
|
9
|
-
#
|
|
10
|
-
# @param request [EM::HttpRequest]
|
|
11
|
-
# @param method [Symbol, String] HTTP method
|
|
12
|
-
# @param args [Array] the rest of the positional arguments
|
|
13
|
-
def add(request, method, *args, &block)
|
|
14
|
-
queue << {
|
|
15
|
-
request: request,
|
|
16
|
-
method: method,
|
|
17
|
-
args: args,
|
|
18
|
-
block: block
|
|
19
|
-
}
|
|
20
|
-
end
|
|
21
|
-
|
|
22
|
-
# Run all requests on queue with `EM::Synchrony::Multi`, wrapping
|
|
23
|
-
# it in a reactor and fiber if needed.
|
|
24
|
-
def run
|
|
25
|
-
result = nil
|
|
26
|
-
if !EM.reactor_running?
|
|
27
|
-
EM.run do
|
|
28
|
-
Fiber.new do
|
|
29
|
-
result = perform
|
|
30
|
-
EM.stop
|
|
31
|
-
end.resume
|
|
32
|
-
end
|
|
33
|
-
else
|
|
34
|
-
result = perform
|
|
35
|
-
end
|
|
36
|
-
result
|
|
37
|
-
end
|
|
38
|
-
|
|
39
|
-
private
|
|
40
|
-
|
|
41
|
-
# The request queue.
|
|
42
|
-
def queue
|
|
43
|
-
@queue ||= []
|
|
44
|
-
end
|
|
45
|
-
|
|
46
|
-
# Main `EM::Synchrony::Multi` performer.
|
|
47
|
-
def perform
|
|
48
|
-
multi = ::EM::Synchrony::Multi.new
|
|
49
|
-
|
|
50
|
-
queue.each do |item|
|
|
51
|
-
method = "a#{item[:method]}".to_sym
|
|
52
|
-
|
|
53
|
-
req = item[:request].send(method, *item[:args])
|
|
54
|
-
req.callback(&item[:block])
|
|
55
|
-
|
|
56
|
-
req_name = "req_#{multi.requests.size}".to_sym
|
|
57
|
-
multi.add(req_name, req)
|
|
58
|
-
end
|
|
59
|
-
|
|
60
|
-
# Clear the queue, so parallel manager objects can be reused.
|
|
61
|
-
@queue = []
|
|
62
|
-
|
|
63
|
-
# Block fiber until all requests have returned.
|
|
64
|
-
multi.perform
|
|
65
|
-
end
|
|
66
|
-
end
|
|
67
|
-
end
|
|
68
|
-
end
|
|
69
|
-
end
|
|
@@ -1,153 +0,0 @@
|
|
|
1
|
-
# frozen_string_literal: true
|
|
2
|
-
|
|
3
|
-
require 'uri'
|
|
4
|
-
|
|
5
|
-
module Faraday
|
|
6
|
-
class Adapter
|
|
7
|
-
# EventMachine Synchrony adapter.
|
|
8
|
-
class EMSynchrony < Faraday::Adapter
|
|
9
|
-
include EMHttp::Options
|
|
10
|
-
|
|
11
|
-
dependency do
|
|
12
|
-
require 'em-synchrony/em-http'
|
|
13
|
-
require 'em-synchrony/em-multi'
|
|
14
|
-
require 'fiber'
|
|
15
|
-
|
|
16
|
-
require 'faraday/adapter/em_synchrony/parallel_manager'
|
|
17
|
-
|
|
18
|
-
if Faraday::Adapter::EMSynchrony.loaded?
|
|
19
|
-
begin
|
|
20
|
-
require 'openssl'
|
|
21
|
-
rescue LoadError
|
|
22
|
-
warn 'Warning: no such file to load -- openssl. ' \
|
|
23
|
-
'Make sure it is installed if you want HTTPS support'
|
|
24
|
-
else
|
|
25
|
-
require 'em-http/version'
|
|
26
|
-
if EventMachine::HttpRequest::VERSION < '1.1.6'
|
|
27
|
-
require 'faraday/adapter/em_http_ssl_patch'
|
|
28
|
-
end
|
|
29
|
-
end
|
|
30
|
-
end
|
|
31
|
-
end
|
|
32
|
-
|
|
33
|
-
self.supports_parallel = true
|
|
34
|
-
|
|
35
|
-
# @return [ParallelManager]
|
|
36
|
-
def self.setup_parallel_manager(_options = nil)
|
|
37
|
-
ParallelManager.new
|
|
38
|
-
end
|
|
39
|
-
|
|
40
|
-
def call(env)
|
|
41
|
-
super
|
|
42
|
-
request = create_request(env)
|
|
43
|
-
|
|
44
|
-
http_method = env[:method].to_s.downcase.to_sym
|
|
45
|
-
|
|
46
|
-
if env[:parallel_manager]
|
|
47
|
-
# Queue requests for parallel execution.
|
|
48
|
-
execute_parallel_request(env, request, http_method)
|
|
49
|
-
else
|
|
50
|
-
# Execute single request.
|
|
51
|
-
execute_single_request(env, request, http_method)
|
|
52
|
-
end
|
|
53
|
-
|
|
54
|
-
@app.call env
|
|
55
|
-
rescue Errno::ECONNREFUSED
|
|
56
|
-
raise Faraday::ConnectionFailed, $ERROR_INFO
|
|
57
|
-
rescue EventMachine::Connectify::CONNECTError => e
|
|
58
|
-
if e.message.include?('Proxy Authentication Required')
|
|
59
|
-
raise Faraday::ConnectionFailed,
|
|
60
|
-
%(407 "Proxy Authentication Required")
|
|
61
|
-
end
|
|
62
|
-
|
|
63
|
-
raise Faraday::ConnectionFailed, e
|
|
64
|
-
rescue Errno::ETIMEDOUT => e
|
|
65
|
-
raise Faraday::TimeoutError, e
|
|
66
|
-
rescue RuntimeError => e
|
|
67
|
-
if e.message == 'connection closed by server'
|
|
68
|
-
raise Faraday::ConnectionFailed, e
|
|
69
|
-
end
|
|
70
|
-
|
|
71
|
-
raise Faraday::TimeoutError, e if e.message.include?('timeout error')
|
|
72
|
-
|
|
73
|
-
raise
|
|
74
|
-
rescue StandardError => e
|
|
75
|
-
if defined?(OpenSSL) && e.is_a?(OpenSSL::SSL::SSLError)
|
|
76
|
-
raise Faraday::SSLError, e
|
|
77
|
-
end
|
|
78
|
-
|
|
79
|
-
raise
|
|
80
|
-
end
|
|
81
|
-
|
|
82
|
-
def create_request(env)
|
|
83
|
-
EventMachine::HttpRequest.new(
|
|
84
|
-
Utils::URI(env[:url].to_s),
|
|
85
|
-
connection_config(env).merge(@connection_options)
|
|
86
|
-
)
|
|
87
|
-
end
|
|
88
|
-
|
|
89
|
-
private
|
|
90
|
-
|
|
91
|
-
def execute_parallel_request(env, request, http_method)
|
|
92
|
-
env[:parallel_manager].add(request, http_method,
|
|
93
|
-
request_config(env)) do |resp|
|
|
94
|
-
if (req = env[:request]).stream_response?
|
|
95
|
-
warn "Streaming downloads for #{self.class.name} " \
|
|
96
|
-
'are not yet implemented.'
|
|
97
|
-
req.on_data.call(resp.response, resp.response.bytesize)
|
|
98
|
-
end
|
|
99
|
-
|
|
100
|
-
save_response(env, resp.response_header.status,
|
|
101
|
-
resp.response) do |resp_headers|
|
|
102
|
-
resp.response_header.each do |name, value|
|
|
103
|
-
resp_headers[name.to_sym] = value
|
|
104
|
-
end
|
|
105
|
-
end
|
|
106
|
-
|
|
107
|
-
# Finalize the response object with values from `env`.
|
|
108
|
-
env[:response].finish(env)
|
|
109
|
-
end
|
|
110
|
-
end
|
|
111
|
-
|
|
112
|
-
def execute_single_request(env, request, http_method)
|
|
113
|
-
block = -> { request.send(http_method, request_config(env)) }
|
|
114
|
-
client = call_block(block)
|
|
115
|
-
|
|
116
|
-
raise client.error if client&.error
|
|
117
|
-
|
|
118
|
-
if env[:request].stream_response?
|
|
119
|
-
warn "Streaming downloads for #{self.class.name} " \
|
|
120
|
-
'are not yet implemented.'
|
|
121
|
-
env[:request].on_data.call(
|
|
122
|
-
client.response,
|
|
123
|
-
client.response.bytesize
|
|
124
|
-
)
|
|
125
|
-
end
|
|
126
|
-
status = client.response_header.status
|
|
127
|
-
reason = client.response_header.http_reason
|
|
128
|
-
save_response(env, status, client.response, nil, reason) do |headers|
|
|
129
|
-
client.response_header.each do |name, value|
|
|
130
|
-
headers[name.to_sym] = value
|
|
131
|
-
end
|
|
132
|
-
end
|
|
133
|
-
end
|
|
134
|
-
|
|
135
|
-
def call_block(block)
|
|
136
|
-
client = nil
|
|
137
|
-
|
|
138
|
-
if EM.reactor_running?
|
|
139
|
-
client = block.call
|
|
140
|
-
else
|
|
141
|
-
EM.run do
|
|
142
|
-
Fiber.new do
|
|
143
|
-
client = block.call
|
|
144
|
-
EM.stop
|
|
145
|
-
end.resume
|
|
146
|
-
end
|
|
147
|
-
end
|
|
148
|
-
|
|
149
|
-
client
|
|
150
|
-
end
|
|
151
|
-
end
|
|
152
|
-
end
|
|
153
|
-
end
|
|
@@ -1,152 +0,0 @@
|
|
|
1
|
-
# frozen_string_literal: true
|
|
2
|
-
|
|
3
|
-
module Faraday
|
|
4
|
-
class Adapter
|
|
5
|
-
# HTTPClient adapter.
|
|
6
|
-
class HTTPClient < Faraday::Adapter
|
|
7
|
-
dependency 'httpclient'
|
|
8
|
-
|
|
9
|
-
def build_connection(env)
|
|
10
|
-
@client ||= ::HTTPClient.new.tap do |cli|
|
|
11
|
-
# enable compression
|
|
12
|
-
cli.transparent_gzip_decompression = true
|
|
13
|
-
end
|
|
14
|
-
|
|
15
|
-
if (req = env[:request])
|
|
16
|
-
if (proxy = req[:proxy])
|
|
17
|
-
configure_proxy @client, proxy
|
|
18
|
-
end
|
|
19
|
-
|
|
20
|
-
if (bind = req[:bind])
|
|
21
|
-
configure_socket @client, bind
|
|
22
|
-
end
|
|
23
|
-
|
|
24
|
-
configure_timeouts @client, req
|
|
25
|
-
end
|
|
26
|
-
|
|
27
|
-
if env[:url].scheme == 'https' && (ssl = env[:ssl])
|
|
28
|
-
configure_ssl @client, ssl
|
|
29
|
-
end
|
|
30
|
-
|
|
31
|
-
configure_client @client
|
|
32
|
-
|
|
33
|
-
@client
|
|
34
|
-
end
|
|
35
|
-
|
|
36
|
-
def call(env)
|
|
37
|
-
super
|
|
38
|
-
|
|
39
|
-
# TODO: Don't stream yet.
|
|
40
|
-
# https://github.com/nahi/httpclient/pull/90
|
|
41
|
-
env[:body] = env[:body].read if env[:body].respond_to? :read
|
|
42
|
-
|
|
43
|
-
connection(env) do |http|
|
|
44
|
-
resp = http.request env[:method], env[:url],
|
|
45
|
-
body: env[:body],
|
|
46
|
-
header: env[:request_headers]
|
|
47
|
-
|
|
48
|
-
if (req = env[:request]).stream_response?
|
|
49
|
-
warn "Streaming downloads for #{self.class.name} " \
|
|
50
|
-
'are not yet implemented.'
|
|
51
|
-
req.on_data.call(resp.body, resp.body.bytesize)
|
|
52
|
-
end
|
|
53
|
-
save_response env, resp.status, resp.body, resp.headers, resp.reason
|
|
54
|
-
|
|
55
|
-
@app.call env
|
|
56
|
-
end
|
|
57
|
-
rescue ::HTTPClient::TimeoutError, Errno::ETIMEDOUT
|
|
58
|
-
raise Faraday::TimeoutError, $ERROR_INFO
|
|
59
|
-
rescue ::HTTPClient::BadResponseError => e
|
|
60
|
-
if e.message.include?('status 407')
|
|
61
|
-
raise Faraday::ConnectionFailed,
|
|
62
|
-
%(407 "Proxy Authentication Required ")
|
|
63
|
-
end
|
|
64
|
-
|
|
65
|
-
raise Faraday::ClientError, $ERROR_INFO
|
|
66
|
-
rescue Errno::EADDRNOTAVAIL, Errno::ECONNREFUSED, IOError, SocketError
|
|
67
|
-
raise Faraday::ConnectionFailed, $ERROR_INFO
|
|
68
|
-
rescue StandardError => e
|
|
69
|
-
if defined?(::OpenSSL::SSL::SSLError) && \
|
|
70
|
-
e.is_a?(::OpenSSL::SSL::SSLError)
|
|
71
|
-
raise Faraday::SSLError, e
|
|
72
|
-
end
|
|
73
|
-
|
|
74
|
-
raise
|
|
75
|
-
end
|
|
76
|
-
|
|
77
|
-
# @param bind [Hash]
|
|
78
|
-
def configure_socket(client, bind)
|
|
79
|
-
client.socket_local.host = bind[:host]
|
|
80
|
-
client.socket_local.port = bind[:port]
|
|
81
|
-
end
|
|
82
|
-
|
|
83
|
-
# Configure proxy URI and any user credentials.
|
|
84
|
-
#
|
|
85
|
-
# @param proxy [Hash]
|
|
86
|
-
def configure_proxy(client, proxy)
|
|
87
|
-
client.proxy = proxy[:uri]
|
|
88
|
-
return unless proxy[:user] && proxy[:password]
|
|
89
|
-
|
|
90
|
-
client.set_proxy_auth(proxy[:user], proxy[:password])
|
|
91
|
-
end
|
|
92
|
-
|
|
93
|
-
# @param ssl [Hash]
|
|
94
|
-
def configure_ssl(client, ssl)
|
|
95
|
-
ssl_config = client.ssl_config
|
|
96
|
-
ssl_config.verify_mode = ssl_verify_mode(ssl)
|
|
97
|
-
ssl_config.cert_store = ssl_cert_store(ssl)
|
|
98
|
-
|
|
99
|
-
ssl_config.add_trust_ca ssl[:ca_file] if ssl[:ca_file]
|
|
100
|
-
ssl_config.add_trust_ca ssl[:ca_path] if ssl[:ca_path]
|
|
101
|
-
ssl_config.client_cert = ssl[:client_cert] if ssl[:client_cert]
|
|
102
|
-
ssl_config.client_key = ssl[:client_key] if ssl[:client_key]
|
|
103
|
-
ssl_config.verify_depth = ssl[:verify_depth] if ssl[:verify_depth]
|
|
104
|
-
end
|
|
105
|
-
|
|
106
|
-
# @param req [Hash]
|
|
107
|
-
def configure_timeouts(client, req)
|
|
108
|
-
if (sec = request_timeout(:open, req))
|
|
109
|
-
client.connect_timeout = sec
|
|
110
|
-
end
|
|
111
|
-
|
|
112
|
-
if (sec = request_timeout(:write, req))
|
|
113
|
-
client.send_timeout = sec
|
|
114
|
-
end
|
|
115
|
-
|
|
116
|
-
return unless (sec = request_timeout(:read, req))
|
|
117
|
-
|
|
118
|
-
client.receive_timeout = sec
|
|
119
|
-
end
|
|
120
|
-
|
|
121
|
-
def configure_client(client)
|
|
122
|
-
@config_block&.call(client)
|
|
123
|
-
end
|
|
124
|
-
|
|
125
|
-
# @param ssl [Hash]
|
|
126
|
-
# @return [OpenSSL::X509::Store]
|
|
127
|
-
def ssl_cert_store(ssl)
|
|
128
|
-
return ssl[:cert_store] if ssl[:cert_store]
|
|
129
|
-
|
|
130
|
-
# Memoize the cert store so that the same one is passed to
|
|
131
|
-
# HTTPClient each time, to avoid resyncing SSL sessions when
|
|
132
|
-
# it's changed
|
|
133
|
-
@ssl_cert_store ||= begin
|
|
134
|
-
# Use the default cert store by default, i.e. system ca certs
|
|
135
|
-
OpenSSL::X509::Store.new.tap(&:set_default_paths)
|
|
136
|
-
end
|
|
137
|
-
end
|
|
138
|
-
|
|
139
|
-
# @param ssl [Hash]
|
|
140
|
-
def ssl_verify_mode(ssl)
|
|
141
|
-
ssl[:verify_mode] || begin
|
|
142
|
-
if ssl.fetch(:verify, true)
|
|
143
|
-
OpenSSL::SSL::VERIFY_PEER |
|
|
144
|
-
OpenSSL::SSL::VERIFY_FAIL_IF_NO_PEER_CERT
|
|
145
|
-
else
|
|
146
|
-
OpenSSL::SSL::VERIFY_NONE
|
|
147
|
-
end
|
|
148
|
-
end
|
|
149
|
-
end
|
|
150
|
-
end
|
|
151
|
-
end
|
|
152
|
-
end
|
|
@@ -1,132 +0,0 @@
|
|
|
1
|
-
# frozen_string_literal: true
|
|
2
|
-
|
|
3
|
-
module Faraday
|
|
4
|
-
class Adapter
|
|
5
|
-
# Patron adapter.
|
|
6
|
-
class Patron < Faraday::Adapter
|
|
7
|
-
dependency 'patron'
|
|
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
|
-
|
|
24
|
-
def call(env)
|
|
25
|
-
super
|
|
26
|
-
# TODO: support streaming requests
|
|
27
|
-
env[:body] = env[:body].read if env[:body].respond_to? :read
|
|
28
|
-
|
|
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
|
|
36
|
-
end
|
|
37
|
-
end
|
|
38
|
-
|
|
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)
|
|
43
|
-
end
|
|
44
|
-
# Remove the "HTTP/1.1 200", leaving just the reason phrase
|
|
45
|
-
reason_phrase = response.status_line.gsub(/^.* \d{3} /, '')
|
|
46
|
-
|
|
47
|
-
save_response(env, response.status, response.body,
|
|
48
|
-
response.headers, reason_phrase)
|
|
49
|
-
|
|
50
|
-
@app.call env
|
|
51
|
-
rescue ::Patron::TimeoutError => e
|
|
52
|
-
if connection_timed_out_message?(e.message)
|
|
53
|
-
raise Faraday::ConnectionFailed, e
|
|
54
|
-
end
|
|
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 ")
|
|
61
|
-
end
|
|
62
|
-
|
|
63
|
-
raise Faraday::ConnectionFailed, e
|
|
64
|
-
end
|
|
65
|
-
|
|
66
|
-
if loaded? && defined?(::Patron::Request::VALID_ACTIONS)
|
|
67
|
-
# HAX: helps but doesn't work completely
|
|
68
|
-
# https://github.com/toland/patron/issues/34
|
|
69
|
-
::Patron::Request::VALID_ACTIONS.tap do |actions|
|
|
70
|
-
if actions[0].is_a?(Symbol)
|
|
71
|
-
actions << :patch unless actions.include? :patch
|
|
72
|
-
actions << :options unless actions.include? :options
|
|
73
|
-
else
|
|
74
|
-
# Patron 0.4.20 and up
|
|
75
|
-
actions << 'PATCH' unless actions.include? 'PATCH'
|
|
76
|
-
actions << 'OPTIONS' unless actions.include? 'OPTIONS'
|
|
77
|
-
end
|
|
78
|
-
end
|
|
79
|
-
end
|
|
80
|
-
|
|
81
|
-
def configure_ssl(session, ssl)
|
|
82
|
-
if ssl.fetch(:verify, true)
|
|
83
|
-
session.cacert = ssl[:ca_file]
|
|
84
|
-
else
|
|
85
|
-
session.insecure = true
|
|
86
|
-
end
|
|
87
|
-
end
|
|
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
|
-
|
|
112
|
-
private
|
|
113
|
-
|
|
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
|
|
124
|
-
|
|
125
|
-
def connection_timed_out_message?(message)
|
|
126
|
-
CURL_TIMEOUT_MESSAGES.any? do |curl_message|
|
|
127
|
-
message.include?(curl_message)
|
|
128
|
-
end
|
|
129
|
-
end
|
|
130
|
-
end
|
|
131
|
-
end
|
|
132
|
-
end
|
data/lib/faraday/adapter/rack.rb
DELETED
|
@@ -1,75 +0,0 @@
|
|
|
1
|
-
# frozen_string_literal: true
|
|
2
|
-
|
|
3
|
-
module Faraday
|
|
4
|
-
class Adapter
|
|
5
|
-
# Sends requests to a Rack app.
|
|
6
|
-
#
|
|
7
|
-
# @example
|
|
8
|
-
#
|
|
9
|
-
# class MyRackApp
|
|
10
|
-
# def call(env)
|
|
11
|
-
# [200, {'Content-Type' => 'text/html'}, ["hello world"]]
|
|
12
|
-
# end
|
|
13
|
-
# end
|
|
14
|
-
#
|
|
15
|
-
# Faraday.new do |conn|
|
|
16
|
-
# conn.adapter :rack, MyRackApp.new
|
|
17
|
-
# end
|
|
18
|
-
class Rack < Faraday::Adapter
|
|
19
|
-
dependency 'rack/test'
|
|
20
|
-
|
|
21
|
-
# not prefixed with "HTTP_"
|
|
22
|
-
SPECIAL_HEADERS = %w[CONTENT_LENGTH CONTENT_TYPE].freeze
|
|
23
|
-
|
|
24
|
-
def initialize(faraday_app, rack_app)
|
|
25
|
-
super(faraday_app)
|
|
26
|
-
mock_session = ::Rack::MockSession.new(rack_app)
|
|
27
|
-
@session = ::Rack::Test::Session.new(mock_session)
|
|
28
|
-
end
|
|
29
|
-
|
|
30
|
-
def call(env)
|
|
31
|
-
super
|
|
32
|
-
rack_env = build_rack_env(env)
|
|
33
|
-
|
|
34
|
-
env[:request_headers]&.each do |name, value|
|
|
35
|
-
name = name.upcase.tr('-', '_')
|
|
36
|
-
name = "HTTP_#{name}" unless SPECIAL_HEADERS.include? name
|
|
37
|
-
rack_env[name] = value
|
|
38
|
-
end
|
|
39
|
-
|
|
40
|
-
timeout = request_timeout(:open, env[:request])
|
|
41
|
-
timeout ||= request_timeout(:read, env[:request])
|
|
42
|
-
response = if timeout
|
|
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)
|
|
54
|
-
end
|
|
55
|
-
|
|
56
|
-
save_response(env, response.status, response.body, response.headers)
|
|
57
|
-
@app.call env
|
|
58
|
-
end
|
|
59
|
-
|
|
60
|
-
private
|
|
61
|
-
|
|
62
|
-
def execute_request(env, rack_env)
|
|
63
|
-
@session.request(env[:url].to_s, rack_env)
|
|
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
|
|
73
|
-
end
|
|
74
|
-
end
|
|
75
|
-
end
|
|
@@ -1,15 +0,0 @@
|
|
|
1
|
-
# frozen_string_literal: true
|
|
2
|
-
|
|
3
|
-
module Faraday
|
|
4
|
-
class Adapter
|
|
5
|
-
# Typhoeus adapter. This class is just a stub, the real adapter is in
|
|
6
|
-
# https://github.com/typhoeus/typhoeus/blob/master/lib/typhoeus/adapters/faraday.rb
|
|
7
|
-
class Typhoeus < Faraday::Adapter
|
|
8
|
-
# Needs to define this method in order to support Typhoeus <= 1.3.0
|
|
9
|
-
def call; end
|
|
10
|
-
|
|
11
|
-
dependency 'typhoeus'
|
|
12
|
-
dependency 'typhoeus/adapters/faraday'
|
|
13
|
-
end
|
|
14
|
-
end
|
|
15
|
-
end
|