Files
gitlab-foss/spec/features/dashboard/projects_spec.rb
2025-06-17 12:20:51 +00:00

290 lines
9.7 KiB
Ruby

# frozen_string_literal: true
require 'spec_helper'
RSpec.describe 'Dashboard Projects', :js, feature_category: :groups_and_projects do
let_it_be(:user) { create(:user) }
let_it_be(:project, reload: true) { create(:project, :repository, creator: build(:user)) } # ensure creator != owner to avoid N+1 false-positive
let_it_be(:project2) { create(:project, :public) }
let_it_be(:personal_project) { create(:project, namespace: user.namespace) }
let_it_be(:personal_project_with_stars) { create(:project, namespace: user.namespace, star_count: 10) }
let_it_be(:pipeline) { create(:ci_pipeline, :success, project: project, sha: project.commit.sha, ref: project.default_branch) }
before do
project.add_developer(user)
sign_in(user)
end
it 'mounts JS app and defaults to contributed tab' do
visit dashboard_projects_path
wait_for_requests
expect(page).to have_content('Projects')
expect(page).to have_selector('a[aria-selected="true"]', text: 'Contributed')
end
it_behaves_like "an autodiscoverable RSS feed with current_user's feed token" do
before do
visit dashboard_projects_path
wait_for_requests
end
end
it_behaves_like 'a "Your work" page with sidebar and breadcrumbs', :dashboard_projects_path, :projects
it_behaves_like 'page with product usage data collection banner' do
let(:page_path) { dashboard_projects_path }
end
it 'links to the "Explore projects" page' do
visit dashboard_projects_path
wait_for_requests
expect(page).to have_link("Explore projects", href: starred_explore_projects_path)
end
context 'when user has access to the project' do
it 'shows role badge' do
visit member_dashboard_projects_path
wait_for_requests
within_testid("projects-list-item-#{project.id}") do
expect(find_by_testid('user-access-role')).to have_content('Developer')
end
end
end
context 'when last_activity_at and update_at are present', time_travel_to: '2025-01-27T09:44:07Z' do
let_it_be(:project_with_last_activity) do
create(
:project,
namespace: user.namespace,
last_repository_updated_at: 1.hour.ago,
last_activity_at: Time.current
)
end
it 'shows the last_activity_at attribute as the update date', quarantine: 'https://gitlab.com/gitlab-org/gitlab/-/issues/514342' do
visit member_dashboard_projects_path
wait_for_requests
within_testid("projects-list-item-#{project_with_last_activity.id}") do
expect(page).to have_xpath("//time[@datetime='#{project_with_last_activity.last_activity_at.iso8601}']")
end
end
end
context 'when last_activity_at is missing', time_travel_to: '2025-01-27T09:44:07Z' do
it 'shows the updated_at attribute as the update date' do
visit member_dashboard_projects_path
wait_for_requests
within_testid("projects-list-item-#{project.id}") do
expect(page).to have_xpath("//time[@datetime='#{project.updated_at.iso8601}']")
end
end
end
it 'shows personal projects on personal projects tab' do
visit personal_dashboard_projects_path
wait_for_requests
expect(page).not_to have_content(project.name)
expect(page).to have_content(personal_project.name)
end
it 'sorts projects by most stars when sorting by most stars' do
visit personal_dashboard_projects_path(sort: :stars_desc)
wait_for_requests
expect(first('[data-testid*="projects-list-item"]')).to have_content(personal_project_with_stars.title)
end
context 'when a project is archived' do
let_it_be(:archived_project) { create(:project, :archived, namespace: user.namespace) }
let(:personal_tab) { find(".nav-item:nth-child(3)") }
let(:member_tab) { find(".nav-item:nth-child(4)") }
it 'is not included in the personal projects or member count' do
visit dashboard_projects_path
wait_for_requests
within "ul.gl-tabs-nav" do
expect(personal_tab).to have_text("Personal 2")
expect(member_tab).to have_text("Member 3")
personal_tab.click
expect(personal_tab).to have_text("Personal 2")
expect(page).not_to have_content(archived_project.name)
member_tab.click
expect(member_tab).to have_text("Member 3")
expect(page).not_to have_content(archived_project.name)
end
end
end
context 'when on Member projects tab' do
it 'shows all projects you are a member of' do
visit member_dashboard_projects_path
wait_for_requests
expect(page).to have_content(project.name)
expect(page).to have_content(personal_project.name)
expect(page).to have_content(personal_project_with_stars.name)
expect(find('a[aria-selected="true"]')).to have_content('3')
end
end
context 'when on Starred projects tab' do
it_behaves_like 'a "Your work" page with sidebar and breadcrumbs', :starred_dashboard_projects_path, :projects
it 'shows the empty state when there are no starred projects' do
visit(starred_dashboard_projects_path)
wait_for_requests
expect(page).to have_text("You haven't starred any projects yet.")
end
it 'shows only starred projects' do
user.toggle_star(project2)
visit(starred_dashboard_projects_path)
wait_for_requests
expect(page).not_to have_content(project.name)
expect(page).to have_content(project2.name)
expect(find('a[aria-selected="true"]')).to have_content('1')
end
end
describe 'with a pipeline' do
it 'shows that the last pipeline passed' do
visit member_dashboard_projects_path
wait_for_requests
within_testid("projects-list-item-#{project.id}") do
expect(page).to have_css("[data-testid='ci-icon']")
expect(page).to have_css('[data-testid="status_success_borderless-icon"]')
expect(page).to have_link('Status: Passed')
end
end
context 'guest user of project and project has private pipelines' do
let_it_be(:guest_user) { create(:user) }
let_it_be(:project_with_private_pipelines) { create(:project, namespace: user.namespace, public_builds: false) }
before_all do
project_with_private_pipelines.add_guest(guest_user)
end
before do
sign_in(guest_user)
end
it 'does not show the pipeline status' do
visit member_dashboard_projects_path
wait_for_requests
within_testid("projects-list-item-#{project_with_private_pipelines.id}") do
expect(page).not_to have_css("[data-testid='ci-icon']")
end
end
end
context "when last_pipeline is missing" do
it 'does not show the pipeline status' do
visit member_dashboard_projects_path
wait_for_requests
within_testid("projects-list-item-#{personal_project.id}") do
expect(page).not_to have_css("[data-testid='ci-icon']")
end
end
end
end
context 'when project has topics' do
let_it_be(:project_with_topics) { create(:project, namespace: user.namespace, topic_list: 'topic1') }
it 'shows project topics' do
visit member_dashboard_projects_path
wait_for_requests
within_testid("projects-list-item-#{project_with_topics.id}") do
expect(page).to have_link('topic1', href: topic_explore_projects_path(topic_name: 'topic1'))
end
end
end
context 'when project does not have topics' do
it 'does not show project topics' do
visit member_dashboard_projects_path
wait_for_requests
within_testid("projects-list-item-#{project.id}") do
expect(page).not_to have_selector('[data-testid="project-topics"]')
end
end
end
context 'last push widget', :use_clean_rails_memory_store_caching do
before do
event = create(:push_event, project: project, author: user)
create(:push_event_payload, event: event, ref: 'feature', action: :created)
Users::LastPushEventService.new(user).cache_last_push_event(event)
visit dashboard_projects_path
end
it 'shows "Create merge request" button' do
expect(page).to have_content 'You pushed to feature'
within('#content-body') do
find_link('Create merge request', visible: false).click
end
expect(page).to have_selector('.merge-request-form')
expect(page).to have_current_path project_new_merge_request_path(project), ignore_query: true
expect(find('#merge_request_target_project_id', visible: false).value).to eq project.id.to_s
expect(page).to have_content "From feature into master"
end
end
it 'avoids an N+1 query in dashboard index' do
visit member_dashboard_projects_path
wait_for_requests
control = ActiveRecord::QueryRecorder.new do
visit member_dashboard_projects_path
wait_for_requests
end
new_project = create(:project, :repository, name: 'new project')
create(:ci_pipeline, :with_job, status: :success, project: new_project, ref: new_project.commit.sha)
new_project.add_developer(user)
# There are a few known N+1 queries: https://gitlab.com/gitlab-org/gitlab/-/issues/214037
# - User#max_member_access_for_project_ids
# - ProjectsHelper#load_pipeline_status / Ci::CommitWithPipeline#last_pipeline
# - Ci::Pipeline#detailed_status
expect do
visit member_dashboard_projects_path
wait_for_requests
end.not_to exceed_query_limit(control).with_threshold(4)
end
context 'for delayed deletion' do
let_it_be(:project) { create(:project, :archived, namespace: user.namespace, marked_for_deletion_at: Date.current) }
it 'renders Restore button', :js do
visit inactive_dashboard_projects_path
wait_for_requests
within_testid("projects-list-item-#{project.id}") do
click_button 'Actions'
expect(page).to have_button('Restore')
end
end
end
end