Welcome to mirror list, hosted at ThFree Co, Russian Federation.

http_helper.rb « lib « gitlab-shell « ruby - gitlab.com/gitlab-org/gitaly.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
blob: f721e9ae6d45299a1a769fbbc036eb24aee8db59 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
require_relative 'httpunix'
require_relative 'gitlab_logger'
require_relative 'gitlab_net/errors'

module HTTPHelper
  CONTENT_TYPE_JSON = 'application/json'.freeze

  protected

  def config
    @config ||= GitlabConfig.new
  end

  def base_api_endpoint
    "#{config.gitlab_url}/api/v4"
  end

  def internal_api_endpoint
    "#{base_api_endpoint}/internal"
  end

  def http_client_for(uri, options = {})
    http = if uri.is_a?(URI::HTTPUNIX)
             Net::HTTPUNIX.new(uri.hostname)
           else
             Net::HTTP.new(uri.host, uri.port)
           end

    http.read_timeout = options[:read_timeout] || read_timeout

    if uri.is_a?(URI::HTTPS)
      http.use_ssl = true
      http.cert_store = cert_store
      http.verify_mode = OpenSSL::SSL::VERIFY_NONE if config.http_settings.self_signed_cert
    end

    http
  end

  def http_request_for(method, uri, params: {}, headers: {}, options: {})
    request_klass = method == :get ? Net::HTTP::Get : Net::HTTP::Post
    request = request_klass.new(uri.request_uri, headers)

    user = config.http_settings.user
    password = config.http_settings.password
    request.basic_auth(user, password) if user && password

    if options[:json]
      request.body = options[:json].merge(secret_token: secret_token).to_json
    else
      request.set_form_data(params.merge(secret_token: secret_token))
    end

    if uri.is_a?(URI::HTTPUNIX)
      # The HTTPUNIX HTTP client does not set a correct Host header. This can
      # lead to 400 Bad Request responses.
      request['Host'] = 'localhost'
    end

    request
  end

  def request(method, url, params: {}, headers: {}, options: {})
    $logger.debug('Performing request', method: method.to_s.upcase, url: url)

    uri = URI.parse(url)
    http = http_client_for(uri, options)
    request = http_request_for(method, uri,
                               params: params,
                               headers: headers,
                               options: options)

    begin
      start_time = Time.new
      response = http.start { http.request(request) }
    rescue => e
      $logger.warn('Failed to connect', method: method.to_s.upcase, url: url, error: e)
      raise GitlabNet::ApiUnreachableError
    ensure
      fields = { method: method.to_s.upcase, url: url, duration: Time.new - start_time, gitaly_embedded: GITALY_EMBEDDED }
      $logger.info('finished HTTP request', fields)
    end

    case response
    when Net::HTTPSuccess, Net::HTTPMultipleChoices
      $logger.debug('Received response', code: response.code, body: response.body)
    else
      $logger.error('Call failed', method: method.to_s.upcase, url: url, code: response.code, body: response.body)
    end

    response
  end

  def get(url, headers: {}, options: {})
    request(:get, url, headers: headers, options: options)
  end

  def post(url, params, headers: {}, options: {})
    request(:post, url, params: params, headers: headers, options: options)
  end

  def cert_store
    @cert_store ||= begin
      store = OpenSSL::X509::Store.new
      store.set_default_paths

      ca_file = config.http_settings.ca_file
      store.add_file(ca_file) if present?(ca_file)

      ca_path = config.http_settings.ca_path
      store.add_path(ca_path) if present?(ca_path)

      store
    end
  end

  def present?(str)
    !str.nil? && !str.empty?
  end

  def secret_token
    @secret_token ||= File.read config.secret_file
  end

  def read_timeout
    config.http_settings.read_timeout
  end
end