Commit 71e99cfa authored by Serena Fang's avatar Serena Fang Committed by Luke Duncalfe
parent 17c4304e
---
title: Allow SCIM deprovisioning for minimal access users
merge_request: 57178
author:
type: fixed
...@@ -69,7 +69,7 @@ module API ...@@ -69,7 +69,7 @@ module API
parsed_hash = parser.update_params parsed_hash = parser.update_params
if parser.deprovision_user? if parser.deprovision_user?
deprovision(identity) patch_deprovision(identity)
elsif reprovisionable?(identity) && parser.reprovision_user? elsif reprovisionable?(identity) && parser.reprovision_user?
reprovision(identity) reprovision(identity)
elsif parsed_hash[:extern_uid] elsif parsed_hash[:extern_uid]
...@@ -103,14 +103,28 @@ module API ...@@ -103,14 +103,28 @@ module API
group.scim_identities.with_extern_uid(extern_uid).first group.scim_identities.with_extern_uid(extern_uid).first
end end
def deprovision(identity) # delete_deprovision handles the response and returns either no_content! or a detailed error message.
::EE::Gitlab::Scim::DeprovisionService.new(identity).execute def delete_deprovision(identity)
service = ::EE::Gitlab::Scim::DeprovisionService.new(identity).execute
true if service.success?
rescue => e no_content!
logger.error(identity: identity, error: e.class.name, message: e.message, source: "#{__FILE__}:#{__LINE__}") else
logger.error(identity: identity, error: service.class.name, message: service.message, source: "#{__FILE__}:#{__LINE__}")
scim_error!(message: service.message)
end
end
false # The method that calls patch_deprovision, update_scim_user, expects a truthy/falsey value, and then continues to handle the request.
def patch_deprovision(identity)
service = ::EE::Gitlab::Scim::DeprovisionService.new(identity).execute
if service.success?
true
else
logger.error(identity: identity, error: service.class.name, message: service.message, source: "#{__FILE__}:#{__LINE__}")
false
end
end end
def reprovision(identity) def reprovision(identity)
...@@ -212,11 +226,7 @@ module API ...@@ -212,11 +226,7 @@ module API
scim_not_found!(message: "Resource #{params[:id]} not found") unless identity scim_not_found!(message: "Resource #{params[:id]} not found") unless identity
destroyed = deprovision(identity) delete_deprovision(identity)
scim_not_found!(message: "Resource #{params[:id]} not found") unless destroyed
no_content!
end end
end end
end end
......
...@@ -13,23 +13,29 @@ module EE ...@@ -13,23 +13,29 @@ module EE
end end
def execute def execute
return error(_("Could not remove %{user} from %{group}. User is not a group member.") % { user: user.name, group: group.name }) unless group_membership
return error(_("Could not remove %{user} from %{group}. Cannot remove last group owner.") % { user: user.name, group: group.name }) if group.last_owner?(user)
ScimIdentity.transaction do ScimIdentity.transaction do
identity.update!(active: false) identity.update!(active: false)
remove_group_access remove_group_access
end end
ServiceResponse.success(message: _("User %{user} was removed from %{group}.") % { user: user.name, group: group.name })
end end
private private
def remove_group_access def remove_group_access
return unless group_membership
return if group.last_owner?(user)
::Members::DestroyService.new(user).execute(group_membership) ::Members::DestroyService.new(user).execute(group_membership)
end end
def group_membership def group_membership
@group_membership ||= group.group_member(user) @group_membership ||= group.all_group_members.with_user(user).first
end
def error(message)
ServiceResponse.error(message: message)
end end
end end
end end
......
...@@ -4,36 +4,89 @@ require 'spec_helper' ...@@ -4,36 +4,89 @@ require 'spec_helper'
RSpec.describe ::EE::Gitlab::Scim::DeprovisionService do RSpec.describe ::EE::Gitlab::Scim::DeprovisionService do
describe '#execute' do describe '#execute' do
let_it_be(:identity) { create(:scim_identity, active: true) } let(:identity) { create(:scim_identity, active: true) }
let_it_be(:group) { identity.group } let(:group) { identity.group }
let_it_be(:user) { identity.user } let(:user) { identity.user }
let(:service) { described_class.new(identity) } let(:service) { described_class.new(identity) }
it 'deactivates scim identity' do context 'when user is successfully removed' do
service.execute before do
create(:group_member, group: group, user: user, access_level: GroupMember::REPORTER)
end
expect(identity.active).to be false it 'deactivates scim identity' do
end expect { service.execute }.to change { identity.active }.from(true).to(false)
end
it 'removes group access' do
service.execute
it 'removes group access' do expect(group.all_group_members.pluck(:user_id)).not_to include(user.id)
create(:group_member, group: group, user: user, access_level: GroupMember::REPORTER) end
service.execute it 'returns the successful deprovision message' do
response = service.execute
expect(group.members.pluck(:user_id)).not_to include(user.id) expect(response.message).to include("User #{user.name} was removed from #{group.name}.")
end
end end
it 'does not remove the last owner' do context 'with minimal access role' do
create(:group_member, group: group, user: user, access_level: GroupMember::OWNER) before do
stub_licensed_features(minimal_access_role: true)
create(:group_member, group: group, user: user, access_level: ::Gitlab::Access::MINIMAL_ACCESS)
end
it 'deactivates scim identity' do
expect { service.execute }.to change { identity.active }.from(true).to(false)
end
service.execute it 'removes group access' do
service.execute
expect(identity.group.members.pluck(:user_id)).to include(user.id) expect(group.all_group_members.pluck(:user_id)).not_to include(user.id)
end
it 'returns the successful deprovision message' do
response = service.execute
expect(response.message).to include("User #{user.name} was removed from #{group.name}.")
end
end end
it 'does not change group membership when the user is not a member' do context 'when user is not successfully removed' do
expect { service.execute }.not_to change { group.members.count } context 'when user is the last owner' do
before do
create(:group_member, group: group, user: user, access_level: GroupMember::OWNER)
end
it 'does not remove the last owner' do
service.execute
expect(identity.group.members.pluck(:user_id)).to include(user.id)
end
it 'returns the last group owner error' do
response = service.execute
expect(response.error?).to be true
expect(response.errors).to include("Could not remove #{user.name} from #{group.name}. Cannot remove last group owner.")
end
end
context 'when user is not a group member' do
it 'does not change group membership when the user is not a member' do
expect { service.execute }.not_to change { group.members.count }
end
it 'returns the group membership error' do
response = service.execute
expect(response.error?).to be true
expect(response.errors).to include("Could not remove #{user.name} from #{group.name}. User is not a group member.")
end
end
end end
end end
end end
...@@ -38,5 +38,16 @@ RSpec.describe ::EE::Gitlab::Scim::ReprovisionService do ...@@ -38,5 +38,16 @@ RSpec.describe ::EE::Gitlab::Scim::ReprovisionService do
expect { service.execute }.not_to change { group.members.count } expect { service.execute }.not_to change { group.members.count }
end end
context 'with minimal access user' do
before do
stub_licensed_features(minimal_access_role: true)
create(:group_member, group: group, user: user, access_level: ::Gitlab::Access::MINIMAL_ACCESS)
end
it 'does not change group membership when the user is already a member' do
expect { service.execute }.not_to change { group.all_group_members.count }
end
end
end end
end end
...@@ -12,7 +12,7 @@ RSpec.describe API::Scim do ...@@ -12,7 +12,7 @@ RSpec.describe API::Scim do
before do before do
stub_licensed_features(group_allowed_email_domains: true, group_saml: true) stub_licensed_features(group_allowed_email_domains: true, group_saml: true)
group.add_owner(user) group.add_developer(user)
create(:saml_provider, group: group, default_membership_role: Gitlab::Access::DEVELOPER) create(:saml_provider, group: group, default_membership_role: Gitlab::Access::DEVELOPER)
end end
...@@ -337,6 +337,27 @@ RSpec.describe API::Scim do ...@@ -337,6 +337,27 @@ RSpec.describe API::Scim do
expect(identity.reload.active).to be false expect(identity.reload.active).to be false
end end
context 'with owner' do
params = { Operations: [{ 'op': 'Replace', 'path': 'active', 'value': 'False' }] }.to_query
before do
group.add_owner(user)
call_patch_api(params)
end
it 'responds with 412' do
expect(response).to have_gitlab_http_status(:precondition_failed)
end
it 'returns the last group owner error' do
expect(response.body).to include("Error updating #{user.name}")
end
it 'does not deactivate the identity' do
expect(identity.reload.active).to be true
end
end
context 'Reprovision user' do context 'Reprovision user' do
let_it_be(:params) { { Operations: [{ 'op': 'Replace', 'path': 'active', 'value': 'true' }] }.to_query } let_it_be(:params) { { Operations: [{ 'op': 'Replace', 'path': 'active', 'value': 'true' }] }.to_query }
...@@ -450,6 +471,25 @@ RSpec.describe API::Scim do ...@@ -450,6 +471,25 @@ RSpec.describe API::Scim do
end end
end end
context 'with owner' do
before do
group.add_owner(user)
delete scim_api("scim/v2/groups/#{group.full_path}/Users/#{identity.extern_uid}")
end
it 'responds with 412' do
expect(response).to have_gitlab_http_status(:precondition_failed)
end
it 'returns the last group owner error' do
expect(response.body).to include("Could not remove #{user.name} from #{group.name}. Cannot remove last group owner.")
end
it 'does not deactivate the identity' do
expect(identity.reload.active).to be true
end
end
it 'responds with 404 if there is no user' do it 'responds with 404 if there is no user' do
delete scim_api("scim/v2/groups/#{group.full_path}/Users/123") delete scim_api("scim/v2/groups/#{group.full_path}/Users/123")
......
...@@ -8726,6 +8726,12 @@ msgstr "" ...@@ -8726,6 +8726,12 @@ msgstr ""
msgid "Could not load usage counts. Please refresh the page to try again." msgid "Could not load usage counts. Please refresh the page to try again."
msgstr "" msgstr ""
msgid "Could not remove %{user} from %{group}. Cannot remove last group owner."
msgstr ""
msgid "Could not remove %{user} from %{group}. User is not a group member."
msgstr ""
msgid "Could not remove the trigger." msgid "Could not remove the trigger."
msgstr "" msgstr ""
...@@ -33165,6 +33171,9 @@ msgstr "" ...@@ -33165,6 +33171,9 @@ msgstr ""
msgid "User %{username} was successfully removed." msgid "User %{username} was successfully removed."
msgstr "" msgstr ""
msgid "User %{user} was removed from %{group}."
msgstr ""
msgid "User ID" msgid "User ID"
msgstr "" msgstr ""
......
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