issues_spec.rb 24.4 KB
Newer Older
Nihad Abbasov's avatar
Nihad Abbasov committed
1 2
require 'spec_helper'

Jeroen van Baarsen's avatar
Jeroen van Baarsen committed
3
describe API::API, api: true  do
4
  include ApiHelpers
5
  let(:user)        { create(:user) }
6
  let(:user2)       { create(:user) }
7 8 9 10
  let(:non_member)  { create(:user) }
  let(:author)      { create(:author) }
  let(:assignee)    { create(:assignee) }
  let(:admin)       { create(:user, :admin) }
11
  let!(:project)    { create(:project, :public, creator_id: user.id, namespace: user.namespace ) }
12 13 14 15 16 17 18 19
  let!(:closed_issue) do
    create :closed_issue,
           author: user,
           assignee: user,
           project: project,
           state: :closed,
           milestone: milestone
  end
20 21 22 23 24 25 26
  let!(:confidential_issue) do
    create :issue,
           :confidential,
           project: project,
           author: author,
           assignee: assignee
  end
27 28 29 30 31 32 33
  let!(:issue) do
    create :issue,
           author: user,
           assignee: user,
           project: project,
           milestone: milestone
  end
34 35 36
  let!(:label) do
    create(:label, title: 'label', color: '#FFAABB', project: project)
  end
jubianchi's avatar
jubianchi committed
37
  let!(:label_link) { create(:label_link, label: label, target: issue) }
38 39 40 41
  let!(:milestone) { create(:milestone, title: '1.0.0', project: project) }
  let!(:empty_milestone) do
    create(:milestone, title: '2.0.0', project: project)
  end
42
  let!(:note) { create(:note_on_issue, author: user, project: project, noteable: issue) }
43

Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
44
  before { project.team << [user, :reporter] }
Nihad Abbasov's avatar
Nihad Abbasov committed
45 46

  describe "GET /issues" do
47 48 49
    context "when unauthenticated" do
      it "should return authentication error" do
        get api("/issues")
50
        expect(response.status).to eq(401)
51
      end
Nihad Abbasov's avatar
Nihad Abbasov committed
52 53
    end

54
    context "when authenticated" do
Nihad Abbasov's avatar
Nihad Abbasov committed
55
      it "should return an array of issues" do
Robert Speicher's avatar
Robert Speicher committed
56
        get api("/issues", user)
57 58 59
        expect(response.status).to eq(200)
        expect(json_response).to be_an Array
        expect(json_response.first['title']).to eq(issue.title)
Nihad Abbasov's avatar
Nihad Abbasov committed
60
      end
61

62 63
      it "should add pagination headers and keep query params" do
        get api("/issues?state=closed&per_page=3", user)
64
        expect(response.headers['Link']).to eq(
65
          '<http://www.example.com/api/v3/issues?page=1&per_page=3&private_token=%s&state=closed>; rel="first", <http://www.example.com/api/v3/issues?page=1&per_page=3&private_token=%s&state=closed>; rel="last"' % [user.private_token, user.private_token]
66
        )
67
      end
jubianchi's avatar
jubianchi committed
68 69 70

      it 'should return an array of closed issues' do
        get api('/issues?state=closed', user)
71 72 73 74
        expect(response.status).to eq(200)
        expect(json_response).to be_an Array
        expect(json_response.length).to eq(1)
        expect(json_response.first['id']).to eq(closed_issue.id)
jubianchi's avatar
jubianchi committed
75 76 77 78
      end

      it 'should return an array of opened issues' do
        get api('/issues?state=opened', user)
79 80 81 82
        expect(response.status).to eq(200)
        expect(json_response).to be_an Array
        expect(json_response.length).to eq(1)
        expect(json_response.first['id']).to eq(issue.id)
jubianchi's avatar
jubianchi committed
83 84 85 86
      end

      it 'should return an array of all issues' do
        get api('/issues?state=all', user)
87 88 89 90 91
        expect(response.status).to eq(200)
        expect(json_response).to be_an Array
        expect(json_response.length).to eq(2)
        expect(json_response.first['id']).to eq(issue.id)
        expect(json_response.second['id']).to eq(closed_issue.id)
jubianchi's avatar
jubianchi committed
92
      end
jubianchi's avatar
jubianchi committed
93 94 95

      it 'should return an array of labeled issues' do
        get api("/issues?labels=#{label.title}", user)
96 97 98 99
        expect(response.status).to eq(200)
        expect(json_response).to be_an Array
        expect(json_response.length).to eq(1)
        expect(json_response.first['labels']).to eq([label.title])
jubianchi's avatar
jubianchi committed
100 101 102 103
      end

      it 'should return an array of labeled issues when at least one label matches' do
        get api("/issues?labels=#{label.title},foo,bar", user)
104 105 106 107
        expect(response.status).to eq(200)
        expect(json_response).to be_an Array
        expect(json_response.length).to eq(1)
        expect(json_response.first['labels']).to eq([label.title])
jubianchi's avatar
jubianchi committed
108 109 110 111
      end

      it 'should return an empty array if no issue matches labels' do
        get api('/issues?labels=foo,bar', user)
112 113 114
        expect(response.status).to eq(200)
        expect(json_response).to be_an Array
        expect(json_response.length).to eq(0)
jubianchi's avatar
jubianchi committed
115 116 117 118
      end

      it 'should return an array of labeled issues matching given state' do
        get api("/issues?labels=#{label.title}&state=opened", user)
119 120 121 122 123
        expect(response.status).to eq(200)
        expect(json_response).to be_an Array
        expect(json_response.length).to eq(1)
        expect(json_response.first['labels']).to eq([label.title])
        expect(json_response.first['state']).to eq('opened')
jubianchi's avatar
jubianchi committed
124 125 126 127
      end

      it 'should return an empty array if no issue matches labels and state filters' do
        get api("/issues?labels=#{label.title}&state=closed", user)
128 129 130
        expect(response.status).to eq(200)
        expect(json_response).to be_an Array
        expect(json_response.length).to eq(0)
jubianchi's avatar
jubianchi committed
131
      end
Nihad Abbasov's avatar
Nihad Abbasov committed
132 133 134 135
    end
  end

  describe "GET /projects/:id/issues" do
136 137 138
    let(:base_url) { "/projects/#{project.id}" }
    let(:title) { milestone.title }

139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163
    it 'should return project issues without confidential issues for non project members' do
      get api("#{base_url}/issues", non_member)
      expect(response.status).to eq(200)
      expect(json_response).to be_an Array
      expect(json_response.length).to eq(2)
      expect(json_response.first['title']).to eq(issue.title)
    end

    it 'should return project confidential issues for author' do
      get api("#{base_url}/issues", author)
      expect(response.status).to eq(200)
      expect(json_response).to be_an Array
      expect(json_response.length).to eq(3)
      expect(json_response.first['title']).to eq(issue.title)
    end

    it 'should return project confidential issues for assignee' do
      get api("#{base_url}/issues", assignee)
      expect(response.status).to eq(200)
      expect(json_response).to be_an Array
      expect(json_response.length).to eq(3)
      expect(json_response.first['title']).to eq(issue.title)
    end

    it 'should return project issues with confidential issues for project members' do
164
      get api("#{base_url}/issues", user)
165 166
      expect(response.status).to eq(200)
      expect(json_response).to be_an Array
167 168 169 170 171 172 173 174 175
      expect(json_response.length).to eq(3)
      expect(json_response.first['title']).to eq(issue.title)
    end

    it 'should return project confidential issues for admin' do
      get api("#{base_url}/issues", admin)
      expect(response.status).to eq(200)
      expect(json_response).to be_an Array
      expect(json_response.length).to eq(3)
176
      expect(json_response.first['title']).to eq(issue.title)
Nihad Abbasov's avatar
Nihad Abbasov committed
177
    end
jubianchi's avatar
jubianchi committed
178 179

    it 'should return an array of labeled project issues' do
180
      get api("#{base_url}/issues?labels=#{label.title}", user)
181 182 183 184
      expect(response.status).to eq(200)
      expect(json_response).to be_an Array
      expect(json_response.length).to eq(1)
      expect(json_response.first['labels']).to eq([label.title])
jubianchi's avatar
jubianchi committed
185 186 187
    end

    it 'should return an array of labeled project issues when at least one label matches' do
188
      get api("#{base_url}/issues?labels=#{label.title},foo,bar", user)
189 190 191 192
      expect(response.status).to eq(200)
      expect(json_response).to be_an Array
      expect(json_response.length).to eq(1)
      expect(json_response.first['labels']).to eq([label.title])
jubianchi's avatar
jubianchi committed
193 194 195
    end

    it 'should return an empty array if no project issue matches labels' do
196
      get api("#{base_url}/issues?labels=foo,bar", user)
197 198 199
      expect(response.status).to eq(200)
      expect(json_response).to be_an Array
      expect(json_response.length).to eq(0)
200 201 202 203
    end

    it 'should return an empty array if no issue matches milestone' do
      get api("#{base_url}/issues?milestone=#{empty_milestone.title}", user)
204 205 206
      expect(response.status).to eq(200)
      expect(json_response).to be_an Array
      expect(json_response.length).to eq(0)
jubianchi's avatar
jubianchi committed
207
    end
208 209 210

    it 'should return an empty array if milestone does not exist' do
      get api("#{base_url}/issues?milestone=foo", user)
211 212 213
      expect(response.status).to eq(200)
      expect(json_response).to be_an Array
      expect(json_response.length).to eq(0)
214 215 216 217
    end

    it 'should return an array of issues in given milestone' do
      get api("#{base_url}/issues?milestone=#{title}", user)
218 219 220 221 222
      expect(response.status).to eq(200)
      expect(json_response).to be_an Array
      expect(json_response.length).to eq(2)
      expect(json_response.first['id']).to eq(issue.id)
      expect(json_response.second['id']).to eq(closed_issue.id)
223 224 225 226 227
    end

    it 'should return an array of issues matching state in milestone' do
      get api("#{base_url}/issues?milestone=#{milestone.title}"\
              '&state=closed', user)
228 229 230 231
      expect(response.status).to eq(200)
      expect(json_response).to be_an Array
      expect(json_response.length).to eq(1)
      expect(json_response.first['id']).to eq(closed_issue.id)
232
    end
233 234 235 236 237
  end

  describe "GET /projects/:id/issues/:issue_id" do
    it 'exposes known attributes' do
      get api("/projects/#{project.id}/issues/#{issue.id}", user)
238 239

      expect(response.status).to eq(200)
240 241 242 243 244 245 246 247 248 249 250 251 252
      expect(json_response['id']).to eq(issue.id)
      expect(json_response['iid']).to eq(issue.iid)
      expect(json_response['project_id']).to eq(issue.project.id)
      expect(json_response['title']).to eq(issue.title)
      expect(json_response['description']).to eq(issue.description)
      expect(json_response['state']).to eq(issue.state)
      expect(json_response['created_at']).to be_present
      expect(json_response['updated_at']).to be_present
      expect(json_response['labels']).to eq(issue.label_names)
      expect(json_response['milestone']).to be_a Hash
      expect(json_response['assignee']).to be_a Hash
      expect(json_response['author']).to be_a Hash
      expect(json_response['user_notes_count']).to be(1)
253
    end
Nihad Abbasov's avatar
Nihad Abbasov committed
254 255

    it "should return a project issue by id" do
256
      get api("/projects/#{project.id}/issues/#{issue.id}", user)
257

258 259 260
      expect(response.status).to eq(200)
      expect(json_response['title']).to eq(issue.title)
      expect(json_response['iid']).to eq(issue.iid)
Nihad Abbasov's avatar
Nihad Abbasov committed
261
    end
262

263 264
    it 'should return a project issue by iid' do
      get api("/projects/#{project.id}/issues?iid=#{issue.iid}", user)
265 266 267 268
      expect(response.status).to eq 200
      expect(json_response.first['title']).to eq issue.title
      expect(json_response.first['id']).to eq issue.id
      expect(json_response.first['iid']).to eq issue.iid
269 270
    end

271 272
    it "should return 404 if issue id not found" do
      get api("/projects/#{project.id}/issues/54321", user)
273
      expect(response.status).to eq(404)
274
    end
275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309

    context 'confidential issues' do
      it "should return 404 for non project members" do
        get api("/projects/#{project.id}/issues/#{confidential_issue.id}", non_member)
        expect(response.status).to eq(404)
      end

      it "should return confidential issue for project members" do
        get api("/projects/#{project.id}/issues/#{confidential_issue.id}", user)
        expect(response.status).to eq(200)
        expect(json_response['title']).to eq(confidential_issue.title)
        expect(json_response['iid']).to eq(confidential_issue.iid)
      end

      it "should return confidential issue for author" do
        get api("/projects/#{project.id}/issues/#{confidential_issue.id}", author)
        expect(response.status).to eq(200)
        expect(json_response['title']).to eq(confidential_issue.title)
        expect(json_response['iid']).to eq(confidential_issue.iid)
      end

      it "should return confidential issue for assignee" do
        get api("/projects/#{project.id}/issues/#{confidential_issue.id}", assignee)
        expect(response.status).to eq(200)
        expect(json_response['title']).to eq(confidential_issue.title)
        expect(json_response['iid']).to eq(confidential_issue.iid)
      end

      it "should return confidential issue for admin" do
        get api("/projects/#{project.id}/issues/#{confidential_issue.id}", admin)
        expect(response.status).to eq(200)
        expect(json_response['title']).to eq(confidential_issue.title)
        expect(json_response['iid']).to eq(confidential_issue.iid)
      end
    end
Nihad Abbasov's avatar
Nihad Abbasov committed
310 311 312 313
  end

  describe "POST /projects/:id/issues" do
    it "should create a new project issue" do
314
      post api("/projects/#{project.id}/issues", user),
315
        title: 'new issue', labels: 'label, label2'
316 317 318 319
      expect(response.status).to eq(201)
      expect(json_response['title']).to eq('new issue')
      expect(json_response['description']).to be_nil
      expect(json_response['labels']).to eq(['label', 'label2'])
Nihad Abbasov's avatar
Nihad Abbasov committed
320
    end
321 322 323

    it "should return a 400 bad request if title not given" do
      post api("/projects/#{project.id}/issues", user), labels: 'label, label2'
324
      expect(response.status).to eq(400)
325
    end
326

327
    it 'should return 400 on invalid label names' do
328 329 330
      post api("/projects/#{project.id}/issues", user),
           title: 'new issue',
           labels: 'label, ?'
331 332
      expect(response.status).to eq(400)
      expect(json_response['message']['labels']['?']['title']).to eq(['is invalid'])
333
    end
334 335 336 337

    it 'should return 400 if title is too long' do
      post api("/projects/#{project.id}/issues", user),
           title: 'g' * 256
338 339
      expect(response.status).to eq(400)
      expect(json_response['message']['title']).to eq([
340
        'is too long (maximum is 255 characters)'
341
      ])
342
    end
343 344

    context 'when an admin or owner makes the request' do
345 346
      it 'accepts the creation date to be set' do
        creation_time = 2.weeks.ago
347
        post api("/projects/#{project.id}/issues", user),
348
          title: 'new issue', labels: 'label, label2', created_at: creation_time
349 350

        expect(response.status).to eq(201)
351
        expect(Time.parse(json_response['created_at'])).to be_within(1.second).of(creation_time)
352 353
      end
    end
Nihad Abbasov's avatar
Nihad Abbasov committed
354 355
  end

356 357 358 359 360 361 362 363
  describe 'POST /projects/:id/issues with spam filtering' do
    before do
      Grape::Endpoint.before_each do |endpoint|
        allow(endpoint).to receive(:check_for_spam?).and_return(true)
        allow(endpoint).to receive(:is_spam?).and_return(true)
      end
    end

364 365 366 367 368 369 370
    let(:params) do
      {
        title: 'new issue',
        description: 'content here',
        labels: 'label, label2'
      }
    end
371

372 373
    it "should not create a new project issue" do
      expect { post api("/projects/#{project.id}/issues", user), params }.not_to change(Issue, :count)
374 375
      expect(response.status).to eq(400)
      expect(json_response['message']).to eq({ "error" => "Spam detected" })
376

377 378 379
      spam_logs = SpamLog.all
      expect(spam_logs.count).to eq(1)
      expect(spam_logs[0].title).to eq('new issue')
380
      expect(spam_logs[0].description).to eq('content here')
381 382 383 384 385 386
      expect(spam_logs[0].user).to eq(user)
      expect(spam_logs[0].noteable_type).to eq('Issue')
      expect(spam_logs[0].project_id).to eq(project.id)
    end
  end

Andrew8xx8's avatar
Andrew8xx8 committed
387
  describe "PUT /projects/:id/issues/:issue_id to update only title" do
Nihad Abbasov's avatar
Nihad Abbasov committed
388
    it "should update a project issue" do
389
      put api("/projects/#{project.id}/issues/#{issue.id}", user),
Andrew8xx8's avatar
Andrew8xx8 committed
390
        title: 'updated title'
391
      expect(response.status).to eq(200)
Andrew8xx8's avatar
Andrew8xx8 committed
392

393
      expect(json_response['title']).to eq('updated title')
Andrew8xx8's avatar
Andrew8xx8 committed
394
    end
395 396 397 398

    it "should return 404 error if issue id not found" do
      put api("/projects/#{project.id}/issues/44444", user),
        title: 'updated title'
399
      expect(response.status).to eq(404)
400
    end
401

402
    it 'should return 400 on invalid label names' do
403 404 405
      put api("/projects/#{project.id}/issues/#{issue.id}", user),
          title: 'updated title',
          labels: 'label, ?'
406 407
      expect(response.status).to eq(400)
      expect(json_response['message']['labels']['?']['title']).to eq(['is invalid'])
408
    end
409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437

    context 'confidential issues' do
      it "should return 403 for non project members" do
        put api("/projects/#{project.id}/issues/#{confidential_issue.id}", non_member),
          title: 'updated title'
        expect(response.status).to eq(403)
      end

      it "should update a confidential issue for project members" do
        put api("/projects/#{project.id}/issues/#{confidential_issue.id}", user),
          title: 'updated title'
        expect(response.status).to eq(200)
        expect(json_response['title']).to eq('updated title')
      end

      it "should update a confidential issue for author" do
        put api("/projects/#{project.id}/issues/#{confidential_issue.id}", author),
          title: 'updated title'
        expect(response.status).to eq(200)
        expect(json_response['title']).to eq('updated title')
      end

      it "should update a confidential issue for admin" do
        put api("/projects/#{project.id}/issues/#{confidential_issue.id}", admin),
          title: 'updated title'
        expect(response.status).to eq(200)
        expect(json_response['title']).to eq('updated title')
      end
    end
438 439 440 441 442 443 444 445 446
  end

  describe 'PUT /projects/:id/issues/:issue_id to update labels' do
    let!(:label) { create(:label, title: 'dummy', project: project) }
    let!(:label_link) { create(:label_link, label: label, target: issue) }

    it 'should not update labels if not present' do
      put api("/projects/#{project.id}/issues/#{issue.id}", user),
          title: 'updated title'
447 448
      expect(response.status).to eq(200)
      expect(json_response['labels']).to eq([label.title])
449 450 451 452 453
    end

    it 'should remove all labels' do
      put api("/projects/#{project.id}/issues/#{issue.id}", user),
          labels: ''
454 455
      expect(response.status).to eq(200)
      expect(json_response['labels']).to eq([])
456 457 458 459 460
    end

    it 'should update labels' do
      put api("/projects/#{project.id}/issues/#{issue.id}", user),
          labels: 'foo,bar'
461 462 463
      expect(response.status).to eq(200)
      expect(json_response['labels']).to include 'foo'
      expect(json_response['labels']).to include 'bar'
464 465 466 467 468
    end

    it 'should return 400 on invalid label names' do
      put api("/projects/#{project.id}/issues/#{issue.id}", user),
          labels: 'label, ?'
469 470
      expect(response.status).to eq(400)
      expect(json_response['message']['labels']['?']['title']).to eq(['is invalid'])
471 472 473 474 475
    end

    it 'should allow special label names' do
      put api("/projects/#{project.id}/issues/#{issue.id}", user),
          labels: 'label:foo, label-bar,label_bar,label/bar'
476 477 478 479 480
      expect(response.status).to eq(200)
      expect(json_response['labels']).to include 'label:foo'
      expect(json_response['labels']).to include 'label-bar'
      expect(json_response['labels']).to include 'label_bar'
      expect(json_response['labels']).to include 'label/bar'
481
    end
482 483 484 485

    it 'should return 400 if title is too long' do
      put api("/projects/#{project.id}/issues/#{issue.id}", user),
          title: 'g' * 256
486 487
      expect(response.status).to eq(400)
      expect(json_response['message']['title']).to eq([
488
        'is too long (maximum is 255 characters)'
489
      ])
490
    end
Andrew8xx8's avatar
Andrew8xx8 committed
491 492 493 494 495 496
  end

  describe "PUT /projects/:id/issues/:issue_id to update state and label" do
    it "should update a project issue" do
      put api("/projects/#{project.id}/issues/#{issue.id}", user),
        labels: 'label2', state_event: "close"
497
      expect(response.status).to eq(200)
Andrew8xx8's avatar
Andrew8xx8 committed
498

499 500
      expect(json_response['labels']).to include 'label2'
      expect(json_response['state']).to eq "closed"
Nihad Abbasov's avatar
Nihad Abbasov committed
501
    end
502 503 504 505 506 507 508 509 510 511 512 513

    context 'when an admin or owner makes the request' do
      it 'accepts the update date to be set' do
        update_time = 2.weeks.ago
        put api("/projects/#{project.id}/issues/#{issue.id}", user),
          labels: 'label3', state_event: 'close', updated_at: update_time
        expect(response.status).to eq(200)

        expect(json_response['labels']).to include 'label3'
        expect(Time.parse(json_response['updated_at'])).to be_within(1.second).of(update_time)
      end
    end
Nihad Abbasov's avatar
Nihad Abbasov committed
514 515 516
  end

  describe "DELETE /projects/:id/issues/:issue_id" do
517
    it "rejects a non member from deleting an issue" do
518 519
      delete api("/projects/#{project.id}/issues/#{issue.id}", non_member)
      expect(response.status).to be(403)
520 521
    end

522
    it "rejects a developer from deleting an issue" do
523 524 525
      delete api("/projects/#{project.id}/issues/#{issue.id}", author)
      expect(response.status).to be(403)
    end
526

527 528 529 530 531 532 533 534 535
    context "when the user is project owner" do
      let(:owner)     { create(:user) }
      let(:project)   { create(:project, namespace: owner.namespace) }

      it "deletes the issue if an admin requests it" do
        delete api("/projects/#{project.id}/issues/#{issue.id}", owner)
        expect(response.status).to eq(200)
        expect(json_response['state']).to eq 'opened'
      end
Nihad Abbasov's avatar
Nihad Abbasov committed
536 537
    end
  end
538 539 540 541 542 543 544

  describe '/projects/:id/issues/:issue_id/move' do
    let!(:target_project) { create(:project, path: 'project2', creator_id: user.id, namespace: user.namespace ) }
    let!(:target_project2) { create(:project, creator_id: non_member.id, namespace: non_member.namespace ) }

    it 'moves an issue' do
      post api("/projects/#{project.id}/issues/#{issue.id}/move", user),
545
               to_project_id: target_project.id
546 547 548 549 550

      expect(response.status).to eq(201)
      expect(json_response['project_id']).to eq(target_project.id)
    end

551 552 553 554
    context 'when source and target projects are the same' do
      it 'returns 400 when trying to move an issue' do
        post api("/projects/#{project.id}/issues/#{issue.id}/move", user),
                 to_project_id: project.id
555

556 557 558
        expect(response.status).to eq(400)
        expect(json_response['message']).to eq('Cannot move issue to project it originates from!')
      end
559 560
    end

561 562 563 564
    context 'when the user does not have the permission to move issues' do
      it 'returns 400 when trying to move an issue' do
        post api("/projects/#{project.id}/issues/#{issue.id}/move", user),
                 to_project_id: target_project2.id
565

566 567 568
        expect(response.status).to eq(400)
        expect(json_response['message']).to eq('Cannot move issue due to insufficient permissions!')
      end
569 570 571 572
    end

    it 'moves the issue to another namespace if I am admin' do
      post api("/projects/#{project.id}/issues/#{issue.id}/move", admin),
573
               to_project_id: target_project2.id
574 575 576 577 578

      expect(response.status).to eq(201)
      expect(json_response['project_id']).to eq(target_project2.id)
    end

579 580 581 582
    context 'when issue does not exist' do
      it 'returns 404 when trying to move an issue' do
        post api("/projects/#{project.id}/issues/123/move", user),
                 to_project_id: target_project.id
583

584 585
        expect(response.status).to eq(404)
      end
586 587
    end

588 589 590 591
    context 'when source project does not exist' do
      it 'returns 404 when trying to move an issue' do
        post api("/projects/123/issues/#{issue.id}/move", user),
                 to_project_id: target_project.id
592

593 594 595 596 597 598 599 600 601 602 603
        expect(response.status).to eq(404)
      end
    end

    context 'when target project does not exist' do
      it 'returns 404 when trying to move an issue' do
        post api("/projects/#{project.id}/issues/#{issue.id}/move", user),
                 to_project_id: 123

        expect(response.status).to eq(404)
      end
604 605
    end
  end
606

607
  describe 'POST :id/issues/:issue_id/subscription' do
608
    it 'subscribes to an issue' do
609
      post api("/projects/#{project.id}/issues/#{issue.id}/subscription", user2)
610 611 612 613 614 615

      expect(response.status).to eq(201)
      expect(json_response['subscribed']).to eq(true)
    end

    it 'returns 304 if already subscribed' do
616
      post api("/projects/#{project.id}/issues/#{issue.id}/subscription", user)
617 618 619

      expect(response.status).to eq(304)
    end
620 621 622 623 624 625

    it 'returns 404 if the issue is not found' do
      post api("/projects/#{project.id}/issues/123/subscription", user)

      expect(response.status).to eq(404)
    end
626 627 628 629 630 631

    it 'returns 404 if the issue is confidential' do
      post api("/projects/#{project.id}/issues/#{confidential_issue.id}/subscription", non_member)

      expect(response.status).to eq(404)
    end
632 633
  end

634
  describe 'DELETE :id/issues/:issue_id/subscription' do
635
    it 'unsubscribes from an issue' do
636
      delete api("/projects/#{project.id}/issues/#{issue.id}/subscription", user)
637

638
      expect(response.status).to eq(200)
639 640 641 642
      expect(json_response['subscribed']).to eq(false)
    end

    it 'returns 304 if not subscribed' do
643
      delete api("/projects/#{project.id}/issues/#{issue.id}/subscription", user2)
644 645 646

      expect(response.status).to eq(304)
    end
647 648

    it 'returns 404 if the issue is not found' do
649
      delete api("/projects/#{project.id}/issues/123/subscription", user)
650 651 652

      expect(response.status).to eq(404)
    end
653 654 655 656 657 658

    it 'returns 404 if the issue is confidential' do
      delete api("/projects/#{project.id}/issues/#{confidential_issue.id}/subscription", non_member)

      expect(response.status).to eq(404)
    end
659
  end
Nihad Abbasov's avatar
Nihad Abbasov committed
660
end