faraday 0.16.2 → 0.17.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.
- checksums.yaml +4 -4
- data/LICENSE.md +1 -1
- data/README.md +347 -18
- data/lib/faraday.rb +175 -93
- data/lib/faraday/adapter.rb +22 -36
- data/lib/faraday/adapter/em_http.rb +99 -142
- data/lib/faraday/adapter/em_http_ssl_patch.rb +17 -23
- data/lib/faraday/adapter/em_synchrony.rb +60 -104
- data/lib/faraday/adapter/em_synchrony/parallel_manager.rb +15 -18
- data/lib/faraday/adapter/excon.rb +55 -100
- data/lib/faraday/adapter/httpclient.rb +39 -61
- data/lib/faraday/adapter/net_http.rb +51 -104
- data/lib/faraday/adapter/net_http_persistent.rb +27 -48
- data/lib/faraday/adapter/patron.rb +35 -54
- data/lib/faraday/adapter/rack.rb +12 -28
- data/lib/faraday/adapter/test.rb +53 -86
- data/lib/faraday/adapter/typhoeus.rb +1 -4
- data/lib/faraday/autoload.rb +36 -47
- data/lib/faraday/connection.rb +179 -321
- data/lib/faraday/error.rb +32 -80
- data/lib/faraday/middleware.rb +28 -4
- data/lib/faraday/options.rb +186 -35
- data/lib/faraday/parameters.rb +197 -4
- data/lib/faraday/rack_builder.rb +56 -67
- data/lib/faraday/request.rb +36 -68
- data/lib/faraday/request/authorization.rb +30 -42
- data/lib/faraday/request/basic_authentication.rb +7 -14
- data/lib/faraday/request/instrumentation.rb +27 -45
- data/lib/faraday/request/multipart.rb +48 -79
- data/lib/faraday/request/retry.rb +170 -197
- data/lib/faraday/request/token_authentication.rb +10 -15
- data/lib/faraday/request/url_encoded.rb +23 -41
- data/lib/faraday/response.rb +16 -23
- data/lib/faraday/response/logger.rb +69 -22
- data/lib/faraday/response/raise_error.rb +14 -36
- data/lib/faraday/upload_io.rb +67 -0
- data/lib/faraday/utils.rb +245 -28
- metadata +5 -22
- data/lib/faraday/adapter_registry.rb +0 -28
- data/lib/faraday/dependency_loader.rb +0 -37
- data/lib/faraday/deprecated_class.rb +0 -28
- data/lib/faraday/encoders/flat_params_encoder.rb +0 -94
- data/lib/faraday/encoders/nested_params_encoder.rb +0 -171
- data/lib/faraday/file_part.rb +0 -128
- data/lib/faraday/logging/formatter.rb +0 -92
- data/lib/faraday/middleware_registry.rb +0 -129
- data/lib/faraday/options/connection_options.rb +0 -22
- data/lib/faraday/options/env.rb +0 -181
- data/lib/faraday/options/proxy_options.rb +0 -28
- data/lib/faraday/options/request_options.rb +0 -21
- data/lib/faraday/options/ssl_options.rb +0 -59
- data/lib/faraday/param_part.rb +0 -53
- data/lib/faraday/utils/headers.rb +0 -139
- data/lib/faraday/utils/params_hash.rb +0 -61
- data/spec/external_adapters/faraday_specs_setup.rb +0 -14
@@ -1,49 +1,43 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
1
|
require 'openssl'
|
4
2
|
require 'em-http'
|
5
3
|
|
6
|
-
# EventMachine patch to make SSL work.
|
7
4
|
module EmHttpSslPatch
|
8
5
|
def ssl_verify_peer(cert_string)
|
6
|
+
cert = nil
|
9
7
|
begin
|
10
|
-
|
8
|
+
cert = OpenSSL::X509::Certificate.new(cert_string)
|
11
9
|
rescue OpenSSL::X509::CertificateError
|
12
10
|
return false
|
13
11
|
end
|
14
12
|
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
13
|
+
@last_seen_cert = cert
|
14
|
+
|
15
|
+
if certificate_store.verify(@last_seen_cert)
|
16
|
+
begin
|
17
|
+
certificate_store.add_cert(@last_seen_cert)
|
18
|
+
rescue OpenSSL::X509::StoreError => e
|
19
|
+
raise e unless e.message == 'cert already in hash table'
|
20
|
+
end
|
21
|
+
true
|
22
|
+
else
|
23
|
+
raise OpenSSL::SSL::SSLError.new(%(unable to verify the server certificate for "#{host}"))
|
24
24
|
end
|
25
|
-
true
|
26
25
|
end
|
27
26
|
|
28
27
|
def ssl_handshake_completed
|
29
28
|
return true unless verify_peer?
|
30
29
|
|
31
|
-
unless
|
32
|
-
raise OpenSSL::SSL::SSLError
|
33
|
-
|
30
|
+
unless OpenSSL::SSL.verify_certificate_identity(@last_seen_cert, host)
|
31
|
+
raise OpenSSL::SSL::SSLError.new(%(host "#{host}" does not match the server certificate))
|
32
|
+
else
|
33
|
+
true
|
34
34
|
end
|
35
|
-
|
36
|
-
true
|
37
35
|
end
|
38
36
|
|
39
37
|
def verify_peer?
|
40
38
|
parent.connopts.tls[:verify_peer]
|
41
39
|
end
|
42
40
|
|
43
|
-
def verified_cert_identity?
|
44
|
-
OpenSSL::SSL.verify_certificate_identity(@last_seen_cert, host)
|
45
|
-
end
|
46
|
-
|
47
41
|
def host
|
48
42
|
parent.uri.host
|
49
43
|
end
|
@@ -1,10 +1,7 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
1
|
require 'uri'
|
4
2
|
|
5
3
|
module Faraday
|
6
4
|
class Adapter
|
7
|
-
# EventMachine Synchrony adapter.
|
8
5
|
class EMSynchrony < Faraday::Adapter
|
9
6
|
include EMHttp::Options
|
10
7
|
|
@@ -16,8 +13,7 @@ module Faraday
|
|
16
13
|
|
17
14
|
self.supports_parallel = true
|
18
15
|
|
19
|
-
|
20
|
-
def self.setup_parallel_manager(_options = nil)
|
16
|
+
def self.setup_parallel_manager(options = {})
|
21
17
|
ParallelManager.new
|
22
18
|
end
|
23
19
|
|
@@ -27,110 +23,73 @@ module Faraday
|
|
27
23
|
|
28
24
|
http_method = env[:method].to_s.downcase.to_sym
|
29
25
|
|
26
|
+
# Queue requests for parallel execution.
|
30
27
|
if env[:parallel_manager]
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
@app.call env
|
39
|
-
rescue Errno::ECONNREFUSED
|
40
|
-
raise Faraday::ConnectionFailed, $ERROR_INFO
|
41
|
-
rescue EventMachine::Connectify::CONNECTError => e
|
42
|
-
if e.message.include?('Proxy Authentication Required')
|
43
|
-
raise Faraday::ConnectionFailed,
|
44
|
-
%(407 "Proxy Authentication Required")
|
45
|
-
end
|
46
|
-
|
47
|
-
raise Faraday::ConnectionFailed, e
|
48
|
-
rescue Errno::ETIMEDOUT => e
|
49
|
-
raise Faraday::TimeoutError, e
|
50
|
-
rescue RuntimeError => e
|
51
|
-
if e.message == 'connection closed by server'
|
52
|
-
raise Faraday::ConnectionFailed, e
|
53
|
-
end
|
54
|
-
|
55
|
-
raise Faraday::TimeoutError, e if e.message.include?('timeout error')
|
56
|
-
|
57
|
-
raise
|
58
|
-
rescue StandardError => e
|
59
|
-
if defined?(OpenSSL) && e.is_a?(OpenSSL::SSL::SSLError)
|
60
|
-
raise Faraday::SSLError, e
|
61
|
-
end
|
62
|
-
|
63
|
-
raise
|
64
|
-
end
|
65
|
-
|
66
|
-
def create_request(env)
|
67
|
-
EventMachine::HttpRequest.new(
|
68
|
-
Utils::URI(env[:url].to_s),
|
69
|
-
connection_config(env).merge(@connection_options)
|
70
|
-
)
|
71
|
-
end
|
28
|
+
env[:parallel_manager].add(request, http_method, request_config(env)) do |resp|
|
29
|
+
save_response(env, resp.response_header.status, resp.response) do |resp_headers|
|
30
|
+
resp.response_header.each do |name, value|
|
31
|
+
resp_headers[name.to_sym] = value
|
32
|
+
end
|
33
|
+
end
|
72
34
|
|
73
|
-
|
35
|
+
# Finalize the response object with values from `env`.
|
36
|
+
env[:response].finish(env)
|
37
|
+
end
|
74
38
|
|
75
|
-
|
76
|
-
|
77
|
-
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
39
|
+
# Execute single request.
|
40
|
+
else
|
41
|
+
client = nil
|
42
|
+
block = lambda { request.send(http_method, request_config(env)) }
|
43
|
+
|
44
|
+
if !EM.reactor_running?
|
45
|
+
EM.run do
|
46
|
+
Fiber.new {
|
47
|
+
client = block.call
|
48
|
+
EM.stop
|
49
|
+
}.resume
|
50
|
+
end
|
51
|
+
else
|
52
|
+
client = block.call
|
82
53
|
end
|
83
54
|
|
84
|
-
|
85
|
-
|
86
|
-
|
55
|
+
raise client.error if client.error
|
56
|
+
|
57
|
+
status = client.response_header.status
|
58
|
+
reason = client.response_header.http_reason
|
59
|
+
save_response(env, status, client.response, nil, reason) do |resp_headers|
|
60
|
+
client.response_header.each do |name, value|
|
87
61
|
resp_headers[name.to_sym] = value
|
88
62
|
end
|
89
63
|
end
|
90
|
-
|
91
|
-
# Finalize the response object with values from `env`.
|
92
|
-
env[:response].finish(env)
|
93
64
|
end
|
94
|
-
end
|
95
|
-
|
96
|
-
def execute_single_request(env, request, http_method)
|
97
|
-
block = -> { request.send(http_method, request_config(env)) }
|
98
|
-
client = call_block(block)
|
99
65
|
|
100
|
-
|
101
|
-
|
102
|
-
|
103
|
-
|
104
|
-
|
105
|
-
|
106
|
-
|
107
|
-
|
108
|
-
)
|
66
|
+
@app.call env
|
67
|
+
rescue Errno::ECONNREFUSED
|
68
|
+
raise Error::ConnectionFailed, $!
|
69
|
+
rescue EventMachine::Connectify::CONNECTError => err
|
70
|
+
if err.message.include?("Proxy Authentication Required")
|
71
|
+
raise Error::ConnectionFailed, %{407 "Proxy Authentication Required "}
|
72
|
+
else
|
73
|
+
raise Error::ConnectionFailed, err
|
109
74
|
end
|
110
|
-
|
111
|
-
|
112
|
-
|
113
|
-
|
114
|
-
|
115
|
-
|
75
|
+
rescue Errno::ETIMEDOUT => err
|
76
|
+
raise Error::TimeoutError, err
|
77
|
+
rescue RuntimeError => err
|
78
|
+
if err.message == "connection closed by server"
|
79
|
+
raise Error::ConnectionFailed, err
|
80
|
+
else
|
81
|
+
raise
|
116
82
|
end
|
117
|
-
|
118
|
-
|
119
|
-
|
120
|
-
client = nil
|
121
|
-
|
122
|
-
if EM.reactor_running?
|
123
|
-
client = block.call
|
83
|
+
rescue => err
|
84
|
+
if defined?(OpenSSL) && OpenSSL::SSL::SSLError === err
|
85
|
+
raise Faraday::SSLError, err
|
124
86
|
else
|
125
|
-
|
126
|
-
Fiber.new do
|
127
|
-
client = block.call
|
128
|
-
EM.stop
|
129
|
-
end.resume
|
130
|
-
end
|
87
|
+
raise
|
131
88
|
end
|
89
|
+
end
|
132
90
|
|
133
|
-
|
91
|
+
def create_request(env)
|
92
|
+
EventMachine::HttpRequest.new(Utils::URI(env[:url].to_s), connection_config(env).merge(@connection_options))
|
134
93
|
end
|
135
94
|
end
|
136
95
|
end
|
@@ -138,13 +97,10 @@ end
|
|
138
97
|
|
139
98
|
require 'faraday/adapter/em_synchrony/parallel_manager'
|
140
99
|
|
141
|
-
|
142
|
-
|
143
|
-
|
144
|
-
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
require 'faraday/adapter/em_http_ssl_patch'
|
149
|
-
end
|
150
|
-
end
|
100
|
+
begin
|
101
|
+
require 'openssl'
|
102
|
+
rescue LoadError
|
103
|
+
warn "Warning: no such file to load -- openssl. Make sure it is installed if you want HTTPS support"
|
104
|
+
else
|
105
|
+
require 'faraday/adapter/em_http_ssl_patch'
|
106
|
+
end if Faraday::Adapter::EMSynchrony.loaded?
|
@@ -1,21 +1,16 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
1
|
module Faraday
|
4
2
|
class Adapter
|
5
3
|
class EMSynchrony < Faraday::Adapter
|
6
|
-
# A parallel manager for EMSynchrony.
|
7
4
|
class ParallelManager
|
8
|
-
|
9
|
-
#
|
10
|
-
#
|
11
|
-
# @param method [Symbol, String] HTTP method
|
12
|
-
# @param args [Array] the rest of the positional arguments
|
5
|
+
|
6
|
+
# Add requests to queue. The `request` argument should be a
|
7
|
+
# `EM::HttpRequest` object.
|
13
8
|
def add(request, method, *args, &block)
|
14
9
|
queue << {
|
15
|
-
request
|
16
|
-
method
|
17
|
-
args
|
18
|
-
block
|
10
|
+
:request => request,
|
11
|
+
:method => method,
|
12
|
+
:args => args,
|
13
|
+
:block => block
|
19
14
|
}
|
20
15
|
end
|
21
16
|
|
@@ -24,18 +19,19 @@ module Faraday
|
|
24
19
|
def run
|
25
20
|
result = nil
|
26
21
|
if !EM.reactor_running?
|
27
|
-
EM.run
|
22
|
+
EM.run {
|
28
23
|
Fiber.new do
|
29
24
|
result = perform
|
30
25
|
EM.stop
|
31
26
|
end.resume
|
32
|
-
|
27
|
+
}
|
33
28
|
else
|
34
29
|
result = perform
|
35
30
|
end
|
36
31
|
result
|
37
32
|
end
|
38
33
|
|
34
|
+
|
39
35
|
private
|
40
36
|
|
41
37
|
# The request queue.
|
@@ -63,7 +59,8 @@ module Faraday
|
|
63
59
|
# Block fiber until all requests have returned.
|
64
60
|
multi.perform
|
65
61
|
end
|
66
|
-
|
67
|
-
|
68
|
-
|
69
|
-
end
|
62
|
+
|
63
|
+
end # ParallelManager
|
64
|
+
end # EMSynchrony
|
65
|
+
end # Adapter
|
66
|
+
end # Faraday
|
@@ -1,47 +1,74 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
1
|
module Faraday
|
4
2
|
class Adapter
|
5
|
-
# Excon adapter.
|
6
3
|
class Excon < Faraday::Adapter
|
7
4
|
dependency 'excon'
|
8
5
|
|
9
6
|
def call(env)
|
10
7
|
super
|
11
8
|
|
12
|
-
opts =
|
13
|
-
|
14
|
-
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
|
9
|
+
opts = {}
|
10
|
+
if env[:url].scheme == 'https' && ssl = env[:ssl]
|
11
|
+
opts[:ssl_verify_peer] = !!ssl.fetch(:verify, true)
|
12
|
+
opts[:ssl_ca_path] = ssl[:ca_path] if ssl[:ca_path]
|
13
|
+
opts[:ssl_ca_file] = ssl[:ca_file] if ssl[:ca_file]
|
14
|
+
opts[:client_cert] = ssl[:client_cert] if ssl[:client_cert]
|
15
|
+
opts[:client_key] = ssl[:client_key] if ssl[:client_key]
|
16
|
+
opts[:certificate] = ssl[:certificate] if ssl[:certificate]
|
17
|
+
opts[:private_key] = ssl[:private_key] if ssl[:private_key]
|
18
|
+
opts[:ssl_version] = ssl[:version] if ssl[:version]
|
19
|
+
opts[:ssl_min_version] = ssl[:min_version] if ssl[:min_version]
|
20
|
+
opts[:ssl_max_version] = ssl[:max_version] if ssl[:max_version]
|
21
|
+
|
22
|
+
# https://github.com/geemus/excon/issues/106
|
23
|
+
# https://github.com/jruby/jruby-ossl/issues/19
|
24
|
+
opts[:nonblock] = false
|
25
|
+
end
|
20
26
|
|
21
|
-
req = env[:request]
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
req
|
27
|
+
if ( req = env[:request] )
|
28
|
+
if req[:timeout]
|
29
|
+
opts[:read_timeout] = req[:timeout]
|
30
|
+
opts[:connect_timeout] = req[:timeout]
|
31
|
+
opts[:write_timeout] = req[:timeout]
|
26
32
|
end
|
27
|
-
end
|
28
33
|
|
29
|
-
|
30
|
-
|
31
|
-
|
34
|
+
if req[:open_timeout]
|
35
|
+
opts[:connect_timeout] = req[:open_timeout]
|
36
|
+
end
|
32
37
|
|
33
|
-
|
34
|
-
|
35
|
-
|
38
|
+
if req[:proxy]
|
39
|
+
opts[:proxy] = {
|
40
|
+
:host => req[:proxy][:uri].host,
|
41
|
+
:hostname => req[:proxy][:uri].hostname,
|
42
|
+
:port => req[:proxy][:uri].port,
|
43
|
+
:scheme => req[:proxy][:uri].scheme,
|
44
|
+
:user => req[:proxy][:user],
|
45
|
+
:password => req[:proxy][:password]
|
46
|
+
}
|
47
|
+
end
|
48
|
+
end
|
36
49
|
|
37
|
-
|
50
|
+
conn = create_connection(env, opts)
|
38
51
|
|
39
|
-
|
40
|
-
|
41
|
-
|
52
|
+
resp = conn.request \
|
53
|
+
:method => env[:method].to_s.upcase,
|
54
|
+
:headers => env[:request_headers],
|
55
|
+
:body => read_body(env)
|
56
|
+
|
57
|
+
save_response(env, resp.status.to_i, resp.body, resp.headers, resp.reason_phrase)
|
58
|
+
|
59
|
+
@app.call env
|
60
|
+
rescue ::Excon::Errors::SocketError => err
|
61
|
+
if err.message =~ /\btimeout\b/
|
62
|
+
raise Error::TimeoutError, err
|
63
|
+
elsif err.message =~ /\bcertificate\b/
|
64
|
+
raise Faraday::SSLError, err
|
65
|
+
else
|
66
|
+
raise Error::ConnectionFailed, err
|
67
|
+
end
|
68
|
+
rescue ::Excon::Errors::Timeout => err
|
69
|
+
raise Error::TimeoutError, err
|
42
70
|
end
|
43
71
|
|
44
|
-
# @return [Excon]
|
45
72
|
def create_connection(env, opts)
|
46
73
|
::Excon.new(env[:url].to_s, opts.merge(@connection_options))
|
47
74
|
end
|
@@ -50,78 +77,6 @@ module Faraday
|
|
50
77
|
def read_body(env)
|
51
78
|
env[:body].respond_to?(:read) ? env[:body].read : env[:body]
|
52
79
|
end
|
53
|
-
|
54
|
-
private
|
55
|
-
|
56
|
-
def opts_from_env(env)
|
57
|
-
opts = {}
|
58
|
-
amend_opts_with_ssl!(opts, env[:ssl]) if needs_ssl_settings?(env)
|
59
|
-
|
60
|
-
if (req = env[:request])
|
61
|
-
amend_opts_with_timeouts!(opts, req)
|
62
|
-
amend_opts_with_proxy_settings!(opts, req)
|
63
|
-
end
|
64
|
-
|
65
|
-
opts
|
66
|
-
end
|
67
|
-
|
68
|
-
def needs_ssl_settings?(env)
|
69
|
-
env[:url].scheme == 'https' && env[:ssl]
|
70
|
-
end
|
71
|
-
|
72
|
-
OPTS_KEYS = [
|
73
|
-
%i[client_cert client_cert],
|
74
|
-
%i[client_key client_key],
|
75
|
-
%i[certificate certificate],
|
76
|
-
%i[private_key private_key],
|
77
|
-
%i[ssl_ca_path ca_path],
|
78
|
-
%i[ssl_ca_file ca_file],
|
79
|
-
%i[ssl_version version],
|
80
|
-
%i[ssl_min_version min_version],
|
81
|
-
%i[ssl_max_version max_version]
|
82
|
-
].freeze
|
83
|
-
|
84
|
-
def amend_opts_with_ssl!(opts, ssl)
|
85
|
-
opts[:ssl_verify_peer] = !!ssl.fetch(:verify, true)
|
86
|
-
# https://github.com/geemus/excon/issues/106
|
87
|
-
# https://github.com/jruby/jruby-ossl/issues/19
|
88
|
-
opts[:nonblock] = false
|
89
|
-
|
90
|
-
OPTS_KEYS.each do |(key_in_opts, key_in_ssl)|
|
91
|
-
next unless ssl[key_in_ssl]
|
92
|
-
|
93
|
-
opts[key_in_opts] = ssl[key_in_ssl]
|
94
|
-
end
|
95
|
-
end
|
96
|
-
|
97
|
-
def amend_opts_with_timeouts!(opts, req)
|
98
|
-
timeout = req[:timeout]
|
99
|
-
return unless timeout
|
100
|
-
|
101
|
-
opts[:read_timeout] = timeout
|
102
|
-
opts[:connect_timeout] = timeout
|
103
|
-
opts[:write_timeout] = timeout
|
104
|
-
|
105
|
-
open_timeout = req[:open_timeout]
|
106
|
-
return unless open_timeout
|
107
|
-
|
108
|
-
opts[:connect_timeout] = open_timeout
|
109
|
-
end
|
110
|
-
|
111
|
-
def amend_opts_with_proxy_settings!(opts, req)
|
112
|
-
opts[:proxy] = proxy_settings_for_opts(req[:proxy]) if req[:proxy]
|
113
|
-
end
|
114
|
-
|
115
|
-
def proxy_settings_for_opts(proxy)
|
116
|
-
{
|
117
|
-
host: proxy[:uri].host,
|
118
|
-
hostname: proxy[:uri].hostname,
|
119
|
-
port: proxy[:uri].port,
|
120
|
-
scheme: proxy[:uri].scheme,
|
121
|
-
user: proxy[:user],
|
122
|
-
password: proxy[:password]
|
123
|
-
}
|
124
|
-
end
|
125
80
|
end
|
126
81
|
end
|
127
82
|
end
|