BigW Consortium Gitlab

git_http_client_controller.rb 4.34 KB
Newer Older
Jacob Vosmaer committed
1 2 3 4 5 6
# This file should be identical in GitLab Community Edition and Enterprise Edition

class Projects::GitHttpClientController < Projects::ApplicationController
  include ActionController::HttpAuthentication::Basic
  include KerberosSpnegoHelper

7 8 9 10 11
  attr_reader :authentication_result

  delegate :actor, :authentication_abilities, to: :authentication_result, allow_nil: true

  alias_method :user, :actor
Jacob Vosmaer committed
12 13 14 15 16 17 18 19 20

  # Git clients will not know what authenticity token to send along
  skip_before_action :verify_authenticity_token
  skip_before_action :repository
  before_action :authenticate_user
  before_action :ensure_project_found!

  private

21 22 23 24 25 26 27 28
  def download_request?
    raise NotImplementedError
  end

  def upload_request?
    raise NotImplementedError
  end

Jacob Vosmaer committed
29
  def authenticate_user
30 31
    @authentication_result = Gitlab::Auth::Result.new

Jacob Vosmaer committed
32 33
    if allow_basic_auth? && basic_auth_provided?
      login, password = user_name_and_password(request)
34

35
      if handle_basic_authentication(login, password)
Jacob Vosmaer committed
36 37 38
        return # Allow access
      end
    elsif allow_kerberos_spnego_auth? && spnego_provided?
39
      kerberos_user = find_kerberos_user
40

41
      if kerberos_user
42
        @authentication_result = Gitlab::Auth::Result.new(
43
          kerberos_user, nil, :kerberos, Gitlab::Auth.full_authentication_abilities)
Jacob Vosmaer committed
44 45 46 47

        send_final_spnego_response
        return # Allow access
      end
48 49 50 51
    elsif project && download_request? && Guest.can?(:download_code, project)
      @authentication_result = Gitlab::Auth::Result.new(nil, project, :none, [:download_code])

      return # Allow access
Jacob Vosmaer committed
52 53 54 55
    end

    send_challenges
    render plain: "HTTP Basic: Access denied\n", status: 401
56
  rescue Gitlab::Auth::MissingPersonalTokenError
57
    render_missing_personal_token
Jacob Vosmaer committed
58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78
  end

  def basic_auth_provided?
    has_basic_credentials?(request)
  end

  def send_challenges
    challenges = []
    challenges << 'Basic realm="GitLab"' if allow_basic_auth?
    challenges << spnego_challenge if allow_kerberos_spnego_auth?
    headers['Www-Authenticate'] = challenges.join("\n") if challenges.any?
  end

  def ensure_project_found!
    render_not_found if project.blank?
  end

  def project
    return @project if defined?(@project)

    project_id, _ = project_id_with_suffix
Douwe Maan committed
79 80 81 82 83 84
    @project =
      if project_id.blank?
        nil
      else
        Project.find_by_full_path("#{params[:namespace_id]}/#{project_id}")
      end
Jacob Vosmaer committed
85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104
  end

  # This method returns two values so that we can parse
  # params[:project_id] (untrusted input!) in exactly one place.
  def project_id_with_suffix
    id = params[:project_id] || ''

    %w[.wiki.git .git].each do |suffix|
      if id.end_with?(suffix)
        # Be careful to only remove the suffix from the end of 'id'.
        # Accidentally removing it from the middle is how security
        # vulnerabilities happen!
        return [id.slice(0, id.length - suffix.length), suffix]
      end
    end

    # Something is wrong with params[:project_id]; do not pass it on.
    [nil, nil]
  end

105
  def render_missing_personal_token
106 107
    render plain: "HTTP Basic: Access denied\n" \
                  "You have 2FA enabled, please use a personal access token for Git over HTTP.\n" \
108 109
                  "You can generate one at #{profile_personal_access_tokens_url}",
           status: 401
110 111
  end

Jacob Vosmaer committed
112
  def repository
113 114 115 116 117 118
    wiki? ? project.wiki.repository : project.repository
  end

  def wiki?
    return @wiki if defined?(@wiki)

Jacob Vosmaer committed
119
    _, suffix = project_id_with_suffix
120
    @wiki = suffix == '.wiki.git'
Jacob Vosmaer committed
121 122 123 124 125 126
  end

  def render_not_found
    render plain: 'Not Found', status: :not_found
  end

127 128 129
  def handle_basic_authentication(login, password)
    @authentication_result = Gitlab::Auth.find_for_git_client(
      login, password, project: project, ip: request.ip)
130

131
    return false unless @authentication_result.success?
132

133 134
    if download_request?
      authentication_has_download_access?
135
    else
136
      authentication_has_upload_access?
137
    end
138 139
  end

140
  def ci?
141
    authentication_result.ci?(project)
142 143
  end

144 145 146 147 148 149 150
  def authentication_has_download_access?
    has_authentication_ability?(:download_code) || has_authentication_ability?(:build_download_code)
  end

  def authentication_has_upload_access?
    has_authentication_ability?(:push_code)
  end
151

152 153
  def has_authentication_ability?(capability)
    (authentication_abilities || []).include?(capability)
Jacob Vosmaer committed
154
  end
155

156 157
  def authentication_project
    authentication_result.project
158
  end
Jacob Vosmaer committed
159
end