Commit a687282f authored by Taylan Develioglu's avatar Taylan Develioglu

Return message when personal access token creation fails in internal API

This is a follow-up to the discussion in
https://gitlab.com/gitlab-org/gitlab-shell/-/merge_requests/397
to move expiry date validation onto the API side. See:

https://gitlab.com/gitlab-org/gitlab-shell/-/merge_requests/397#note_391081491)

Before this change the internal api would return a 500 error to the user
when creation of the personal access token fails, for example, when a
maximum token lifetime is enforced.

After this change, the internal api will return a more meaningful error
message to the user.
parent ac26955d
---
title: Return message when personal access token creation fails in internal API
merge_request: 40073
author: Taylan Develioglu
type: changed
...@@ -277,4 +277,57 @@ RSpec.describe API::Internal::Base do ...@@ -277,4 +277,57 @@ RSpec.describe API::Internal::Base do
) )
end end
end end
describe 'POST /internal/personal_access_token' do
let_it_be(:user) { create(:user) }
let_it_be(:key) { create(:key, user: user) }
let(:instance_level_max_personal_access_token_lifetime) { nil }
let(:secret_token) { Gitlab::Shell.secret_token }
before do
stub_licensed_features(personal_access_token_expiration_policy: !!instance_level_max_personal_access_token_lifetime)
stub_application_setting(max_personal_access_token_lifetime: instance_level_max_personal_access_token_lifetime)
end
context 'with a max token lifetime on the instance' do
let(:instance_level_max_personal_access_token_lifetime) { 10 }
it 'returns an error message when the expiry date exceeds the max token lifetime' do
post api('/internal/personal_access_token'),
params: {
secret_token: secret_token,
key_id: key.id,
name: 'newtoken',
scopes: %w(read_api read_repository),
expires_at: (instance_level_max_personal_access_token_lifetime + 1).days.from_now.to_date.to_s
}
aggregate_failures do
expect(json_response['success']).to eq(false)
expect(json_response['message']).to eq("Failed to create token: Expires at is invalid")
end
end
it 'returns a valid token when the expiry date does not exceed the max token lifetime' do
expires_at = instance_level_max_personal_access_token_lifetime.days.from_now.to_date.to_s
post api('/internal/personal_access_token'),
params: {
secret_token: secret_token,
key_id: key.id,
name: 'newtoken',
scopes: %w(read_api read_repository),
expires_at: expires_at
}
aggregate_failures do
expect(json_response['success']).to eq(true)
expect(json_response['token']).to match(/\A\S{20}\z/)
expect(json_response['scopes']).to match_array(%w(read_api read_repository))
expect(json_response['expires_at']).to eq(expires_at)
end
end
end
end
end end
...@@ -241,14 +241,16 @@ module API ...@@ -241,14 +241,16 @@ module API
break { success: false, message: "Invalid token expiry date: '#{params[:expires_at]}'" } break { success: false, message: "Invalid token expiry date: '#{params[:expires_at]}'" }
end end
access_token = nil result = ::PersonalAccessTokens::CreateService.new(
user, name: params[:name], scopes: params[:scopes], expires_at: expires_at
).execute
::Users::UpdateService.new(current_user, user: user).execute! do |user| unless result.status == :success
access_token = user.personal_access_tokens.create!( break { success: false, message: "Failed to create token: #{result.message}" }
name: params[:name], scopes: params[:scopes], expires_at: expires_at
)
end end
access_token = result.payload[:personal_access_token]
{ success: true, token: access_token.token, scopes: access_token.scopes, expires_at: access_token.expires_at } { success: true, token: access_token.token, scopes: access_token.scopes, expires_at: access_token.expires_at }
end end
......
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