repository_spec.rb 82.4 KB
Newer Older
1 2
require 'spec_helper'

3
describe Repository do
4
  include RepoHelpers
5
  TestBlob = Struct.new(:path)
6

7
  let(:project) { create(:project, :repository) }
8
  let(:repository) { project.repository }
9
  let(:broken_repository) { create(:project, :broken_storage).repository }
10
  let(:user) { create(:user) }
11
  let(:git_user) { Gitlab::Git::User.from_gitlab(user) }
12

13
  let(:message) { 'Test message' }
14

Rubén Dávila's avatar
Rubén Dávila committed
15
  let(:merge_commit) do
16
    merge_request = create(:merge_request, source_branch: 'feature', target_branch: 'master', source_project: project)
17 18 19 20

    merge_commit_id = repository.merge(user,
                                       merge_request.diff_head_sha,
                                       merge_request,
21
                                       message)
22

23
    repository.commit(merge_commit_id)
24
  end
25

26 27
  let(:author_email) { 'user@example.org' }
  let(:author_name) { 'John Doe' }
28

29 30 31
  def expect_to_raise_storage_error
    expect { yield }.to raise_error do |exception|
      storage_exceptions = [Gitlab::Git::Storage::Inaccessible, Gitlab::Git::CommandError, GRPC::Unavailable]
32 33 34
      known_exception = storage_exceptions.select { |e| exception.is_a?(e) }

      expect(known_exception).not_to be_nil
35 36 37
    end
  end

38
  describe '#branch_names_contains' do
39 40 41
    shared_examples '#branch_names_contains' do
      set(:project) { create(:project, :repository) }
      let(:repository) { project.repository }
42

43
      subject { repository.branch_names_contains(sample_commit.id) }
44

45 46 47 48 49 50 51 52 53
      it { is_expected.to include('master') }
      it { is_expected.not_to include('feature') }
      it { is_expected.not_to include('fix') }

      describe 'when storage is broken', :broken_storage  do
        it 'should raise a storage error' do
          expect_to_raise_storage_error do
            broken_repository.branch_names_contains(sample_commit.id)
          end
54 55 56
        end
      end
    end
57 58 59 60 61 62 63 64

    context 'when gitaly is enabled' do
      it_behaves_like '#branch_names_contains'
    end

    context 'when gitaly is disabled', :skip_gitaly_mock do
      it_behaves_like '#branch_names_contains'
    end
65
  end
66

67
  describe '#tag_names_contains' do
68 69
    shared_examples '#tag_names_contains' do
      subject { repository.tag_names_contains(sample_commit.id) }
70

71 72 73 74 75 76 77
      it { is_expected.to include('v1.1.0') }
      it { is_expected.not_to include('v1.0.0') }
    end

    context 'when gitaly is enabled' do
      it_behaves_like '#tag_names_contains'
    end
78

79 80 81
    context 'when gitaly is enabled', :skip_gitaly_mock do
      it_behaves_like '#tag_names_contains'
    end
82 83
  end

84
  describe 'tags_sorted_by' do
haseeb's avatar
haseeb committed
85 86
    context 'name_desc' do
      subject { repository.tags_sorted_by('name_desc').map(&:name) }
87 88 89 90

      it { is_expected.to eq(['v1.1.0', 'v1.0.0']) }
    end

haseeb's avatar
haseeb committed
91 92 93 94 95 96
    context 'name_asc' do
      subject { repository.tags_sorted_by('name_asc').map(&:name) }

      it { is_expected.to eq(['v1.0.0', 'v1.1.0']) }
    end

97 98 99 100 101 102 103 104 105 106 107
    context 'updated' do
      let(:tag_a) { repository.find_tag('v1.0.0') }
      let(:tag_b) { repository.find_tag('v1.1.0') }

      context 'desc' do
        subject { repository.tags_sorted_by('updated_desc').map(&:name) }

        before do
          double_first = double(committed_date: Time.now)
          double_last = double(committed_date: Time.now - 1.second)

108 109
          allow(tag_a).to receive(:dereferenced_target).and_return(double_first)
          allow(tag_b).to receive(:dereferenced_target).and_return(double_last)
110
          allow(repository).to receive(:tags).and_return([tag_a, tag_b])
111 112 113 114 115 116 117 118 119 120 121 122
        end

        it { is_expected.to eq(['v1.0.0', 'v1.1.0']) }
      end

      context 'asc' do
        subject { repository.tags_sorted_by('updated_asc').map(&:name) }

        before do
          double_first = double(committed_date: Time.now - 1.second)
          double_last = double(committed_date: Time.now)

123 124
          allow(tag_a).to receive(:dereferenced_target).and_return(double_last)
          allow(tag_b).to receive(:dereferenced_target).and_return(double_first)
125
          allow(repository).to receive(:tags).and_return([tag_a, tag_b])
126 127 128 129
        end

        it { is_expected.to eq(['v1.1.0', 'v1.0.0']) }
      end
130 131 132 133 134 135 136 137 138

      context 'annotated tag pointing to a blob' do
        let(:annotated_tag_name) { 'annotated-tag' }

        subject { repository.tags_sorted_by('updated_asc').map(&:name) }

        before do
          options = { message: 'test tag message\n',
                      tagger: { name: 'John Smith', email: 'john@gmail.com' } }
139 140 141 142

          Gitlab::GitalyClient::StorageSettings.allow_disk_access do
            repository.rugged.tags.create(annotated_tag_name, 'a48e4fc218069f68ef2e769dd8dfea3991362175', options)
          end
143 144 145 146 147 148 149 150 151 152 153 154 155 156

          double_first = double(committed_date: Time.now - 1.second)
          double_last = double(committed_date: Time.now)

          allow(tag_a).to receive(:dereferenced_target).and_return(double_last)
          allow(tag_b).to receive(:dereferenced_target).and_return(double_first)
        end

        it { is_expected.to eq(['v1.1.0', 'v1.0.0', annotated_tag_name]) }

        after do
          repository.rugged.tags.delete(annotated_tag_name)
        end
      end
157 158 159
    end
  end

160
  describe '#ref_name_for_sha' do
161
    it 'returns the ref' do
162 163
      allow(repository.raw_repository).to receive(:ref_name_for_sha)
        .and_return('refs/environments/production/77')
164

165
      expect(repository.ref_name_for_sha('bla', '0' * 40)).to eq 'refs/environments/production/77'
166 167 168
    end
  end

169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188
  describe '#ref_exists?' do
    context 'when ref exists' do
      it 'returns true' do
        expect(repository.ref_exists?('refs/heads/master')).to be true
      end
    end

    context 'when ref does not exist' do
      it 'returns false' do
        expect(repository.ref_exists?('refs/heads/non-existent')).to be false
      end
    end

    context 'when ref format is incorrect' do
      it 'returns false' do
        expect(repository.ref_exists?('refs/heads/invalid:master')).to be false
      end
    end
  end

189
  describe '#last_commit_for_path' do
190 191
    shared_examples 'getting last commit for path' do
      subject { repository.last_commit_for_path(sample_commit.id, '.gitignore').id }
192

193
      it { is_expected.to eq('c1acaa58bbcbc3eafe538cb8274ba387047b69f8') }
194

195
      describe 'when storage is broken', :broken_storage  do
196 197 198 199 200 201
        it 'should raise a storage error' do
          expect_to_raise_storage_error do
            broken_repository.last_commit_id_for_path(sample_commit.id, '.gitignore')
          end
        end
      end
202 203 204 205 206 207
    end

    context 'when Gitaly feature last_commit_for_path is enabled' do
      it_behaves_like 'getting last commit for path'
    end

208
    context 'when Gitaly feature last_commit_for_path is disabled', :skip_gitaly_mock do
209 210
      it_behaves_like 'getting last commit for path'
    end
211
  end
212

Hiroyuki Sato's avatar
Hiroyuki Sato committed
213
  describe '#last_commit_id_for_path' do
214 215
    shared_examples 'getting last commit ID for path' do
      subject { repository.last_commit_id_for_path(sample_commit.id, '.gitignore') }
Hiroyuki Sato's avatar
Hiroyuki Sato committed
216

217 218 219 220 221 222 223 224 225 226 227
      it "returns last commit id for a given path" do
        is_expected.to eq('c1acaa58bbcbc3eafe538cb8274ba387047b69f8')
      end

      it "caches last commit id for a given path" do
        cache = repository.send(:cache)
        key = "last_commit_id_for_path:#{sample_commit.id}:#{Digest::SHA1.hexdigest('.gitignore')}"

        expect(cache).to receive(:fetch).with(key).and_return('c1acaa5')
        is_expected.to eq('c1acaa5')
      end
228

229
      describe 'when storage is broken', :broken_storage  do
230 231 232 233 234 235
        it 'should raise a storage error' do
          expect_to_raise_storage_error do
            broken_repository.last_commit_for_path(sample_commit.id, '.gitignore').id
          end
        end
      end
Hiroyuki Sato's avatar
Hiroyuki Sato committed
236
    end
Hiroyuki Sato's avatar
Hiroyuki Sato committed
237

238 239 240
    context 'when Gitaly feature last_commit_for_path is enabled' do
      it_behaves_like 'getting last commit ID for path'
    end
241

242
    context 'when Gitaly feature last_commit_for_path is disabled', :skip_gitaly_mock do
243
      it_behaves_like 'getting last commit ID for path'
Hiroyuki Sato's avatar
Hiroyuki Sato committed
244 245 246
    end
  end

247
  describe '#commits' do
Tiago Botelho's avatar
Tiago Botelho committed
248 249 250 251
    context 'when neither the all flag nor a ref are specified' do
      it 'returns every commit from default branch' do
        expect(repository.commits(limit: 60).size).to eq(37)
      end
252 253
    end

Tiago Botelho's avatar
Tiago Botelho committed
254 255 256 257
    context 'when ref is passed' do
      it 'returns every commit from the specified ref' do
        expect(repository.commits('master', limit: 60).size).to eq(37)
      end
258

Tiago Botelho's avatar
Tiago Botelho committed
259 260 261 262 263 264 265 266 267 268 269 270 271
      context 'when all' do
        it 'returns every commit from the repository' do
          expect(repository.commits('master', limit: 60, all: true).size).to eq(60)
        end
      end

      context 'with path' do
        it 'sets follow when it is a single path' do
          expect(Gitlab::Git::Commit).to receive(:where).with(a_hash_including(follow: true)).and_call_original.twice

          repository.commits('master', limit: 1, path: 'README.md')
          repository.commits('master', limit: 1, path: ['README.md'])
        end
272

Tiago Botelho's avatar
Tiago Botelho committed
273 274
        it 'does not set follow when it is multiple paths' do
          expect(Gitlab::Git::Commit).to receive(:where).with(a_hash_including(follow: false)).and_call_original
275

Tiago Botelho's avatar
Tiago Botelho committed
276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292
          repository.commits('master', limit: 1, path: ['README.md', 'CHANGELOG'])
        end
      end

      context 'without path' do
        it 'does not set follow' do
          expect(Gitlab::Git::Commit).to receive(:where).with(a_hash_including(follow: false)).and_call_original

          repository.commits('master', limit: 1)
        end
      end
    end

    context "when 'all' flag is set" do
      it 'returns every commit from the repository' do
        expect(repository.commits(all: true, limit: 60).size).to eq(60)
      end
293 294 295
    end
  end

296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317
  describe '#new_commits' do
    let(:new_refs) do
      double(:git_rev_list, new_refs: %w[
        c1acaa58bbcbc3eafe538cb8274ba387047b69f8
        5937ac0a7beb003549fc5fd26fc247adbce4a52e
      ])
    end

    it 'delegates to Gitlab::Git::RevList' do
      expect(Gitlab::Git::RevList).to receive(:new).with(
        repository.raw,
        newrev: 'aaaabbbbccccddddeeeeffffgggghhhhiiiijjjj').and_return(new_refs)

      commits = repository.new_commits('aaaabbbbccccddddeeeeffffgggghhhhiiiijjjj')

      expect(commits).to eq([
        repository.commit('c1acaa58bbcbc3eafe538cb8274ba387047b69f8'),
        repository.commit('5937ac0a7beb003549fc5fd26fc247adbce4a52e')
      ])
    end
  end

318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365
  describe '#commits_by' do
    set(:project) { create(:project, :repository) }

    shared_examples 'batch commits fetching' do
      let(:oids) { TestEnv::BRANCH_SHA.values }

      subject { project.repository.commits_by(oids: oids) }

      it 'finds each commit' do
        expect(subject).not_to include(nil)
        expect(subject.size).to eq(oids.size)
      end

      it 'returns only Commit instances' do
        expect(subject).to all( be_a(Commit) )
      end

      context 'when some commits are not found ' do
        let(:oids) do
          ['deadbeef'] + TestEnv::BRANCH_SHA.values.first(10)
        end

        it 'returns only found commits' do
          expect(subject).not_to include(nil)
          expect(subject.size).to eq(10)
        end
      end

      context 'when no oids are passed' do
        let(:oids) { [] }

        it 'does not call #batch_by_oid' do
          expect(Gitlab::Git::Commit).not_to receive(:batch_by_oid)

          subject
        end
      end
    end

    context 'when Gitaly list_commits_by_oid is enabled' do
      it_behaves_like 'batch commits fetching'
    end

    context 'when Gitaly list_commits_by_oid is enabled', :disable_gitaly do
      it_behaves_like 'batch commits fetching'
    end
  end

366
  describe '#find_commits_by_message' do
367 368 369 370 371 372 373 374 375 376 377 378 379 380
    shared_examples 'finding commits by message' do
      it 'returns commits with messages containing a given string' do
        commit_ids = repository.find_commits_by_message('submodule').map(&:id)

        expect(commit_ids).to include(
          '5937ac0a7beb003549fc5fd26fc247adbce4a52e',
          '6f6d7e7ed97bb5f0054f2b1df789b39ca89b6ff9',
          'cfe32cf61b73a0d5e9f13e774abde7ff789b1660'
        )
        expect(commit_ids).not_to include('913c66a37b4a45b9769037c55c2d238bd0942d2e')
      end

      it 'is case insensitive' do
        commit_ids = repository.find_commits_by_message('SUBMODULE').map(&:id)
381

382 383
        expect(commit_ids).to include('5937ac0a7beb003549fc5fd26fc247adbce4a52e')
      end
384
    end
385

386 387
    context 'when Gitaly commits_by_message feature is enabled' do
      it_behaves_like 'finding commits by message'
388 389
    end

390
    context 'when Gitaly commits_by_message feature is disabled', :skip_gitaly_mock do
391
      it_behaves_like 'finding commits by message'
392 393
    end

394
    describe 'when storage is broken', :broken_storage do
395 396 397
      it 'should raise a storage error' do
        expect_to_raise_storage_error { broken_repository.find_commits_by_message('s') }
      end
398
    end
399 400
  end

401
  describe '#blob_at' do
402 403 404 405 406 407
    context 'blank sha' do
      subject { repository.blob_at(Gitlab::Git::BLANK_SHA, '.gitignore') }

      it { is_expected.to be_nil }
    end
  end
408

409
  describe '#merged_to_root_ref?' do
410
    context 'merged branch without ff' do
411
      subject { repository.merged_to_root_ref?('branch-merged') }
412 413 414

      it { is_expected.to be_truthy }
    end
415

416 417
    # If the HEAD was ff then it will be false
    context 'merged with ff' do
418 419 420 421
      subject { repository.merged_to_root_ref?('improve/awesome') }

      it { is_expected.to be_truthy }
    end
422

423 424 425 426 427
    context 'not merged branch' do
      subject { repository.merged_to_root_ref?('not-merged-branch') }

      it { is_expected.to be_falsey }
    end
428 429 430 431 432 433

    context 'default branch' do
      subject { repository.merged_to_root_ref?('master') }

      it { is_expected.to be_falsey }
    end
434 435
  end

436
  describe '#can_be_merged?' do
437 438 439
    shared_examples 'can be merged' do
      context 'mergeable branches' do
        subject { repository.can_be_merged?('0b4bc9a49b562e85de7cc9e834518ea6828729b9', 'master') }
440

441 442
        it { is_expected.to be_truthy }
      end
443

444
      context 'non-mergeable branches without conflict sides missing' do
445
        subject { repository.can_be_merged?('bb5206fee213d983da88c47f9cf4cc6caf9c66dc', 'feature') }
446

447 448
        it { is_expected.to be_falsey }
      end
449

450 451 452 453 454 455
      context 'non-mergeable branches with conflict sides missing' do
        subject { repository.can_be_merged?('conflict-missing-side', 'conflict-start') }

        it { is_expected.to be_falsey }
      end

456 457
      context 'non merged branch' do
        subject { repository.merged_to_root_ref?('fix') }
458

459 460 461 462 463 464 465 466
        it { is_expected.to be_falsey }
      end

      context 'non existent branch' do
        subject { repository.merged_to_root_ref?('non_existent_branch') }

        it { is_expected.to be_nil }
      end
467 468
    end

469 470 471
    context 'when Gitaly can_be_merged feature is enabled' do
      it_behaves_like 'can be merged'
    end
472

473 474
    context 'when Gitaly can_be_merged feature is disabled', :disable_gitaly do
      it_behaves_like 'can be merged'
475
    end
476 477
  end

478 479 480
  describe '#commit' do
    context 'when ref exists' do
      it 'returns commit object' do
481 482
        expect(repository.commit('master'))
          .to be_an_instance_of Commit
483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506
      end
    end

    context 'when ref does not exist' do
      it 'returns nil' do
        expect(repository.commit('non-existent-ref')).to be_nil
      end
    end

    context 'when ref is not valid' do
      context 'when preceding tree element exists' do
        it 'returns nil' do
          expect(repository.commit('master:ref')).to be_nil
        end
      end

      context 'when preceding tree element does not exist' do
        it 'returns nil' do
          expect(repository.commit('non-existent:ref')).to be_nil
        end
      end
    end
  end

Douwe Maan's avatar
Douwe Maan committed
507
  describe "#create_dir" do
508 509
    it "commits a change that creates a new directory" do
      expect do
510
        repository.create_dir(user, 'newdir',
511
          message: 'Create newdir', branch_name: 'master')
512
      end.to change { repository.count_commits(ref: 'master') }.by(1)
513 514 515 516 517

      newdir = repository.tree('master', 'newdir')
      expect(newdir.path).to eq('newdir')
    end

518
    context "when committing to another project" do
519
      let(:forked_project) { create(:project, :repository) }
520 521 522

      it "creates a fork and commit to the forked project" do
        expect do
523
          repository.create_dir(user, 'newdir',
524 525
            message: 'Create newdir', branch_name: 'patch',
            start_branch_name: 'master', start_project: forked_project)
526
        end.to change { repository.count_commits(ref: 'master') }.by(0)
527 528 529 530 531 532 533 534 535

        expect(repository.branch_exists?('patch')).to be_truthy
        expect(forked_project.repository.branch_exists?('patch')).to be_falsy

        newdir = repository.tree('patch', 'newdir')
        expect(newdir.path).to eq('newdir')
      end
    end

536 537 538
    context "when an author is specified" do
      it "uses the given email/name to set the commit's author" do
        expect do
539
          repository.create_dir(user, 'newdir',
540 541 542
            message: 'Add newdir',
            branch_name: 'master',
            author_email: author_email, author_name: author_name)
543
        end.to change { repository.count_commits(ref: 'master') }.by(1)
544 545 546 547 548 549 550 551 552

        last_commit = repository.commit

        expect(last_commit.author_email).to eq(author_email)
        expect(last_commit.author_name).to eq(author_name)
      end
    end
  end

553 554
  describe "#create_file" do
    it 'commits new file successfully' do
555
      expect do
556 557 558
        repository.create_file(user, 'NEWCHANGELOG', 'Changelog!',
                               message: 'Create changelog',
                               branch_name: 'master')
559
      end.to change { repository.count_commits(ref: 'master') }.by(1)
560

561
      blob = repository.blob_at('master', 'NEWCHANGELOG')
562

563
      expect(blob.data).to eq('Changelog!')
564
    end
565

566
    it 'creates new file and dir when file_path has a forward slash' do
567
      expect do
568 569
        repository.create_file(user, 'new_dir/new_file.txt', 'File!',
                               message: 'Create new_file with new_dir',
570
                               branch_name: 'master')
571
      end.to change { repository.count_commits(ref: 'master') }.by(1)
572

573 574
      expect(repository.tree('master', 'new_dir').path).to eq('new_dir')
      expect(repository.blob_at('master', 'new_dir/new_file.txt').data).to eq('File!')
575 576
    end

577
    it 'respects the autocrlf setting' do
578
      repository.create_file(user, 'hello.txt', "Hello,\r\nWorld",
579
                             message: 'Add hello world',
580
                             branch_name: 'master')
581 582 583 584 585 586

      blob = repository.blob_at('master', 'hello.txt')

      expect(blob.data).to eq("Hello,\nWorld")
    end

587 588 589
    context "when an author is specified" do
      it "uses the given email/name to set the commit's author" do
        expect do
590
          repository.create_file(user, 'NEWREADME', 'README!',
591 592 593 594
                                 message: 'Add README',
                                 branch_name: 'master',
                                 author_email: author_email,
                                 author_name: author_name)
595
        end.to change { repository.count_commits(ref: 'master') }.by(1)
596 597 598 599 600 601 602

        last_commit = repository.commit

        expect(last_commit.author_email).to eq(author_email)
        expect(last_commit.author_name).to eq(author_name)
      end
    end
603 604
  end

605
  describe "#update_file" do
606 607 608 609 610
    it 'updates file successfully' do
      expect do
        repository.update_file(user, 'CHANGELOG', 'Changelog!',
                               message: 'Update changelog',
                               branch_name: 'master')
611
      end.to change { repository.count_commits(ref: 'master') }.by(1)
612 613 614 615 616 617

      blob = repository.blob_at('master', 'CHANGELOG')

      expect(blob.data).to eq('Changelog!')
    end

618
    it 'updates filename successfully' do
619 620
      expect do
        repository.update_file(user, 'NEWLICENSE', 'Copyright!',
621
                                     branch_name: 'master',
622
                                     previous_path: 'LICENSE',
623
                                     message: 'Changes filename')
624
      end.to change { repository.count_commits(ref: 'master') }.by(1)
625 626 627 628 629 630

      files = repository.ls_files('master')

      expect(files).not_to include('LICENSE')
      expect(files).to include('NEWLICENSE')
    end
631 632 633 634

    context "when an author is specified" do
      it "uses the given email/name to set the commit's author" do
        expect do
635 636 637 638 639 640
          repository.update_file(user, 'README', 'Updated README!',
                                 branch_name: 'master',
                                 previous_path: 'README',
                                 message: 'Update README',
                                 author_email: author_email,
                                 author_name: author_name)
641
        end.to change { repository.count_commits(ref: 'master') }.by(1)
642 643 644 645 646 647 648 649 650

        last_commit = repository.commit

        expect(last_commit.author_email).to eq(author_email)
        expect(last_commit.author_name).to eq(author_name)
      end
    end
  end

Douwe Maan's avatar
Douwe Maan committed
651
  describe "#delete_file" do
652 653
    it 'removes file successfully' do
      expect do
654
        repository.delete_file(user, 'README',
655
          message: 'Remove README', branch_name: 'master')
656
      end.to change { repository.count_commits(ref: 'master') }.by(1)
657 658 659 660 661 662 663

      expect(repository.blob_at('master', 'README')).to be_nil
    end

    context "when an author is specified" do
      it "uses the given email/name to set the commit's author" do
        expect do
664
          repository.delete_file(user, 'README',
665 666
            message: 'Remove README', branch_name: 'master',
            author_email: author_email, author_name: author_name)
667
        end.to change { repository.count_commits(ref: 'master') }.by(1)
668 669 670 671 672 673 674 675 676

        last_commit = repository.commit

        expect(last_commit.author_email).to eq(author_email)
        expect(last_commit.author_name).to eq(author_name)
      end
    end
  end

677
  shared_examples "search_files_by_content" do
678
    let(:results) { repository.search_files_by_content('feature', 'master') }
679 680 681 682
    subject { results }

    it { is_expected.to be_an Array }

683
    it 'regex-escapes the query string' do
684
      results = repository.search_files_by_content("test\\", 'master')
685 686 687 688

      expect(results.first).not_to start_with('fatal:')
    end

689
    it 'properly handles an unmatched parenthesis' do
690
      results = repository.search_files_by_content("test(", 'master')
691 692 693 694

      expect(results.first).not_to start_with('fatal:')
    end

Valery Sizov's avatar
Valery Sizov committed
695
    it 'properly handles when query is not present' do
696
      results = repository.search_files_by_content('', 'master')
Valery Sizov's avatar
Valery Sizov committed
697 698 699 700 701

      expect(results).to match_array([])
    end

    it 'properly handles query when repo is empty' do
702
      repository = create(:project, :empty_repo).repository
703
      results = repository.search_files_by_content('test', 'master')
Valery Sizov's avatar
Valery Sizov committed
704 705 706 707

      expect(results).to match_array([])
    end

708
    describe 'when storage is broken', :broken_storage  do
709 710 711 712 713 714 715
      it 'should raise a storage error' do
        expect_to_raise_storage_error do
          broken_repository.search_files_by_content('feature', 'master')
        end
      end
    end

716 717 718 719
    describe 'result' do
      subject { results.first }

      it { is_expected.to be_an String }
720
      it { expect(subject.lines[2]).to eq("master:CHANGELOG\x00190\x00  - Feature: Replace teams with group membership\n") }
721 722
    end
  end
723

724
  shared_examples "search_files_by_name" do
725 726 727 728 729 730
    let(:results) { repository.search_files_by_name('files', 'master') }

    it 'returns result' do
      expect(results.first).to eq('files/html/500.html')
    end

Andrew McCallum's avatar
Andrew McCallum committed
731 732
    it 'ignores leading slashes' do
      results = repository.search_files_by_name('/files', 'master')
733

734 735 736
      expect(results.first).to eq('files/html/500.html')
    end

737
    it 'properly handles when query is only slashes' do
738
      results = repository.search_files_by_name('//', 'master')
739

740
      expect(results).to match_array([])
741 742
    end

743 744 745 746 747 748 749
    it 'properly handles when query is not present' do
      results = repository.search_files_by_name('', 'master')

      expect(results).to match_array([])
    end

    it 'properly handles query when repo is empty' do
750
      repository = create(:project, :empty_repo).repository
751 752 753 754 755

      results = repository.search_files_by_name('test', 'master')

      expect(results).to match_array([])
    end
756

757
    describe 'when storage is broken', :broken_storage  do
758 759 760 761 762 763
      it 'should raise a storage error' do
        expect_to_raise_storage_error { broken_repository.search_files_by_name('files', 'master') }
      end
    end
  end

764 765 766 767 768 769 770 771 772 773
  describe 'with gitaly enabled' do
    it_behaves_like 'search_files_by_content'
    it_behaves_like 'search_files_by_name'
  end

  describe 'with gitaly disabled', :disable_gitaly do
    it_behaves_like 'search_files_by_content'
    it_behaves_like 'search_files_by_name'
  end

774
  describe '#async_remove_remote' do
775 776 777 778 779 780 781 782 783 784 785 786 787 788
    before do
      masterrev = repository.find_branch('master').dereferenced_target
      create_remote_branch('joe', 'remote_branch', masterrev)
    end

    context 'when worker is scheduled successfully' do
      before do
        masterrev = repository.find_branch('master').dereferenced_target
        create_remote_branch('remote_name', 'remote_branch', masterrev)

        allow(RepositoryRemoveRemoteWorker).to receive(:perform_async).and_return('1234')
      end

      it 'returns job_id' do
789
        expect(repository.async_remove_remote('joe')).to eq('1234')
790 791 792 793 794 795 796 797 798
      end
    end

    context 'when worker does not schedule successfully' do
      before do
        allow(RepositoryRemoveRemoteWorker).to receive(:perform_async).and_return(nil)
      end

      it 'returns nil' do
799
        expect(Rails.logger).to receive(:info).with("Remove remote job failed to create for #{project.id} with remote name joe.")
800

801
        expect(repository.async_remove_remote('joe')).to be_nil
802 803 804 805
      end
    end
  end

806
  describe '#fetch_ref' do
807
    let(:broken_repository) { create(:project, :broken_storage).repository }
808

809
    describe 'when storage is broken', :broken_storage  do
810
      it 'should raise a storage error' do
811 812 813
        expect_to_raise_storage_error do
          broken_repository.fetch_ref(broken_repository, source_ref: '1', target_ref: '2')
        end
814 815
      end
    end
816 817
  end

818
  describe '#create_ref' do
819
    it 'redirects the call to write_ref' do
820 821
      ref, ref_path = '1', '2'

822
      expect(repository.raw_repository).to receive(:write_ref).with(ref_path, ref)
823 824 825 826 827

      repository.create_ref(ref, ref_path)
    end
  end

828
  describe "#changelog", :use_clean_rails_memory_store_caching do
829 830
    it 'accepts changelog' do
      expect(repository.tree).to receive(:blobs).and_return([TestBlob.new('changelog')])
831

832
      expect(repository.changelog.path).to eq('changelog')
833
    end
834

835 836
    it 'accepts news instead of changelog' do
      expect(repository.tree).to receive(:blobs).and_return([TestBlob.new('news')])
837

838
      expect(repository.changelog.path).to eq('news')
839
    end
840

841 842
    it 'accepts history instead of changelog' do
      expect(repository.tree).to receive(:blobs).and_return([TestBlob.new('history')])
843

844
      expect(repository.changelog.path).to eq('history')
845 846
    end

847 848
    it 'accepts changes instead of changelog' do
      expect(repository.tree).to receive(:blobs).and_return([TestBlob.new('changes')])
849

850
      expect(repository.changelog.path).to eq('changes')
851 852
    end

853 854 855
    it 'is case-insensitive' do
      expect(repository.tree).to receive(:blobs).and_return([TestBlob.new('CHANGELOG')])

856
      expect(repository.changelog.path).to eq('CHANGELOG')
857 858 859
    end
  end

860
  describe "#license_blob", :use_clean_rails_memory_store_caching do
861
    before do
862
      repository.delete_file(
863
        user, 'LICENSE', message: 'Remove LICENSE', branch_name: 'master')
864 865
    end

866
    it 'handles when HEAD points to non-existent ref' do
867
      repository.create_file(
868
        user, 'LICENSE', 'Copyright!',
869
        message: 'Add LICENSE', branch_name: 'master')
870

871
      allow(repository).to receive(:root_ref).and_raise(Gitlab::Git::Repository::NoRepository)
872 873 874 875

      expect(repository.license_blob).to be_nil
    end

876
    it 'looks in the root_ref only' do
877
      repository.delete_file(user, 'LICENSE',
878
        message: 'Remove LICENSE', branch_name: 'markdown')
879
      repository.create_file(user, 'LICENSE',
880
        Licensee::License.new('mit').content,
881
        message: 'Add LICENSE', branch_name: 'markdown')
882 883 884 885

      expect(repository.license_blob).to be_nil
    end

886
    it 'detects license file with no recognizable open-source license content' do
887 888
      repository.create_file(user, 'LICENSE', 'Copyright!',
        message: 'Add LICENSE', branch_name: 'master')
889

890
      expect(repository.license_blob.path).to eq('LICENSE')
891 892
    end

893 894
    %w[LICENSE LICENCE LiCensE LICENSE.md LICENSE.foo COPYING COPYING.md].each do |filename|
      it "detects '#{filename}'" do
895
        repository.create_file(user, filename,
896
          Licensee::License.new('mit').content,
897
          message: "Add #{filename}", branch_name: 'master')
898

899 900
        expect(repository.license_blob.name).to eq(filename)
      end
901 902 903
    end
  end

904
  describe '#license_key', :use_clean_rails_memory_store_caching do
905
    before do
906
      repository.delete_file(user, 'LICENSE',
907
        message: 'Remove LICENSE', branch_name: 'master')
908
    end
909

910
    it 'returns nil when no license is detected' do
911 912 913
      expect(repository.license_key).to be_nil
    end

914 915 916
    it 'returns nil when the repository does not exist' do
      expect(repository).to receive(:exists?).and_return(false)

917 918 919
      expect(repository.license_key).to be_nil
    end

920
    it 'returns nil when the content is not recognizable' do
921
      repository.create_file(user, 'LICENSE', 'Gitlab B.V.',
922
        message: 'Add LICENSE', branch_name: 'master')
923 924

      expect(repository.license_key).to be_nil
925
    end
926

927 928 929 930 931 932
    it 'returns nil when the commit SHA does not exist' do
      allow(repository.head_commit).to receive(:sha).and_return('1' * 40)

      expect(repository.license_key).to be_nil
    end

933
    it 'returns nil when master does not exist' do
934 935
      repository.rm_branch(user, 'master')

936
      expect(repository.license_key).to be_nil
937 938
    end

939
    it 'returns the license key' do
940
      repository.create_file(user, 'LICENSE',
941
        Licensee::License.new('mit').content,
942
        message: 'Add LICENSE', branch_name: 'master')
943

944
      expect(repository.license_key).to eq('mit')
945
    end
946
  end
947

948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964
  describe '#license' do
    before do
      repository.delete_file(user, 'LICENSE',
        message: 'Remove LICENSE', branch_name: 'master')
    end

    it 'returns nil when no license is detected' do
      expect(repository.license).to be_nil
    end

    it 'returns nil when the repository does not exist' do
      expect(repository).to receive(:exists?).and_return(false)

      expect(repository.license).to be_nil
    end

    it 'returns nil when the content is not recognizable' do
965
      repository.create_file(user, 'LICENSE', 'Gitlab B.V.',
966 967 968 969 970 971 972 973 974 975 976 977 978 979 980
        message: 'Add LICENSE', branch_name: 'master')

      expect(repository.license).to be_nil
    end

    it 'returns the license' do
      license = Licensee::License.new('mit')
      repository.create_file(user, 'LICENSE',
        license.content,
        message: 'Add LICENSE', branch_name: 'master')

      expect(repository.license).to eq(license)
    end
  end

981
  describe "#gitlab_ci_yml", :use_clean_rails_memory_store_caching do
982 983 984 985
    it 'returns valid file' do
      files = [TestBlob.new('file'), TestBlob.new('.gitlab-ci.yml'), TestBlob.new('copying')]
      expect(repository.tree).to receive(:blobs).and_return(files)

986
      expect(repository.gitlab_ci_yml.path).to eq('.gitlab-ci.yml')
987 988 989 990 991 992 993 994
    end

    it 'returns nil if not exists' do
      expect(repository.tree).to receive(:blobs).and_return([])
      expect(repository.gitlab_ci_yml).to be_nil
    end

    it 'returns nil for empty repository' do
995
      allow(repository).to receive(:root_ref).and_raise(Gitlab::Git::Repository::NoRepository)
996 997 998 999
      expect(repository.gitlab_ci_yml).to be_nil
    end
  end

1000
  describe '#add_branch' do
1001 1002
    let(:branch_name) { 'new_feature' }
    let(:target) { 'master' }
1003

1004
    subject { repository.add_branch(user, branch_name, target) }
1005

1006 1007 1008 1009
    it "calls Gitaly's OperationService" do
      expect_any_instance_of(Gitlab::GitalyClient::OperationService)
        .to receive(:user_create_branch).with(branch_name, user, target)
        .and_return(nil)
1010

1011
      subject
1012 1013
    end

1014 1015 1016 1017
    it 'creates_the_branch' do
      expect(subject.name).to eq(branch_name)
      expect(repository.find_branch(branch_name)).not_to be_nil
    end
1018

1019 1020
    context 'with a non-existing target' do
      let(:target) { 'fake-target' }
1021

1022 1023 1024
      it "returns false and doesn't create the branch" do
        expect(subject).to be(false)
        expect(repository.find_branch(branch_name)).to be_nil
1025 1026 1027 1028
      end
    end
  end

1029
  describe '#find_branch' do
1030 1031 1032
    context 'fresh_repo is true' do
      it 'delegates the call to raw_repository' do
        expect(repository.raw_repository).to receive(:find_branch).with('master', true)
1033

1034
        repository.find_branch('master', fresh_repo: true)
1035 1036 1037
      end
    end

1038 1039 1040
    context 'fresh_repo is false' do
      it 'delegates the call to raw_repository' do
        expect(repository.raw_repository).to receive(:find_branch).with('master', false)
1041

1042
        repository.find_branch('master', fresh_repo: false)
1043 1044 1045 1046
      end
    end
  end

1047
  describe '#update_branch_with_hooks' do
1048
    let(:old_rev) { '0b4bc9a49b562e85de7cc9e834518ea6828729b9' } # git rev-parse feature
1049
    let(:new_rev) { 'a74ae73c1ccde9b974a70e82b901588071dc142a' } # commit whose parent is old_rev
1050 1051 1052
    let(:updating_ref) { 'refs/heads/feature' }
    let(:target_project) { project }
    let(:target_repository) { target_project.repository }
1053

1054 1055 1056 1057 1058 1059 1060
    around do |example|
      # TODO Gitlab::Git::OperationService will be moved to gitaly-ruby and disappear from this repo
      Gitlab::GitalyClient::StorageSettings.allow_disk_access do
        example.run
      end
    end

1061
    context 'when pre hooks were successful' do
1062
      before do
1063 1064
        service = Gitlab::Git::HooksService.new
        expect(Gitlab::Git::HooksService).to receive(:new).and_return(service)
1065
        expect(service).to receive(:execute)
1066
          .with(git_user, target_repository.raw_repository, old_rev, new_rev, updating_ref)
1067
          .and_yield(service).and_return(true)
1068
      end
1069

1070
      it 'runs without errors' do
1071
        expect do
1072
          Gitlab::Git::OperationService.new(git_user, repository.raw_repository).with_branch('feature') do
1073 1074
            new_rev
          end
1075 1076
        end.not_to raise_error
      end
1077

1078
      it 'ensures the autocrlf Git option is set to :input' do
1079
        service = Gitlab::Git::OperationService.new(git_user, repository.raw_repository)
1080 1081

        expect(service).to receive(:update_autocrlf_option)
1082

1083
        service.with_branch('feature') { new_rev }
1084
      end
1085 1086 1087

      context "when the branch wasn't empty" do
        it 'updates the head' do
1088
          expect(repository.find_branch('feature').dereferenced_target.id).to eq(old_rev)
1089

1090
          Gitlab::Git::OperationService.new(git_user, repository.raw_repository).with_branch('feature') do
1091 1092 1093
            new_rev
          end

1094
          expect(repository.find_branch('feature').dereferenced_target.id).to eq(new_rev)
1095 1096
        end
      end
1097 1098 1099 1100 1101 1102 1103 1104

      context 'when target project does not have the commit' do
        let(:target_project) { create(:project, :empty_repo) }
        let(:old_rev) { Gitlab::Git::BLANK_SHA }
        let(:new_rev) { project.commit('feature').sha }
        let(:updating_ref) { 'refs/heads/master' }

        it 'fetch_ref and create the branch' do
1105
          expect(target_project.repository.raw_repository).to receive(:fetch_ref)
1106 1107
            .and_call_original

1108
          Gitlab::Git::OperationService.new(git_user, target_repository.raw_repository)
1109 1110
            .with_branch(
              'master',
1111
              start_repository: project.repository.raw_repository,
1112 1113 1114 1115 1116 1117 1118 1119 1120 1121 1122 1123
              start_branch_name: 'feature') { new_rev }

          expect(target_repository.branch_names).to contain_exactly('master')
        end
      end

      context 'when target project already has the commit' do
        let(:target_project) { create(:project, :repository) }

        it 'does not fetch_ref and just pass the commit' do
          expect(target_repository).not_to receive(:fetch_ref)

1124
          Gitlab::Git::OperationService.new(git_user, target_repository.raw_repository)
1125
            .with_branch('feature', start_repository: project.repository.raw_repository) { new_rev }
1126 1127
        end
      end
1128 1129
    end

1130 1131 1132 1133
    context 'when temporary ref failed to be created from other project' do
      let(:target_project) { create(:project, :empty_repo) }

      before do
1134
        expect(target_project.repository.raw_repository).to receive(:run_git)
1135 1136 1137 1138 1139 1140 1141 1142
      end

      it 'raises Rugged::ReferenceError' do
        raise_reference_error = raise_error(Rugged::ReferenceError) do |err|
          expect(err.cause).to be_nil
        end

        expect do
1143
          Gitlab::Git::OperationService.new(git_user, target_project.repository.raw_repository)
1144
            .with_branch('feature',
1145
                         start_repository: project.repository.raw_repository,
1146 1147 1148 1149 1150
                         &:itself)
        end.to raise_reference_error
      end
    end

1151
    context 'when the update adds more than one commit' do
1152
      let(:old_rev) { '33f3729a45c02fc67d00adb1b8bca394b0e761d9' }
1153

1154
      it 'runs without errors' do
1155
        # old_rev is an ancestor of new_rev
1156
        expect(repository.merge_base(old_rev, new_rev)).to eq(old_rev)
1157 1158 1159 1160

        # old_rev is not a direct ancestor (parent) of new_rev
        expect(repository.rugged.lookup(new_rev).parent_ids).not_to include(old_rev)

1161 1162 1163
        branch = 'feature-ff-target'
        repository.add_branch(user, branch, old_rev)

1164
        expect do
1165
          Gitlab::Git::OperationService.new(git_user, repository.raw_repository).with_branch(branch) do
1166 1167 1168
            new_rev
          end
        end.not_to raise_error
1169 1170 1171 1172
      end
    end

    context 'when the update would remove commits from the target branch' do
1173 1174
      let(:branch) { 'master' }
      let(:old_rev) { repository.find_branch(branch).dereferenced_target.sha }
1175

1176
      it 'raises an exception' do
1177
        # The 'master' branch is NOT an ancestor of new_rev.
1178
        expect(repository.merge_base(old_rev, new_rev)).not_to eq(old_rev)
1179 1180 1181 1182

        # Updating 'master' to new_rev would lose the commits on 'master' that
        # are not contained in new_rev. This should not be allowed.
        expect do
1183
          Gitlab::Git::OperationService.new(git_user, repository.raw_repository).with_branch(branch) do
1184 1185
            new_rev
          end
1186
        end.to raise_error(Gitlab::Git::CommitError)
1187 1188 1189
      end
    end

1190
    context 'when pre hooks failed' do
1191
      it 'gets an error' do
1192
        allow_any_instance_of(Gitlab::Git::Hook).to receive(:trigger).and_return([false, ''])
1193 1194

        expect do
1195
          Gitlab::Git::OperationService.new(git_user, repository.raw_repository).with_branch('feature') do
1196 1197
            new_rev
          end
1198
        end.to raise_error(Gitlab::Git::PreReceiveError)
1199 1200
      end
    end
1201 1202 1203 1204 1205 1206

    context 'when target branch is different from source branch' do
      before do
        allow_any_instance_of(Gitlab::Git::Hook).to receive(:trigger).and_return([true, ''])
      end

1207 1208
      subject do
        Gitlab::Git::OperationService.new(git_user, repository.raw_repository).with_branch('new-feature') do
Jacob Vosmaer's avatar
Jacob Vosmaer committed
1209
          new_rev
1210
        end
1211
      end
1212 1213 1214 1215 1216

      it 'returns branch_created as true' do
        expect(subject).not_to be_repo_created
        expect(subject).to     be_branch_created
      end
1217 1218 1219 1220 1221 1222 1223 1224
    end

    context 'when repository is empty' do
      before do
        allow_any_instance_of(Gitlab::Git::Hook).to receive(:trigger).and_return([true, ''])
      end

      it 'expires creation and branch cache' do
1225
        empty_repository = create(:project, :empty_repo).repository
1226 1227 1228 1229 1230 1231

        expect(empty_repository).to receive(:expire_exists_cache)
        expect(empty_repository).to receive(:expire_root_ref_cache)
        expect(empty_repository).to receive(:expire_emptiness_caches)
        expect(empty_repository).to receive(:expire_branches_cache)

1232
        empty_repository.create_file(user, 'CHANGELOG', 'Changelog!',
1233
                                     message: 'Updates file content',
1234
                                     branch_name: 'master')
1235 1236
      end
    end
1237
  end
1238

1239
  describe '#exists?' do
1240
    it 'returns true when a repository exists' do
1241
      expect(repository.exists?).to be(true)
1242 1243
    end

1244
    it 'returns false if no full path can be constructed' do
1245
      allow(repository).to receive(:full_path).and_return(nil)
1246

1247
      expect(repository.exists?).to be(false)
1248
    end
1249

1250
    context 'with broken storage', :broken_storage do
1251 1252 1253 1254
      it 'should raise a storage error' do
        expect_to_raise_storage_error { broken_repository.exists? }
      end
    end
1255
  end
1256

1257
  describe '#has_visible_content?' do
1258 1259 1260 1261 1262 1263 1264
    before do
      # If raw_repository.has_visible_content? gets called more than once then
      # caching is broken. We don't want that.
      expect(repository.raw_repository).to receive(:has_visible_content?)
        .once
        .and_return(result)
    end
1265

1266 1267
    context 'when true' do
      let(:result) { true }
1268

1269 1270 1271 1272 1273
      it 'returns true and caches it' do
        expect(repository.has_visible_content?).to eq(true)
        # Second call hits the cache
        expect(repository.has_visible_content?).to eq(true)
      end
1274 1275
    end

1276 1277
    context 'when false' do
      let(:result) { false }
1278

1279 1280 1281 1282
      it 'returns false and caches it' do
        expect(repository.has_visible_content?).to eq(false)
        # Second call hits the cache
        expect(repository.has_visible_content?).to eq(false)
1283
      end
1284 1285 1286
    end
  end

1287 1288 1289 1290 1291 1292 1293 1294 1295
  describe '#branch_exists?' do
    it 'uses branch_names' do
      allow(repository).to receive(:branch_names).and_return(['foobar'])

      expect(repository.branch_exists?('foobar')).to eq(true)
      expect(repository.branch_exists?('master')).to eq(false)
    end
  end

1296 1297 1298 1299 1300 1301 1302 1303 1304
  describe '#tag_exists?' do
    it 'uses tag_names' do
      allow(repository).to receive(:tag_names).and_return(['foobar'])

      expect(repository.tag_exists?('foobar')).to eq(true)
      expect(repository.tag_exists?('master')).to eq(false)
    end
  end

1305 1306 1307 1308 1309 1310 1311 1312 1313 1314 1315 1316 1317 1318 1319 1320
  describe '#branch_names', :use_clean_rails_memory_store_caching do
    let(:fake_branch_names) { ['foobar'] }

    it 'gets cached across Repository instances' do
      allow(repository.raw_repository).to receive(:branch_names).once.and_return(fake_branch_names)

      expect(repository.branch_names).to eq(fake_branch_names)

      fresh_repository = Project.find(project.id).repository
      expect(fresh_repository.object_id).not_to eq(repository.object_id)

      expect(fresh_repository.raw_repository).not_to receive(:branch_names)
      expect(fresh_repository.branch_names).to eq(fake_branch_names)
    end
  end

1321
  describe '#update_autocrlf_option' do
1322 1323 1324 1325 1326 1327 1328
    around do |example|
      # TODO Gitlab::Git::OperationService will be moved to gitaly-ruby and disappear from this repo
      Gitlab::GitalyClient::StorageSettings.allow_disk_access do
        example.run
      end
    end

1329 1330 1331 1332 1333 1334
    describe 'when autocrlf is not already set to :input' do
      before do
        repository.raw_repository.autocrlf = true
      end

      it 'sets autocrlf to :input' do
1335
        Gitlab::Git::OperationService.new(nil, repository.raw_repository).send(:update_autocrlf_option)
1336 1337 1338 1339 1340 1341 1342 1343 1344 1345 1346

        expect(repository.raw_repository.autocrlf).to eq(:input)
      end
    end

    describe 'when autocrlf is already set to :input' do
      before do
        repository.raw_repository.autocrlf = :input
      end

      it 'does nothing' do
1347 1348
        expect(repository.raw_repository).not_to receive(:autocrlf=)
          .with(:input)
1349

1350
        Gitlab::Git::OperationService.new(nil, repository.raw_repository).send(:update_autocrlf_option)
1351 1352 1353 1354
      end
    end
  end

1355 1356 1357 1358
  describe '#empty?' do
    let(:empty_repository) { create(:project_empty_repo).repository }

    it 'returns true for an empty repository' do
1359
      expect(empty_repository).to be_empty
1360 1361 1362
    end

    it 'returns false for a non-empty repository' do
1363
      expect(repository).not_to be_empty
1364 1365 1366
    end

    it 'caches the output' do
1367
      expect(repository.raw_repository).to receive(:has_visible_content?).once
1368 1369 1370 1371 1372 1373 1374 1375 1376 1377 1378 1379

      repository.empty?
      repository.empty?
    end
  end

  describe '#root_ref' do
    it 'returns a branch name' do
      expect(repository.root_ref).to be_an_instance_of(String)
    end

    it 'caches the output' do
1380 1381 1382
      expect(repository.raw_repository).to receive(:root_ref)
        .once
        .and_return('master')
1383 1384 1385 1386 1387 1388 1389 1390 1391 1392

      repository.root_ref
      repository.root_ref
    end
  end

  describe '#expire_root_ref_cache' do
    it 'expires the root reference cache' do
      repository.root_ref

1393 1394 1395
      expect(repository.raw_repository).to receive(:root_ref)
        .once
        .and_return('foo')
1396 1397 1398 1399 1400 1401 1402

      repository.expire_root_ref_cache

      expect(repository.root_ref).to eq('foo')
    end
  end

1403
  describe '#expire_branch_cache' do
1404 1405 1406 1407 1408
    # This method is private but we need it for testing purposes. Sadly there's
    # no other proper way of testing caching operations.
    let(:cache) { repository.send(:cache) }

    it 'expires the cache for all branches' do
1409 1410 1411
      expect(cache).to receive(:expire)
        .at_least(repository.branches.length * 2)
        .times
1412 1413 1414 1415 1416

      repository.expire_branch_cache
    end

    it 'expires the cache for all branches when the root branch is given' do
1417 1418 1419
      expect(cache).to receive(:expire)
        .at_least(repository.branches.length * 2)
        .times
1420 1421 1422 1423 1424

      repository.expire_branch_cache(repository.root_ref)
    end

    it 'expires the cache for a specific branch' do
1425
      expect(cache).to receive(:expire).twice
1426 1427

      repository.expire_branch_cache('foo')
1428 1429
    end
  end
Valery Sizov's avatar
Valery Sizov committed
1430

1431 1432 1433
  describe '#expire_emptiness_caches' do
    let(:cache) { repository.send(:cache) }

1434 1435 1436
    it 'expires the caches for an empty repository' do
      allow(repository).to receive(:empty?).and_return(true)

1437
      expect(cache).to receive(:expire).with(:has_visible_content?)
1438 1439 1440

      repository.expire_emptiness_caches
    end
1441 1442 1443 1444

    it 'does not expire the cache for a non-empty repository' do
      allow(repository).to receive(:empty?).and_return(false)

1445
      expect(cache).not_to receive(:expire).with(:has_visible_content?)
1446 1447 1448

      repository.expire_emptiness_caches
    end
1449 1450 1451 1452 1453 1454

    it 'expires the memoized repository cache' do
      allow(repository.raw_repository).to receive(:expire_has_local_branches_cache).and_call_original

      repository.expire_emptiness_caches
    end
1455 1456
  end

1457
  describe 'skip_merges option' do
1458
    subject { repository.commits(Gitlab::Git::BRANCH_REF_PREFIX + "'test'", limit: 100, skip_merges: true).map { |k| k.id } }
1459 1460 1461

    it { is_expected.not_to include('e56497bb5f03a90a51293fc6d516788730953899') }
  end
1462

1463
  describe '#merge' do
1464 1465
    let(:merge_request) { create(:merge_request, source_branch: 'feature', target_branch: 'master', source_project: project) }

1466
    let(:message) { 'Test \r\n\r\n message' }
1467

1468 1469 1470 1471 1472
    shared_examples '#merge' do
      it 'merges the code and returns the commit id' do
        expect(merge_commit).to be_present
        expect(repository.blob_at(merge_commit.id, 'files/ruby/feature.rb')).to be_present
      end
1473

1474 1475
      it 'sets the `in_progress_merge_commit_sha` flag for the given merge request' do
        merge_commit_id = merge(repository, user, merge_request, message)
1476

1477 1478
        expect(merge_request.in_progress_merge_commit_sha).to eq(merge_commit_id)
      end
1479

1480 1481
      it 'removes carriage returns from commit message' do
        merge_commit_id = merge(repository, user, merge_request, message)
1482

1483 1484
        expect(repository.commit(merge_commit_id).message).to eq(message.delete("\r"))
      end
1485
    end
1486

1487 1488 1489
    context 'with gitaly' do
      it_behaves_like '#merge'
    end
1490

1491 1492
    context 'without gitaly', :skip_gitaly_mock do
      it_behaves_like '#merge'
1493 1494
    end

1495 1496
    def merge(repository, user, merge_request, message)
      repository.merge(user, merge_request.diff_head_sha, merge_request, message)
1497
    end
1498 1499
  end

1500
  describe '#ff_merge' do
1501 1502 1503
    before do
      repository.add_branch(user, 'ff-target', 'feature~5')
    end
1504

1505
    it 'merges the code and return the commit id' do
1506
      merge_request = create(:merge_request, source_branch: 'feature', target_branch: 'ff-target', source_project: project)
1507 1508 1509 1510 1511 1512 1513 1514 1515 1516 1517
      merge_commit_id = repository.ff_merge(user,
                                            merge_request.diff_head_sha,
                                            merge_request.target_branch,
                                            merge_request: merge_request)
      merge_commit = repository.commit(merge_commit_id)

      expect(merge_commit).to be_present
      expect(repository.blob_at(merge_commit.id, 'files/ruby/feature.rb')).to be_present
    end

    it 'sets the `in_progress_merge_commit_sha` flag for the given merge request' do
1518
      merge_request = create(:merge_request, source_branch: 'feature', target_branch: 'ff-target', source_project: project)
1519 1520 1521 1522 1523 1524 1525 1526 1527
      merge_commit_id = repository.ff_merge(user,
                                            merge_request.diff_head_sha,
                                            merge_request.target_branch,
                                            merge_request: merge_request)

      expect(merge_request.in_progress_merge_commit_sha).to eq(merge_commit_id)
    end
  end

1528
  describe '#revert' do
1529 1530 1531 1532 1533 1534 1535 1536 1537
    shared_examples 'reverting a commit' do
      let(:new_image_commit) { repository.commit('33f3729a45c02fc67d00adb1b8bca394b0e761d9') }
      let(:update_image_commit) { repository.commit('2f63565e7aac07bcdadb654e253078b727143ec4') }
      let(:message) { 'revert message' }

      context 'when there is a conflict' do
        it 'raises an error' do
          expect { repository.revert(user, new_image_commit, 'master', message) }.to raise_error(Gitlab::Git::Repository::CreateTreeError)
        end
1538 1539
      end

1540 1541 1542
      context 'when commit was already reverted' do
        it 'raises an error' do
          repository.revert(user, update_image_commit, 'master', message)
1543

1544 1545
          expect { repository.revert(user, update_image_commit, 'master', message) }.to raise_error(Gitlab::Git::Repository::CreateTreeError)
        end
1546 1547
      end

1548 1549 1550 1551 1552 1553 1554 1555 1556 1557 1558 1559 1560 1561
      context 'when commit can be reverted' do
        it 'reverts the changes' do
          expect(repository.revert(user, update_image_commit, 'master', message)).to be_truthy
        end
      end

      context 'reverting a merge commit' do
        it 'reverts the changes' do
          merge_commit
          expect(repository.blob_at_branch('master', 'files/ruby/feature.rb')).to be_present

          repository.revert(user, merge_commit, 'master', message)
          expect(repository.blob_at_branch('master', 'files/ruby/feature.rb')).not_to be_present
        end
1562 1563 1564
      end
    end

1565 1566 1567
    context 'when Gitaly revert feature is enabled' do
      it_behaves_like 'reverting a commit'
    end
1568

1569 1570
    context 'when Gitaly revert feature is disabled', :disable_gitaly do
      it_behaves_like 'reverting a commit'
1571 1572
    end
  end
1573

1574
  describe '#cherry_pick' do
1575 1576 1577 1578 1579 1580 1581 1582 1583 1584
    shared_examples 'cherry-picking a commit' do
      let(:conflict_commit) { repository.commit('c642fe9b8b9f28f9225d7ea953fe14e74748d53b') }
      let(:pickable_commit) { repository.commit('7d3b0f7cff5f37573aea97cebfd5692ea1689924') }
      let(:pickable_merge) { repository.commit('e56497bb5f03a90a51293fc6d516788730953899') }
      let(:message) { 'cherry-pick message' }

      context 'when there is a conflict' do
        it 'raises an error' do
          expect { repository.cherry_pick(user, conflict_commit, 'master', message) }.to raise_error(Gitlab::Git::Repository::CreateTreeError)
        end
1585 1586
      end

1587 1588 1589
      context 'when commit was already cherry-picked' do
        it 'raises an error' do
          repository.cherry_pick(user, pickable_commit, 'master', message)
1590

1591 1592
          expect { repository.cherry_pick(user, pickable_commit, 'master', message) }.to raise_error(Gitlab::Git::Repository::CreateTreeError)
        end
1593 1594
      end

1595 1596 1597 1598
      context 'when commit can be cherry-picked' do
        it 'cherry-picks the changes' do
          expect(repository.cherry_pick(user, pickable_commit, 'master', message)).to be_truthy
        end
1599 1600
      end

1601 1602 1603
      context 'cherry-picking a merge commit' do
        it 'cherry-picks the changes' do
          expect(repository.blob_at_branch('improve/awesome', 'foo/bar/.gitkeep')).to be_nil
1604

1605 1606
          cherry_pick_commit_sha = repository.cherry_pick(user, pickable_merge, 'improve/awesome', message)
          cherry_pick_commit_message = project.commit(cherry_pick_commit_sha).message
1607

1608 1609 1610
          expect(repository.blob_at_branch('improve/awesome', 'foo/bar/.gitkeep')).not_to be_nil
          expect(cherry_pick_commit_message).to eq(message)
        end
1611 1612
      end
    end
1613 1614 1615 1616 1617 1618 1619 1620

    context 'when Gitaly cherry_pick feature is enabled' do
      it_behaves_like 'cherry-picking a commit'
    end

    context 'when Gitaly cherry_pick feature is disabled', :disable_gitaly do
      it_behaves_like 'cherry-picking a commit'
    end
1621 1622
  end

1623 1624 1625 1626 1627 1628 1629
  describe '#before_delete' do
    describe 'when a repository does not exist' do
      before do
        allow(repository).to receive(:exists?).and_return(false)
      end

      it 'does not flush caches that depend on repository data' do
1630
        expect(repository).not_to receive(:expire_cache)
1631 1632 1633 1634

        repository.before_delete
      end

1635 1636 1637 1638 1639 1640 1641 1642 1643 1644 1645 1646
      it 'flushes the tags cache' do
        expect(repository).to receive(:expire_tags_cache)

        repository.before_delete
      end

      it 'flushes the branches cache' do
        expect(repository).to receive(:expire_branches_cache)

        repository.before_delete
      end

1647 1648 1649 1650 1651 1652 1653 1654 1655 1656 1657
      it 'flushes the root ref cache' do
        expect(repository).to receive(:expire_root_ref_cache)

        repository.before_delete
      end

      it 'flushes the emptiness caches' do
        expect(repository).to receive(:expire_emptiness_caches)

        repository.before_delete
      end
1658 1659

      it 'flushes the exists cache' do
1660
        expect(repository).to receive(:expire_exists_cache).twice
1661 1662 1663

        repository.before_delete
      end
1664 1665 1666 1667 1668 1669 1670
    end

    describe 'when a repository exists' do
      before do
        allow(repository).to receive(:exists?).and_return(true)
      end

1671 1672 1673 1674 1675 1676 1677 1678 1679 1680 1681 1682
      it 'flushes the tags cache' do
        expect(repository).to receive(:expire_tags_cache)

        repository.before_delete
      end

      it 'flushes the branches cache' do
        expect(repository).to receive(:expire_branches_cache)

        repository.before_delete
      end

1683 1684 1685 1686 1687 1688 1689 1690 1691 1692 1693 1694 1695 1696 1697 1698 1699 1700 1701 1702 1703 1704 1705 1706 1707 1708 1709 1710
      it 'flushes the root ref cache' do
        expect(repository).to receive(:expire_root_ref_cache)

        repository.before_delete
      end

      it 'flushes the emptiness caches' do
        expect(repository).to receive(:expire_emptiness_caches)

        repository.before_delete
      end
    end
  end

  describe '#before_change_head' do
    it 'flushes the branch cache' do
      expect(repository).to receive(:expire_branch_cache)

      repository.before_change_head
    end

    it 'flushes the root ref cache' do
      expect(repository).to receive(:expire_root_ref_cache)

      repository.before_change_head
    end
  end

1711 1712 1713 1714 1715 1716 1717 1718 1719 1720
  describe '#after_change_head' do
    it 'flushes the readme cache' do
      expect(repository).to receive(:expire_method_caches).with([
        :readme,
        :changelog,
        :license,
        :contributing,
        :gitignore,
        :koding,
        :gitlab_ci,
1721 1722
        :avatar,
        :issue_template,
1723 1724
        :merge_request_template,
        :xcode_config
1725 1726 1727 1728 1729 1730
      ])

      repository.after_change_head
    end
  end

1731
  describe '#before_push_tag' do
1732
    it 'flushes the cache' do
1733 1734 1735
      expect(repository).to receive(:expire_statistics_caches)
      expect(repository).to receive(:expire_emptiness_caches)
      expect(repository).to receive(:expire_tags_cache)
1736

1737
      repository.before_push_tag
1738 1739 1740 1741
    end
  end

  describe '#after_import' do
1742 1743
    it 'flushes and builds the cache' do
      expect(repository).to receive(:expire_content_cache)
1744 1745 1746

      repository.after_import
    end
1747 1748 1749
  end

  describe '#after_push_commit' do
1750
    it 'expires statistics caches' do
1751 1752
      expect(repository).to receive(:expire_statistics_caches)
        .and_call_original
1753

1754 1755 1756
      expect(repository).to receive(:expire_branch_cache)
        .with('master')
        .and_call_original
1757

1758
      repository.after_push_commit('master')
1759 1760 1761 1762
    end
  end

  describe '#after_create_branch' do
1763
    it 'expires the branch caches' do
1764
      expect(repository).to receive(:expire_branches_cache)
1765 1766 1767 1768 1769 1770

      repository.after_create_branch
    end
  end

  describe '#after_remove_branch' do
1771
    it 'expires the branch caches' do
1772
      expect(repository).to receive(:expire_branches_cache)
1773 1774 1775 1776

      repository.after_remove_branch
    end
  end
1777

1778 1779 1780 1781 1782 1783
  describe '#after_create' do
    it 'flushes the exists cache' do
      expect(repository).to receive(:expire_exists_cache)

      repository.after_create
    end
1784 1785 1786 1787 1788 1789 1790 1791 1792 1793 1794 1795

    it 'flushes the root ref cache' do
      expect(repository).to receive(:expire_root_ref_cache)

      repository.after_create
    end

    it 'flushes the emptiness caches' do
      expect(repository).to receive(:expire_emptiness_caches)

      repository.after_create
    end
1796 1797
  end

1798 1799 1800 1801 1802 1803 1804 1805 1806 1807
  describe "#copy_gitattributes" do
    it 'returns true with a valid ref' do
      expect(repository.copy_gitattributes('master')).to be_truthy
    end

    it 'returns false with an invalid ref' do
      expect(repository.copy_gitattributes('invalid')).to be_falsey
    end
  end

1808 1809
  describe '#before_remove_tag' do
    it 'flushes the tag cache' do
1810 1811
      expect(repository).to receive(:expire_tags_cache).and_call_original
      expect(repository).to receive(:expire_statistics_caches).and_call_original
1812 1813 1814 1815 1816 1817 1818

      repository.before_remove_tag
    end
  end

  describe '#branch_count' do
    it 'returns the number of branches' do
Douwe Maan's avatar
Douwe Maan committed
1819
      expect(repository.branch_count).to be_an(Integer)
Kim "BKC" Carlbäcker's avatar
Kim "BKC" Carlbäcker committed
1820 1821

      # NOTE: Until rugged goes away, make sure rugged and gitaly are in sync
1822 1823 1824
      rugged_count = Gitlab::GitalyClient::StorageSettings.allow_disk_access do
        repository.raw_repository.rugged.branches.count
      end
Kim "BKC" Carlbäcker's avatar
Kim "BKC" Carlbäcker committed
1825

1826
      expect(repository.branch_count).to eq(rugged_count)
1827 1828 1829 1830 1831
    end
  end

  describe '#tag_count' do
    it 'returns the number of tags' do
Douwe Maan's avatar
Douwe Maan committed
1832
      expect(repository.tag_count).to be_an(Integer)
Kim "BKC" Carlbäcker's avatar
Kim "BKC" Carlbäcker committed
1833

1834
      # NOTE: Until rugged goes away, make sure rugged and gitaly are in sync
1835 1836 1837
      rugged_count = Gitlab::GitalyClient::StorageSettings.allow_disk_access do
        repository.raw_repository.rugged.tags.count
      end
Kim "BKC" Carlbäcker's avatar
Kim "BKC" Carlbäcker committed
1838

1839
      expect(repository.tag_count).to eq(rugged_count)
1840 1841 1842
    end
  end

1843
  describe '#expire_branches_cache' do
1844
    it 'expires the cache' do
1845
      expect(repository).to receive(:expire_method_caches)
1846
        .with(%i(branch_names branch_count has_visible_content?))
1847
        .and_call_original
1848

1849
      repository.expire_branches_cache
1850 1851 1852
    end
  end

1853
  describe '#expire_tags_cache' do
1854
    it 'expires the cache' do
1855 1856 1857
      expect(repository).to receive(:expire_method_caches)
        .with(%i(tag_names tag_count))
        .and_call_original
1858

1859
      repository.expire_tags_cache
1860 1861
    end
  end
1862

1863
  describe '#add_tag' do
1864
    let(:user) { build_stubbed(:user) }
1865

1866 1867 1868 1869
    shared_examples 'adding tag' do
      context 'with a valid target' do
        it 'creates the tag' do
          repository.add_tag(user, '8.5', 'master', 'foo')
1870

1871 1872 1873 1874 1875
          tag = repository.find_tag('8.5')
          expect(tag).to be_present
          expect(tag.message).to eq('foo')
          expect(tag.dereferenced_target.id).to eq(repository.commit('master').id)
        end
1876

1877 1878 1879 1880 1881 1882
        it 'returns a Gitlab::Git::Tag object' do
          tag = repository.add_tag(user, '8.5', 'master', 'foo')

          expect(tag).to be_a(Gitlab::Git::Tag)
        end
      end
1883

1884 1885 1886 1887
      context 'with an invalid target' do
        it 'returns false' do
          expect(repository.add_tag(user, '8.5', 'bar', 'foo')).to be false
        end
1888
      end
1889
    end
1890

1891 1892 1893
    context 'when Gitaly operation_user_add_tag feature is enabled' do
      it_behaves_like 'adding tag'
    end
1894

1895
    context 'when Gitaly operation_user_add_tag feature is disabled', :disable_gitaly do
1896 1897 1898
      it_behaves_like 'adding tag'

      it 'passes commit SHA to pre-receive and update hooks and tag SHA to post-receive hook' do
1899 1900 1901
        pre_receive_hook = Gitlab::Git::Hook.new('pre-receive', project)
        update_hook = Gitlab::Git::Hook.new('update', project)
        post_receive_hook = Gitlab::Git::Hook.new('post-receive', project)
1902

1903 1904
        allow(Gitlab::Git::Hook).to receive(:new)
          .and_return(pre_receive_hook, update_hook, post_receive_hook)
1905

1906 1907 1908
        allow(pre_receive_hook).to receive(:trigger).and_call_original
        allow(update_hook).to receive(:trigger).and_call_original
        allow(post_receive_hook).to receive(:trigger).and_call_original
1909

1910 1911
        tag = repository.add_tag(user, '8.5', 'master', 'foo')

1912 1913 1914
        commit_sha = repository.commit('master').id
        tag_sha = tag.target

1915
        expect(pre_receive_hook).to have_received(:trigger)
1916
          .with(anything, anything, anything, commit_sha, anything)
1917
        expect(update_hook).to have_received(:trigger)
1918
          .with(anything, anything, anything, commit_sha, anything)
1919
        expect(post_receive_hook).to have_received(:trigger)
1920
          .with(anything, anything, anything, tag_sha, anything)
1921 1922
      end
    end
1923 1924
  end

1925
  describe '#rm_branch' do
1926 1927 1928 1929
    shared_examples "user deleting a branch" do
      it 'removes a branch' do
        expect(repository).to receive(:before_remove_branch)
        expect(repository).to receive(:after_remove_branch)
1930

1931 1932 1933
        repository.rm_branch(user, 'feature')
      end
    end
1934

1935 1936
    context 'with gitaly enabled' do
      it_behaves_like "user deleting a branch"
1937

1938 1939 1940
      context 'when pre hooks failed' do
        before do
          allow_any_instance_of(Gitlab::GitalyClient::OperationService)
1941
            .to receive(:user_delete_branch).and_raise(Gitlab::Git::PreReceiveError)
1942 1943 1944 1945 1946
        end

        it 'gets an error and does not delete the branch' do
          expect do
            repository.rm_branch(user, 'feature')
1947
          end.to raise_error(Gitlab::Git::PreReceiveError)
1948 1949 1950 1951 1952 1953

          expect(repository.find_branch('feature')).not_to be_nil
        end
      end
    end

1954
    context 'with gitaly disabled', :disable_gitaly do
1955 1956 1957 1958 1959 1960 1961 1962 1963 1964 1965 1966 1967 1968 1969
      it_behaves_like "user deleting a branch"

      let(:old_rev) { '0b4bc9a49b562e85de7cc9e834518ea6828729b9' } # git rev-parse feature
      let(:blank_sha) { '0000000000000000000000000000000000000000' }

      context 'when pre hooks were successful' do
        it 'runs without errors' do
          expect_any_instance_of(Gitlab::Git::HooksService).to receive(:execute)
            .with(git_user, repository.raw_repository, old_rev, blank_sha, 'refs/heads/feature')

          expect { repository.rm_branch(user, 'feature') }.not_to raise_error
        end

        it 'deletes the branch' do
          allow_any_instance_of(Gitlab::Git::Hook).to receive(:trigger).and_return([true, nil])
1970

1971 1972 1973 1974 1975 1976 1977 1978 1979 1980 1981 1982
          expect { repository.rm_branch(user, 'feature') }.not_to raise_error

          expect(repository.find_branch('feature')).to be_nil
        end
      end

      context 'when pre hooks failed' do
        it 'gets an error' do
          allow_any_instance_of(Gitlab::Git::Hook).to receive(:trigger).and_return([false, ''])

          expect do
            repository.rm_branch(user, 'feature')
1983
          end.to raise_error(Gitlab::Git::PreReceiveError)
1984 1985 1986 1987 1988 1989 1990
        end

        it 'does not delete the branch' do
          allow_any_instance_of(Gitlab::Git::Hook).to receive(:trigger).and_return([false, ''])

          expect do
            repository.rm_branch(user, 'feature')
1991
          end.to raise_error(Gitlab::Git::PreReceiveError)
1992 1993 1994
          expect(repository.find_branch('feature')).not_to be_nil
        end
      end
1995 1996
    end
  end
1997 1998

  describe '#rm_tag' do
1999 2000 2001
    shared_examples 'removing tag' do
      it 'removes a tag' do
        expect(repository).to receive(:before_remove_tag)
2002

2003
        repository.rm_tag(build_stubbed(:user), 'v1.1.0')
Lin Jen-Shin's avatar
Lin Jen-Shin committed
2004

2005 2006 2007 2008 2009 2010 2011
        expect(repository.find_tag('v1.1.0')).to be_nil
      end
    end

    context 'when Gitaly operation_user_delete_tag feature is enabled' do
      it_behaves_like 'removing tag'
    end
2012

2013
    context 'when Gitaly operation_user_delete_tag feature is disabled', :skip_gitaly_mock do
2014
      it_behaves_like 'removing tag'
2015 2016
    end
  end
2017 2018

  describe '#avatar' do
2019
    it 'returns nil if repo does not exist' do
2020
      allow(repository).to receive(:root_ref).and_raise(Gitlab::Git::Repository::NoRepository)
2021 2022 2023 2024

      expect(repository.avatar).to eq(nil)
    end

2025
    it 'returns the first avatar file found in the repository' do
2026 2027 2028
      expect(repository).to receive(:file_on_head)
        .with(:avatar)
        .and_return(double(:tree, path: 'logo.png'))
2029 2030 2031 2032 2033

      expect(repository.avatar).to eq('logo.png')
    end

    it 'caches the output' do
2034 2035 2036 2037
      expect(repository).to receive(:file_on_head)
        .with(:avatar)
        .once
        .and_return(double(:tree, path: 'logo.png'))
2038

2039
      2.times { expect(repository.avatar).to eq('logo.png') }
2040 2041 2042
    end
  end

2043
  describe '#expire_exists_cache' do
2044 2045
    let(:cache) { repository.send(:cache) }

2046 2047 2048 2049
    it 'expires the cache' do
      expect(cache).to receive(:expire).with(:exists?)

      repository.expire_exists_cache
2050
    end
2051
  end
2052

2053 2054
  describe '#xcode_project?' do
    before do
2055
      allow(repository).to receive(:tree).with(:head).and_return(double(:tree, trees: [tree]))
2056 2057
    end

2058 2059
    context 'when the root contains a *.xcodeproj directory' do
      let(:tree) { double(:tree, path: 'Foo.xcodeproj') }
2060 2061 2062 2063 2064 2065

      it 'returns true' do
        expect(repository.xcode_project?).to be_truthy
      end
    end

2066 2067
    context 'when the root contains a *.xcworkspace directory' do
      let(:tree) { double(:tree, path: 'Foo.xcworkspace') }
2068 2069 2070 2071 2072 2073

      it 'returns true' do
        expect(repository.xcode_project?).to be_truthy
      end
    end

2074 2075
    context 'when the root contains no Xcode config directory' do
      let(:tree) { double(:tree, path: 'Foo') }
2076 2077 2078 2079 2080 2081 2082

      it 'returns false' do
        expect(repository.xcode_project?).to be_falsey
      end
    end
  end

2083
  describe "#keep_around" do
2084 2085 2086
    it "does not fail if we attempt to reference bad commit" do
      expect(repository.kept_around?('abc1234')).to be_falsey
    end
2087

2088 2089 2090 2091 2092
    it "stores a reference to the specified commit sha so it isn't garbage collected" do
      repository.keep_around(sample_commit.id)

      expect(repository.kept_around?(sample_commit.id)).to be_truthy
    end
2093 2094 2095 2096

    it "attempting to call keep_around on truncated ref does not fail" do
      repository.keep_around(sample_commit.id)
      ref = repository.send(:keep_around_ref_name, sample_commit.id)
2097 2098 2099 2100

      path = Gitlab::GitalyClient::StorageSettings.allow_disk_access do
        File.join(repository.path, ref)
      end
2101 2102 2103 2104 2105 2106 2107 2108 2109 2110
      # Corrupt the reference
      File.truncate(path, 0)

      expect(repository.kept_around?(sample_commit.id)).to be_falsey

      repository.keep_around(sample_commit.id)

      expect(repository.kept_around?(sample_commit.id)).to be_falsey

      File.delete(path)
2111
    end
2112
  end
2113

2114
  describe '#update_ref' do
2115 2116 2117 2118 2119 2120 2121
    around do |example|
      # TODO Gitlab::Git::OperationService will be moved to gitaly-ruby and disappear from this repo
      Gitlab::GitalyClient::StorageSettings.allow_disk_access do
        example.run
      end
    end

2122
    it 'can create a ref' do
2123
      Gitlab::Git::OperationService.new(nil, repository.raw_repository).send(:update_ref, 'refs/heads/foobar', 'refs/heads/master', Gitlab::Git::BLANK_SHA)
2124 2125 2126

      expect(repository.find_branch('foobar')).not_to be_nil
    end
2127

2128 2129
    it 'raises CommitError when the ref update fails' do
      expect do
2130 2131
        Gitlab::Git::OperationService.new(nil, repository.raw_repository).send(:update_ref, 'refs/heads/master', 'refs/heads/master', Gitlab::Git::BLANK_SHA)
      end.to raise_error(Gitlab::Git::CommitError)
2132 2133
    end
  end
2134

2135
  describe '#contribution_guide', :use_clean_rails_memory_store_caching do
2136
    it 'returns and caches the output' do
2137 2138 2139 2140
      expect(repository).to receive(:file_on_head)
        .with(:contributing)
        .and_return(Gitlab::Git::Tree.new(path: 'CONTRIBUTING.md'))
        .once
2141 2142

      2.times do
2143 2144
        expect(repository.contribution_guide)
          .to be_an_instance_of(Gitlab::Git::Tree)
2145
      end
2146 2147
    end
  end
2148

2149
  describe '#gitignore', :use_clean_rails_memory_store_caching do
2150
    it 'returns and caches the output' do
2151 2152 2153 2154
      expect(repository).to receive(:file_on_head)
        .with(:gitignore)
        .and_return(Gitlab::Git::Tree.new(path: '.gitignore'))
        .once
2155

2156
      2.times do
2157
        expect(repository.gitignore).to be_an_instance_of(Gitlab::Git::Tree)
2158 2159
      end
    end
2160
  end
2161

2162
  describe '#koding_yml', :use_clean_rails_memory_store_caching do
2163
    it 'returns and caches the output' do
2164 2165 2166 2167
      expect(repository).to receive(:file_on_head)
        .with(:koding)
        .and_return(Gitlab::Git::Tree.new(path: '.koding.yml'))
        .once
2168

2169
      2.times do
2170
        expect(repository.koding_yml).to be_an_instance_of(Gitlab::Git::Tree)
2171
      end
2172 2173
    end
  end
2174

2175
  describe '#readme', :use_clean_rails_memory_store_caching do
2176 2177
    context 'with a non-existing repository' do
      it 'returns nil' do
2178
        allow(repository).to receive(:tree).with(:head).and_return(nil)
2179

2180 2181 2182
        expect(repository.readme).to be_nil
      end
    end
2183

2184
    context 'with an existing repository' do
2185 2186 2187 2188 2189 2190 2191 2192 2193 2194 2195 2196
      context 'when no README exists' do
        it 'returns nil' do
          allow_any_instance_of(Tree).to receive(:readme).and_return(nil)

          expect(repository.readme).to be_nil
        end
      end

      context 'when a README exists' do
        it 'returns the README' do
          expect(repository.readme).to be_an_instance_of(ReadmeBlob)
        end
2197
      end
2198 2199
    end
  end
2200

2201 2202
  describe '#expire_statistics_caches' do
    it 'expires the caches' do
2203 2204
      expect(repository).to receive(:expire_method_caches)
        .with(%i(size commit_count))
2205

2206 2207 2208
      repository.expire_statistics_caches
    end
  end
2209

2210 2211
  describe '#expire_all_method_caches' do
    it 'expires the caches of all methods' do
2212 2213
      expect(repository).to receive(:expire_method_caches)
        .with(Repository::CACHED_METHODS)
2214

2215 2216
      repository.expire_all_method_caches
    end
2217 2218 2219 2220 2221 2222 2223 2224 2225

    it 'all cache_method definitions are in the lists of method caches' do
      methods = repository.methods.map do |method|
        match = /^_uncached_(.*)/.match(method)
        match[1].to_sym if match
      end.compact

      expect(methods).to match_array(Repository::CACHED_METHODS + Repository::MEMOIZED_CACHED_METHODS)
    end
2226 2227 2228 2229 2230 2231 2232 2233 2234 2235
  end

  describe '#file_on_head' do
    context 'with a non-existing repository' do
      it 'returns nil' do
        expect(repository).to receive(:tree).with(:head).and_return(nil)

        expect(repository.file_on_head(:readme)).to be_nil
      end
    end
2236

2237 2238 2239
    context 'with a repository that has no blobs' do
      it 'returns nil' do
        expect_any_instance_of(Tree).to receive(:blobs).and_return([])
2240

2241 2242 2243 2244 2245 2246
        expect(repository.file_on_head(:readme)).to be_nil
      end
    end

    context 'with an existing repository' do
      it 'returns a Gitlab::Git::Tree' do
2247 2248
        expect(repository.file_on_head(:readme))
          .to be_an_instance_of(Gitlab::Git::Tree)
2249 2250 2251 2252
      end
    end
  end

2253 2254 2255 2256 2257 2258
  describe '#head_tree' do
    context 'with an existing repository' do
      it 'returns a Tree' do
        expect(repository.head_tree).to be_an_instance_of(Tree)
      end
    end
2259

2260 2261 2262
    context 'with a non-existing repository' do
      it 'returns nil' do
        expect(repository).to receive(:head_commit).and_return(nil)
2263

2264
        expect(repository.head_tree).to be_nil
2265
      end
2266 2267
    end
  end
2268

2269 2270 2271 2272 2273
  describe '#tree' do
    context 'using a non-existing repository' do
      before do
        allow(repository).to receive(:head_commit).and_return(nil)
      end
2274

2275 2276 2277 2278 2279 2280 2281
      it 'returns nil' do
        expect(repository.tree(:head)).to be_nil
      end

      it 'returns nil when using a path' do
        expect(repository.tree(:head, 'README.md')).to be_nil
      end
2282 2283
    end

2284 2285 2286 2287 2288 2289
    context 'using an existing repository' do
      it 'returns a Tree' do
        expect(repository.tree(:head)).to be_an_instance_of(Tree)
      end
    end
  end
2290

2291 2292 2293 2294
  describe '#size' do
    context 'with a non-existing repository' do
      it 'returns 0' do
        expect(repository).to receive(:exists?).and_return(false)
2295

2296
        expect(repository.size).to eq(0.0)
2297
      end
2298
    end
2299

2300 2301 2302 2303
    context 'with an existing repository' do
      it 'returns the repository size as a Float' do
        expect(repository.size).to be_an_instance_of(Float)
      end
2304 2305
    end
  end
2306

2307 2308
  describe '#local_branches' do
    it 'returns the local branches' do
2309
      masterrev = repository.find_branch('master').dereferenced_target
2310
      create_remote_branch('joe', 'remote_branch', masterrev)
2311
      repository.add_branch(user, 'local_branch', masterrev.id)
2312 2313 2314 2315 2316 2317

      expect(repository.local_branches.any? { |branch| branch.name == 'remote_branch' }).to eq(false)
      expect(repository.local_branches.any? { |branch| branch.name == 'local_branch' }).to eq(true)
    end
  end

2318 2319
  describe '#remote_branches' do
    it 'returns the remote branches' do
2320
      masterrev = repository.find_branch('master').dereferenced_target
2321
      create_remote_branch('joe', 'remote_branch', masterrev)
2322
      repository.add_branch(user, 'local_branch', masterrev.id)
2323 2324 2325 2326 2327 2328 2329 2330

      expect(repository.remote_branches('joe').any? { |branch| branch.name == 'local_branch' }).to eq(false)
      expect(repository.remote_branches('joe').any? { |branch| branch.name == 'remote_branch' }).to eq(true)
    end
  end

  describe '#upstream_branches' do
    it 'returns branches from the upstream remote' do
2331
      masterrev = repository.find_branch('master').dereferenced_target
2332 2333 2334 2335 2336 2337 2338 2339
      create_remote_branch('upstream', 'upstream_branch', masterrev)

      expect(repository.upstream_branches.size).to eq(1)
      expect(repository.upstream_branches.first).to be_an_instance_of(Gitlab::Git::Branch)
      expect(repository.upstream_branches.first.name).to eq('upstream_branch')
    end
  end

2340 2341 2342 2343 2344 2345 2346
  describe '#commit_count' do
    context 'with a non-existing repository' do
      it 'returns 0' do
        expect(repository).to receive(:root_ref).and_return(nil)

        expect(repository.commit_count).to eq(0)
      end
2347 2348
    end

2349 2350
    context 'with an existing repository' do
      it 'returns the commit count' do
Douwe Maan's avatar
Douwe Maan committed
2351
        expect(repository.commit_count).to be_an(Integer)
2352 2353 2354
      end
    end
  end
2355

2356
  describe '#commit_count_for_ref' do
2357
    let(:project) { create :project }
2358

2359 2360
    context 'with a non-existing repository' do
      it 'returns 0' do
2361 2362 2363 2364 2365 2366 2367 2368
        expect(project.repository.commit_count_for_ref('master')).to eq(0)
      end
    end

    context 'with empty repository' do
      it 'returns 0' do
        project.create_repository
        expect(project.repository.commit_count_for_ref('master')).to eq(0)
2369 2370 2371 2372 2373 2374 2375 2376 2377 2378
      end
    end

    context 'when searching for the root ref' do
      it 'returns the same count as #commit_count' do
        expect(repository.commit_count_for_ref(repository.root_ref)).to eq(repository.commit_count)
      end
    end
  end

2379 2380 2381 2382 2383 2384 2385 2386 2387
  describe '#diverging_commit_counts' do
    it 'returns the commit counts behind and ahead of default branch' do
      result = repository.diverging_commit_counts(
        repository.find_branch('fix'))

      expect(result).to eq(behind: 29, ahead: 2)
    end
  end

2388 2389
  describe '#refresh_method_caches' do
    it 'refreshes the caches of the given types' do
2390 2391
      expect(repository).to receive(:expire_method_caches)
        .with(%i(rendered_readme license_blob license_key license))
2392

2393
      expect(repository).to receive(:rendered_readme)
2394 2395
      expect(repository).to receive(:license_blob)
      expect(repository).to receive(:license_key)
2396
      expect(repository).to receive(:license)
2397

2398
      repository.refresh_method_caches(%i(readme license))
2399 2400
    end
  end
2401

Douwe Maan's avatar
Douwe Maan committed
2402 2403
  describe '#gitlab_ci_yml_for' do
    before do
2404
      repository.create_file(User.last, '.gitlab-ci.yml', 'CONTENT', message: 'Add .gitlab-ci.yml', branch_name: 'master')
Douwe Maan's avatar
Douwe Maan committed
2405 2406 2407 2408 2409 2410 2411 2412 2413 2414 2415 2416 2417 2418 2419 2420 2421
    end

    context 'when there is a .gitlab-ci.yml at the commit' do
      it 'returns the content' do
        expect(repository.gitlab_ci_yml_for(repository.commit.sha)).to eq('CONTENT')
      end
    end

    context 'when there is no .gitlab-ci.yml at the commit' do
      it 'returns nil' do
        expect(repository.gitlab_ci_yml_for(repository.commit.parent.sha)).to be_nil
      end
    end
  end

  describe '#route_map_for' do
    before do
2422
      repository.create_file(User.last, '.gitlab/route-map.yml', 'CONTENT', message: 'Add .gitlab/route-map.yml', branch_name: 'master')
Douwe Maan's avatar
Douwe Maan committed
2423 2424 2425 2426 2427 2428 2429 2430 2431 2432 2433 2434 2435 2436 2437
    end

    context 'when there is a .gitlab/route-map.yml at the commit' do
      it 'returns the content' do
        expect(repository.route_map_for(repository.commit.sha)).to eq('CONTENT')
      end
    end

    context 'when there is no .gitlab/route-map.yml at the commit' do
      it 'returns nil' do
        expect(repository.route_map_for(repository.commit.parent.sha)).to be_nil
      end
    end
  end

2438 2439 2440 2441 2442 2443 2444 2445 2446 2447
  describe '#after_sync' do
    it 'expires repository cache' do
      expect(repository).to receive(:expire_all_method_caches)
      expect(repository).to receive(:expire_branch_cache)
      expect(repository).to receive(:expire_content_cache)

      repository.after_sync
    end
  end

2448
  def create_remote_branch(remote_name, branch_name, target)
2449 2450 2451
    rugged = Gitlab::GitalyClient::StorageSettings.allow_disk_access do
      repository.rugged
    end
2452
    rugged.references.create("refs/remotes/#{remote_name}/#{branch_name}", target.id)
2453
  end
2454

2455
  describe '#ancestor?' do
2456 2457
    let(:commit) { repository.commit }
    let(:ancestor) { commit.parents.first }
2458

2459
    shared_examples '#ancestor?' do
2460
      it 'it is an ancestor' do
2461
        expect(repository.ancestor?(ancestor.id, commit.id)).to eq(true)
2462 2463 2464
      end

      it 'it is not an ancestor' do
2465
        expect(repository.ancestor?(commit.id, ancestor.id)).to eq(false)
2466 2467 2468
      end

      it 'returns false on nil-values' do
2469 2470 2471
        expect(repository.ancestor?(nil, commit.id)).to eq(false)
        expect(repository.ancestor?(ancestor.id, nil)).to eq(false)
        expect(repository.ancestor?(nil, nil)).to eq(false)
2472
      end
2473

2474 2475 2476
      it 'returns false for invalid commit IDs' do
        expect(repository.ancestor?(commit.id, Gitlab::Git::BLANK_SHA)).to eq(false)
        expect(repository.ancestor?( Gitlab::Git::BLANK_SHA, commit.id)).to eq(false)
2477
      end
2478
    end
2479

2480 2481 2482
    context 'with Gitaly enabled' do
      it_behaves_like('#ancestor?')
    end
2483

2484 2485
    context 'with Gitaly disabled', :skip_gitaly_mock do
      it_behaves_like('#ancestor?')
2486 2487
    end
  end
2488

2489 2490 2491 2492 2493 2494 2495 2496 2497 2498 2499 2500 2501 2502 2503 2504 2505 2506 2507 2508 2509 2510 2511 2512 2513 2514
  describe '#archive_metadata' do
    let(:ref) { 'master' }
    let(:storage_path) { '/tmp' }

    let(:prefix) { [project.path, ref].join('-') }
    let(:filename) { prefix + '.tar.gz' }

    subject(:result) { repository.archive_metadata(ref, storage_path, append_sha: false) }

    context 'with hashed storage disabled' do
      let(:project) { create(:project, :repository, :legacy_storage) }

      it 'uses the project path to generate the filename' do
        expect(result['ArchivePrefix']).to eq(prefix)
        expect(File.basename(result['ArchivePath'])).to eq(filename)
      end
    end

    context 'with hashed storage enabled' do
      it 'uses the project path to generate the filename' do
        expect(result['ArchivePrefix']).to eq(prefix)
        expect(File.basename(result['ArchivePath'])).to eq(filename)
      end
    end
  end

2515 2516 2517 2518 2519 2520 2521 2522 2523 2524 2525 2526 2527 2528 2529 2530 2531 2532 2533
  describe 'commit cache' do
    set(:project) { create(:project, :repository) }

    it 'caches based on SHA' do
      # Gets the commit oid, and warms the cache
      oid = project.commit.id

      expect(Gitlab::Git::Commit).not_to receive(:find).once

      project.commit_by(oid: oid)
    end

    it 'caches nil values' do
      expect(Gitlab::Git::Commit).to receive(:find).once

      project.commit_by(oid: '1' * 40)
      project.commit_by(oid: '1' * 40)
    end
  end
2534 2535 2536 2537 2538 2539 2540 2541 2542 2543 2544 2545 2546 2547 2548 2549 2550 2551 2552 2553

  describe '#raw_repository' do
    subject { repository.raw_repository }

    it 'returns a Gitlab::Git::Repository representation of the repository' do
      expect(subject).to be_a(Gitlab::Git::Repository)
      expect(subject.relative_path).to eq(project.disk_path + '.git')
      expect(subject.gl_repository).to eq("project-#{project.id}")
    end

    context 'with a wiki repository' do
      let(:repository) { project.wiki.repository }

      it 'creates a Gitlab::Git::Repository with the proper attributes' do
        expect(subject).to be_a(Gitlab::Git::Repository)
        expect(subject.relative_path).to eq(project.disk_path + '.wiki.git')
        expect(subject.gl_repository).to eq("wiki-#{project.id}")
      end
    end
  end
2554 2555 2556 2557 2558 2559 2560 2561 2562 2563 2564 2565 2566 2567 2568 2569 2570 2571 2572 2573 2574 2575 2576 2577 2578 2579 2580 2581 2582 2583 2584 2585 2586 2587 2588 2589 2590 2591 2592 2593 2594 2595 2596 2597 2598 2599 2600 2601 2602 2603 2604 2605 2606 2607 2608 2609 2610 2611 2612 2613 2614 2615 2616 2617 2618 2619 2620 2621 2622 2623 2624 2625 2626 2627 2628 2629 2630 2631 2632 2633 2634 2635 2636 2637 2638 2639 2640 2641 2642 2643 2644 2645 2646 2647 2648 2649 2650 2651 2652 2653 2654 2655 2656 2657 2658 2659 2660

  describe '#contributors' do
    let(:author_a) { build(:author, email: 'tiagonbotelho@hotmail.com', name: 'tiagonbotelho') }
    let(:author_b) { build(:author, email: 'gitlab@winniehell.de', name: 'Winnie') }
    let(:author_c) { build(:author, email: 'douwe@gitlab.com', name: 'Douwe Maan') }
    let(:stubbed_commits) do
      [build(:commit, author: author_a),
       build(:commit, author: author_a),
       build(:commit, author: author_b),
       build(:commit, author: author_c),
       build(:commit, author: author_c),
       build(:commit, author: author_c)]
    end
    let(:order_by) { nil }
    let(:sort) { nil }

    before do
      allow(repository).to receive(:commits).with(nil, limit: 2000, offset: 0, skip_merges: true).and_return(stubbed_commits)
    end

    subject { repository.contributors(order_by: order_by, sort: sort) }

    def expect_contributors(*contributors)
      expect(subject.map(&:email)).to eq(contributors.map(&:email))
    end

    it 'returns the array of Gitlab::Contributor for the repository' do
      expect_contributors(author_a, author_b, author_c)
    end

    context 'order_by email' do
      let(:order_by) { 'email' }

      context 'asc' do
        let(:sort) { 'asc' }

        it 'returns all the contributors ordered by email asc case insensitive' do
          expect_contributors(author_c, author_b, author_a)
        end
      end

      context 'desc' do
        let(:sort) { 'desc' }

        it 'returns all the contributors ordered by email desc case insensitive' do
          expect_contributors(author_a, author_b, author_c)
        end
      end
    end

    context 'order_by name' do
      let(:order_by) { 'name' }

      context 'asc' do
        let(:sort) { 'asc' }

        it 'returns all the contributors ordered by name asc case insensitive' do
          expect_contributors(author_c, author_a, author_b)
        end
      end

      context 'desc' do
        let(:sort) { 'desc' }

        it 'returns all the contributors ordered by name desc case insensitive' do
          expect_contributors(author_b, author_a, author_c)
        end
      end
    end

    context 'order_by commits' do
      let(:order_by) { 'commits' }

      context 'asc' do
        let(:sort) { 'asc' }

        it 'returns all the contributors ordered by commits asc' do
          expect_contributors(author_b, author_a, author_c)
        end
      end

      context 'desc' do
        let(:sort) { 'desc' }

        it 'returns all the contributors ordered by commits desc' do
          expect_contributors(author_c, author_a, author_b)
        end
      end
    end

    context 'invalid ordering' do
      let(:order_by) { 'unknown' }

      it 'returns the contributors unsorted' do
        expect_contributors(author_a, author_b, author_c)
      end
    end

    context 'invalid sorting' do
      let(:order_by) { 'name' }
      let(:sort) { 'unknown' }

      it 'returns the contributors unsorted' do
        expect_contributors(author_a, author_b, author_c)
      end
    end
  end
2661
end