# frozen_string_literal: true require 'spec_helper' RSpec.describe 'Admin::Users', feature_category: :user_management do include Spec::Support::Helpers::Features::AdminUsersHelpers include Spec::Support::Helpers::ModalHelpers let_it_be(:user, reload: true) { create(:omniauth_user, provider: 'twitter', extern_uid: '123456') } let_it_be(:current_user) { create(:admin) } before do sign_in(current_user) gitlab_enable_admin_mode_sign_in(current_user) end describe 'GET /admin/users', :js do before do visit admin_users_path end it "is ok" do expect(page).to have_current_path(admin_users_path, ignore_query: true) end it "has users list" do current_user.reload expect(page).to have_content(current_user.email) expect(page).to have_content(current_user.name) expect(page).to have_content(current_user.created_at.strftime('%b %d, %Y')) expect(page).to have_content(user.email) expect(page).to have_content(user.name) expect(page).to have_content('Projects') click_user_dropdown_toggle(user.id) expect(page).to have_button('Block') expect(page).to have_button('Deactivate') expect(page).to have_button('Delete user') expect(page).to have_button('Delete user and contributions') end it 'clicking edit user takes us to edit page', :aggregate_failures do page.within("[data-testid='user-actions-#{user.id}']") do click_link 'Edit' end expect(page).to have_content('Name') expect(page).to have_content('Password') end describe 'view extra user information' do it 'shows the user popover on hover', :js, quarantine: 'https://gitlab.com/gitlab-org/gitlab/-/issues/11290' do expect(page).not_to have_selector('#__BV_popover_1__') first_user_link = page.first('.js-user-link') first_user_link.hover expect(page).to have_selector('#__BV_popover_1__') end end context 'user project count' do before do project = create(:project) project.add_maintainer(current_user) end it 'displays count of users projects' do visit admin_users_path expect(page.find("[data-testid='user-project-count-#{current_user.id}']").text).to eq("1") end end describe 'tabs' do it 'has multiple tabs to filter users' do expect(page).to have_link('Active', href: admin_users_path) expect(page).to have_link('Admins', href: admin_users_path(filter: 'admins')) expect(page).to have_link('2FA Enabled', href: admin_users_path(filter: 'two_factor_enabled')) expect(page).to have_link('2FA Disabled', href: admin_users_path(filter: 'two_factor_disabled')) expect(page).to have_link('External', href: admin_users_path(filter: 'external')) expect(page).to have_link('Blocked', href: admin_users_path(filter: 'blocked')) expect(page).to have_link('Deactivated', href: admin_users_path(filter: 'deactivated')) expect(page).to have_link('Without projects', href: admin_users_path(filter: 'wop')) end context '`Pending approval` tab' do before do visit admin_users_path end it 'shows the `Pending approval` tab' do expect(page).to have_link('Pending approval', href: admin_users_path(filter: 'blocked_pending_approval')) end end end describe 'search and sort' do before_all do create(:user, name: 'Foo Bar', last_activity_on: 3.days.ago) create(:user, name: 'Foo Baz', last_activity_on: 2.days.ago) create(:user, name: 'Dmitriy') end it 'searches users by name' do visit admin_users_path(search_query: 'Foo') expect(page).to have_content('Foo Bar') expect(page).to have_content('Foo Baz') expect(page).not_to have_content('Dmitriy') end it 'sorts users by name' do visit admin_users_path sort_by('Name') expect(first_row.text).to include('Dmitriy') expect(second_row.text).to include('Foo Bar') end it 'sorts search results only' do visit admin_users_path(search_query: 'Foo') sort_by('Name') expect(page).not_to have_content('Dmitriy') expect(first_row.text).to include('Foo Bar') expect(second_row.text).to include('Foo Baz') end it 'searches with respect of sorting' do visit admin_users_path(sort: 'name_asc') fill_in :search_query, with: 'Foo' click_button('Search users') expect(first_row.text).to include('Foo Bar') expect(second_row.text).to include('Foo Baz') end it 'sorts users by recent last activity' do visit admin_users_path(search_query: 'Foo') sort_by('Recent last activity') expect(first_row.text).to include('Foo Baz') expect(second_row.text).to include('Foo Bar') end it 'sorts users by oldest last activity' do visit admin_users_path(search_query: 'Foo') sort_by('Oldest last activity') expect(first_row.text).to include('Foo Bar') expect(second_row.text).to include('Foo Baz') end end describe 'Two-factor Authentication filters' do it 'counts users who have enabled 2FA' do create(:user, :two_factor) visit admin_users_path page.within('.filter-two-factor-enabled .gl-tab-counter-badge') do expect(page).to have_content('1') end end it 'filters by users who have enabled 2FA' do user = create(:user, :two_factor) visit admin_users_path click_link '2FA Enabled' expect(page).to have_content(user.email) end it 'counts users who have not enabled 2FA' do visit admin_users_path page.within('.filter-two-factor-disabled .gl-tab-counter-badge') do expect(page).to have_content('2') # Including admin end end it 'filters by users who have not enabled 2FA' do visit admin_users_path click_link '2FA Disabled' expect(page).to have_content(user.email) end end describe 'Pending approval filter' do it 'counts users who are pending approval' do create_list(:user, 2, :blocked_pending_approval) visit admin_users_path page.within('.filter-blocked-pending-approval .gl-tab-counter-badge') do expect(page).to have_content('2') end end it 'filters by users who are pending approval' do user = create(:user, :blocked_pending_approval) visit admin_users_path click_link 'Pending approval' expect(page).to have_content(user.email) end end context 'when blocking/unblocking a user' do it 'shows confirmation and allows blocking and unblocking', :js do expect(page).to have_content(user.email) click_action_in_user_dropdown(user.id, 'Block') wait_for_requests expect(page).to have_content('Block user') expect(page).to have_content('Blocking user has the following effects') expect(page).to have_content('User will not be able to login') expect(page).to have_content('Owned groups will be left') find('.modal-footer button', text: 'Block').click wait_for_requests expect(page).to have_content('Successfully blocked') expect(page).not_to have_content(user.email) click_link 'Blocked' wait_for_requests expect(page).to have_content(user.email) click_action_in_user_dropdown(user.id, 'Unblock') expect(page).to have_content('Unblock user') expect(page).to have_content('You can always block their account again if needed.') find('.modal-footer button', text: 'Unblock').click wait_for_requests expect(page).to have_content('Successfully unblocked') expect(page).not_to have_content(user.email) end end context 'when deactivating/re-activating a user' do it 'shows confirmation and allows deactivating and re-activating', :js do expect(page).to have_content(user.email) click_action_in_user_dropdown(user.id, 'Deactivate') expect(page).to have_content('Deactivate user') expect(page).to have_content('Deactivating a user has the following effects') expect(page).to have_content('The user will be logged out') expect(page).to have_content('Personal projects, group and user history will be left intact') find('.modal-footer button', text: 'Deactivate').click wait_for_requests expect(page).to have_content('Successfully deactivated') expect(page).not_to have_content(user.email) click_link 'Deactivated' wait_for_requests expect(page).to have_content(user.email) click_action_in_user_dropdown(user.id, 'Activate') expect(page).to have_content('Activate user') expect(page).to have_content('You can always deactivate their account again if needed.') find('.modal-footer button', text: 'Activate').click wait_for_requests expect(page).to have_content('Successfully activated') expect(page).not_to have_content(user.email) end end context 'when a user is locked', time_travel_to: '2020-02-25 10:30:45 -0700' do let_it_be(:locked_user) { create(:user, locked_at: DateTime.parse('2020-02-25 10:30:00 -0700')) } it "displays `Locked` badge next to user" do expect(page).to have_content("#{locked_user.name} Locked") end it 'allows a user to be unlocked from the `User administration dropdown', :js do accept_gl_confirm("Unlock user #{locked_user.name}?", button_text: 'Unlock') do click_action_in_user_dropdown(locked_user.id, 'Unlock') end expect(page).not_to have_content("#{locked_user.name} (Locked)") end end describe 'internal users' do context 'when showing a `Ghost User`' do let_it_be(:ghost_user) { create(:user, :ghost) } it 'does not render actions dropdown' do expect(page).not_to have_css("[data-testid='user-actions-#{ghost_user.id}'] [data-testid='dropdown-toggle']") end end context 'when showing a `Bot User`' do let_it_be(:bot_user) { create(:user, user_type: :alert_bot) } it 'does not render actions dropdown' do expect(page).not_to have_css("[data-testid='user-actions-#{bot_user.id}'] [data-testid='dropdown-toggle']") end end end context 'user group count', :js do before do group = create(:group) group.add_developer(current_user) project = create(:project, group: create(:group)) project.add_reporter(current_user) end it 'displays count of the users authorized groups' do visit admin_users_path wait_for_requests expect(page.find("[data-testid='user-group-count-#{current_user.id}']").text).to eq("2") end end end describe 'GET /admin/users/new' do let_it_be(:user_username) { 'bang' } before do visit new_admin_user_path fill_in 'user_name', with: 'Big Bang' fill_in 'user_username', with: user_username fill_in 'user_email', with: 'bigbang@mail.com' end it 'creates new user' do expect { click_button 'Create user' }.to change { User.count }.by(1) end it 'applies defaults to user' do click_button 'Create user' user = User.find_by(username: 'bang') expect(user.projects_limit) .to eq(Gitlab.config.gitlab.default_projects_limit) expect(user.can_create_group) .to eq(Gitlab::CurrentSettings.can_create_group) expect(user.private_profile) .to eq(Gitlab::CurrentSettings.user_defaults_to_private_profile) end it 'creates user with valid data' do click_button 'Create user' user = User.find_by(username: 'bang') expect(user.name).to eq('Big Bang') expect(user.email).to eq('bigbang@mail.com') end it 'calls send mail' do expect_next_instance_of(NotificationService) do |instance| expect(instance).to receive(:new_user) end click_button 'Create user' end it 'sends valid email to user with email & password' do perform_enqueued_jobs do click_button 'Create user' end user = User.find_by(username: 'bang') email = ActionMailer::Base.deliveries.last expect(email.subject).to have_content('Account was created') expect(email.text_part.body).to have_content(user.email) expect(email.text_part.body).to have_content('password') end context 'username contains spaces' do let_it_be(:user_username) { 'Bing bang' } it "doesn't create the user and shows an error message" do expect { click_button 'Create user' }.to change { User.count }.by(0) expect(page).to have_content('The form contains the following error') expect(page).to have_content('Username can contain only letters, digits') end end context 'with new users set to external enabled' do context 'with regex to match internal user email address set', :js do before do stub_application_setting(user_default_external: true) stub_application_setting(user_default_internal_regex: '\.internal@') visit new_admin_user_path end it 'automatically unchecks external for matching email' do expects_external_to_be_checked expects_warning_to_be_hidden fill_in 'user_email', with: 'test.internal@domain.ch' expects_external_to_be_unchecked expects_warning_to_be_shown fill_in 'user_email', with: 'test@domain.ch' expects_external_to_be_checked expects_warning_to_be_hidden uncheck 'user_external' expects_warning_to_be_hidden end it 'creates an internal user' do user_name = 'tester1' fill_in 'user_email', with: 'test.internal@domain.ch' fill_in 'user_name', with: 'tester1 name' fill_in 'user_username', with: user_name expects_external_to_be_unchecked expects_warning_to_be_shown click_button 'Create user' new_user = User.find_by(username: user_name) expect(new_user.external).to be_falsy end def expects_external_to_be_checked expect(find('#user_external')).to be_checked end def expects_external_to_be_unchecked expect(find('#user_external')).not_to be_checked end def expects_warning_to_be_hidden expect(find('#warning_external_automatically_set', visible: :all)[:class]).to include 'hidden' end def expects_warning_to_be_shown expect(find('#warning_external_automatically_set')[:class]).not_to include 'hidden' end end end end describe 'GET /admin/users/:id/projects' do let_it_be(:group) { create(:group) } let_it_be(:project) { create(:project, group: group) } before do group.add_developer(user) visit projects_admin_user_path(user) end it 'lists groups' do within(:css, '.gl-mb-3 + .gl-card') do expect(page).to have_content 'Groups' expect(page).to have_link group.name, href: admin_group_path(group) end end it 'allows navigation to the group details' do within(:css, '.gl-mb-3 + .gl-card') do click_link group.name end expect(page).to have_content "Group: #{group.name}" expect(page).to have_content project.name end it 'shows the group access level' do within(:css, '.gl-mb-3 + .gl-card') do expect(page).to have_content 'Developer' end end it 'allows group membership to be revoked', :js do page.within(first('.group_member')) do find('.btn[data-testid="remove-user"]').click end accept_gl_confirm(button_text: 'Remove') wait_for_requests expect(page).not_to have_selector('.group_member') end end describe 'show breadcrumbs' do it do visit admin_user_path(user) check_breadcrumb(user.name) visit projects_admin_user_path(user) check_breadcrumb(user.name) visit keys_admin_user_path(user) check_breadcrumb(user.name) visit admin_user_impersonation_tokens_path(user) check_breadcrumb(user.name) visit admin_user_identities_path(user) check_breadcrumb(user.name) visit new_admin_user_identity_path(user) check_breadcrumb('New Identity') visit admin_user_identities_path(user) find('.table').find(:link, 'Edit').click check_breadcrumb('Edit Identity') end def check_breadcrumb(content) expect(find('[data-testid="breadcrumb-current-link"]')).to have_content(content) end end describe 'GET /admin/users/:id/edit' do before do visit edit_admin_user_path(user) end describe 'Update user' do before do fill_in 'user_name', with: 'Big Bang' fill_in 'user_email', with: 'bigbang@mail.com' fill_in 'user_password', with: 'AValidPassword1' fill_in 'user_password_confirmation', with: 'AValidPassword1' choose 'user_access_level_admin' check 'Private profile' click_button 'Save changes' end it 'shows page with new data' do expect(page).to have_content('bigbang@mail.com') expect(page).to have_content('Big Bang') end it 'changes user entry' do user.reload expect(user.name).to eq('Big Bang') expect(user.admin?).to be_truthy expect(user.password_expires_at).to be <= Time.zone.now expect(user.private_profile).to eq(true) end end describe 'update username to non ascii char' do it do fill_in 'user_username', with: '\u3042\u3044' click_button('Save') page.within '#error_explanation' do expect(page).to have_content('Username') end expect(page).to have_selector(%(form[action="/admin/users/#{user.username}"])) end end end def first_row page.all('[role="row"]')[1] end def second_row page.all('[role="row"]')[2] end def sort_by(option) page.within('.filtered-search-block') do find('.gl-dropdown').click find('.gl-dropdown-item', text: option).click end end end