Commit 0e7fe044 authored by Sean McGivern's avatar Sean McGivern

Allow read_api scope in rate limiter

Previously, requests using a token that only had read_api access would
be treated as unauthenticated by the rate limiter. This ensures that
they are (correctly) treated as authenticated.

This does point to a general problem with the rate limiting code:
because it happens in a middleware before any application code, it's
hard to keep in sync with the different authentication methods the
application may allow. This applies to things like:

1. How the credentials are sent (Private-Token header, Authorization
   header, basic auth, etc.).
2. What credentials we accept (a personal access token, a personal
   access token with read_api scope, a job token, a deploy token, etc.).
3. Which endpoints those apply to.

The last one is mostly a concern for performance: we might pre-emptively
try authenticating using a method that isn't valid for a particular
endpoint. From a rate limiting perspective it doesn't matter so much, as
we're treating a request with valid credentials for a different endpoint
as authenticated for the current endpoint, which is wrong but not a
significant problem.
parent ec056fa7
...@@ -92,10 +92,10 @@ module Gitlab ...@@ -92,10 +92,10 @@ module Gitlab
# We only allow Private Access Tokens with `api` scope to be used by web # We only allow Private Access Tokens with `api` scope to be used by web
# requests on RSS feeds or ICS files for backwards compatibility. # requests on RSS feeds or ICS files for backwards compatibility.
# It is also used by GraphQL/API requests. # It is also used by GraphQL/API requests.
def find_user_from_web_access_token(request_format) def find_user_from_web_access_token(request_format, scopes: [:api])
return unless access_token && valid_web_access_format?(request_format) return unless access_token && valid_web_access_format?(request_format)
validate_access_token!(scopes: [:api]) validate_access_token!(scopes: scopes)
::PersonalAccessTokens::LastUsedService.new(access_token).execute ::PersonalAccessTokens::LastUsedService.new(access_token).execute
......
...@@ -30,7 +30,7 @@ module Gitlab ...@@ -30,7 +30,7 @@ module Gitlab
end end
def find_sessionless_user(request_format) def find_sessionless_user(request_format)
find_user_from_web_access_token(request_format) || find_user_from_web_access_token(request_format, scopes: [:api, :read_api]) ||
find_user_from_feed_token(request_format) || find_user_from_feed_token(request_format) ||
find_user_from_static_object_token(request_format) || find_user_from_static_object_token(request_format) ||
find_user_from_basic_auth_job || find_user_from_basic_auth_job ||
......
...@@ -6,7 +6,8 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -6,7 +6,8 @@ RSpec.describe Gitlab::Auth::AuthFinders do
include described_class include described_class
include HttpBasicAuthHelpers include HttpBasicAuthHelpers
let(:user) { create(:user) } # Create the feed_token and static_object_token for the user
let_it_be(:user) { create(:user).tap(&:feed_token).tap(&:static_object_token) }
let(:env) do let(:env) do
{ {
'rack.input' => '' 'rack.input' => ''
...@@ -65,7 +66,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -65,7 +66,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do
end end
describe '#find_user_from_bearer_token' do describe '#find_user_from_bearer_token' do
let(:job) { create(:ci_build, user: user) } let_it_be_with_reload(:job) { create(:ci_build, user: user) }
subject { find_user_from_bearer_token } subject { find_user_from_bearer_token }
...@@ -91,7 +92,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -91,7 +92,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do
end end
context 'with a personal access token' do context 'with a personal access token' do
let(:pat) { create(:personal_access_token, user: user) } let_it_be(:pat) { create(:personal_access_token, user: user) }
let(:token) { pat.token } let(:token) { pat.token }
before do before do
...@@ -148,7 +149,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -148,7 +149,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do
end end
it 'returns nil if valid feed_token and disabled' do it 'returns nil if valid feed_token and disabled' do
allow(Gitlab::CurrentSettings).to receive(:disable_feed_token).and_return(true) stub_application_setting(disable_feed_token: true)
set_param(:feed_token, user.feed_token) set_param(:feed_token, user.feed_token)
expect(find_user_from_feed_token(:rss)).to be_nil expect(find_user_from_feed_token(:rss)).to be_nil
...@@ -166,7 +167,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -166,7 +167,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do
end end
context 'when rss_token param is provided' do context 'when rss_token param is provided' do
it 'returns user if valid rssd_token' do it 'returns user if valid rss_token' do
set_param(:rss_token, user.feed_token) set_param(:rss_token, user.feed_token)
expect(find_user_from_feed_token(:rss)).to eq user expect(find_user_from_feed_token(:rss)).to eq user
...@@ -347,7 +348,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -347,7 +348,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do
end end
describe '#find_user_from_access_token' do describe '#find_user_from_access_token' do
let(:personal_access_token) { create(:personal_access_token, user: user) } let_it_be(:personal_access_token) { create(:personal_access_token, user: user) }
before do before do
set_header('SCRIPT_NAME', 'url.atom') set_header('SCRIPT_NAME', 'url.atom')
...@@ -386,7 +387,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -386,7 +387,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do
end end
context 'when using a non-prefixed access token' do context 'when using a non-prefixed access token' do
let(:personal_access_token) { create(:personal_access_token, :no_prefix, user: user) } let_it_be(:personal_access_token) { create(:personal_access_token, :no_prefix, user: user) }
it 'returns user' do it 'returns user' do
set_header('HTTP_AUTHORIZATION', "Bearer #{personal_access_token.token}") set_header('HTTP_AUTHORIZATION', "Bearer #{personal_access_token.token}")
...@@ -398,7 +399,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -398,7 +399,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do
end end
describe '#find_user_from_web_access_token' do describe '#find_user_from_web_access_token' do
let(:personal_access_token) { create(:personal_access_token, user: user) } let_it_be_with_reload(:personal_access_token) { create(:personal_access_token, user: user) }
before do before do
set_header(described_class::PRIVATE_TOKEN_HEADER, personal_access_token.token) set_header(described_class::PRIVATE_TOKEN_HEADER, personal_access_token.token)
...@@ -449,6 +450,22 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -449,6 +450,22 @@ RSpec.describe Gitlab::Auth::AuthFinders do
expect(find_user_from_web_access_token(:api)).to be_nil expect(find_user_from_web_access_token(:api)).to be_nil
end end
context 'when the token has read_api scope' do
before do
personal_access_token.update!(scopes: ['read_api'])
set_header('SCRIPT_NAME', '/api/endpoint')
end
it 'raises InsufficientScopeError by default' do
expect { find_user_from_web_access_token(:api) }.to raise_error(Gitlab::Auth::InsufficientScopeError)
end
it 'finds the user when the read_api scope is passed' do
expect(find_user_from_web_access_token(:api, scopes: [:api, :read_api])).to eq(user)
end
end
context 'when relative_url_root is set' do context 'when relative_url_root is set' do
before do before do
stub_config_setting(relative_url_root: '/relative_root') stub_config_setting(relative_url_root: '/relative_root')
...@@ -464,7 +481,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -464,7 +481,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do
end end
describe '#find_personal_access_token' do describe '#find_personal_access_token' do
let(:personal_access_token) { create(:personal_access_token, user: user) } let_it_be(:personal_access_token) { create(:personal_access_token, user: user) }
before do before do
set_header('SCRIPT_NAME', 'url.atom') set_header('SCRIPT_NAME', 'url.atom')
...@@ -534,7 +551,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -534,7 +551,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do
end end
context 'access token is valid' do context 'access token is valid' do
let(:personal_access_token) { create(:personal_access_token, user: user) } let_it_be(:personal_access_token) { create(:personal_access_token, user: user) }
let(:route_authentication_setting) { { basic_auth_personal_access_token: true } } let(:route_authentication_setting) { { basic_auth_personal_access_token: true } }
it 'finds the token from basic auth' do it 'finds the token from basic auth' do
...@@ -555,7 +572,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -555,7 +572,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do
end end
context 'route_setting is not set' do context 'route_setting is not set' do
let(:personal_access_token) { create(:personal_access_token, user: user) } let_it_be(:personal_access_token) { create(:personal_access_token, user: user) }
it 'returns nil' do it 'returns nil' do
auth_header_with(personal_access_token.token) auth_header_with(personal_access_token.token)
...@@ -565,7 +582,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -565,7 +582,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do
end end
context 'route_setting is not correct' do context 'route_setting is not correct' do
let(:personal_access_token) { create(:personal_access_token, user: user) } let_it_be(:personal_access_token) { create(:personal_access_token, user: user) }
let(:route_authentication_setting) { { basic_auth_personal_access_token: false } } let(:route_authentication_setting) { { basic_auth_personal_access_token: false } }
it 'returns nil' do it 'returns nil' do
...@@ -611,8 +628,9 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -611,8 +628,9 @@ RSpec.describe Gitlab::Auth::AuthFinders do
context 'with CI username' do context 'with CI username' do
let(:username) { ::Gitlab::Auth::CI_JOB_USER } let(:username) { ::Gitlab::Auth::CI_JOB_USER }
let(:user) { create(:user) }
let(:build) { create(:ci_build, user: user, status: :running) } let_it_be(:user) { create(:user) }
let_it_be(:build) { create(:ci_build, user: user, status: :running) }
it 'returns nil without password' do it 'returns nil without password' do
set_basic_auth_header(username, nil) set_basic_auth_header(username, nil)
...@@ -645,11 +663,11 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -645,11 +663,11 @@ RSpec.describe Gitlab::Auth::AuthFinders do
describe '#validate_access_token!' do describe '#validate_access_token!' do
subject { validate_access_token! } subject { validate_access_token! }
let(:personal_access_token) { create(:personal_access_token, user: user) } let_it_be_with_reload(:personal_access_token) { create(:personal_access_token, user: user) }
context 'with a job token' do context 'with a job token' do
let_it_be(:job) { create(:ci_build, user: user, status: :running) }
let(:route_authentication_setting) { { job_token_allowed: true } } let(:route_authentication_setting) { { job_token_allowed: true } }
let(:job) { create(:ci_build, user: user, status: :running) }
before do before do
env['HTTP_AUTHORIZATION'] = "Bearer #{job.token}" env['HTTP_AUTHORIZATION'] = "Bearer #{job.token}"
...@@ -671,7 +689,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -671,7 +689,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do
end end
it 'returns Gitlab::Auth::ExpiredError if token expired' do it 'returns Gitlab::Auth::ExpiredError if token expired' do
personal_access_token.expires_at = 1.day.ago personal_access_token.update!(expires_at: 1.day.ago)
expect { validate_access_token! }.to raise_error(Gitlab::Auth::ExpiredError) expect { validate_access_token! }.to raise_error(Gitlab::Auth::ExpiredError)
end end
...@@ -688,7 +706,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -688,7 +706,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do
end end
context 'with impersonation token' do context 'with impersonation token' do
let(:personal_access_token) { create(:personal_access_token, :impersonation, user: user) } let_it_be(:personal_access_token) { create(:personal_access_token, :impersonation, user: user) }
context 'when impersonation is disabled' do context 'when impersonation is disabled' do
before do before do
...@@ -704,7 +722,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -704,7 +722,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do
end end
describe '#find_user_from_job_token' do describe '#find_user_from_job_token' do
let(:job) { create(:ci_build, user: user, status: :running) } let_it_be(:job) { create(:ci_build, user: user, status: :running) }
let(:route_authentication_setting) { { job_token_allowed: true } } let(:route_authentication_setting) { { job_token_allowed: true } }
subject { find_user_from_job_token } subject { find_user_from_job_token }
...@@ -866,7 +884,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do ...@@ -866,7 +884,7 @@ RSpec.describe Gitlab::Auth::AuthFinders do
end end
describe '#find_runner_from_token' do describe '#find_runner_from_token' do
let(:runner) { create(:ci_runner) } let_it_be(:runner) { create(:ci_runner) }
context 'with API requests' do context 'with API requests' do
before do before do
......
...@@ -47,7 +47,10 @@ RSpec.describe Gitlab::Auth::RequestAuthenticator do ...@@ -47,7 +47,10 @@ RSpec.describe Gitlab::Auth::RequestAuthenticator do
let!(:job_token_user) { build(:user) } let!(:job_token_user) { build(:user) }
it 'returns access_token user first' do it 'returns access_token user first' do
allow_any_instance_of(described_class).to receive(:find_user_from_web_access_token).and_return(access_token_user) allow_any_instance_of(described_class).to receive(:find_user_from_web_access_token)
.with(anything, scopes: [:api, :read_api])
.and_return(access_token_user)
allow_any_instance_of(described_class).to receive(:find_user_from_feed_token).and_return(feed_token_user) allow_any_instance_of(described_class).to receive(:find_user_from_feed_token).and_return(feed_token_user)
expect(subject.find_sessionless_user(:api)).to eq access_token_user expect(subject.find_sessionless_user(:api)).to eq access_token_user
......
...@@ -170,10 +170,10 @@ RSpec.describe 'Rack Attack global throttles' do ...@@ -170,10 +170,10 @@ RSpec.describe 'Rack Attack global throttles' do
end end
describe 'API requests authenticated with personal access token', :api do describe 'API requests authenticated with personal access token', :api do
let(:user) { create(:user) } let_it_be(:user) { create(:user) }
let(:token) { create(:personal_access_token, user: user) } let_it_be(:token) { create(:personal_access_token, user: user) }
let(:other_user) { create(:user) } let_it_be(:other_user) { create(:user) }
let(:other_user_token) { create(:personal_access_token, user: other_user) } let_it_be(:other_user_token) { create(:personal_access_token, user: other_user) }
let(:throttle_setting_prefix) { 'throttle_authenticated_api' } let(:throttle_setting_prefix) { 'throttle_authenticated_api' }
let(:api_partial_url) { '/todos' } let(:api_partial_url) { '/todos' }
...@@ -190,6 +190,41 @@ RSpec.describe 'Rack Attack global throttles' do ...@@ -190,6 +190,41 @@ RSpec.describe 'Rack Attack global throttles' do
it_behaves_like 'rate-limited token-authenticated requests' it_behaves_like 'rate-limited token-authenticated requests'
end end
context 'with the token in the OAuth headers' do
let(:request_args) { api_get_args_with_token_headers(api_partial_url, oauth_token_headers(token)) }
let(:other_user_request_args) { api_get_args_with_token_headers(api_partial_url, oauth_token_headers(other_user_token)) }
it_behaves_like 'rate-limited token-authenticated requests'
end
context 'with the token in basic auth' do
let(:request_args) { api_get_args_with_token_headers(api_partial_url, basic_auth_headers(user, token)) }
let(:other_user_request_args) { api_get_args_with_token_headers(api_partial_url, basic_auth_headers(other_user, other_user_token)) }
it_behaves_like 'rate-limited token-authenticated requests'
end
context 'with a read_api scope' do
before do
token.update!(scopes: ['read_api'])
other_user_token.update!(scopes: ['read_api'])
end
context 'with the token in the headers' do
let(:request_args) { api_get_args_with_token_headers(api_partial_url, personal_access_token_headers(token)) }
let(:other_user_request_args) { api_get_args_with_token_headers(api_partial_url, personal_access_token_headers(other_user_token)) }
it_behaves_like 'rate-limited token-authenticated requests'
end
context 'with the token in the OAuth headers' do
let(:request_args) { api_get_args_with_token_headers(api_partial_url, oauth_token_headers(token)) }
let(:other_user_request_args) { api_get_args_with_token_headers(api_partial_url, oauth_token_headers(other_user_token)) }
it_behaves_like 'rate-limited token-authenticated requests'
end
end
end end
describe 'API requests authenticated with OAuth token', :api do describe 'API requests authenticated with OAuth token', :api do
...@@ -217,6 +252,15 @@ RSpec.describe 'Rack Attack global throttles' do ...@@ -217,6 +252,15 @@ RSpec.describe 'Rack Attack global throttles' do
it_behaves_like 'rate-limited token-authenticated requests' it_behaves_like 'rate-limited token-authenticated requests'
end end
context 'with a read_api scope' do
let(:read_token) { Doorkeeper::AccessToken.create!(application_id: application.id, resource_owner_id: user.id, scopes: "read_api") }
let(:other_user_read_token) { Doorkeeper::AccessToken.create!(application_id: other_user_application.id, resource_owner_id: other_user.id, scopes: "read_api") }
let(:request_args) { api_get_args_with_token_headers(api_partial_url, oauth_token_headers(read_token)) }
let(:other_user_request_args) { api_get_args_with_token_headers(api_partial_url, oauth_token_headers(other_user_read_token)) }
it_behaves_like 'rate-limited token-authenticated requests'
end
end end
describe '"web" (non-API) requests authenticated with RSS token' do describe '"web" (non-API) requests authenticated with RSS token' do
......
...@@ -21,6 +21,11 @@ module RackAttackSpecHelpers ...@@ -21,6 +21,11 @@ module RackAttackSpecHelpers
{ 'AUTHORIZATION' => "Bearer #{oauth_access_token.token}" } { 'AUTHORIZATION' => "Bearer #{oauth_access_token.token}" }
end end
def basic_auth_headers(user, personal_access_token)
encoded_login = ["#{user.username}:#{personal_access_token.token}"].pack('m0')
{ 'AUTHORIZATION' => "Basic #{encoded_login}" }
end
def expect_rejection(&block) def expect_rejection(&block)
yield yield
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment