Welcome to mirror list, hosted at ThFree Co, Russian Federation.

gitlab.com/gitlab-org/gitlab-foss.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
path: root/spec/lib
diff options
context:
space:
mode:
authorGitLab Bot <gitlab-bot@gitlab.com>2024-01-24 15:06:49 +0300
committerGitLab Bot <gitlab-bot@gitlab.com>2024-01-24 15:06:49 +0300
commitaf5193aa3cc159fd7d08be988bd00f1921077117 (patch)
tree664d9936ab99208c49b8721877be87b7e420d5bd /spec/lib
parent4dcdd5bebb55bd5522ec180070d4d265e00943b5 (diff)
Add latest changes from gitlab-org/gitlab@masterHEADmaster
Diffstat (limited to 'spec/lib')
-rw-r--r--spec/lib/gitlab/ci/config/external/mapper/normalizer_spec.rb9
-rw-r--r--spec/lib/gitlab/ci/config/external/mapper_spec.rb18
-rw-r--r--spec/lib/gitlab/database/loose_foreign_keys_spec.rb4
-rw-r--r--spec/lib/gitlab/pages/url_builder_spec.rb12
-rw-r--r--spec/lib/gitlab/url_blocker_spec.rb1009
5 files changed, 31 insertions, 1021 deletions
diff --git a/spec/lib/gitlab/ci/config/external/mapper/normalizer_spec.rb b/spec/lib/gitlab/ci/config/external/mapper/normalizer_spec.rb
index 09212833d84..6c80df084d1 100644
--- a/spec/lib/gitlab/ci/config/external/mapper/normalizer_spec.rb
+++ b/spec/lib/gitlab/ci/config/external/mapper/normalizer_spec.rb
@@ -40,5 +40,14 @@ RSpec.describe Gitlab::Ci::Config::External::Mapper::Normalizer, feature_categor
{ remote: 'https://example.com/.gitlab-ci.yml' }]
)
end
+
+ context 'when the location value is an invalid type' do
+ let(:locations) { [123] }
+
+ it 'raises an error' do
+ expect { process }.to raise_error(
+ Gitlab::Ci::Config::External::Mapper::InvalidTypeError, /Each include must be a hash or a string/)
+ end
+ end
end
end
diff --git a/spec/lib/gitlab/ci/config/external/mapper_spec.rb b/spec/lib/gitlab/ci/config/external/mapper_spec.rb
index d67b0ff8895..d6aa39c4849 100644
--- a/spec/lib/gitlab/ci/config/external/mapper_spec.rb
+++ b/spec/lib/gitlab/ci/config/external/mapper_spec.rb
@@ -174,6 +174,15 @@ RSpec.describe Gitlab::Ci::Config::External::Mapper, feature_category: :pipeline
it_behaves_like 'logging config file fetch', 'config_file_fetch_project_content_duration_s', 1
end
+
+ context 'when the include value is a Boolean' do
+ let(:values) { { include: true } }
+
+ it 'raises an error' do
+ expect { process }.to raise_error(
+ Gitlab::Ci::Config::External::Mapper::InvalidTypeError, /Each include must be a hash or a string/)
+ end
+ end
end
context "when 'include' is defined as an array" do
@@ -186,6 +195,15 @@ RSpec.describe Gitlab::Ci::Config::External::Mapper, feature_category: :pipeline
expect(subject).to all(respond_to(:valid?))
expect(subject).to all(respond_to(:content))
end
+
+ context 'when an include value is an Array' do
+ let(:values) { { include: [remote_url, [local_file]] } }
+
+ it 'raises an error' do
+ expect { process }.to raise_error(
+ Gitlab::Ci::Config::External::Mapper::InvalidTypeError, /Each include must be a hash or a string/)
+ end
+ end
end
context "when 'include' is defined as an array of hashes" do
diff --git a/spec/lib/gitlab/database/loose_foreign_keys_spec.rb b/spec/lib/gitlab/database/loose_foreign_keys_spec.rb
index 1824a50cb28..297103e550d 100644
--- a/spec/lib/gitlab/database/loose_foreign_keys_spec.rb
+++ b/spec/lib/gitlab/database/loose_foreign_keys_spec.rb
@@ -143,6 +143,10 @@ RSpec.describe Gitlab::Database::LooseForeignKeys do
described_class.instance_variable_set(:@loose_foreign_keys_yaml, loose_foreign_keys_yaml)
end
+ after do
+ described_class.instance_variable_set(:@loose_foreign_keys_yaml, nil)
+ end
+
it 'raises Gitlab::Database::GitlabSchema::UnknownSchemaError error' do
expect { subject }.to raise_error(Gitlab::Database::GitlabSchema::UnknownSchemaError)
end
diff --git a/spec/lib/gitlab/pages/url_builder_spec.rb b/spec/lib/gitlab/pages/url_builder_spec.rb
index 863c4481c9e..1a97ca01c3e 100644
--- a/spec/lib/gitlab/pages/url_builder_spec.rb
+++ b/spec/lib/gitlab/pages/url_builder_spec.rb
@@ -63,12 +63,6 @@ RSpec.describe Gitlab::Pages::UrlBuilder, feature_category: :pages do
it { is_expected.to eq('http://group.example.com/project') }
end
- context 'when project is upper cased' do
- let(:full_path) { 'group/Project' }
-
- it { is_expected.to eq('http://group.example.com/project') }
- end
-
context 'when project is in a nested group page' do
let(:full_path) { 'group/subgroup/project' }
@@ -133,12 +127,6 @@ RSpec.describe Gitlab::Pages::UrlBuilder, feature_category: :pages do
it { is_expected.to eq('http://example.com/group/project') }
end
- context 'when project is upper cased' do
- let(:full_path) { 'group/Project' }
-
- it { is_expected.to eq('http://example.com/group/project') }
- end
-
context 'when project is in a nested group page' do
let(:full_path) { 'group/subgroup/project' }
diff --git a/spec/lib/gitlab/url_blocker_spec.rb b/spec/lib/gitlab/url_blocker_spec.rb
deleted file mode 100644
index 9e98cdc05eb..00000000000
--- a/spec/lib/gitlab/url_blocker_spec.rb
+++ /dev/null
@@ -1,1009 +0,0 @@
-# frozen_string_literal: true
-
-require 'spec_helper'
-
-RSpec.describe Gitlab::UrlBlocker, :stub_invalid_dns_only, feature_category: :shared do
- include StubRequests
-
- let(:schemes) { %w[http https] }
-
- # This test ensures backward compatibliity for the validate! method.
- # We shoud refactor all callers of validate! to handle a Result object:
- # https://gitlab.com/gitlab-org/gitlab/-/issues/410890
- describe '#validate!' do
- let(:options) { { schemes: schemes } }
-
- subject { described_class.validate!(import_url, **options) }
-
- shared_examples 'validates URI and hostname' do
- it 'runs the url validations' do
- uri, hostname = subject
-
- expect(uri).to eq(Addressable::URI.parse(expected_uri))
- expect(hostname).to eq(expected_hostname)
- end
- end
-
- context 'when the URL hostname is a domain' do
- context 'when domain can be resolved' do
- let(:import_url) { 'https://example.org' }
-
- before do
- stub_dns(import_url, ip_address: '93.184.216.34')
- end
-
- it_behaves_like 'validates URI and hostname' do
- let(:expected_uri) { 'https://93.184.216.34' }
- let(:expected_hostname) { 'example.org' }
- let(:expected_use_proxy) { false }
- end
- end
- end
- end
-
- describe '#validate_url_with_proxy!' do
- let(:options) { { schemes: schemes } }
-
- subject { described_class.validate_url_with_proxy!(import_url, **options) }
-
- shared_examples 'validates URI and hostname' do
- it 'runs the url validations' do
- expect(subject.uri).to eq(Addressable::URI.parse(expected_uri))
- expect(subject.hostname).to eq(expected_hostname)
- expect(subject.use_proxy).to eq(expected_use_proxy)
- end
- end
-
- shared_context 'when instance configured to deny all requests' do
- before do
- allow(Gitlab::CurrentSettings).to receive(:current_application_settings?).and_return(true)
- stub_application_setting(deny_all_requests_except_allowed: true)
- end
- end
-
- shared_examples 'a URI denied by `deny_all_requests_except_allowed`' do
- context 'when instance setting is enabled' do
- include_context 'when instance configured to deny all requests'
-
- it 'blocks the request' do
- expect { subject }.to raise_error(Gitlab::HTTP_V2::UrlBlocker::BlockedUrlError)
- end
- end
-
- context 'when instance setting is not enabled' do
- it 'does not block the request' do
- expect { subject }.not_to raise_error
- end
- end
-
- context 'when passed as an argument' do
- let(:options) { super().merge(deny_all_requests_except_allowed: arg_value) }
-
- context 'when argument is a proc that evaluates to true' do
- let(:arg_value) { proc { true } }
-
- it 'blocks the request' do
- expect { subject }.to raise_error(Gitlab::HTTP_V2::UrlBlocker::BlockedUrlError)
- end
- end
-
- context 'when argument is a proc that evaluates to false' do
- let(:arg_value) { proc { false } }
-
- it 'does not block the request' do
- expect { subject }.not_to raise_error
- end
- end
-
- context 'when argument is true' do
- let(:arg_value) { true }
-
- it 'blocks the request' do
- expect { subject }.to raise_error(Gitlab::HTTP_V2::UrlBlocker::BlockedUrlError)
- end
- end
-
- context 'when argument is false' do
- let(:arg_value) { false }
-
- it 'does not block the request' do
- expect { subject }.not_to raise_error
- end
- end
- end
- end
-
- shared_examples 'a URI exempt from `deny_all_requests_except_allowed`' do
- include_context 'when instance configured to deny all requests'
-
- it 'does not block the request' do
- expect { subject }.not_to raise_error
- end
- end
-
- context 'when URI is nil' do
- let(:import_url) { nil }
-
- it_behaves_like 'validates URI and hostname' do
- let(:expected_uri) { nil }
- let(:expected_hostname) { nil }
- let(:expected_use_proxy) { true }
- end
-
- it_behaves_like 'a URI exempt from `deny_all_requests_except_allowed`'
- end
-
- context 'when URI is internal' do
- let(:import_url) { 'http://localhost' }
-
- before do
- stub_dns(import_url, ip_address: '127.0.0.1')
- end
-
- it_behaves_like 'validates URI and hostname' do
- let(:expected_uri) { 'http://127.0.0.1' }
- let(:expected_hostname) { 'localhost' }
- let(:expected_use_proxy) { false }
- end
-
- it_behaves_like 'a URI exempt from `deny_all_requests_except_allowed`'
- end
-
- context 'when URI is for a local object storage' do
- let(:import_url) { "#{host}/external-diffs/merge_request_diffs/mr-1/diff-1" }
- let(:enabled_object_storage_setting) do
- {
- 'enabled' => true,
- 'object_store' =>
- {
- 'enabled' => true,
- 'connection' => {
- 'endpoint' => host
- }
- }
- }
- end
-
- before do
- allow(Settings).to receive(:external_diffs).and_return(enabled_object_storage_setting)
- end
-
- context 'when allow_object_storage is true' do
- let(:options) { { allow_object_storage: true, schemes: schemes } }
-
- context 'with a local domain name' do
- let(:host) { 'http://review-minio-svc.svc:9000' }
-
- before do
- stub_dns(host, ip_address: '127.0.0.1')
- end
-
- it_behaves_like 'validates URI and hostname' do
- let(:expected_uri) { 'http://127.0.0.1:9000/external-diffs/merge_request_diffs/mr-1/diff-1' }
- let(:expected_hostname) { 'review-minio-svc.svc' }
- let(:expected_use_proxy) { false }
- end
-
- it_behaves_like 'a URI exempt from `deny_all_requests_except_allowed`'
- end
-
- context 'with an IP address' do
- let(:host) { 'http://127.0.0.1:9000' }
-
- it_behaves_like 'validates URI and hostname' do
- let(:expected_uri) { 'http://127.0.0.1:9000/external-diffs/merge_request_diffs/mr-1/diff-1' }
- let(:expected_hostname) { nil }
- let(:expected_use_proxy) { false }
- end
-
- it_behaves_like 'a URI exempt from `deny_all_requests_except_allowed`'
- end
-
- context 'when LFS object storage is enabled' do
- let(:lfs_config) do
- {
- 'enabled' => lfs_enabled,
- # This nesting of settings is necessary to trigger the bug
- 'object_store' => GitlabSettings::Options.build({ 'enabled' => true })
- }
- end
-
- let(:config) do
- {
- 'gitlab' => Gitlab.config.gitlab,
- 'repositories' => { 'storages' => { 'default' => 'test' } },
- 'lfs' => GitlabSettings::Options.build(lfs_config)
- }
- end
-
- let(:host) { 'http://127.0.0.1:9000' }
- let(:settings) { GitlabSettings::Options.build(config) }
-
- before do
- allow(Gitlab).to receive(:config).and_return(settings)
- settings.repositories.storages.default
- end
-
- context 'when LFS is disabled' do
- let(:lfs_enabled) { false }
-
- it 'raises an error' do
- expect { subject }.to raise_error(Gitlab::HTTP_V2::UrlBlocker::BlockedUrlError)
- end
- end
-
- context 'when LFS is enabled with no connection endpoint' do
- let(:lfs_enabled) { true }
-
- it 'raises an error' do
- expect { subject }.to raise_error(Gitlab::HTTP_V2::UrlBlocker::BlockedUrlError)
- end
- end
- end
- end
-
- context 'when allow_object_storage is false' do
- context 'with a local domain name' do
- let(:host) { 'http://review-minio-svc.svc:9000' }
-
- before do
- stub_dns(host, ip_address: '127.0.0.1')
- end
-
- it 'raises an error' do
- expect { subject }.to raise_error(Gitlab::HTTP_V2::UrlBlocker::BlockedUrlError)
- end
- end
-
- context 'with an IP address' do
- let(:host) { 'http://127.0.0.1:9000' }
-
- it 'raises an error' do
- expect { subject }.to raise_error(Gitlab::HTTP_V2::UrlBlocker::BlockedUrlError)
- end
- end
- end
- end
-
- context 'when resolving runs into a timeout' do
- let(:import_url) { 'http://example.com' }
-
- before do
- stub_const("#{described_class}::GETADDRINFO_TIMEOUT_SECONDS", 1)
- allow(Addrinfo).to receive(:getaddrinfo) { sleep 2 }
- end
-
- it 'raises an error due to DNS timeout' do
- expect { subject }.to raise_error(Gitlab::HTTP_V2::UrlBlocker::BlockedUrlError, "execution expired")
- end
- end
-
- context 'when the URL hostname is a domain' do
- context 'when domain can be resolved' do
- let(:import_url) { 'https://example.org' }
-
- before do
- stub_dns(import_url, ip_address: '93.184.216.34')
- end
-
- it_behaves_like 'validates URI and hostname' do
- let(:expected_uri) { 'https://93.184.216.34' }
- let(:expected_hostname) { 'example.org' }
- let(:expected_use_proxy) { false }
- end
-
- it_behaves_like 'a URI denied by `deny_all_requests_except_allowed`'
- end
-
- context 'when domain cannot be resolved' do
- let(:import_url) { 'http://foobar.x' }
-
- before do
- stub_env('RSPEC_ALLOW_INVALID_URLS', 'false')
- end
-
- it 'raises an error' do
- expect { subject }.to raise_error(Gitlab::HTTP_V2::UrlBlocker::BlockedUrlError)
- end
-
- context 'with HTTP_PROXY' do
- let(:import_url) { 'http://foobar.x' }
-
- before do
- stub_env('http_proxy', 'http://proxy.example.com')
- end
-
- it_behaves_like 'validates URI and hostname' do
- let(:expected_uri) { import_url }
- let(:expected_hostname) { nil }
- let(:expected_use_proxy) { true }
- end
-
- context 'with no_proxy' do
- before do
- stub_env('no_proxy', 'foobar.x')
- end
-
- it_behaves_like 'validates URI and hostname' do
- let(:expected_uri) { import_url }
- let(:expected_hostname) { nil }
- let(:expected_use_proxy) { false }
- end
- end
- end
- end
-
- context 'when domain is too long' do
- let(:import_url) { "https://example#{'a' * 1024}.com" }
-
- it 'raises an error' do
- expect { subject }.to raise_error(Gitlab::HTTP_V2::UrlBlocker::BlockedUrlError)
- end
- end
- end
-
- context 'when the URL hostname is an IP address' do
- let(:import_url) { 'https://93.184.216.34' }
-
- it_behaves_like 'validates URI and hostname' do
- let(:expected_uri) { import_url }
- let(:expected_hostname) { nil }
- let(:expected_use_proxy) { false }
- end
-
- it_behaves_like 'a URI denied by `deny_all_requests_except_allowed`'
-
- context 'when the address is invalid' do
- let(:import_url) { 'http://1.1.1.1.1' }
-
- it 'raises an error' do
- stub_env('RSPEC_ALLOW_INVALID_URLS', 'false')
-
- expect { subject }.to raise_error(Gitlab::HTTP_V2::UrlBlocker::BlockedUrlError)
- end
- end
- end
-
- context 'when DNS rebinding protection with IP allowed' do
- let(:import_url) { 'http://a.192.168.0.120.3times.127.0.0.1.1time.repeat.rebind.network:9121/scrape?target=unix:///var/opt/gitlab/redis/redis.socket&amp;check-keys=*' }
-
- before do
- stub_dns(import_url, ip_address: '192.168.0.120')
-
- allow(Gitlab::UrlBlockers::UrlAllowlist).to receive(:ip_allowed?).and_return(true)
- end
-
- it_behaves_like 'validates URI and hostname' do
- let(:expected_uri) { 'http://192.168.0.120:9121/scrape?target=unix:///var/opt/gitlab/redis/redis.socket&amp;check-keys=*' }
- let(:expected_hostname) { 'a.192.168.0.120.3times.127.0.0.1.1time.repeat.rebind.network' }
- let(:expected_use_proxy) { false }
- end
-
- it_behaves_like 'a URI exempt from `deny_all_requests_except_allowed`'
-
- context 'with HTTP_PROXY' do
- before do
- stub_env('http_proxy', 'http://proxy.example.com')
- end
-
- it_behaves_like 'validates URI and hostname' do
- let(:expected_uri) { import_url }
- let(:expected_hostname) { nil }
- let(:expected_use_proxy) { true }
- end
-
- context 'when domain is in no_proxy env' do
- before do
- stub_env('no_proxy', 'a.192.168.0.120.3times.127.0.0.1.1time.repeat.rebind.network')
- end
-
- it_behaves_like 'validates URI and hostname' do
- let(:expected_uri) { 'http://192.168.0.120:9121/scrape?target=unix:///var/opt/gitlab/redis/redis.socket&amp;check-keys=*' }
- let(:expected_hostname) { 'a.192.168.0.120.3times.127.0.0.1.1time.repeat.rebind.network' }
- let(:expected_use_proxy) { false }
- end
- end
- end
- end
-
- context 'with disabled DNS rebinding protection' do
- let(:options) { { dns_rebind_protection: false, schemes: schemes } }
-
- context 'when URI is internal' do
- let(:import_url) { 'http://localhost' }
-
- it_behaves_like 'validates URI and hostname' do
- let(:expected_uri) { import_url }
- let(:expected_hostname) { nil }
- let(:expected_use_proxy) { false }
- end
-
- it_behaves_like 'a URI exempt from `deny_all_requests_except_allowed`'
- end
-
- context 'when the URL hostname is a domain' do
- let(:import_url) { 'https://example.org' }
-
- before do
- stub_env('RSPEC_ALLOW_INVALID_URLS', 'false')
- end
-
- context 'when domain can be resolved' do
- it_behaves_like 'validates URI and hostname' do
- let(:expected_uri) { import_url }
- let(:expected_hostname) { nil }
- let(:expected_use_proxy) { false }
- end
-
- it_behaves_like 'a URI denied by `deny_all_requests_except_allowed`'
- end
-
- context 'when domain cannot be resolved' do
- let(:import_url) { 'http://foobar.x' }
-
- it_behaves_like 'validates URI and hostname' do
- let(:expected_uri) { import_url }
- let(:expected_hostname) { nil }
- let(:expected_use_proxy) { false }
- end
-
- it_behaves_like 'a URI denied by `deny_all_requests_except_allowed`'
- end
- end
-
- context 'when the URL hostname is an IP address' do
- let(:import_url) { 'https://93.184.216.34' }
-
- it_behaves_like 'validates URI and hostname' do
- let(:expected_uri) { import_url }
- let(:expected_hostname) { nil }
- let(:expected_use_proxy) { false }
- end
-
- it_behaves_like 'a URI denied by `deny_all_requests_except_allowed`'
-
- context 'when it is invalid' do
- let(:import_url) { 'http://1.1.1.1.1' }
-
- it_behaves_like 'validates URI and hostname' do
- let(:expected_uri) { import_url }
- let(:expected_hostname) { nil }
- let(:expected_use_proxy) { false }
- end
-
- it_behaves_like 'a URI denied by `deny_all_requests_except_allowed`'
- end
- end
- end
- end
-
- describe '#blocked_url?' do
- let(:ports) { Project::VALID_IMPORT_PORTS }
-
- it 'allows imports from configured web host and port' do
- import_url = "http://#{Gitlab.host_with_port}/t.git"
- expect(described_class.blocked_url?(import_url, schemes: schemes)).to be false
- end
-
- it 'allows mirroring from configured SSH host and port' do
- import_url = "ssh://#{Gitlab.config.gitlab_shell.ssh_host}:#{Gitlab.config.gitlab_shell.ssh_port}/t.git"
- expect(described_class.blocked_url?(import_url, schemes: schemes)).to be false
- end
-
- it 'returns true for bad localhost hostname' do
- expect(described_class.blocked_url?('https://localhost:65535/foo/foo.git', schemes: schemes)).to be true
- end
-
- it 'returns true for bad port' do
- expect(described_class.blocked_url?('https://gitlab.com:25/foo/foo.git', ports: ports, schemes: schemes)).to be true
- end
-
- it 'returns true for bad scheme' do
- expect(described_class.blocked_url?('https://gitlab.com/foo/foo.git', schemes: ['https'])).to be false
- expect(described_class.blocked_url?('https://gitlab.com/foo/foo.git', schemes: ['http'])).to be true
- end
-
- it 'returns true for bad protocol on configured web/SSH host and ports' do
- web_url = "javascript://#{Gitlab.host_with_port}/t.git%0aalert(1)"
- expect(described_class.blocked_url?(web_url, schemes: schemes)).to be true
-
- ssh_url = "javascript://#{Gitlab.config.gitlab_shell.ssh_host}:#{Gitlab.config.gitlab_shell.ssh_port}/t.git%0aalert(1)"
- expect(described_class.blocked_url?(ssh_url, schemes: schemes)).to be true
- end
-
- it 'returns true for localhost IPs' do
- expect(described_class.blocked_url?('https://[0:0:0:0:0:0:0:0]/foo/foo.git', schemes: schemes)).to be true
- expect(described_class.blocked_url?('https://0.0.0.0/foo/foo.git', schemes: schemes)).to be true
- expect(described_class.blocked_url?('https://[::]/foo/foo.git', schemes: schemes)).to be true
- end
-
- it 'returns true for loopback IP' do
- expect(described_class.blocked_url?('https://127.0.0.2/foo/foo.git', schemes: schemes)).to be true
- expect(described_class.blocked_url?('https://127.0.0.1/foo/foo.git', schemes: schemes)).to be true
- expect(described_class.blocked_url?('https://[::1]/foo/foo.git', schemes: schemes)).to be true
- end
-
- it 'returns true for alternative version of 127.0.0.1 (0177.1)' do
- expect(described_class.blocked_url?('https://0177.1:65535/foo/foo.git', schemes: schemes)).to be true
- end
-
- it 'returns true for alternative version of 127.0.0.1 (017700000001)' do
- expect(described_class.blocked_url?('https://017700000001:65535/foo/foo.git', schemes: schemes)).to be true
- end
-
- it 'returns true for alternative version of 127.0.0.1 (0x7f.1)' do
- expect(described_class.blocked_url?('https://0x7f.1:65535/foo/foo.git', schemes: schemes)).to be true
- end
-
- it 'returns true for alternative version of 127.0.0.1 (0x7f.0.0.1)' do
- expect(described_class.blocked_url?('https://0x7f.0.0.1:65535/foo/foo.git', schemes: schemes)).to be true
- end
-
- it 'returns true for alternative version of 127.0.0.1 (0x7f000001)' do
- expect(described_class.blocked_url?('https://0x7f000001:65535/foo/foo.git', schemes: schemes)).to be true
- end
-
- it 'returns true for alternative version of 127.0.0.1 (2130706433)' do
- expect(described_class.blocked_url?('https://2130706433:65535/foo/foo.git', schemes: schemes)).to be true
- end
-
- it 'returns true for alternative version of 127.0.0.1 (127.000.000.001)' do
- expect(described_class.blocked_url?('https://127.000.000.001:65535/foo/foo.git', schemes: schemes)).to be true
- end
-
- it 'returns true for alternative version of 127.0.0.1 (127.0.1)' do
- expect(described_class.blocked_url?('https://127.0.1:65535/foo/foo.git', schemes: schemes)).to be true
- end
-
- context 'with ipv6 mapped address' do
- it 'returns true for localhost IPs' do
- expect(described_class.blocked_url?('https://[0:0:0:0:0:ffff:0.0.0.0]/foo/foo.git', schemes: schemes)).to be true
- expect(described_class.blocked_url?('https://[::ffff:0.0.0.0]/foo/foo.git', schemes: schemes)).to be true
- expect(described_class.blocked_url?('https://[::ffff:0:0]/foo/foo.git', schemes: schemes)).to be true
- end
-
- it 'returns true for loopback IPs' do
- expect(described_class.blocked_url?('https://[0:0:0:0:0:ffff:127.0.0.1]/foo/foo.git', schemes: schemes)).to be true
- expect(described_class.blocked_url?('https://[::ffff:127.0.0.1]/foo/foo.git', schemes: schemes)).to be true
- expect(described_class.blocked_url?('https://[::ffff:7f00:1]/foo/foo.git', schemes: schemes)).to be true
- expect(described_class.blocked_url?('https://[0:0:0:0:0:ffff:127.0.0.2]/foo/foo.git', schemes: schemes)).to be true
- expect(described_class.blocked_url?('https://[::ffff:127.0.0.2]/foo/foo.git', schemes: schemes)).to be true
- expect(described_class.blocked_url?('https://[::ffff:7f00:2]/foo/foo.git', schemes: schemes)).to be true
- end
- end
-
- it 'returns true for a non-alphanumeric hostname' do
- aggregate_failures do
- expect(described_class).to be_blocked_url('ssh://-oProxyCommand=whoami/a', schemes: ['ssh'])
-
- # The leading character here is a Unicode "soft hyphen"
- expect(described_class).to be_blocked_url('ssh://­oProxyCommand=whoami/a', schemes: ['ssh'])
-
- # Unicode alphanumerics are allowed
- expect(described_class).not_to be_blocked_url('ssh://ğitlab.com/a', schemes: ['ssh'])
- end
- end
-
- it 'returns true for invalid URL' do
- expect(described_class.blocked_url?('http://:8080', schemes: schemes)).to be true
- end
-
- it 'returns false for legitimate URL' do
- expect(described_class.blocked_url?('https://gitlab.com/foo/foo.git', schemes: schemes)).to be false
- end
-
- describe 'allow_local_network' do
- let(:shared_address_space_ips) { ['100.64.0.0', '100.64.127.127', '100.64.255.255'] }
-
- let(:local_ips) do
- [
- '192.168.1.2',
- '[0:0:0:0:0:ffff:192.168.1.2]',
- '[::ffff:c0a8:102]',
- '10.0.0.2',
- '[0:0:0:0:0:ffff:10.0.0.2]',
- '[::ffff:a00:2]',
- '172.16.0.2',
- '[0:0:0:0:0:ffff:172.16.0.2]',
- '[::ffff:ac10:20]',
- '[feef::1]',
- '[fee2::]',
- '[fc00:bf8b:e62c:abcd:abcd:aaaa:aaaa:aaaa]',
- *shared_address_space_ips
- ]
- end
-
- let(:limited_broadcast_address_variants) do
- [
- '255.255.255.255', # "normal" dotted decimal
- '0377.0377.0377.0377', # Octal
- '0377.00000000377.00377.0000377', # Still octal
- '0xff.0xff.0xff.0xff', # hex
- '0xffffffff', # still hex
- '0xBaaaaaaaaaaaaaaaaffffffff', # padded hex
- '255.255.255.255:65535', # with a port
- '4294967295', # as an integer / dword
- '[::ffff:ffff:ffff]', # short IPv6
- '[0000:0000:0000:0000:0000:ffff:ffff:ffff]' # long IPv6
- ]
- end
-
- let(:fake_domain) { 'www.fakedomain.fake' }
-
- shared_examples 'allows local requests' do |url_blocker_attributes|
- it 'does not block urls from private networks' do
- local_ips.each do |ip|
- stub_domain_resolv(fake_domain, ip) do
- expect(described_class).not_to be_blocked_url("http://#{fake_domain}", **url_blocker_attributes)
- end
-
- expect(described_class).not_to be_blocked_url("http://#{ip}", **url_blocker_attributes)
- end
- end
-
- it 'allows localhost endpoints' do
- expect(described_class).not_to be_blocked_url('http://0.0.0.0', **url_blocker_attributes)
- expect(described_class).not_to be_blocked_url('http://localhost', **url_blocker_attributes)
- expect(described_class).not_to be_blocked_url('http://127.0.0.1', **url_blocker_attributes)
- end
-
- it 'allows loopback endpoints' do
- expect(described_class).not_to be_blocked_url('http://127.0.0.2', **url_blocker_attributes)
- end
-
- it 'allows IPv4 link-local endpoints' do
- expect(described_class).not_to be_blocked_url('http://169.254.169.254', **url_blocker_attributes)
- expect(described_class).not_to be_blocked_url('http://169.254.168.100', **url_blocker_attributes)
- end
-
- it 'allows IPv6 link-local endpoints' do
- expect(described_class).not_to be_blocked_url('http://[0:0:0:0:0:ffff:169.254.169.254]', **url_blocker_attributes)
- expect(described_class).not_to be_blocked_url('http://[::ffff:169.254.169.254]', **url_blocker_attributes)
- expect(described_class).not_to be_blocked_url('http://[::ffff:a9fe:a9fe]', **url_blocker_attributes)
- expect(described_class).not_to be_blocked_url('http://[0:0:0:0:0:ffff:169.254.168.100]', **url_blocker_attributes)
- expect(described_class).not_to be_blocked_url('http://[::ffff:169.254.168.100]', **url_blocker_attributes)
- expect(described_class).not_to be_blocked_url('http://[::ffff:a9fe:a864]', **url_blocker_attributes)
- expect(described_class).not_to be_blocked_url('http://[fe80::c800:eff:fe74:8]', **url_blocker_attributes)
- end
-
- it 'allows limited broadcast address 255.255.255.255 and variants' do
- limited_broadcast_address_variants.each do |variant|
- expect(described_class).not_to be_blocked_url("https://#{variant}", **url_blocker_attributes), "Expected #{variant} to be allowed"
- end
- end
- end
-
- context 'when true (default)' do
- it_behaves_like 'allows local requests', { allow_localhost: true, allow_local_network: true, schemes: %w[http https] }
- end
-
- context 'when false' do
- it 'blocks urls from private networks' do
- local_ips.each do |ip|
- stub_domain_resolv(fake_domain, ip) do
- expect(described_class).to be_blocked_url("http://#{fake_domain}", allow_local_network: false, schemes: schemes)
- end
-
- expect(described_class).to be_blocked_url("http://#{ip}", allow_local_network: false, schemes: schemes)
- end
- end
-
- it 'blocks IPv4 link-local endpoints' do
- expect(described_class).to be_blocked_url('http://169.254.169.254', allow_local_network: false, schemes: schemes)
- expect(described_class).to be_blocked_url('http://169.254.168.100', allow_local_network: false, schemes: schemes)
- end
-
- it 'blocks IPv6 link-local endpoints' do
- expect(described_class).to be_blocked_url('http://[0:0:0:0:0:ffff:169.254.169.254]', allow_local_network: false, schemes: schemes)
- expect(described_class).to be_blocked_url('http://[::ffff:169.254.169.254]', allow_local_network: false, schemes: schemes)
- expect(described_class).to be_blocked_url('http://[::ffff:a9fe:a9fe]', allow_local_network: false, schemes: schemes)
- expect(described_class).to be_blocked_url('http://[0:0:0:0:0:ffff:169.254.168.100]', allow_local_network: false, schemes: schemes)
- expect(described_class).to be_blocked_url('http://[::ffff:169.254.168.100]', allow_local_network: false, schemes: schemes)
- expect(described_class).to be_blocked_url('http://[::ffff:a9fe:a864]', allow_local_network: false, schemes: schemes)
- expect(described_class).to be_blocked_url('http://[fe80::c800:eff:fe74:8]', allow_local_network: false, schemes: schemes)
- end
-
- it 'blocks limited broadcast address 255.255.255.255 and variants' do
- # Raise BlockedUrlError for invalid URLs.
- # The padded hex version, for example, is a valid URL on Mac but
- # not on Ubuntu.
- stub_env('RSPEC_ALLOW_INVALID_URLS', 'false')
-
- limited_broadcast_address_variants.each do |variant|
- expect(described_class).to be_blocked_url("https://#{variant}", allow_local_network: false, schemes: schemes), "Expected #{variant} to be blocked"
- end
- end
-
- context 'when local domain/IP is allowed' do
- let(:url_blocker_attributes) do
- {
- allow_localhost: false,
- allow_local_network: false,
- schemes: schemes
- }
- end
-
- before do
- allow(ApplicationSetting).to receive(:current).and_return(ApplicationSetting.new)
- stub_application_setting(outbound_local_requests_whitelist: allowlist)
- end
-
- context 'with IPs in allowlist' do
- let(:allowlist) do
- [
- '0.0.0.0',
- '127.0.0.1',
- '127.0.0.2',
- '192.168.1.1',
- *local_ips,
- '0:0:0:0:0:ffff:169.254.169.254',
- '::ffff:a9fe:a9fe',
- '::ffff:169.254.168.100',
- '::ffff:a9fe:a864',
- 'fe80::c800:eff:fe74:8',
- '255.255.255.255',
-
- # garbage IPs
- '45645632345',
- 'garbage456:more345gar:bage'
- ]
- end
-
- it_behaves_like 'allows local requests', { allow_localhost: false, allow_local_network: false, schemes: %w[http https] }
-
- it 'allows IP when dns_rebind_protection is disabled' do
- url = "http://example.com"
- attrs = url_blocker_attributes.merge(dns_rebind_protection: false)
-
- stub_domain_resolv('example.com', '192.168.1.2') do
- expect(described_class).not_to be_blocked_url(url, **attrs)
- end
-
- stub_domain_resolv('example.com', '192.168.1.3') do
- expect(described_class).to be_blocked_url(url, **attrs)
- end
- end
-
- it 'allows the limited broadcast address 255.255.255.255' do
- expect(described_class).not_to be_blocked_url('http://255.255.255.255', **url_blocker_attributes)
- end
- end
-
- context 'with domains in allowlist' do
- let(:allowlist) do
- [
- 'www.example.com',
- 'example.com',
- 'xn--itlab-j1a.com',
- 'garbage$^$%#$^&$'
- ]
- end
-
- it 'allows domains present in allowlist' do
- domain = 'example.com'
- subdomain1 = 'www.example.com'
- subdomain2 = 'subdomain.example.com'
-
- stub_domain_resolv(domain, '192.168.1.1') do
- expect(described_class).not_to be_blocked_url("http://#{domain}",
- **url_blocker_attributes)
- end
-
- stub_domain_resolv(subdomain1, '192.168.1.1') do
- expect(described_class).not_to be_blocked_url("http://#{subdomain1}",
- **url_blocker_attributes)
- end
-
- # subdomain2 is not part of the allowlist so it should be blocked
- stub_domain_resolv(subdomain2, '192.168.1.1') do
- expect(described_class).to be_blocked_url("http://#{subdomain2}",
- **url_blocker_attributes)
- end
- end
-
- it 'works with unicode and idna encoded domains' do
- unicode_domain = 'ğitlab.com'
- idna_encoded_domain = 'xn--itlab-j1a.com'
-
- stub_domain_resolv(unicode_domain, '192.168.1.1') do
- expect(described_class).not_to be_blocked_url("http://#{unicode_domain}",
- **url_blocker_attributes)
- end
-
- stub_domain_resolv(idna_encoded_domain, '192.168.1.1') do
- expect(described_class).not_to be_blocked_url("http://#{idna_encoded_domain}",
- **url_blocker_attributes)
- end
- end
-
- shared_examples 'dns rebinding checks' do
- shared_examples 'allowlists the domain' do
- let(:allowlist) { [domain] }
- let(:url) { "http://#{domain}" }
-
- before do
- stub_env('RSPEC_ALLOW_INVALID_URLS', 'false')
- end
-
- it do
- expect(described_class).not_to be_blocked_url(url, dns_rebind_protection: dns_rebind_value, schemes: schemes)
- end
- end
-
- describe 'dns_rebinding_setting' do
- context 'when enabled' do
- let(:dns_rebind_value) { true }
-
- it_behaves_like 'allowlists the domain'
- end
-
- context 'when disabled' do
- let(:dns_rebind_value) { false }
-
- it_behaves_like 'allowlists the domain'
- end
- end
- end
-
- context 'when the domain cannot be resolved' do
- let(:domain) { 'foobar.x' }
-
- it_behaves_like 'dns rebinding checks'
- end
-
- context 'when the domain can be resolved' do
- let(:domain) { 'example.com' }
-
- before do
- stub_dns(url, ip_address: '93.184.216.34')
- end
-
- it_behaves_like 'dns rebinding checks'
- end
- end
-
- context 'with ports' do
- let(:allowlist) do
- ["127.0.0.1:2000"]
- end
-
- it 'allows domain with port when resolved ip has port allowed' do
- stub_domain_resolv("www.resolve-domain.com", '127.0.0.1') do
- expect(described_class).not_to be_blocked_url("http://www.resolve-domain.com:2000", **url_blocker_attributes)
- end
- end
- end
- end
- end
- end
-
- describe 'enforce_user' do
- context 'when false (default)' do
- it 'does not block urls with a non-alphanumeric username' do
- expect(described_class).not_to be_blocked_url('ssh://-oProxyCommand=whoami@example.com/a', schemes: ['ssh'])
-
- # The leading character here is a Unicode "soft hyphen"
- expect(described_class).not_to be_blocked_url('ssh://­oProxyCommand=whoami@example.com/a', schemes: ['ssh'])
-
- # Unicode alphanumerics are allowed
- expect(described_class).not_to be_blocked_url('ssh://ğitlab@example.com/a', schemes: ['ssh'])
- end
- end
-
- context 'when true' do
- it 'blocks urls with a non-alphanumeric username' do
- aggregate_failures do
- expect(described_class).to be_blocked_url('ssh://-oProxyCommand=whoami@example.com/a', enforce_user: true, schemes: ['ssh'])
-
- # The leading character here is a Unicode "soft hyphen"
- expect(described_class).to be_blocked_url('ssh://­oProxyCommand=whoami@example.com/a', enforce_user: true, schemes: ['ssh'])
-
- # Unicode alphanumerics are allowed
- expect(described_class).not_to be_blocked_url('ssh://ğitlab@example.com/a', enforce_user: true, schemes: ['ssh'])
- end
- end
- end
- end
-
- context 'when ascii_only is true' do
- it 'returns true for unicode domain' do
- expect(described_class.blocked_url?('https://𝕘itⅼαƄ.com/foo/foo.bar', ascii_only: true, schemes: schemes)).to be true
- end
-
- it 'returns true for unicode tld' do
- expect(described_class.blocked_url?('https://gitlab.ᴄοm/foo/foo.bar', ascii_only: true, schemes: schemes)).to be true
- end
-
- it 'returns true for unicode path' do
- expect(described_class.blocked_url?('https://gitlab.com/𝒇οο/𝒇οο.Ƅαꮁ', ascii_only: true, schemes: schemes)).to be true
- end
-
- it 'returns true for IDNA deviations' do
- expect(described_class.blocked_url?('https://mißile.com/foo/foo.bar', ascii_only: true, schemes: schemes)).to be true
- expect(described_class.blocked_url?('https://miςςile.com/foo/foo.bar', ascii_only: true, schemes: schemes)).to be true
- expect(described_class.blocked_url?('https://git‍lab.com/foo/foo.bar', ascii_only: true, schemes: schemes)).to be true
- expect(described_class.blocked_url?('https://git‌lab.com/foo/foo.bar', ascii_only: true, schemes: schemes)).to be true
- end
- end
-
- it 'blocks urls with invalid ip address' do
- stub_env('RSPEC_ALLOW_INVALID_URLS', 'false')
-
- expect(described_class).to be_blocked_url('http://8.8.8.8.8', schemes: schemes)
- end
-
- it 'blocks urls whose hostname cannot be resolved' do
- stub_env('RSPEC_ALLOW_INVALID_URLS', 'false')
-
- expect(described_class).to be_blocked_url('http://foobar.x', schemes: schemes)
- end
-
- context 'when gitlab is running on a non-default port' do
- let(:gitlab_port) { 3000 }
-
- before do
- stub_config(gitlab: { protocol: 'http', host: 'gitlab.local', port: gitlab_port })
- end
-
- it 'returns true for url targeting the wrong port' do
- stub_domain_resolv('gitlab.local', '127.0.0.1') do
- expect(described_class).to be_blocked_url("http://gitlab.local/foo", schemes: schemes)
- end
- end
-
- it 'does not block url on gitlab port' do
- stub_domain_resolv('gitlab.local', '127.0.0.1') do
- expect(described_class).not_to be_blocked_url("http://gitlab.local:#{gitlab_port}/foo", schemes: schemes)
- end
- end
- end
-
- def stub_domain_resolv(domain, ip, port = 80)
- address = instance_double(Addrinfo,
- ip_address: ip,
- ipv4_private?: true,
- ipv6_linklocal?: false,
- ipv4_loopback?: false,
- ipv6_loopback?: false,
- ipv4?: false,
- ip_port: port
- )
- allow(Addrinfo).to receive(:getaddrinfo).with(domain, port, any_args).and_return([address])
- allow(address).to receive(:ipv6_v4mapped?).and_return(false)
-
- yield
-
- allow(Addrinfo).to receive(:getaddrinfo).and_call_original
- end
- end
-
- describe '#validate_hostname' do
- let(:ip_addresses) do
- [
- '2001:db8:1f70::999:de8:7648:6e8',
- 'FE80::C800:EFF:FE74:8',
- '::ffff:127.0.0.1',
- '::ffff:169.254.168.100',
- '::ffff:7f00:1',
- '0:0:0:0:0:ffff:0.0.0.0',
- 'localhost',
- '127.0.0.1',
- '127.000.000.001',
- '0x7f000001',
- '0x7f.0.0.1',
- '0x7f.0.0.1',
- '017700000001',
- '0177.1',
- '2130706433',
- '::',
- '::1'
- ]
- end
-
- it 'does not raise error for valid Ip addresses' do
- ip_addresses.each do |ip|
- expect { described_class.send(:validate_hostname, ip) }.not_to raise_error
- end
- end
- end
-end