Module: SimpleAuthSdk::Mixins::HTTPProxy

Included in:
API::AuthenticationEndpoints, SimpleAuthSdk::Mixins
Defined in:
lib/simple_auth_sdk/mixins/httpproxy.rb

Overview

HTTPProxy module provides methods for making HTTP requests using the rest-client gem. It includes functionality for retrying requests, encoding URIs, and handling various HTTP methods. You can integrate this module into your classes to easily make HTTP requests with retry logic.

Constant Summary collapse

DEFAULT_RETRIES =
3
MAX_ALLOWED_RETRIES =
10
MAX_REQUEST_RETRY_JITTER =
250
MAX_REQUEST_RETRY_DELAY =
1000
MIN_REQUEST_RETRY_DELAY =
250
BASE_DELAY =
100

Instance Attribute Summary collapse

Instance Method Summary collapse

Instance Attribute Details

#base_uriObject



200
201
202
# File 'lib/simple_auth_sdk/mixins/httpproxy.rb', line 200

def base_uri
  SimpleAuthSdk.configuration.base_uri
end

#headersHash

Returns the headers to be sent with each request.

Returns:

  • (Hash)

    the headers to be sent with each request.



17
# File 'lib/simple_auth_sdk/mixins/httpproxy.rb', line 17

attr_writer :headers

#retry_countInteger

Returns the number of retries for failed requests.

Returns:

  • (Integer)

    the number of retries for failed requests.



25
26
27
# File 'lib/simple_auth_sdk/mixins/httpproxy.rb', line 25

def retry_count
  @retry_count
end

#timeoutInteger

Returns the timeout in seconds for HTTP requests.

Returns:

  • (Integer)

    the timeout in seconds for HTTP requests.



21
22
23
# File 'lib/simple_auth_sdk/mixins/httpproxy.rb', line 21

def timeout
  @timeout
end

Instance Method Details

#add_headers(h = {}) ⇒ void

This method returns an undefined value.

Merges additional headers into the existing headers hash.

Parameters:

  • h (Hash) (defaults to: {})

    the headers to merge.

Raises:

  • (ArgumentError)

    if headers are not a hash.



100
101
102
103
104
105
# File 'lib/simple_auth_sdk/mixins/httpproxy.rb', line 100

def add_headers(h = {})
  raise ArgumentError, 'Headers must be an object which responds to #to_hash' unless h.respond_to?(:to_hash)

  @headers ||= {}
  @headers.merge!(h.to_hash)
end

#call(method, url, timeout, headers, body = nil) ⇒ RestClient::Response

Executes an HTTP request using rest-client.

Parameters:

  • method (Symbol)

    the HTTP method to use (e.g., :get, :post).

  • url (String)

    the full URL for the request.

  • timeout (Integer)

    the timeout for the request in seconds.

  • headers (Hash)

    the headers to include in the request.

  • body (String, nil) (defaults to: nil)

    the request body, if applicable.

Returns:

  • (RestClient::Response)

    the HTTP response.

Raises:



181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
# File 'lib/simple_auth_sdk/mixins/httpproxy.rb', line 181

def call(method, url, timeout, headers, body = nil)
  RestClient::Request.execute(
    method: method,
    url: url,
    timeout: timeout,
    headers: headers,
    payload: body
  )
rescue RestClient::Exception => e
  case e
  when RestClient::RequestTimeout
    raise SimpleAuthSdk::RequestTimeout, e.message
  else
    e.response
  end
end

#encode_uri(uri) ⇒ String

Encodes the URI based on whether a base_uri is set.

Parameters:

  • uri (String)

    the URI to encode.

Returns:

  • (String)

    the full encoded URI.



82
83
84
85
# File 'lib/simple_auth_sdk/mixins/httpproxy.rb', line 82

def encode_uri(uri)
  path = base_uri ? Addressable::URI.new(path: uri).normalized_path : Addressable::URI.escape(uri)
  url(path)
end

#methodHash, String

Defines HTTP methods that will be proxied to HTTP class methods. These methods will make an HTTP request with the corresponding HTTP method.

Parameters:

  • auth_token (String, nil)

    the authentication token to be used for the request. If nil, no Authorization header is set.

  • uri (String)

    the URI to make the request to.

  • body (Hash)

    the request body.

  • extra_headers (Hash)

    additional headers for the request.

Returns:

  • (Hash, String)

    the parsed response body, or raw response if JSON parsing fails.



42
43
44
45
46
47
48
49
50
51
52
53
54
55
# File 'lib/simple_auth_sdk/mixins/httpproxy.rb', line 42

%i[get post put patch delete delete_with_body].each do |method|
  define_method(method) do |uri, auth_token = nil, body = {}, extra_headers = {}|
    body = body.delete_if { |_, v| v.nil? }

    # Set the Authorization header if auth_token is provided
    headers_with_auth = extra_headers.dup
    headers_with_auth['Authorization'] = "Bearer #{auth_token}" if auth_token

    headers_with_auth['Content-Type'] ||= 'application/json'

    # Proceed with the request, passing the modified headers
    request_with_retry(method, uri, body, headers_with_auth)
  end
end

#request(method, uri, body = {}, extra_headers = {}) ⇒ Hash, String

Makes an HTTP request.

Parameters:

  • method (Symbol)

    the HTTP method to use (e.g., :get, :post).

  • uri (String)

    the URI to make the request to.

  • body (Hash) (defaults to: {})

    the request body.

  • extra_headers (Hash) (defaults to: {})

    additional headers for the request.

Returns:

  • (Hash, String)

    the parsed response body, or raw response if JSON parsing fails.

Raises:



144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
# File 'lib/simple_auth_sdk/mixins/httpproxy.rb', line 144

def request(method, uri, body = {}, extra_headers = {})
  result = case method
           when :get
             @headers ||= {}
             get_headers = @headers.merge({ params: body }).merge(extra_headers)
             call(:get, encode_uri(uri), timeout, get_headers)
           when :delete
             @headers ||= {}
             delete_headers = @headers.merge({ params: body })
             call(:delete, encode_uri(uri), timeout, delete_headers)
           when :delete_with_body
             call(:delete, encode_uri(uri), timeout, headers, body.to_json)
           else
             get_headers = headers.merge(extra_headers)
             call(method, encode_uri(uri), timeout, get_headers, body.to_json)
           end
  case result.code
  when 200...226 then safe_parse_json(result.body)
  when 400       then raise SimpleAuthSdk::BadRequest.new(result.body, code: result.code, headers: result.headers)
  when 401       then raise SimpleAuthSdk::Unauthorized.new(result.body, code: result.code, headers: result.headers)
  when 403       then raise SimpleAuthSdk::AccessDenied.new(result.body, code: result.code, headers: result.headers)
  when 404       then raise SimpleAuthSdk::NotFound.new(result.body, code: result.code, headers: result.headers)
  when 429       then raise SimpleAuthSdk::RateLimitEncountered.new(result.body, code: result.code, headers: result.headers)
  when 500       then raise SimpleAuthSdk::ServerError.new(result.body, code: result.code, headers: result.headers)
  else           raise SimpleAuthSdk::Unsupported.new(result.body, code: result.code, headers: result.headers)
  end
end

#request_with_retry(method, uri, body = {}, extra_headers = {}) ⇒ Hash, String

Makes an HTTP request with retry logic.

Parameters:

  • method (Symbol)

    the HTTP method to use (e.g., :get, :post).

  • uri (String)

    the URI to make the request to.

  • body (Hash) (defaults to: {})

    the request body.

  • extra_headers (Hash) (defaults to: {})

    additional headers for the request.

Returns:

  • (Hash, String)

    the parsed response body, or raw response if JSON parsing fails.



124
125
126
127
128
# File 'lib/simple_auth_sdk/mixins/httpproxy.rb', line 124

def request_with_retry(method, uri, body = {}, extra_headers = {})
  Retryable.retryable(retry_options) do
    request(method, uri, body, extra_headers)
  end
end

#retry_optionsHash

Returns the options for retrying requests, including the number of retries and the sleep timer.

Returns:

  • (Hash)

    a hash containing retry options.



60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
# File 'lib/simple_auth_sdk/mixins/httpproxy.rb', line 60

def retry_options
  sleep_timer = lambda do |attempt|
    wait = BASE_DELAY * (2**attempt - 1) # Exponential delay with each subsequent request attempt.
    wait += rand(wait + 1..wait + MAX_REQUEST_RETRY_JITTER) # Add jitter to the delay window.
    wait = [MAX_REQUEST_RETRY_DELAY, wait].min # Cap delay at MAX_REQUEST_RETRY_DELAY.
    wait = [MIN_REQUEST_RETRY_DELAY, wait].max # Ensure delay is no less than MIN_REQUEST_RETRY_DELAY.
    wait / 1000.to_f.round(2) # Convert ms to seconds
  end

  tries = 1 + [Integer(retry_count || DEFAULT_RETRIES), MAX_ALLOWED_RETRIES].min # Cap retries at MAX_ALLOWED_RETRIES

  {
    tries: tries,
    sleep: sleep_timer,
    on: SimpleAuthSdk::RateLimitEncountered
  }
end

#safe_parse_json(body) ⇒ Hash, String

Safely parses the JSON response body.

Parameters:

  • body (String)

    the response body to parse.

Returns:

  • (Hash, String)

    the parsed JSON or raw body if parsing fails.



111
112
113
114
115
# File 'lib/simple_auth_sdk/mixins/httpproxy.rb', line 111

def safe_parse_json(body)
  JSON.parse(body.to_s)
rescue JSON::ParserError
  body
end

#url(path) ⇒ String

Constructs the full URL by appending the path to the base URI.

Parameters:

  • path (String)

    the path to append to the base URI.

Returns:

  • (String)

    the full URL.



91
92
93
# File 'lib/simple_auth_sdk/mixins/httpproxy.rb', line 91

def url(path)
  "#{base_uri}#{path}"
end