commit_spec.rb 9.99 KB
Newer Older
1 2
require 'spec_helper'

Douwe Maan's avatar
Douwe Maan committed
3
describe Commit, models: true do
Yorick Peterse's avatar
Yorick Peterse committed
4
  let(:project) { create(:project, :public) }
5 6 7 8 9 10 11 12 13 14 15
  let(:commit)  { project.commit }

  describe 'modules' do
    subject { described_class }

    it { is_expected.to include_module(Mentionable) }
    it { is_expected.to include_module(Participable) }
    it { is_expected.to include_module(Referable) }
    it { is_expected.to include_module(StaticModel) }
  end

16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35
  describe '#author' do
    it 'looks up the author in a case-insensitive way' do
      user = create(:user, email: commit.author_email.upcase)
      expect(commit.author).to eq(user)
    end

    it 'caches the author' do
      user = create(:user, email: commit.author_email)
      expect(RequestStore).to receive(:active?).twice.and_return(true)
      expect_any_instance_of(Commit).to receive(:find_author_by_any_email).and_call_original

      expect(commit.author).to eq(user)
      key = "commit_author:#{commit.author_email}"
      expect(RequestStore.store[key]).to eq(user)

      expect(commit.author).to eq(user)
      RequestStore.store.clear
    end
  end

36 37
  describe '#to_reference' do
    it 'returns a String reference to the object' do
38
      expect(commit.to_reference).to eq commit.id
39 40 41 42
    end

    it 'supports a cross-project reference' do
      cross = double('project')
43 44 45 46 47 48 49 50 51 52 53 54
      expect(commit.to_reference(cross)).to eq "#{project.to_reference}@#{commit.id}"
    end
  end

  describe '#reference_link_text' do
    it 'returns a String reference to the object' do
      expect(commit.reference_link_text).to eq commit.short_id
    end

    it 'supports a cross-project reference' do
      cross = double('project')
      expect(commit.reference_link_text(cross)).to eq "#{project.to_reference}@#{commit.short_id}"
55 56
    end
  end
57

58 59
  describe '#title' do
    it "returns no_commit_message when safe_message is blank" do
60 61
      allow(commit).to receive(:safe_message).and_return('')
      expect(commit.title).to eq("--no commit message")
62
    end
63

64
    it "truncates a message without a newline at 80 characters" do
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
65
      message = 'Lorem ipsum dolor sit amet, consectetur adipiscing elit. Donec sodales id felis id blandit. Vivamus egestas lacinia lacus, sed rutrum mauris.'
66

67
      allow(commit).to receive(:safe_message).and_return(message)
68
      expect(commit.title).to eq("#{message[0..79]}…")
69
    end
70

71 72
    it "truncates a message with a newline before 80 characters at the newline" do
      message = commit.safe_message.split(" ").first
73

74 75
      allow(commit).to receive(:safe_message).and_return(message + "\n" + message)
      expect(commit.title).to eq(message)
76
    end
77

Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
78
    it "does not truncates a message with a newline after 80 but less 100 characters" do
79
      message = <<eos
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
80 81 82
Lorem ipsum dolor sit amet, consectetur adipiscing elit. Donec sodales id felis id blandit.
Vivamus egestas lacinia lacus, sed rutrum mauris.
eos
83

84 85
      allow(commit).to receive(:safe_message).and_return(message)
      expect(commit.title).to eq(message.split("\n").first)
86 87
    end
  end
88

89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109
  describe '#full_title' do
    it "returns no_commit_message when safe_message is blank" do
      allow(commit).to receive(:safe_message).and_return('')
      expect(commit.full_title).to eq("--no commit message")
    end

    it "returns entire message if there is no newline" do
      message = 'Lorem ipsum dolor sit amet, consectetur adipiscing elit. Donec sodales id felis id blandit. Vivamus egestas lacinia lacus, sed rutrum mauris.'

      allow(commit).to receive(:safe_message).and_return(message)
      expect(commit.full_title).to eq(message)
    end

    it "returns first line of message if there is a newLine" do
      message = commit.safe_message.split(" ").first

      allow(commit).to receive(:safe_message).and_return(message + "\n" + message)
      expect(commit.full_title).to eq(message)
    end
  end

110 111 112
  describe "delegation" do
    subject { commit }

113 114 115 116 117 118 119 120 121 122 123
    it { is_expected.to respond_to(:message) }
    it { is_expected.to respond_to(:authored_date) }
    it { is_expected.to respond_to(:committed_date) }
    it { is_expected.to respond_to(:committer_email) }
    it { is_expected.to respond_to(:author_email) }
    it { is_expected.to respond_to(:parents) }
    it { is_expected.to respond_to(:date) }
    it { is_expected.to respond_to(:diffs) }
    it { is_expected.to respond_to(:tree) }
    it { is_expected.to respond_to(:id) }
    it { is_expected.to respond_to(:to_patch) }
124
  end
125 126 127

  describe '#closes_issues' do
    let(:issue) { create :issue, project: project }
128 129
    let(:other_project) { create :project, :public }
    let(:other_issue) { create :issue, project: other_project }
130 131 132 133 134 135
    let(:commiter) { create :user }

    before do
      project.team << [commiter, :developer]
      other_project.team << [commiter, :developer]
    end
136 137

    it 'detects issues that this commit is marked as closing' do
138
      ext_ref = "#{other_project.path_with_namespace}##{other_issue.iid}"
139 140 141 142 143 144

      allow(commit).to receive_messages(
        safe_message: "Fixes ##{issue.iid} and #{ext_ref}",
        committer_email: commiter.email
      )

Douwe Maan's avatar
Douwe Maan committed
145 146
      expect(commit.closes_issues).to include(issue)
      expect(commit.closes_issues).to include(other_issue)
147
    end
148 149 150
  end

  it_behaves_like 'a mentionable' do
Douwe Maan's avatar
Douwe Maan committed
151
    subject { create(:project).commit }
152

Douwe Maan's avatar
Douwe Maan committed
153
    let(:author) { create(:user, email: subject.author_email) }
154
    let(:backref_text) { "commit #{subject.id}" }
155 156 157
    let(:set_mentionable_text) do
      ->(txt) { allow(subject).to receive(:safe_message).and_return(txt) }
    end
158 159 160 161

    # Include the subject in the repository stub.
    let(:extra_commits) { [subject] }
  end
162 163

  describe '#hook_attrs' do
Valery Sizov's avatar
Valery Sizov committed
164
    let(:data) { commit.hook_attrs(with_changed_files: true) }
165 166

    it { expect(data).to be_a(Hash) }
167 168 169 170
    it { expect(data[:message]).to include('adds bar folder and branch-test text file to check Repository merged_to_root_ref method') }
    it { expect(data[:timestamp]).to eq('2016-09-27T14:37:46+00:00') }
    it { expect(data[:added]).to eq(["bar/branch-test.txt"]) }
    it { expect(data[:modified]).to eq([]) }
171 172
    it { expect(data[:removed]).to eq([]) }
  end
173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206

  describe '#reverts_commit?' do
    let(:another_commit) { double(:commit, revert_description: "This reverts commit #{commit.sha}") }

    it { expect(commit.reverts_commit?(another_commit)).to be_falsy }

    context 'commit has no description' do
      before { allow(commit).to receive(:description?).and_return(false) }

      it { expect(commit.reverts_commit?(another_commit)).to be_falsy }
    end

    context "another_commit's description does not revert commit" do
      before { allow(commit).to receive(:description).and_return("Foo Bar") }

      it { expect(commit.reverts_commit?(another_commit)).to be_falsy }
    end

    context "another_commit's description reverts commit" do
      before { allow(commit).to receive(:description).and_return("Foo #{another_commit.revert_description} Bar") }

      it { expect(commit.reverts_commit?(another_commit)).to be_truthy }
    end

    context "another_commit's description reverts merged merge request" do
      before do
        revert_description = "This reverts merge request !foo123"
        allow(another_commit).to receive(:revert_description).and_return(revert_description)
        allow(commit).to receive(:description).and_return("Foo #{another_commit.revert_description} Bar")
      end

      it { expect(commit.reverts_commit?(another_commit)).to be_truthy }
    end
  end
207 208

  describe '#status' do
209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254
    context 'without arguments for compound status' do
      shared_examples 'giving the status from pipeline' do
        it do
          expect(commit.status).to eq(Ci::Pipeline.status)
        end
      end

      context 'with pipelines' do
        let!(:pipeline) do
          create(:ci_empty_pipeline, project: project, sha: commit.sha)
        end

        it_behaves_like 'giving the status from pipeline'
      end

      context 'without pipelines' do
        it_behaves_like 'giving the status from pipeline'
      end
    end

    context 'when a particular ref is specified' do
      let!(:pipeline_from_master) do
        create(:ci_empty_pipeline,
               project: project,
               sha: commit.sha,
               ref: 'master',
               status: 'failed')
      end

      let!(:pipeline_from_fix) do
        create(:ci_empty_pipeline,
               project: project,
               sha: commit.sha,
               ref: 'fix',
               status: 'success')
      end

      it 'gives pipelines from a particular branch' do
        expect(commit.status('master')).to eq(pipeline_from_master.status)
        expect(commit.status('fix')).to eq(pipeline_from_fix.status)
      end

      it 'gives compound status if ref is nil' do
        expect(commit.status(nil)).to eq(commit.status)
      end
    end
255
  end
Yorick Peterse's avatar
Yorick Peterse committed
256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291

  describe '#participants' do
    let(:user1) { build(:user) }
    let(:user2) { build(:user) }

    let!(:note1) do
      create(:note_on_commit,
             commit_id: commit.id,
             project: project,
             note: 'foo')
    end

    let!(:note2) do
      create(:note_on_commit,
             commit_id: commit.id,
             project: project,
             note: 'bar')
    end

    before do
      allow(commit).to receive(:author).and_return(user1)
      allow(commit).to receive(:committer).and_return(user2)
    end

    it 'includes the commit author' do
      expect(commit.participants).to include(commit.author)
    end

    it 'includes the committer' do
      expect(commit.participants).to include(commit.committer)
    end

    it 'includes the authors of the commit notes' do
      expect(commit.participants).to include(note1.author, note2.author)
    end
  end
292 293 294 295 296

  describe '#uri_type' do
    it 'returns the URI type at the given path' do
      expect(commit.uri_type('files/html')).to be(:tree)
      expect(commit.uri_type('files/images/logo-black.png')).to be(:raw)
297
      expect(project.commit('video').uri_type('files/videos/intro.mp4')).to be(:raw)
298 299 300 301 302 303 304
      expect(commit.uri_type('files/js/application.js')).to be(:blob)
    end

    it "returns nil if the path doesn't exists" do
      expect(commit.uri_type('this/path/doesnt/exist')).to be_nil
    end
  end
305
end