commits_spec.rb 20.3 KB
Newer Older
1 2 3
require 'spec_helper'
require 'mime/types'

4
describe API::V3::Commits do
5 6 7 8 9 10 11
  let(:user) { create(:user) }
  let(:user2) { create(:user) }
  let!(:project) { create(:project, :repository, creator: user, namespace: user.namespace) }
  let!(:guest) { create(:project_member, :guest, user: user2, project: project) }
  let!(:note) { create(:note_on_commit, author: user, project: project, commit_id: project.repository.commit.id, note: 'a comment on a commit') }
  let!(:another_note) { create(:note_on_commit, author: user, project: project, commit_id: project.repository.commit.id, note: 'another comment on a commit') }

12
  before { project.add_reporter(user) }
13 14 15

  describe "List repository commits" do
    context "authorized user" do
16
      before { project.add_reporter(user2) }
17 18 19 20 21

      it "returns project commits" do
        commit = project.repository.commit
        get v3_api("/projects/#{project.id}/repository/commits", user)

22
        expect(response).to have_gitlab_http_status(200)
23 24 25 26 27 28 29 30 31 32
        expect(json_response).to be_an Array
        expect(json_response.first['id']).to eq(commit.id)
        expect(json_response.first['committer_name']).to eq(commit.committer_name)
        expect(json_response.first['committer_email']).to eq(commit.committer_email)
      end
    end

    context "unauthorized user" do
      it "does not return project commits" do
        get v3_api("/projects/#{project.id}/repository/commits")
33
        expect(response).to have_gitlab_http_status(401)
34 35 36 37 38
      end
    end

    context "since optional parameter" do
      it "returns project commits since provided parameter" do
39
        commits = project.repository.commits("master", limit: 2)
40 41 42 43 44 45 46 47 48 49 50 51
        since = commits.second.created_at

        get v3_api("/projects/#{project.id}/repository/commits?since=#{since.utc.iso8601}", user)

        expect(json_response.size).to eq 2
        expect(json_response.first["id"]).to eq(commits.first.id)
        expect(json_response.second["id"]).to eq(commits.second.id)
      end
    end

    context "until optional parameter" do
      it "returns project commits until provided parameter" do
52
        commits = project.repository.commits("master", limit: 20)
53 54 55 56
        before = commits.second.created_at

        get v3_api("/projects/#{project.id}/repository/commits?until=#{before.utc.iso8601}", user)

57
        if commits.size == 20
58 59 60 61 62 63 64 65 66 67 68 69 70 71
          expect(json_response.size).to eq(20)
        else
          expect(json_response.size).to eq(commits.size - 1)
        end

        expect(json_response.first["id"]).to eq(commits.second.id)
        expect(json_response.second["id"]).to eq(commits.third.id)
      end
    end

    context "invalid xmlschema date parameters" do
      it "returns an invalid parameter error message" do
        get v3_api("/projects/#{project.id}/repository/commits?since=invalid-date", user)

72
        expect(response).to have_gitlab_http_status(400)
73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88
        expect(json_response['error']).to eq('since is invalid')
      end
    end

    context "path optional parameter" do
      it "returns project commits matching provided path parameter" do
        path = 'files/ruby/popen.rb'

        get v3_api("/projects/#{project.id}/repository/commits?path=#{path}", user)

        expect(json_response.size).to eq(3)
        expect(json_response.first["id"]).to eq("570e7b2abdd848b95f2f578043fc23bd6f6fd24d")
      end
    end
  end

89
  describe "POST /projects/:id/repository/commits" do
90 91 92 93 94
    let!(:url) { "/projects/#{project.id}/repository/commits" }

    it 'returns a 403 unauthorized for user without permissions' do
      post v3_api(url, user2)

95
      expect(response).to have_gitlab_http_status(403)
96 97 98 99 100
    end

    it 'returns a 400 bad request if no params are given' do
      post v3_api(url, user)

101
      expect(response).to have_gitlab_http_status(400)
102 103
    end

104
    describe 'create' do
105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135
      let(:message) { 'Created file' }
      let!(:invalid_c_params) do
        {
          branch_name: 'master',
          commit_message: message,
          actions: [
            {
              action: 'create',
              file_path: 'files/ruby/popen.rb',
              content: 'puts 8'
            }
          ]
        }
      end
      let!(:valid_c_params) do
        {
          branch_name: 'master',
          commit_message: message,
          actions: [
            {
              action: 'create',
              file_path: 'foo/bar/baz.txt',
              content: 'puts 8'
            }
          ]
        }
      end

      it 'a new file in project repo' do
        post v3_api(url, user), valid_c_params

136
        expect(response).to have_gitlab_http_status(201)
137 138 139 140 141 142 143 144
        expect(json_response['title']).to eq(message)
        expect(json_response['committer_name']).to eq(user.name)
        expect(json_response['committer_email']).to eq(user.email)
      end

      it 'returns a 400 bad request if file exists' do
        post v3_api(url, user), invalid_c_params

145
        expect(response).to have_gitlab_http_status(400)
146 147
      end

148 149 150
      context 'with project path containing a dot in URL' do
        let!(:user) { create(:user, username: 'foo.bar') }
        let(:url) { "/projects/#{CGI.escape(project.full_path)}/repository/commits" }
151 152 153 154

        it 'a new file in project repo' do
          post v3_api(url, user), valid_c_params

155
          expect(response).to have_gitlab_http_status(201)
156 157 158 159
        end
      end
    end

160
    describe 'delete' do
161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189
      let(:message) { 'Deleted file' }
      let!(:invalid_d_params) do
        {
          branch_name: 'markdown',
          commit_message: message,
          actions: [
            {
              action: 'delete',
              file_path: 'doc/api/projects.md'
            }
          ]
        }
      end
      let!(:valid_d_params) do
        {
          branch_name: 'markdown',
          commit_message: message,
          actions: [
            {
              action: 'delete',
              file_path: 'doc/api/users.md'
            }
          ]
        }
      end

      it 'an existing file in project repo' do
        post v3_api(url, user), valid_d_params

190
        expect(response).to have_gitlab_http_status(201)
191 192 193 194 195 196
        expect(json_response['title']).to eq(message)
      end

      it 'returns a 400 bad request if file does not exist' do
        post v3_api(url, user), invalid_d_params

197
        expect(response).to have_gitlab_http_status(400)
198 199 200
      end
    end

201
    describe 'move' do
202 203 204 205 206 207 208 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
      let(:message) { 'Moved file' }
      let!(:invalid_m_params) do
        {
          branch_name: 'feature',
          commit_message: message,
          actions: [
            {
              action: 'move',
              file_path: 'CHANGELOG',
              previous_path: 'VERSION',
              content: '6.7.0.pre'
            }
          ]
        }
      end
      let!(:valid_m_params) do
        {
          branch_name: 'feature',
          commit_message: message,
          actions: [
            {
              action: 'move',
              file_path: 'VERSION.txt',
              previous_path: 'VERSION',
              content: '6.7.0.pre'
            }
          ]
        }
      end

      it 'an existing file in project repo' do
        post v3_api(url, user), valid_m_params

235
        expect(response).to have_gitlab_http_status(201)
236 237 238 239 240 241
        expect(json_response['title']).to eq(message)
      end

      it 'returns a 400 bad request if file does not exist' do
        post v3_api(url, user), invalid_m_params

242
        expect(response).to have_gitlab_http_status(400)
243 244 245
      end
    end

246
    describe 'update' do
247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277
      let(:message) { 'Updated file' }
      let!(:invalid_u_params) do
        {
          branch_name: 'master',
          commit_message: message,
          actions: [
            {
              action: 'update',
              file_path: 'foo/bar.baz',
              content: 'puts 8'
            }
          ]
        }
      end
      let!(:valid_u_params) do
        {
          branch_name: 'master',
          commit_message: message,
          actions: [
            {
              action: 'update',
              file_path: 'files/ruby/popen.rb',
              content: 'puts 8'
            }
          ]
        }
      end

      it 'an existing file in project repo' do
        post v3_api(url, user), valid_u_params

278
        expect(response).to have_gitlab_http_status(201)
279 280 281 282 283 284
        expect(json_response['title']).to eq(message)
      end

      it 'returns a 400 bad request if file does not exist' do
        post v3_api(url, user), invalid_u_params

285
        expect(response).to have_gitlab_http_status(400)
286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 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
      end
    end

    context "multiple operations" do
      let(:message) { 'Multiple actions' }
      let!(:invalid_mo_params) do
        {
          branch_name: 'master',
          commit_message: message,
          actions: [
            {
              action: 'create',
              file_path: 'files/ruby/popen.rb',
              content: 'puts 8'
            },
            {
              action: 'delete',
              file_path: 'doc/v3_api/projects.md'
            },
            {
              action: 'move',
              file_path: 'CHANGELOG',
              previous_path: 'VERSION',
              content: '6.7.0.pre'
            },
            {
              action: 'update',
              file_path: 'foo/bar.baz',
              content: 'puts 8'
            }
          ]
        }
      end
      let!(:valid_mo_params) do
        {
          branch_name: 'master',
          commit_message: message,
          actions: [
            {
              action: 'create',
              file_path: 'foo/bar/baz.txt',
              content: 'puts 8'
            },
            {
              action: 'delete',
              file_path: 'Gemfile.zip'
            },
            {
              action: 'move',
              file_path: 'VERSION.txt',
              previous_path: 'VERSION',
              content: '6.7.0.pre'
            },
            {
              action: 'update',
              file_path: 'files/ruby/popen.rb',
              content: 'puts 8'
            }
          ]
        }
      end

      it 'are commited as one in project repo' do
        post v3_api(url, user), valid_mo_params

351
        expect(response).to have_gitlab_http_status(201)
352 353 354 355 356 357
        expect(json_response['title']).to eq(message)
      end

      it 'return a 400 bad request if there are any issues' do
        post v3_api(url, user), invalid_mo_params

358
        expect(response).to have_gitlab_http_status(400)
359 360 361 362 363 364 365 366 367
      end
    end
  end

  describe "Get a single commit" do
    context "authorized user" do
      it "returns a commit by sha" do
        get v3_api("/projects/#{project.id}/repository/commits/#{project.repository.commit.id}", user)

368
        expect(response).to have_gitlab_http_status(200)
369 370 371 372 373 374 375 376 377
        expect(json_response['id']).to eq(project.repository.commit.id)
        expect(json_response['title']).to eq(project.repository.commit.title)
        expect(json_response['stats']['additions']).to eq(project.repository.commit.stats.additions)
        expect(json_response['stats']['deletions']).to eq(project.repository.commit.stats.deletions)
        expect(json_response['stats']['total']).to eq(project.repository.commit.stats.total)
      end

      it "returns a 404 error if not found" do
        get v3_api("/projects/#{project.id}/repository/commits/invalid_sha", user)
378
        expect(response).to have_gitlab_http_status(404)
379 380 381 382 383
      end

      it "returns nil for commit without CI" do
        get v3_api("/projects/#{project.id}/repository/commits/#{project.repository.commit.id}", user)

384
        expect(response).to have_gitlab_http_status(200)
385 386 387 388
        expect(json_response['status']).to be_nil
      end

      it "returns status for CI" do
S
Shinya Maeda 已提交
389
        pipeline = project.pipelines.create(source: :push, ref: 'master', sha: project.repository.commit.sha, protected: false)
390 391 392 393
        pipeline.update(status: 'success')

        get v3_api("/projects/#{project.id}/repository/commits/#{project.repository.commit.id}", user)

394
        expect(response).to have_gitlab_http_status(200)
395 396 397 398
        expect(json_response['status']).to eq(pipeline.status)
      end

      it "returns status for CI when pipeline is created" do
S
Shinya Maeda 已提交
399
        project.pipelines.create(source: :push, ref: 'master', sha: project.repository.commit.sha, protected: false)
400 401 402

        get v3_api("/projects/#{project.id}/repository/commits/#{project.repository.commit.id}", user)

403
        expect(response).to have_gitlab_http_status(200)
404 405
        expect(json_response['status']).to eq("created")
      end
406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432

      context 'when stat param' do
        let(:project_id) { project.id }
        let(:commit_id) { project.repository.commit.id }
        let(:route) { "/projects/#{project_id}/repository/commits/#{commit_id}" }

        it 'is not present return stats by default' do
          get v3_api(route, user)

          expect(response).to have_gitlab_http_status(200)
          expect(json_response).to include 'stats'
        end

        it "is false it does not include stats" do
          get v3_api(route, user), stats: false

          expect(response).to have_gitlab_http_status(200)
          expect(json_response).not_to include 'stats'
        end

        it "is true it includes stats" do
          get v3_api(route, user), stats: true

          expect(response).to have_gitlab_http_status(200)
          expect(json_response).to include 'stats'
        end
      end
433 434 435 436 437
    end

    context "unauthorized user" do
      it "does not return the selected commit" do
        get v3_api("/projects/#{project.id}/repository/commits/#{project.repository.commit.id}")
438
        expect(response).to have_gitlab_http_status(401)
439 440 441 442 443 444
      end
    end
  end

  describe "Get the diff of a commit" do
    context "authorized user" do
445
      before { project.add_reporter(user2) }
446 447 448

      it "returns the diff of the selected commit" do
        get v3_api("/projects/#{project.id}/repository/commits/#{project.repository.commit.id}/diff", user)
449
        expect(response).to have_gitlab_http_status(200)
450 451 452 453 454 455 456 457

        expect(json_response).to be_an Array
        expect(json_response.length).to be >= 1
        expect(json_response.first.keys).to include "diff"
      end

      it "returns a 404 error if invalid commit" do
        get v3_api("/projects/#{project.id}/repository/commits/invalid_sha/diff", user)
458
        expect(response).to have_gitlab_http_status(404)
459 460 461 462 463 464
      end
    end

    context "unauthorized user" do
      it "does not return the diff of the selected commit" do
        get v3_api("/projects/#{project.id}/repository/commits/#{project.repository.commit.id}/diff")
465
        expect(response).to have_gitlab_http_status(401)
466 467 468 469 470 471 472 473
      end
    end
  end

  describe 'Get the comments of a commit' do
    context 'authorized user' do
      it 'returns merge_request comments' do
        get v3_api("/projects/#{project.id}/repository/commits/#{project.repository.commit.id}/comments", user)
474
        expect(response).to have_gitlab_http_status(200)
475 476 477 478 479 480 481 482
        expect(json_response).to be_an Array
        expect(json_response.length).to eq(2)
        expect(json_response.first['note']).to eq('a comment on a commit')
        expect(json_response.first['author']['id']).to eq(user.id)
      end

      it 'returns a 404 error if merge_request_id not found' do
        get v3_api("/projects/#{project.id}/repository/commits/1234ab/comments", user)
483
        expect(response).to have_gitlab_http_status(404)
484 485 486 487 488 489
      end
    end

    context 'unauthorized user' do
      it 'does not return the diff of the selected commit' do
        get v3_api("/projects/#{project.id}/repository/commits/1234ab/comments")
490
        expect(response).to have_gitlab_http_status(401)
491 492 493 494 495 496 497 498 499 500 501
      end
    end
  end

  describe 'POST :id/repository/commits/:sha/cherry_pick' do
    let(:master_pickable_commit)  { project.commit('7d3b0f7cff5f37573aea97cebfd5692ea1689924') }

    context 'authorized user' do
      it 'cherry picks a commit' do
        post v3_api("/projects/#{project.id}/repository/commits/#{master_pickable_commit.id}/cherry_pick", user), branch: 'master'

502
        expect(response).to have_gitlab_http_status(201)
503
        expect(json_response['title']).to eq(master_pickable_commit.title)
504
        expect(json_response['message']).to eq(master_pickable_commit.cherry_pick_message(user))
505 506 507 508 509 510 511
        expect(json_response['author_name']).to eq(master_pickable_commit.author_name)
        expect(json_response['committer_name']).to eq(user.name)
      end

      it 'returns 400 if commit is already included in the target branch' do
        post v3_api("/projects/#{project.id}/repository/commits/#{master_pickable_commit.id}/cherry_pick", user), branch: 'markdown'

512
        expect(response).to have_gitlab_http_status(400)
D
Douwe Maan 已提交
513
        expect(json_response['message']).to include('Sorry, we cannot cherry-pick this commit automatically.')
514 515 516
      end

      it 'returns 400 if you are not allowed to push to the target branch' do
517
        project.add_developer(user2)
518 519 520 521
        protected_branch = create(:protected_branch, project: project, name: 'feature')

        post v3_api("/projects/#{project.id}/repository/commits/#{master_pickable_commit.id}/cherry_pick", user2), branch: protected_branch.name

522
        expect(response).to have_gitlab_http_status(400)
523 524 525 526 527 528
        expect(json_response['message']).to eq('You are not allowed to push into this branch')
      end

      it 'returns 400 for missing parameters' do
        post v3_api("/projects/#{project.id}/repository/commits/#{master_pickable_commit.id}/cherry_pick", user)

529
        expect(response).to have_gitlab_http_status(400)
530 531 532 533 534 535
        expect(json_response['error']).to eq('branch is missing')
      end

      it 'returns 404 if commit is not found' do
        post v3_api("/projects/#{project.id}/repository/commits/abcd0123/cherry_pick", user), branch: 'master'

536
        expect(response).to have_gitlab_http_status(404)
537 538 539 540 541 542
        expect(json_response['message']).to eq('404 Commit Not Found')
      end

      it 'returns 404 if branch is not found' do
        post v3_api("/projects/#{project.id}/repository/commits/#{master_pickable_commit.id}/cherry_pick", user), branch: 'foo'

543
        expect(response).to have_gitlab_http_status(404)
544 545 546 547 548 549
        expect(json_response['message']).to eq('404 Branch Not Found')
      end

      it 'returns 400 for missing parameters' do
        post v3_api("/projects/#{project.id}/repository/commits/#{master_pickable_commit.id}/cherry_pick", user)

550
        expect(response).to have_gitlab_http_status(400)
551 552 553 554 555 556 557 558
        expect(json_response['error']).to eq('branch is missing')
      end
    end

    context 'unauthorized user' do
      it 'does not cherry pick the commit' do
        post v3_api("/projects/#{project.id}/repository/commits/#{master_pickable_commit.id}/cherry_pick"), branch: 'master'

559
        expect(response).to have_gitlab_http_status(401)
560 561 562 563 564 565 566 567
      end
    end
  end

  describe 'Post comment to commit' do
    context 'authorized user' do
      it 'returns comment' do
        post v3_api("/projects/#{project.id}/repository/commits/#{project.repository.commit.id}/comments", user), note: 'My comment'
568
        expect(response).to have_gitlab_http_status(201)
569 570 571 572 573 574 575 576 577
        expect(json_response['note']).to eq('My comment')
        expect(json_response['path']).to be_nil
        expect(json_response['line']).to be_nil
        expect(json_response['line_type']).to be_nil
      end

      it 'returns the inline comment' do
        post v3_api("/projects/#{project.id}/repository/commits/#{project.repository.commit.id}/comments", user), note: 'My comment', path: project.repository.commit.raw_diffs.first.new_path, line: 1, line_type: 'new'

578
        expect(response).to have_gitlab_http_status(201)
579 580 581 582 583 584 585 586
        expect(json_response['note']).to eq('My comment')
        expect(json_response['path']).to eq(project.repository.commit.raw_diffs.first.new_path)
        expect(json_response['line']).to eq(1)
        expect(json_response['line_type']).to eq('new')
      end

      it 'returns 400 if note is missing' do
        post v3_api("/projects/#{project.id}/repository/commits/#{project.repository.commit.id}/comments", user)
587
        expect(response).to have_gitlab_http_status(400)
588 589 590 591
      end

      it 'returns 404 if note is attached to non existent commit' do
        post v3_api("/projects/#{project.id}/repository/commits/1234ab/comments", user), note: 'My comment'
592
        expect(response).to have_gitlab_http_status(404)
593 594 595 596 597 598
      end
    end

    context 'unauthorized user' do
      it 'does not return the diff of the selected commit' do
        post v3_api("/projects/#{project.id}/repository/commits/#{project.repository.commit.id}/comments")
599
        expect(response).to have_gitlab_http_status(401)
600 601 602 603
      end
    end
  end
end