Files
gitlab-foss/qa/qa/support/api.rb
2025-07-07 21:10:07 +00:00

209 lines
6.3 KiB
Ruby

# frozen_string_literal: true
module QA
module Support
module API
extend self
HTTP_STATUS_OK = 200
HTTP_STATUS_CREATED = 201
HTTP_STATUS_NO_CONTENT = 204
HTTP_STATUS_ACCEPTED = 202
HTTP_STATUS_PERMANENT_REDIRECT = 308
HTTP_STATUS_BAD_REQUEST = 400
HTTP_STATUS_UNAUTHORIZED = 401
HTTP_STATUS_NOT_FOUND = 404
HTTP_STATUS_TOO_MANY_REQUESTS = 429
HTTP_STATUS_SERVER_ERROR = 500
HTTP_STATUS_SERVICE_UNAVAILABLE = 503
def post(url, payload, args = {})
request_args = {
method: :post,
url: url,
payload: payload,
verify_ssl: false
}.merge(with_canary(args))
with_retry_on_too_many_requests do
RestClient::Request.execute(request_args)
rescue StandardError => e
return_response_or_raise(e)
end
end
def get(url, args = {})
request_args = {
method: :get,
url: url,
verify_ssl: false
}.merge(with_canary(args))
with_retry_on_too_many_requests do
RestClient::Request.execute(request_args)
rescue StandardError => e
return_response_or_raise(e)
end
end
def patch(url, payload = nil, args = {})
request_args = {
method: :patch,
url: url,
payload: payload,
verify_ssl: false
}.merge(with_canary(args))
with_retry_on_too_many_requests do
RestClient::Request.execute(request_args)
rescue StandardError => e
return_response_or_raise(e)
end
end
def put(url, payload = nil, args = {})
request_args = {
method: :put,
url: url,
payload: payload,
verify_ssl: false
}.merge(with_canary(args))
with_retry_on_too_many_requests do
RestClient::Request.execute(request_args)
rescue StandardError => e
return_response_or_raise(e)
end
end
def delete(url, args = {})
request_args = {
method: :delete,
url: url,
verify_ssl: false
}.merge(with_canary(args))
with_retry_on_too_many_requests do
RestClient::Request.execute(request_args)
rescue StandardError => e
return_response_or_raise(e)
end
end
def head(url, args = {})
request_args = {
method: :head,
url: url,
verify_ssl: false
}.merge(with_canary(args))
with_retry_on_too_many_requests do
RestClient::Request.execute(request_args)
rescue StandardError => e
return_response_or_raise(e)
end
end
def masked_url(url)
url.sub(/private_token=[^&]*/, "private_token=[****]")
end
# Returns the response body with secrets masked.
#
# @param [String] response the response body as the string value of a JSON hash
# @param [Array<Symbol>] mask_by_key the keys of the JSON parsed response body whose values will be masked
# @return [Hash] the response body with the specified secrets values replaced with `****`
def masked_parsed_response(response, mask_by_key:)
Helpers::Masker.mask(parse_body(response), by_key: Array(mask_by_key))
end
# Merges the gitlab_canary cookie into existing cookies for mixed environment testing.
#
# @param [Hash] args the existing args passed to method
# @return [Hash] args or args with merged canary cookie if it exists
def with_canary(args)
canary_cookie = QA::Runtime::Env.canary_cookie
return args if canary_cookie.empty?
args.deep_merge(cookies: QA::Runtime::Env.canary_cookie)
end
def with_retry_on_too_many_requests
response = nil
Support::Retrier.retry_until(log: false, message: "Retrying upon receiving 429 HTTP status") do
response = yield
if response.code == HTTP_STATUS_TOO_MANY_REQUESTS
response_wait = response.headers[:retry_after].to_i
wait_seconds = response_wait <= 0 ? 60 : response_wait
QA::Runtime::Logger.warn("Received 429 - Too many requests. Waiting for #{wait_seconds} seconds.")
sleep wait_seconds
end
response.code != HTTP_STATUS_TOO_MANY_REQUESTS
end
response
end
def parse_body(response)
JSON.parse(response.body, symbolize_names: true)
end
def return_response_or_raise(error)
raise error, masked_url(error.to_s) unless error.respond_to?(:response) && error.response
error.response
end
def auto_paginated_response(url, attempts: 0)
pages = []
with_paginated_response_body(url, attempts: attempts) { |response| pages << response }
pages.flatten
end
def with_paginated_response_body(url, attempts: 0)
not_ok_error = ->(resp) do
raise "Failed to GET #{masked_url(url)} - (#{resp.code}): `#{resp}`."
end
loop do
response = if attempts > 0
Retrier.retry_on_exception(max_attempts: attempts, log: false, sleep_interval: 1) do
get(url).tap { |resp| not_ok_error.call(resp) if resp.code != HTTP_STATUS_OK }
end
else
get(url).tap { |resp| not_ok_error.call(resp) if resp.code != HTTP_STATUS_OK }
end
page, pages, next_page = response.headers.values_at(:x_page, :x_total_pages, :x_next_page)
api_endpoint = url.match(%r{v4/(\S+)\?})[1]
QA::Runtime::Logger.debug("Fetching page (#{page}/#{pages}) for '#{api_endpoint}' ...") unless pages.to_i <= 1
yield parse_body(response)
break if next_page.blank?
url = url.match?(/&page=\d+/) ? url.gsub(/&page=\d+/, "&page=#{next_page}") : "#{url}&page=#{next_page}"
end
end
def success?(response_code)
[HTTP_STATUS_NO_CONTENT, HTTP_STATUS_ACCEPTED, HTTP_STATUS_OK, HTTP_STATUS_CREATED].include?(response_code)
end
def fatal_response?(response_code)
[HTTP_STATUS_UNAUTHORIZED,
HTTP_STATUS_SERVER_ERROR,
HTTP_STATUS_BAD_REQUEST,
HTTP_STATUS_SERVICE_UNAVAILABLE].include?(response_code)
end
end
end
end