295 lines
11 KiB
Ruby
295 lines
11 KiB
Ruby
|
# frozen_string_literal: true
|
||
|
|
||
|
require 'rspec/core/sandbox'
|
||
|
|
||
|
RSpec.configure do |c|
|
||
|
c.around do |ex|
|
||
|
RSpec::Core::Sandbox.sandboxed do |config|
|
||
|
# If there is an example-within-an-example, we want to make sure the inner example
|
||
|
# does not get a reference to the outer example (the real spec) if it calls
|
||
|
# something like `pending`
|
||
|
config.before(:context) { RSpec.current_example = nil }
|
||
|
|
||
|
config.color_mode = :off
|
||
|
|
||
|
ex.run
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
RSpec.describe QA::Specs::Helpers::ContextSelector do
|
||
|
include Helpers::StubENV
|
||
|
include QA::Specs::Helpers::RSpec
|
||
|
|
||
|
before do
|
||
|
QA::Runtime::Scenario.define(:gitlab_address, 'https://staging.gitlab.com')
|
||
|
described_class.configure_rspec
|
||
|
end
|
||
|
|
||
|
describe '.context_matches?' do
|
||
|
it 'returns true when url has .com' do
|
||
|
QA::Runtime::Scenario.define(:gitlab_address, "https://staging.gitlab.com")
|
||
|
|
||
|
expect(described_class.dot_com?).to be_truthy
|
||
|
end
|
||
|
|
||
|
it 'returns false when url does not have .com' do
|
||
|
QA::Runtime::Scenario.define(:gitlab_address, "https://gitlab.test")
|
||
|
|
||
|
expect(described_class.dot_com?).to be_falsey
|
||
|
end
|
||
|
|
||
|
context 'with arguments' do
|
||
|
it 'returns true when :subdomain is set' do
|
||
|
QA::Runtime::Scenario.define(:gitlab_address, "https://staging.gitlab.com")
|
||
|
|
||
|
expect(described_class.dot_com?(subdomain: :staging)).to be_truthy
|
||
|
end
|
||
|
|
||
|
it 'matches multiple subdomains' do
|
||
|
QA::Runtime::Scenario.define(:gitlab_address, "https://staging.gitlab.com")
|
||
|
|
||
|
expect(described_class.context_matches?(subdomain: [:release, :staging])).to be_truthy
|
||
|
expect(described_class.context_matches?(:production, subdomain: [:release, :staging])).to be_truthy
|
||
|
end
|
||
|
|
||
|
it 'matches :production' do
|
||
|
QA::Runtime::Scenario.define(:gitlab_address, "https://gitlab.com/")
|
||
|
|
||
|
expect(described_class.context_matches?(:production)).to be_truthy
|
||
|
end
|
||
|
|
||
|
it 'doesnt match with mismatching switches' do
|
||
|
QA::Runtime::Scenario.define(:gitlab_address, 'https://gitlab.test')
|
||
|
|
||
|
aggregate_failures do
|
||
|
expect(described_class.context_matches?(tld: '.net')).to be_falsey
|
||
|
expect(described_class.context_matches?(:production)).to be_falsey
|
||
|
expect(described_class.context_matches?(subdomain: [:staging])).to be_falsey
|
||
|
expect(described_class.context_matches?(domain: 'example')).to be_falsey
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
it 'returns false for mismatching' do
|
||
|
QA::Runtime::Scenario.define(:gitlab_address, "https://staging.gitlab.com")
|
||
|
|
||
|
expect(described_class.context_matches?(:production)).to be_falsey
|
||
|
end
|
||
|
end
|
||
|
|
||
|
describe 'description and context blocks' do
|
||
|
context 'with environment set' do
|
||
|
it 'can apply to contexts or descriptions' do
|
||
|
group = describe_successfully 'Runs in staging', only: { subdomain: :staging } do
|
||
|
it('runs in staging') {}
|
||
|
end
|
||
|
|
||
|
expect(group.examples[0].execution_result.status).to eq(:passed)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'with different environment set' do
|
||
|
before do
|
||
|
QA::Runtime::Scenario.define(:gitlab_address, 'https://gitlab.com')
|
||
|
described_class.configure_rspec
|
||
|
end
|
||
|
|
||
|
it 'does not run against production' do
|
||
|
group = describe_successfully 'Runs in staging', :something, only: { subdomain: :staging } do
|
||
|
it('runs in staging') {}
|
||
|
end
|
||
|
|
||
|
expect(group.examples[0].execution_result.status).to eq(:pending)
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
it 'runs only in staging' do
|
||
|
group = describe_successfully do
|
||
|
it('runs in staging', only: { subdomain: :staging }) {}
|
||
|
it('doesnt run in staging', only: :production) {}
|
||
|
it('runs in staging also', only: { subdomain: %i[release staging] }) {}
|
||
|
it('runs in any env') {}
|
||
|
end
|
||
|
|
||
|
expect(group.examples[0].execution_result.status).to eq(:passed)
|
||
|
expect(group.examples[1].execution_result.status).to eq(:pending)
|
||
|
expect(group.examples[2].execution_result.status).to eq(:passed)
|
||
|
expect(group.examples[3].execution_result.status).to eq(:passed)
|
||
|
end
|
||
|
|
||
|
context 'custom env' do
|
||
|
before do
|
||
|
QA::Runtime::Scenario.define(:gitlab_address, 'https://release.gitlab.net')
|
||
|
end
|
||
|
|
||
|
it 'runs on a custom environment' do
|
||
|
group = describe_successfully do
|
||
|
it('runs on release gitlab net', only: { tld: '.net', subdomain: :release, domain: 'gitlab' }) {}
|
||
|
it('does not run on release', only: :production) {}
|
||
|
end
|
||
|
|
||
|
expect(group.examples.first.execution_result.status).to eq(:passed)
|
||
|
expect(group.examples.last.execution_result.status).to eq(:pending)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'production' do
|
||
|
before do
|
||
|
QA::Runtime::Scenario.define(:gitlab_address, 'https://gitlab.com/')
|
||
|
end
|
||
|
|
||
|
it 'runs on production' do
|
||
|
group = describe_successfully do
|
||
|
it('runs on prod', only: :production) {}
|
||
|
it('does not run in prod', only: { subdomain: :staging }) {}
|
||
|
it('runs in prod and staging', only: { subdomain: /(staging.)?/, domain: 'gitlab' }) {}
|
||
|
end
|
||
|
|
||
|
expect(group.examples[0].execution_result.status).to eq(:passed)
|
||
|
expect(group.examples[1].execution_result.status).to eq(:pending)
|
||
|
expect(group.examples[2].execution_result.status).to eq(:passed)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
it 'outputs a message for invalid environments' do
|
||
|
group = describe_successfully do
|
||
|
it('will skip', only: :production) {}
|
||
|
end
|
||
|
|
||
|
expect(group.examples.first.execution_result.pending_message).to match(/[Tt]est.*not compatible.*environment/)
|
||
|
end
|
||
|
|
||
|
context 'with pipeline constraints' do
|
||
|
context 'without CI_PROJECT_NAME set' do
|
||
|
before do
|
||
|
stub_env('CI_PROJECT_NAME', nil)
|
||
|
described_class.configure_rspec
|
||
|
end
|
||
|
|
||
|
it 'runs on any pipeline' do
|
||
|
group = describe_successfully do
|
||
|
it('runs given a single named pipeline', only: { pipeline: :nightly }) {}
|
||
|
it('runs given an array of pipelines', only: { pipeline: [:canary, :not_nightly] }) {}
|
||
|
end
|
||
|
|
||
|
aggregate_failures do
|
||
|
expect(group.examples[0].execution_result.status).to eq(:passed)
|
||
|
expect(group.examples[1].execution_result.status).to eq(:passed)
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when a pipeline triggered from the default branch runs in gitlab-qa' do
|
||
|
before do
|
||
|
stub_env('CI_PROJECT_NAME', 'gitlab-qa')
|
||
|
described_class.configure_rspec
|
||
|
end
|
||
|
|
||
|
it 'runs on default branch pipelines' do
|
||
|
group = describe_successfully do
|
||
|
it('runs on master pipeline given a single pipeline', only: { pipeline: :master }) {}
|
||
|
it('runs in master given an array of pipelines', only: { pipeline: [:canary, :master] }) {}
|
||
|
it('does not run in non-default pipelines', only: { pipeline: [:nightly, :not_nightly, :not_master] }) {}
|
||
|
end
|
||
|
|
||
|
aggregate_failures do
|
||
|
expect(group.examples[0].execution_result.status).to eq(:passed)
|
||
|
expect(group.examples[1].execution_result.status).to eq(:passed)
|
||
|
expect(group.examples[2].execution_result.status).to eq(:pending)
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'with CI_PROJECT_NAME set' do
|
||
|
before do
|
||
|
stub_env('CI_PROJECT_NAME', 'NIGHTLY')
|
||
|
described_class.configure_rspec
|
||
|
end
|
||
|
|
||
|
it 'runs on designated pipeline' do
|
||
|
group = describe_successfully do
|
||
|
it('runs on nightly', only: { pipeline: :nightly }) {}
|
||
|
it('does not run in not_nightly', only: { pipeline: :not_nightly }) {}
|
||
|
it('runs on nightly given an array', only: { pipeline: [:canary, :nightly] }) {}
|
||
|
it('does not run in not_nightly given an array', only: { pipeline: [:not_nightly, :canary] }) {}
|
||
|
end
|
||
|
|
||
|
aggregate_failures do
|
||
|
expect(group.examples[0].execution_result.status).to eq(:passed)
|
||
|
expect(group.examples[1].execution_result.status).to eq(:pending)
|
||
|
expect(group.examples[2].execution_result.status).to eq(:passed)
|
||
|
expect(group.examples[3].execution_result.status).to eq(:pending)
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when excluding contexts' do
|
||
|
context 'with job constraints' do
|
||
|
context 'without CI_JOB_NAME set' do
|
||
|
before do
|
||
|
stub_env('CI_JOB_NAME', nil)
|
||
|
described_class.configure_rspec
|
||
|
end
|
||
|
|
||
|
it 'runs in any job' do
|
||
|
group = describe_successfully do
|
||
|
it('runs given a single named job', exclude: { job: 'ee:instance-image' }) {}
|
||
|
it('runs given a single regex pattern', exclude: { job: '.*:instance-image' }) {}
|
||
|
it('runs given an array of jobs', exclude: { job: %w[ee:instance-image qa-schedules-browser_ui-3_create] }) {}
|
||
|
it('runs given an array of regex patterns', exclude: { job: %w[ee:.* qa-schedules-browser_ui.*] }) {}
|
||
|
it('runs given a mix of strings and regex patterns', exclude: { job: %w[ee:instance-image qa-schedules-browser_ui.*] }) {}
|
||
|
end
|
||
|
|
||
|
aggregate_failures do
|
||
|
group.examples.each do |example|
|
||
|
expect(example.execution_result.status).to eq(:passed)
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'with CI_JOB_NAME set' do
|
||
|
before do
|
||
|
stub_env('CI_JOB_NAME', 'ee:instance-image')
|
||
|
described_class.configure_rspec
|
||
|
end
|
||
|
|
||
|
it 'does not run in the specified job' do
|
||
|
group = describe_successfully do
|
||
|
it('skips given a single named job', exclude: { job: 'ee:instance-image' }) {}
|
||
|
it('skips given a single regex pattern', exclude: { job: '.*:instance-image' }) {}
|
||
|
it('skips given an array of jobs', exclude: { job: %w[ee:instance-image qa-schedules-browser_ui-3_create] }) {}
|
||
|
it('skips given an array of regex patterns', exclude: { job: %w[ee:.* qa-schedules-browser_ui.*] }) {}
|
||
|
it('skips given a mix of strings and regex patterns', exclude: { job: %w[ee:instance-image qa-schedules-browser_ui.*] }) {}
|
||
|
end
|
||
|
|
||
|
aggregate_failures do
|
||
|
group.examples.each do |example|
|
||
|
expect(example.execution_result.status).to eq(:pending)
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
it 'runs in jobs that do not match' do
|
||
|
group = describe_successfully do
|
||
|
it('runs given a single named job', exclude: { job: 'ce:instance-image' }) {}
|
||
|
it('runs given a single regex pattern', exclude: { job: '.*:instance-image-quarantine' }) {}
|
||
|
it('runs given an array of jobs', exclude: { job: %w[ce:instance-image qa-schedules-browser_ui-3_create] }) {}
|
||
|
it('runs given an array of regex patterns', exclude: { job: %w[ce:.* qa-schedules-browser_ui.*] }) {}
|
||
|
it('runs given a mix of strings and regex patterns', exclude: { job: %w[ce:instance-image qa-schedules-browser_ui.*] }) {}
|
||
|
end
|
||
|
|
||
|
aggregate_failures do
|
||
|
group.examples.each do |example|
|
||
|
expect(example.execution_result.status).to eq(:passed)
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|