# frozen_string_literal: true

require 'spec_helper'

RSpec.describe "Group Runners" do
  include Spec::Support::Helpers::Features::RunnersHelpers

  let_it_be(:group_owner) { create(:user) }
  let_it_be(:group) { create(:group) }
  let_it_be(:project) { create(:project, group: group) }

  before do
    group.add_owner(group_owner)
    sign_in(group_owner)
  end

  describe "Group runners page", :js do
    let!(:group_registration_token) { group.runners_token }

    describe "runners registration" do
      before do
        visit group_runners_path(group)
      end

      it_behaves_like "shows and resets runner registration token" do
        let(:dropdown_text) { 'Register a group runner' }
        let(:registration_token) { group_registration_token }
      end
    end

    context "with no runners" do
      before do
        visit group_runners_path(group)
      end

      it_behaves_like 'shows no runners registered'

      it 'shows tabs with total counts equal to 0' do
        expect(page).to have_link('All 0')
        expect(page).to have_link('Group 0')
        expect(page).to have_link('Project 0')
      end
    end

    context "with an online group runner" do
      let!(:group_runner) do
        create(:ci_runner, :group, groups: [group], description: 'runner-foo', contacted_at: Time.zone.now)
      end

      before do
        visit group_runners_path(group)
      end

      it_behaves_like 'shows runner in list' do
        let(:runner) { group_runner }
      end

      it_behaves_like 'pauses, resumes and deletes a runner' do
        let(:runner) { group_runner }
      end

      it 'shows a group badge' do
        within_runner_row(group_runner.id) do
          expect(page).to have_selector '.badge', text: s_('Runners|Group')
        end
      end

      it 'can edit runner information' do
        within_runner_row(group_runner.id) do
          expect(find_link('Edit')[:href]).to end_with(edit_group_runner_path(group, group_runner))
        end
      end

      context 'when description does not match' do
        before do
          input_filtered_search_keys('runner-baz')
        end

        it_behaves_like 'shows no runners found'

        it 'shows no runner' do
          expect(page).not_to have_content 'runner-foo'
        end
      end
    end

    context "with an online project runner" do
      let!(:project_runner) do
        create(:ci_runner, :project, projects: [project], description: 'runner-bar', contacted_at: Time.zone.now)
      end

      before do
        visit group_runners_path(group)
      end

      it_behaves_like 'shows runner in list' do
        let(:runner) { project_runner }
      end

      it_behaves_like 'pauses, resumes and deletes a runner' do
        let(:runner) { project_runner }
      end

      it 'shows a project badge' do
        within_runner_row(project_runner.id) do
          expect(page).to have_selector '.badge', text: s_('Runners|Project')
        end
      end

      it 'can edit runner information' do
        within_runner_row(project_runner.id) do
          expect(find_link('Edit')[:href]).to end_with(edit_group_runner_path(group, project_runner))
        end
      end
    end

    context "with an online instance runner" do
      let!(:instance_runner) do
        create(:ci_runner, :instance, description: 'runner-baz', contacted_at: Time.zone.now)
      end

      before do
        visit group_runners_path(group)
      end

      context "when selecting 'Show only inherited'" do
        before do
          find("[data-testid='runner-membership-toggle'] button").click

          wait_for_requests
        end

        it_behaves_like 'shows runner in list' do
          let(:runner) { instance_runner }
        end

        it 'shows runner details page' do
          click_link("##{instance_runner.id} (#{instance_runner.short_sha})")

          expect(current_url).to include(group_runner_path(group, instance_runner))
          expect(page).to have_content "#{s_('Runners|Description')} runner-baz"
        end
      end
    end

    context 'with a multi-project runner' do
      let(:project) { create(:project, group: group) }
      let(:project_2) { create(:project, group: group) }
      let!(:runner) { create(:ci_runner, :project, projects: [project, project_2], description: 'group-runner') }

      it 'user cannot remove the project runner' do
        visit group_runners_path(group)

        within_runner_row(runner.id) do
          expect(page).not_to have_button 'Delete runner'
        end
      end
    end

    describe 'filtered search' do
      before do
        visit group_runners_path(group)
      end

      it 'allows user to search by paused and status', :js do
        focus_filtered_search

        page.within(search_bar_selector) do
          expect(page).to have_link(s_('Runners|Paused'))
          expect(page).to have_content('Status')
        end
      end
    end

    describe 'filter by tag' do
      let!(:runner_1) { create(:ci_runner, :group, groups: [group], description: 'runner-blue', tag_list: ['blue']) }
      let!(:runner_2) { create(:ci_runner, :group, groups: [group], description: 'runner-red', tag_list: ['red']) }

      before do
        visit group_runners_path(group)
      end

      it_behaves_like 'filters by tag' do
        let(:tag) { 'blue' }
        let(:found_runner) { runner_1.description }
        let(:missing_runner) { runner_2.description }
      end
    end
  end

  describe "Group runner show page", :js do
    let!(:group_runner) do
      create(:ci_runner, :group, groups: [group], description: 'runner-foo')
    end

    it 'user views runner details' do
      visit group_runner_path(group, group_runner)

      expect(page).to have_content "#{s_('Runners|Description')} runner-foo"
    end
  end

  describe "Group runner edit page", :js do
    let!(:group_runner) do
      create(:ci_runner, :group, groups: [group])
    end

    before do
      visit edit_group_runner_path(group, group_runner)
      wait_for_requests
    end

    it_behaves_like 'submits edit runner form' do
      let(:runner) { group_runner }
      let(:runner_page_path) { group_runner_path(group, group_runner) }
    end
  end
end