483 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Ruby
		
	
	
	
			
		
		
	
	
			483 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Ruby
		
	
	
	
| # frozen_string_literal: true
 | |
| 
 | |
| require 'spec_helper'
 | |
| 
 | |
| RSpec.describe Gitlab::Utils do
 | |
|   using RSpec::Parameterized::TableSyntax
 | |
| 
 | |
|   delegate :to_boolean, :boolean_to_yes_no, :slugify, :random_string, :which,
 | |
|            :ensure_array_from_string, :to_exclusive_sentence, :bytes_to_megabytes,
 | |
|            :append_path, :check_path_traversal!, :allowlisted?, :check_allowed_absolute_path!, :decode_path, :ms_to_round_sec, to: :described_class
 | |
| 
 | |
|   describe '.check_path_traversal!' do
 | |
|     it 'detects path traversal in string without any separators' do
 | |
|       expect { check_path_traversal!('.') }.to raise_error(/Invalid path/)
 | |
|       expect { check_path_traversal!('..') }.to raise_error(/Invalid path/)
 | |
|     end
 | |
| 
 | |
|     it 'detects path traversal at the start of the string' do
 | |
|       expect { check_path_traversal!('../foo') }.to raise_error(/Invalid path/)
 | |
|       expect { check_path_traversal!('..\\foo') }.to raise_error(/Invalid path/)
 | |
|     end
 | |
| 
 | |
|     it 'detects path traversal at the start of the string, even to just the subdirectory' do
 | |
|       expect { check_path_traversal!('../') }.to raise_error(/Invalid path/)
 | |
|       expect { check_path_traversal!('..\\') }.to raise_error(/Invalid path/)
 | |
|       expect { check_path_traversal!('/../') }.to raise_error(/Invalid path/)
 | |
|       expect { check_path_traversal!('\\..\\') }.to raise_error(/Invalid path/)
 | |
|     end
 | |
| 
 | |
|     it 'detects path traversal in the middle of the string' do
 | |
|       expect { check_path_traversal!('foo/../../bar') }.to raise_error(/Invalid path/)
 | |
|       expect { check_path_traversal!('foo\\..\\..\\bar') }.to raise_error(/Invalid path/)
 | |
|       expect { check_path_traversal!('foo/..\\bar') }.to raise_error(/Invalid path/)
 | |
|       expect { check_path_traversal!('foo\\../bar') }.to raise_error(/Invalid path/)
 | |
|       expect { check_path_traversal!('foo/..\\..\\..\\..\\../bar') }.to raise_error(/Invalid path/)
 | |
|     end
 | |
| 
 | |
|     it 'detects path traversal at the end of the string when slash-terminates' do
 | |
|       expect { check_path_traversal!('foo/../') }.to raise_error(/Invalid path/)
 | |
|       expect { check_path_traversal!('foo\\..\\') }.to raise_error(/Invalid path/)
 | |
|     end
 | |
| 
 | |
|     it 'detects path traversal at the end of the string' do
 | |
|       expect { check_path_traversal!('foo/..') }.to raise_error(/Invalid path/)
 | |
|       expect { check_path_traversal!('foo\\..') }.to raise_error(/Invalid path/)
 | |
|     end
 | |
| 
 | |
|     it 'does nothing for a safe string' do
 | |
|       expect(check_path_traversal!('./foo')).to eq('./foo')
 | |
|       expect(check_path_traversal!('.test/foo')).to eq('.test/foo')
 | |
|       expect(check_path_traversal!('..test/foo')).to eq('..test/foo')
 | |
|       expect(check_path_traversal!('dir/..foo.rb')).to eq('dir/..foo.rb')
 | |
|       expect(check_path_traversal!('dir/.foo.rb')).to eq('dir/.foo.rb')
 | |
|     end
 | |
| 
 | |
|     it 'does nothing for a non-string' do
 | |
|       expect(check_path_traversal!(nil)).to be_nil
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.allowlisted?' do
 | |
|     let(:allowed_paths) { ['/home/foo', '/foo/bar', '/etc/passwd']}
 | |
| 
 | |
|     it 'returns true if path is allowed' do
 | |
|       expect(allowlisted?('/foo/bar', allowed_paths)).to be(true)
 | |
|     end
 | |
| 
 | |
|     it 'returns false if path is not allowed' do
 | |
|       expect(allowlisted?('/test/test', allowed_paths)).to be(false)
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.check_allowed_absolute_path!' do
 | |
|     let(:allowed_paths) { ['/home/foo'] }
 | |
| 
 | |
|     it 'raises an exception if an absolute path is not allowed' do
 | |
|       expect { check_allowed_absolute_path!('/etc/passwd', allowed_paths) }.to raise_error(StandardError)
 | |
|     end
 | |
| 
 | |
|     it 'does nothing for an allowed absolute path' do
 | |
|       expect(check_allowed_absolute_path!('/home/foo', allowed_paths)).to be_nil
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.decode_path' do
 | |
|     it 'returns path unencoded for singled-encoded paths' do
 | |
|       expect(decode_path('%2Fhome%2Fbar%3Fasd%3Dqwe')).to eq('/home/bar?asd=qwe')
 | |
|     end
 | |
| 
 | |
|     it 'returns path when it is unencoded' do
 | |
|       expect(decode_path('/home/bar?asd=qwe')).to eq('/home/bar?asd=qwe')
 | |
|     end
 | |
| 
 | |
|     [
 | |
|       '..%252F..%252F..%252Fetc%252Fpasswd',
 | |
|       '%25252Fresult%25252Fchosennickname%25253D%252522jj%252522'
 | |
|     ].each do |multiple_encoded_path|
 | |
|       it 'raises an exception when the path is multiple-encoded' do
 | |
|         expect { decode_path(multiple_encoded_path) }.to raise_error(/path #{multiple_encoded_path} is not allowed/)
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.slugify' do
 | |
|     {
 | |
|       'TEST' => 'test',
 | |
|       'project_with_underscores' => 'project-with-underscores',
 | |
|       'namespace/project' =>  'namespace-project',
 | |
|       'a' * 70 => 'a' * 63,
 | |
|       'test_trailing_' => 'test-trailing'
 | |
|     }.each do |original, expected|
 | |
|       it "slugifies #{original} to #{expected}" do
 | |
|         expect(slugify(original)).to eq(expected)
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.ms_to_round_sec' do
 | |
|     using RSpec::Parameterized::TableSyntax
 | |
| 
 | |
|     where(:original, :expected) do
 | |
|       1999.8999     | 1.9999
 | |
|       12384         | 12.384
 | |
|       333           | 0.333
 | |
|       1333.33333333 | 1.333333
 | |
|     end
 | |
| 
 | |
|     with_them do
 | |
|       it "returns rounded seconds" do
 | |
|         expect(ms_to_round_sec(original)).to eq(expected)
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.to_exclusive_sentence' do
 | |
|     it 'calls #to_sentence on the array' do
 | |
|       array = double
 | |
| 
 | |
|       expect(array).to receive(:to_sentence)
 | |
| 
 | |
|       to_exclusive_sentence(array)
 | |
|     end
 | |
| 
 | |
|     it 'joins arrays with two elements correctly' do
 | |
|       array = %w(foo bar)
 | |
| 
 | |
|       expect(to_exclusive_sentence(array)).to eq('foo or bar')
 | |
|     end
 | |
| 
 | |
|     it 'joins arrays with more than two elements correctly' do
 | |
|       array = %w(foo bar baz)
 | |
| 
 | |
|       expect(to_exclusive_sentence(array)).to eq('foo, bar, or baz')
 | |
|     end
 | |
| 
 | |
|     it 'localizes the connector words' do
 | |
|       array = %w(foo bar baz)
 | |
| 
 | |
|       expect(described_class).to receive(:_).with(' or ').and_return(' <1> ')
 | |
|       expect(described_class).to receive(:_).with(', or ').and_return(', <2> ')
 | |
|       expect(to_exclusive_sentence(array)).to eq('foo, bar, <2> baz')
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.nlbr' do
 | |
|     it 'replaces new lines with <br>' do
 | |
|       expect(described_class.nlbr("<b>hello</b>\n<i>world</i>".freeze)).to eq("hello<br>world")
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.remove_line_breaks' do
 | |
|     using RSpec::Parameterized::TableSyntax
 | |
| 
 | |
|     where(:original, :expected) do
 | |
|       "foo\nbar\nbaz"     | "foobarbaz"
 | |
|       "foo\r\nbar\r\nbaz" | "foobarbaz"
 | |
|       "foobar"            | "foobar"
 | |
|     end
 | |
| 
 | |
|     with_them do
 | |
|       it "replace line breaks with an empty string" do
 | |
|         expect(described_class.remove_line_breaks(original)).to eq(expected)
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.to_boolean' do
 | |
|     it 'accepts booleans' do
 | |
|       expect(to_boolean(true)).to be(true)
 | |
|       expect(to_boolean(false)).to be(false)
 | |
|     end
 | |
| 
 | |
|     it 'converts a valid value to a boolean' do
 | |
|       expect(to_boolean(true)).to be(true)
 | |
|       expect(to_boolean('true')).to be(true)
 | |
|       expect(to_boolean('YeS')).to be(true)
 | |
|       expect(to_boolean('t')).to be(true)
 | |
|       expect(to_boolean('1')).to be(true)
 | |
|       expect(to_boolean('ON')).to be(true)
 | |
| 
 | |
|       expect(to_boolean('FaLse')).to be(false)
 | |
|       expect(to_boolean('F')).to be(false)
 | |
|       expect(to_boolean('NO')).to be(false)
 | |
|       expect(to_boolean('n')).to be(false)
 | |
|       expect(to_boolean('0')).to be(false)
 | |
|       expect(to_boolean('oFF')).to be(false)
 | |
|     end
 | |
| 
 | |
|     it 'converts an invalid value to nil' do
 | |
|       expect(to_boolean('fals')).to be_nil
 | |
|       expect(to_boolean('yeah')).to be_nil
 | |
|       expect(to_boolean('')).to be_nil
 | |
|       expect(to_boolean(nil)).to be_nil
 | |
|     end
 | |
| 
 | |
|     it 'accepts a default value, and does not return it when a valid value is given' do
 | |
|       expect(to_boolean(true, default: false)).to be(true)
 | |
|       expect(to_boolean('true', default: false)).to be(true)
 | |
|       expect(to_boolean('YeS', default: false)).to be(true)
 | |
|       expect(to_boolean('t', default: false)).to be(true)
 | |
|       expect(to_boolean('1', default: 'any value')).to be(true)
 | |
|       expect(to_boolean('ON', default: 42)).to be(true)
 | |
| 
 | |
|       expect(to_boolean('FaLse', default: true)).to be(false)
 | |
|       expect(to_boolean('F', default: true)).to be(false)
 | |
|       expect(to_boolean('NO', default: true)).to be(false)
 | |
|       expect(to_boolean('n', default: true)).to be(false)
 | |
|       expect(to_boolean('0', default: 'any value')).to be(false)
 | |
|       expect(to_boolean('oFF', default: 42)).to be(false)
 | |
|     end
 | |
| 
 | |
|     it 'accepts a default value, and returns it when an invalid value is given' do
 | |
|       expect(to_boolean('fals', default: true)).to eq(true)
 | |
|       expect(to_boolean('yeah', default: false)).to eq(false)
 | |
|       expect(to_boolean('', default: 'any value')).to eq('any value')
 | |
|       expect(to_boolean(nil, default: 42)).to eq(42)
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.boolean_to_yes_no' do
 | |
|     it 'converts booleans to Yes or No' do
 | |
|       expect(boolean_to_yes_no(true)).to eq('Yes')
 | |
|       expect(boolean_to_yes_no(false)).to eq('No')
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.random_string' do
 | |
|     it 'generates a string' do
 | |
|       expect(random_string).to be_kind_of(String)
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.which' do
 | |
|     it 'finds the full path to an executable binary' do
 | |
|       expect(File).to receive(:executable?).with('/bin/sh').and_return(true)
 | |
| 
 | |
|       expect(which('sh', 'PATH' => '/bin')).to eq('/bin/sh')
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.ensure_array_from_string' do
 | |
|     it 'returns the same array if given one' do
 | |
|       arr = ['a', 4, true, { test: 1 }]
 | |
| 
 | |
|       expect(ensure_array_from_string(arr)).to eq(arr)
 | |
|     end
 | |
| 
 | |
|     it 'turns comma-separated strings into arrays' do
 | |
|       str = 'seven, eight, 9, 10'
 | |
| 
 | |
|       expect(ensure_array_from_string(str)).to eq(%w[seven eight 9 10])
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.bytes_to_megabytes' do
 | |
|     it 'converts bytes to megabytes' do
 | |
|       bytes = 1.megabyte
 | |
| 
 | |
|       expect(bytes_to_megabytes(bytes)).to eq(1)
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.append_path' do
 | |
|     using RSpec::Parameterized::TableSyntax
 | |
| 
 | |
|     where(:host, :path, :result) do
 | |
|       'http://test/'  | '/foo/bar'  |  'http://test/foo/bar'
 | |
|       'http://test/'  | '//foo/bar' |  'http://test/foo/bar'
 | |
|       'http://test//' | '/foo/bar'  |  'http://test/foo/bar'
 | |
|       'http://test'   | 'foo/bar'   |  'http://test/foo/bar'
 | |
|       'http://test//' | ''          |  'http://test/'
 | |
|       'http://test//' | nil         |  'http://test/'
 | |
|       ''              | '/foo/bar'  |  '/foo/bar'
 | |
|       nil             | '/foo/bar'  |  '/foo/bar'
 | |
|     end
 | |
| 
 | |
|     with_them do
 | |
|       it 'makes sure there is only one slash as path separator' do
 | |
|         expect(append_path(host, path)).to eq(result)
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.ensure_utf8_size' do
 | |
|     context 'string is has less bytes than expected' do
 | |
|       it 'backfills string with null characters' do
 | |
|         transformed = described_class.ensure_utf8_size('a' * 10, bytes: 32)
 | |
| 
 | |
|         expect(transformed.bytesize).to eq 32
 | |
|         expect(transformed).to eq(('a' * 10) + ('0' * 22))
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     context 'string size is exactly the one that is expected' do
 | |
|       it 'returns original value' do
 | |
|         transformed = described_class.ensure_utf8_size('a' * 32, bytes: 32)
 | |
| 
 | |
|         expect(transformed).to eq 'a' * 32
 | |
|         expect(transformed.bytesize).to eq 32
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     context 'when string contains a few multi-byte UTF characters' do
 | |
|       it 'backfills string with null characters' do
 | |
|         transformed = described_class.ensure_utf8_size('❤' * 6, bytes: 32)
 | |
| 
 | |
|         expect(transformed).to eq '❤❤❤❤❤❤' + ('0' * 14)
 | |
|         expect(transformed.bytesize).to eq 32
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     context 'when string has multiple multi-byte UTF chars exceeding 32 bytes' do
 | |
|       it 'truncates string to 32 characters and backfills it if needed' do
 | |
|         transformed = described_class.ensure_utf8_size('❤' * 18, bytes: 32)
 | |
| 
 | |
|         expect(transformed).to eq(('❤' * 10) + ('0' * 2))
 | |
|         expect(transformed.bytesize).to eq 32
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.deep_indifferent_access' do
 | |
|     let(:hash) do
 | |
|       { "variables" => [{ "key" => "VAR1", "value" => "VALUE2" }] }
 | |
|     end
 | |
| 
 | |
|     subject { described_class.deep_indifferent_access(hash) }
 | |
| 
 | |
|     it 'allows to access hash keys with symbols' do
 | |
|       expect(subject[:variables]).to be_a(Array)
 | |
|     end
 | |
| 
 | |
|     it 'allows to access array keys with symbols' do
 | |
|       expect(subject[:variables].first[:key]).to eq('VAR1')
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.try_megabytes_to_bytes' do
 | |
|     context 'when the size can be converted to megabytes' do
 | |
|       it 'returns the size in megabytes' do
 | |
|         size = described_class.try_megabytes_to_bytes(1)
 | |
| 
 | |
|         expect(size).to eq(1.megabytes)
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     context 'when the size can not be converted to megabytes' do
 | |
|       it 'returns the input size' do
 | |
|         size = described_class.try_megabytes_to_bytes('foo')
 | |
| 
 | |
|         expect(size).to eq('foo')
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.string_to_ip_object' do
 | |
|     it 'returns nil when string is nil' do
 | |
|       expect(described_class.string_to_ip_object(nil)).to eq(nil)
 | |
|     end
 | |
| 
 | |
|     it 'returns nil when string is invalid IP' do
 | |
|       expect(described_class.string_to_ip_object('invalid ip')).to eq(nil)
 | |
|       expect(described_class.string_to_ip_object('')).to eq(nil)
 | |
|     end
 | |
| 
 | |
|     it 'returns IP object when string is valid IP' do
 | |
|       expect(described_class.string_to_ip_object('192.168.1.1')).to eq(IPAddr.new('192.168.1.1'))
 | |
|       expect(described_class.string_to_ip_object('::ffff:a9fe:a864')).to eq(IPAddr.new('::ffff:a9fe:a864'))
 | |
|       expect(described_class.string_to_ip_object('[::ffff:a9fe:a864]')).to eq(IPAddr.new('::ffff:a9fe:a864'))
 | |
|       expect(described_class.string_to_ip_object('127.0.0.0/28')).to eq(IPAddr.new('127.0.0.0/28'))
 | |
|       expect(described_class.string_to_ip_object('1:0:0:0:0:0:0:0/124')).to eq(IPAddr.new('1:0:0:0:0:0:0:0/124'))
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.parse_url' do
 | |
|     it 'returns Addressable::URI object' do
 | |
|       expect(described_class.parse_url('http://gitlab.com')).to be_instance_of(Addressable::URI)
 | |
|     end
 | |
| 
 | |
|     it 'returns nil when URI cannot be parsed' do
 | |
|       expect(described_class.parse_url('://gitlab.com')).to be nil
 | |
|     end
 | |
| 
 | |
|     it 'returns nil with invalid parameter' do
 | |
|       expect(described_class.parse_url(1)).to be nil
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe 'multiple_key_invert' do
 | |
|     it 'invert keys with array values' do
 | |
|       hash = {
 | |
|         dast: [:vulnerabilities_count, :scanned_resources_count],
 | |
|         sast: [:vulnerabilities_count]
 | |
|       }
 | |
|       expect(described_class.multiple_key_invert(hash)).to eq({
 | |
|         vulnerabilities_count: [:dast, :sast],
 | |
|         scanned_resources_count: [:dast]
 | |
|       })
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.stable_sort_by' do
 | |
|     subject(:sorted_list) { described_class.stable_sort_by(list) { |obj| obj[:priority] } }
 | |
| 
 | |
|     context 'when items have the same priority' do
 | |
|       let(:list) do
 | |
|         [
 | |
|           { name: 'obj 1', priority: 1 },
 | |
|           { name: 'obj 2', priority: 1 },
 | |
|           { name: 'obj 3', priority: 1 }
 | |
|         ]
 | |
|       end
 | |
| 
 | |
|       it 'does not change order in cases of ties' do
 | |
|         expect(sorted_list).to eq(list)
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     context 'when items have different priorities' do
 | |
|       let(:list) do
 | |
|         [
 | |
|           { name: 'obj 1', priority: 2 },
 | |
|           { name: 'obj 2', priority: 1 },
 | |
|           { name: 'obj 3', priority: 3 }
 | |
|         ]
 | |
|       end
 | |
| 
 | |
|       it 'sorts items like the regular sort_by' do
 | |
|         expect(sorted_list).to eq([
 | |
|           { name: 'obj 2', priority: 1 },
 | |
|           { name: 'obj 1', priority: 2 },
 | |
|           { name: 'obj 3', priority: 3 }
 | |
|         ])
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '.valid_brackets?' do
 | |
|     where(:input, :allow_nested, :valid) do
 | |
|       'no brackets'              | true  | true
 | |
|       'no brackets'              | false | true
 | |
|       'user[avatar]'             | true  | true
 | |
|       'user[avatar]'             | false | true
 | |
|       'user[avatar][friends]'    | true  | true
 | |
|       'user[avatar][friends]'    | false | true
 | |
|       'user[avatar[image[url]]]' | true  | true
 | |
|       'user[avatar[image[url]]]' | false | false
 | |
|       'user[avatar[]friends]'    | true  | true
 | |
|       'user[avatar[]friends]'    | false | false
 | |
|       'user[avatar]]'            | true  | false
 | |
|       'user[avatar]]'            | false | false
 | |
|       'user][avatar]]'           | true  | false
 | |
|       'user][avatar]]'           | false | false
 | |
|       'user[avatar'              | true  | false
 | |
|       'user[avatar'              | false | false
 | |
|     end
 | |
| 
 | |
|     with_them do
 | |
|       it { expect(described_class.valid_brackets?(input, allow_nested: allow_nested)).to eq(valid) }
 | |
|     end
 | |
|   end
 | |
| end
 |