issues_controller_test.rb 57.3 KB
Newer Older
1
# Redmine - project management software
2
# Copyright (C) 2006-2011  Jean-Philippe Lang
3 4 5 6 7
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
8
#
9 10 11 12
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
13
#
14 15 16 17
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.

18
require File.expand_path('../../test_helper', __FILE__)
19 20
require 'issues_controller'

edavis10's avatar
edavis10 committed
21
class IssuesControllerTest < ActionController::TestCase
22 23 24 25
  fixtures :projects,
           :users,
           :roles,
           :members,
26
           :member_roles,
27 28
           :issues,
           :issue_statuses,
29
           :versions,
30
           :trackers,
31
           :projects_trackers,
32 33 34
           :issue_categories,
           :enabled_modules,
           :enumerations,
35
           :attachments,
36 37 38
           :workflows,
           :custom_fields,
           :custom_values,
jplang's avatar
jplang committed
39
           :custom_fields_projects,
40
           :custom_fields_trackers,
41 42
           :time_entries,
           :journals,
edavis10's avatar
edavis10 committed
43 44
           :journal_details,
           :queries
45

46 47 48 49 50 51
  def setup
    @controller = IssuesController.new
    @request    = ActionController::TestRequest.new
    @response   = ActionController::TestResponse.new
    User.current = nil
  end
52

53
  def test_index
54
    Setting.default_language = 'en'
55

56 57
    get :index
    assert_response :success
58
    assert_template 'index'
59 60
    assert_not_nil assigns(:issues)
    assert_nil assigns(:project)
61 62 63 64 65
    assert_tag :tag => 'a', :content => /Can't print recipes/
    assert_tag :tag => 'a', :content => /Subproject issue/
    # private projects hidden
    assert_no_tag :tag => 'a', :content => /Issue of a private subproject/
    assert_no_tag :tag => 'a', :content => /Issue on project 2/
66 67
    # project column
    assert_tag :tag => 'th', :content => /Project/
68
  end
69

70 71
  def test_index_should_not_list_issues_when_module_disabled
    EnabledModule.delete_all("name = 'issue_tracking' AND project_id = 1")
72 73 74 75 76 77 78 79
    get :index
    assert_response :success
    assert_template 'index.rhtml'
    assert_not_nil assigns(:issues)
    assert_nil assigns(:project)
    assert_no_tag :tag => 'a', :content => /Can't print recipes/
    assert_tag :tag => 'a', :content => /Subproject issue/
  end
jplang's avatar
jplang committed
80 81 82 83 84 85 86

  def test_index_should_list_visible_issues_only
    get :index, :per_page => 100
    assert_response :success
    assert_not_nil assigns(:issues)
    assert_nil assigns(:issues).detect {|issue| !issue.visible?}
  end
87

88
  def test_index_with_project
89
    Setting.display_subprojects_issues = 0
90 91 92 93
    get :index, :project_id => 1
    assert_response :success
    assert_template 'index.rhtml'
    assert_not_nil assigns(:issues)
94 95 96
    assert_tag :tag => 'a', :content => /Can't print recipes/
    assert_no_tag :tag => 'a', :content => /Subproject issue/
  end
97

98 99 100 101 102 103 104 105 106 107
  def test_index_with_project_and_subprojects
    Setting.display_subprojects_issues = 1
    get :index, :project_id => 1
    assert_response :success
    assert_template 'index.rhtml'
    assert_not_nil assigns(:issues)
    assert_tag :tag => 'a', :content => /Can't print recipes/
    assert_tag :tag => 'a', :content => /Subproject issue/
    assert_no_tag :tag => 'a', :content => /Issue of a private subproject/
  end
108

109 110 111 112 113 114 115 116 117 118
  def test_index_with_project_and_subprojects_should_show_private_subprojects
    @request.session[:user_id] = 2
    Setting.display_subprojects_issues = 1
    get :index, :project_id => 1
    assert_response :success
    assert_template 'index.rhtml'
    assert_not_nil assigns(:issues)
    assert_tag :tag => 'a', :content => /Can't print recipes/
    assert_tag :tag => 'a', :content => /Subproject issue/
    assert_tag :tag => 'a', :content => /Issue of a private subproject/
119
  end
120

121
  def test_index_with_project_and_default_filter
122 123
    get :index, :project_id => 1, :set_filter => 1
    assert_response :success
124
    assert_template 'index'
125
    assert_not_nil assigns(:issues)
126

127 128 129 130 131
    query = assigns(:query)
    assert_not_nil query
    # default filter
    assert_equal({'status_id' => {:operator => 'o', :values => ['']}}, query.filters)
  end
132

133
  def test_index_with_project_and_filter
134
    get :index, :project_id => 1, :set_filter => 1,
jplang's avatar
jplang committed
135 136
      :f => ['tracker_id'],
      :op => {'tracker_id' => '='},
137
      :v => {'tracker_id' => ['1']}
138 139 140
    assert_response :success
    assert_template 'index.rhtml'
    assert_not_nil assigns(:issues)
141

142 143 144 145
    query = assigns(:query)
    assert_not_nil query
    assert_equal({'tracker_id' => {:operator => '=', :values => ['1']}}, query.filters)
  end
146

147 148 149
  def test_index_with_project_and_empty_filters
    get :index, :project_id => 1, :set_filter => 1, :fields => ['']
    assert_response :success
150
    assert_template 'index'
151
    assert_not_nil assigns(:issues)
152

153 154 155 156
    query = assigns(:query)
    assert_not_nil query
    # no filter
    assert_equal({}, query.filters)
157
  end
158

jplang's avatar
jplang committed
159 160 161
  def test_index_with_query
    get :index, :project_id => 1, :query_id => 5
    assert_response :success
162
    assert_template 'index'
jplang's avatar
jplang committed
163 164 165
    assert_not_nil assigns(:issues)
    assert_nil assigns(:issue_count_by_group)
  end
166

167
  def test_index_with_query_grouped_by_tracker
jplang's avatar
jplang committed
168 169
    get :index, :project_id => 1, :query_id => 6
    assert_response :success
170
    assert_template 'index'
jplang's avatar
jplang committed
171
    assert_not_nil assigns(:issues)
172
    assert_not_nil assigns(:issue_count_by_group)
173
  end
174

175 176 177
  def test_index_with_query_grouped_by_list_custom_field
    get :index, :project_id => 1, :query_id => 9
    assert_response :success
tmaruyama's avatar
tmaruyama committed
178
    assert_template 'index'
179
    assert_not_nil assigns(:issues)
180
    assert_not_nil assigns(:issue_count_by_group)
jplang's avatar
jplang committed
181
  end
182

183 184 185
  def test_private_query_should_not_be_available_to_other_users
    q = Query.create!(:name => "private", :user => User.find(2), :is_public => false, :project => nil)
    @request.session[:user_id] = 3
186

187 188 189
    get :index, :query_id => q.id
    assert_response 403
  end
190

191 192 193
  def test_private_query_should_be_available_to_its_user
    q = Query.create!(:name => "private", :user => User.find(2), :is_public => false, :project => nil)
    @request.session[:user_id] = 2
194

195 196 197
    get :index, :query_id => q.id
    assert_response :success
  end
198

199 200 201
  def test_public_query_should_be_available_to_other_users
    q = Query.create!(:name => "private", :user => User.find(2), :is_public => true, :project => nil)
    @request.session[:user_id] = 3
202

203 204 205
    get :index, :query_id => q.id
    assert_response :success
  end
206

jplang's avatar
jplang committed
207 208 209 210
  def test_index_sort_by_field_not_included_in_columns
    Setting.issue_list_default_columns = %w(subject author)
    get :index, :sort => 'tracker'
  end
211

212
  def test_index_csv_with_project
213
    Setting.default_language = 'en'
214

215 216 217 218
    get :index, :format => 'csv'
    assert_response :success
    assert_not_nil assigns(:issues)
    assert_equal 'text/csv', @response.content_type
219
    assert @response.body.starts_with?("#,")
220 221 222 223 224 225

    get :index, :project_id => 1, :format => 'csv'
    assert_response :success
    assert_not_nil assigns(:issues)
    assert_equal 'text/csv', @response.content_type
  end
226

227 228 229 230 231
  def test_index_pdf
    get :index, :format => 'pdf'
    assert_response :success
    assert_not_nil assigns(:issues)
    assert_equal 'application/pdf', @response.content_type
232

233 234 235 236
    get :index, :project_id => 1, :format => 'pdf'
    assert_response :success
    assert_not_nil assigns(:issues)
    assert_equal 'application/pdf', @response.content_type
237

jplang's avatar
jplang committed
238 239 240 241
    get :index, :project_id => 1, :query_id => 6, :format => 'pdf'
    assert_response :success
    assert_not_nil assigns(:issues)
    assert_equal 'application/pdf', @response.content_type
242
  end
243

244 245 246 247 248 249 250
  def test_index_pdf_with_query_grouped_by_list_custom_field
    get :index, :project_id => 1, :query_id => 9, :format => 'pdf'
    assert_response :success
    assert_not_nil assigns(:issues)
    assert_not_nil assigns(:issue_count_by_group)
    assert_equal 'application/pdf', @response.content_type
  end
251

252
  def test_index_sort
jplang's avatar
jplang committed
253
    get :index, :sort => 'tracker,id:desc'
254
    assert_response :success
255

jplang's avatar
jplang committed
256 257 258
    sort_params = @request.session['issues_index_sort']
    assert sort_params.is_a?(String)
    assert_equal 'tracker,id:desc', sort_params
259

jplang's avatar
jplang committed
260 261 262 263
    issues = assigns(:issues)
    assert_not_nil issues
    assert !issues.empty?
    assert_equal issues.sort {|a,b| a.tracker == b.tracker ? b.id <=> a.id : a.tracker <=> b.tracker }.collect(&:id), issues.collect(&:id)
264
  end
265

266 267
  def test_index_with_columns
    columns = ['tracker', 'subject', 'assigned_to']
268
    get :index, :set_filter => 1, :c => columns
269
    assert_response :success
270

271 272 273 274
    # query should use specified columns
    query = assigns(:query)
    assert_kind_of Query, query
    assert_equal columns, query.column_names.map(&:to_s)
275

276 277 278 279
    # columns should be stored in session
    assert_kind_of Hash, session[:query]
    assert_kind_of Array, session[:query][:column_names]
    assert_equal columns, session[:query][:column_names].map(&:to_s)
280 281 282 283 284 285

    # ensure only these columns are kept in the selected columns list
    assert_tag :tag => 'select', :attributes => { :id => 'selected_columns' },
                                 :children => { :count => 3 }
    assert_no_tag :tag => 'option', :attributes => { :value => 'project' },
                                    :parent => { :tag => 'select', :attributes => { :id => "selected_columns" } }
286
  end
287

288 289 290 291
  def test_index_with_custom_field_column
    columns = %w(tracker subject cf_2)
    get :index, :set_filter => 1, :c => columns
    assert_response :success
292

293 294 295 296
    # query should use specified columns
    query = assigns(:query)
    assert_kind_of Query, query
    assert_equal columns, query.column_names.map(&:to_s)
297

298 299 300 301
    assert_tag :td,
      :attributes => {:class => 'cf_2 string'},
      :ancestor => {:tag => 'table', :attributes => {:class => /issues/}}
  end
302

303 304 305 306 307 308
  def test_index_send_html_if_query_is_invalid
    get :index, :f => ['start_date'], :op => {:start_date => '='}
    assert_equal 'text/html', @response.content_type
    assert_template 'index'
  end

309 310 311 312 313 314
  def test_index_send_nothing_if_query_is_invalid
    get :index, :f => ['start_date'], :op => {:start_date => '='}, :format => 'csv'
    assert_equal 'text/csv', @response.content_type
    assert @response.body.blank?
  end

315
  def test_show_by_anonymous
jplang's avatar
jplang committed
316 317
    get :show, :id => 1
    assert_response :success
318
    assert_template 'show'
jplang's avatar
jplang committed
319
    assert_not_nil assigns(:issue)
320
    assert_equal Issue.find(1), assigns(:issue)
321

322 323 324
    # anonymous role is allowed to add a note
    assert_tag :tag => 'form',
               :descendant => { :tag => 'fieldset',
325
                                :child => { :tag => 'legend',
326
                                            :content => /Notes/ } }
jplang's avatar
jplang committed
327
  end
328

329 330 331 332
  def test_show_by_manager
    @request.session[:user_id] = 2
    get :show, :id => 1
    assert_response :success
333

334 335
    assert_tag :tag => 'a',
      :content => /Quote/
336

337 338
    assert_tag :tag => 'form',
               :descendant => { :tag => 'fieldset',
339
                                :child => { :tag => 'legend',
340 341
                                            :content => /Change properties/ } },
               :descendant => { :tag => 'fieldset',
342
                                :child => { :tag => 'legend',
343 344
                                            :content => /Log time/ } },
               :descendant => { :tag => 'fieldset',
345
                                :child => { :tag => 'legend',
346 347
                                            :content => /Notes/ } }
  end
348

349 350 351 352 353 354 355 356 357
  def test_update_form_should_not_display_inactive_enumerations
    @request.session[:user_id] = 2
    get :show, :id => 1
    assert_response :success

    assert ! IssuePriority.find(15).active?
    assert_no_tag :option, :attributes => {:value => '15'},
                           :parent => {:tag => 'select', :attributes => {:id => 'issue_priority_id'} }
  end
358

359 360 361
  def test_update_form_should_allow_attachment_upload
    @request.session[:user_id] = 2
    get :show, :id => 1
362

363 364 365 366 367 368 369
    assert_tag :tag => 'form',
      :attributes => {:id => 'issue-form', :method => 'post', :enctype => 'multipart/form-data'},
      :descendant => {
        :tag => 'input',
        :attributes => {:type => 'file', :name => 'attachments[1][file]'}
      }
  end
370

jplang's avatar
jplang committed
371 372 373 374 375
  def test_show_should_deny_anonymous_access_without_permission
    Role.anonymous.remove_permission!(:view_issues)
    get :show, :id => 1
    assert_response :redirect
  end
376

jplang's avatar
jplang committed
377 378 379 380 381
  def test_show_should_deny_anonymous_access_to_private_issue
    Issue.update_all(["is_private = ?", true], "id = 1")
    get :show, :id => 1
    assert_response :redirect
  end
382

jplang's avatar
jplang committed
383 384 385 386 387 388
  def test_show_should_deny_non_member_access_without_permission
    Role.non_member.remove_permission!(:view_issues)
    @request.session[:user_id] = 9
    get :show, :id => 1
    assert_response 403
  end
389

jplang's avatar
jplang committed
390 391 392 393 394 395
  def test_show_should_deny_non_member_access_to_private_issue
    Issue.update_all(["is_private = ?", true], "id = 1")
    @request.session[:user_id] = 9
    get :show, :id => 1
    assert_response 403
  end
396

jplang's avatar
jplang committed
397 398 399 400 401 402
  def test_show_should_deny_member_access_without_permission
    Role.find(1).remove_permission!(:view_issues)
    @request.session[:user_id] = 2
    get :show, :id => 1
    assert_response 403
  end
403

jplang's avatar
jplang committed
404 405 406 407 408 409
  def test_show_should_deny_member_access_to_private_issue_without_permission
    Issue.update_all(["is_private = ?", true], "id = 1")
    @request.session[:user_id] = 3
    get :show, :id => 1
    assert_response 403
  end
410

jplang's avatar
jplang committed
411 412 413 414 415 416
  def test_show_should_allow_author_access_to_private_issue
    Issue.update_all(["is_private = ?, author_id = 3", true], "id = 1")
    @request.session[:user_id] = 3
    get :show, :id => 1
    assert_response :success
  end
417

jplang's avatar
jplang committed
418 419 420 421 422 423
  def test_show_should_allow_assignee_access_to_private_issue
    Issue.update_all(["is_private = ?, assigned_to_id = 3", true], "id = 1")
    @request.session[:user_id] = 3
    get :show, :id => 1
    assert_response :success
  end
424

jplang's avatar
jplang committed
425 426 427 428 429 430 431
  def test_show_should_allow_member_access_to_private_issue_with_permission
    Issue.update_all(["is_private = ?", true], "id = 1")
    User.find(3).roles_for_project(Project.find(1)).first.update_attribute :issues_visibility, 'all'
    @request.session[:user_id] = 3
    get :show, :id => 1
    assert_response :success
  end
432

433 434 435 436 437
  def test_show_should_not_disclose_relations_to_invisible_issues
    Setting.cross_project_issue_relations = '1'
    IssueRelation.create!(:issue_from => Issue.find(1), :issue_to => Issue.find(2), :relation_type => 'relates')
    # Relation to a private project issue
    IssueRelation.create!(:issue_from => Issue.find(1), :issue_to => Issue.find(4), :relation_type => 'relates')
438

439 440
    get :show, :id => 1
    assert_response :success
441

442 443 444 445 446
    assert_tag :div, :attributes => { :id => 'relations' },
                     :descendant => { :tag => 'a', :content => /#2$/ }
    assert_no_tag :div, :attributes => { :id => 'relations' },
                        :descendant => { :tag => 'a', :content => /#4$/ }
  end
447

448 449 450
  def test_show_atom
    get :show, :id => 2, :format => 'atom'
    assert_response :success
451
    assert_template 'journals/index.rxml'
452
    # Inline image
453
    assert_select 'content', :text => Regexp.new(Regexp.quote('http://test.host/attachments/download/10'))
454
  end
455

jplang's avatar
jplang committed
456
  def test_show_export_to_pdf
457
    get :show, :id => 3, :format => 'pdf'
jplang's avatar
jplang committed
458 459 460 461 462 463
    assert_response :success
    assert_equal 'application/pdf', @response.content_type
    assert @response.body.starts_with?('%PDF')
    assert_not_nil assigns(:issue)
  end

464 465 466 467 468
  def test_get_new
    @request.session[:user_id] = 2
    get :new, :project_id => 1, :tracker_id => 1
    assert_response :success
    assert_template 'new'
469

470
    assert_tag :tag => 'input', :attributes => { :name => 'issue[custom_field_values][2]',
471
                                                 :value => 'Default string' }
472 473 474 475 476

    # Be sure we don't display inactive IssuePriorities
    assert ! IssuePriority.find(15).active?
    assert_no_tag :option, :attributes => {:value => '15'},
                           :parent => {:tag => 'select', :attributes => {:id => 'issue_priority_id'} }
477
  end
478

479 480 481
  def test_get_new_form_should_allow_attachment_upload
    @request.session[:user_id] = 2
    get :new, :project_id => 1, :tracker_id => 1
482

483 484 485 486 487 488 489
    assert_tag :tag => 'form',
      :attributes => {:id => 'issue-form', :method => 'post', :enctype => 'multipart/form-data'},
      :descendant => {
        :tag => 'input',
        :attributes => {:type => 'file', :name => 'attachments[1][file]'}
      }
  end
490 491 492 493 494 495

  def test_get_new_without_tracker_id
    @request.session[:user_id] = 2
    get :new, :project_id => 1
    assert_response :success
    assert_template 'new'
496

497 498 499 500
    issue = assigns(:issue)
    assert_not_nil issue
    assert_equal Project.find(1).trackers.first, issue.tracker
  end
501

502 503 504
  def test_get_new_with_no_default_status_should_display_an_error
    @request.session[:user_id] = 2
    IssueStatus.delete_all
505

506 507
    get :new, :project_id => 1
    assert_response 500
508
    assert_error_tag :content => /No default issue/
509
  end
510

511 512 513
  def test_get_new_with_no_tracker_should_display_an_error
    @request.session[:user_id] = 2
    Tracker.delete_all
514

515 516
    get :new, :project_id => 1
    assert_response 500
517
    assert_error_tag :content => /No tracker/
518
  end
519

520 521
  def test_update_new_form
    @request.session[:user_id] = 2
522
    xhr :post, :new, :project_id => 1,
523
                     :issue => {:tracker_id => 2,
524 525 526 527
                                :subject => 'This is the test_new issue',
                                :description => 'This is the description',
                                :priority_id => 5}
    assert_response :success
528
    assert_template 'attributes'
529

530 531 532 533 534 535
    issue = assigns(:issue)
    assert_kind_of Issue, issue
    assert_equal 1, issue.project_id
    assert_equal 2, issue.tracker_id
    assert_equal 'This is the test_new issue', issue.subject
  end
536

537
  def test_post_create
538
    @request.session[:user_id] = 2
edavis10's avatar
edavis10 committed
539
    assert_difference 'Issue.count' do
540
      post :create, :project_id => 1,
edavis10's avatar
edavis10 committed
541
                 :issue => {:tracker_id => 3,
542
                            :status_id => 2,
edavis10's avatar
edavis10 committed
543 544 545
                            :subject => 'This is the test_new issue',
                            :description => 'This is the description',
                            :priority_id => 5,
546
                            :start_date => '2010-11-07',
edavis10's avatar
edavis10 committed
547 548 549 550
                            :estimated_hours => '',
                            :custom_field_values => {'2' => 'Value for field 2'}}
    end
    assert_redirected_to :controller => 'issues', :action => 'show', :id => Issue.last.id
551

552 553 554
    issue = Issue.find_by_subject('This is the test_new issue')
    assert_not_nil issue
    assert_equal 2, issue.author_id
555
    assert_equal 3, issue.tracker_id
556
    assert_equal 2, issue.status_id
557
    assert_equal Date.parse('2010-11-07'), issue.start_date
558
    assert_nil issue.estimated_hours
559
    v = issue.custom_values.find(:first, :conditions => {:custom_field_id => 2})
560 561
    assert_not_nil v
    assert_equal 'Value for field 2', v.value
562
  end
563

564 565 566 567 568 569 570 571
  def test_post_new_with_group_assignment
    group = Group.find(11)
    project = Project.find(1)
    project.members << Member.new(:principal => group, :roles => [Role.first])

    with_settings :issue_group_assignment => '1' do
      @request.session[:user_id] = 2
      assert_difference 'Issue.count' do
572
        post :create, :project_id => project.id,
573 574 575 576 577 578 579
                      :issue => {:tracker_id => 3,
                                 :status_id => 1,
                                 :subject => 'This is the test_new_with_group_assignment issue',
                                 :assigned_to_id => group.id}
      end
    end
    assert_redirected_to :controller => 'issues', :action => 'show', :id => Issue.last.id
580

581 582 583 584
    issue = Issue.find_by_subject('This is the test_new_with_group_assignment issue')
    assert_not_nil issue
    assert_equal group, issue.assigned_to
  end
585

586 587 588
  def test_post_create_without_start_date
    @request.session[:user_id] = 2
    assert_difference 'Issue.count' do
589
      post :create, :project_id => 1,
590 591 592 593 594 595 596 597 598 599
                 :issue => {:tracker_id => 3,
                            :status_id => 2,
                            :subject => 'This is the test_new issue',
                            :description => 'This is the description',
                            :priority_id => 5,
                            :start_date => '',
                            :estimated_hours => '',
                            :custom_field_values => {'2' => 'Value for field 2'}}
    end
    assert_redirected_to :controller => 'issues', :action => 'show', :id => Issue.last.id
600

601 602 603 604
    issue = Issue.find_by_subject('This is the test_new issue')
    assert_not_nil issue
    assert_nil issue.start_date
  end
605

606
  def test_post_create_and_continue
607
    @request.session[:user_id] = 2
608 609 610 611 612
    assert_difference 'Issue.count' do
      post :create, :project_id => 1,
        :issue => {:tracker_id => 3, :subject => 'This is first issue', :priority_id => 5},
        :continue => ''
    end
613

614 615 616 617
    issue = Issue.first(:order => 'id DESC')
    assert_redirected_to :controller => 'issues', :action => 'new', :project_id => 'ecookbook', :issue => {:tracker_id => 3}
    assert_not_nil flash[:notice], "flash was not set"
    assert flash[:notice].include?("<a href='/issues/#{issue.id}'>##{issue.id}</a>"), "issue link not found in flash: #{flash[:notice]}"
618
  end
619

620
  def test_post_create_without_custom_fields_param
621
    @request.session[:user_id] = 2
edavis10's avatar
edavis10 committed
622
    assert_difference 'Issue.count' do
623
      post :create, :project_id => 1,
edavis10's avatar
edavis10 committed
624 625 626 627 628 629
                 :issue => {:tracker_id => 1,
                            :subject => 'This is the test_new issue',
                            :description => 'This is the description',
                            :priority_id => 5}
    end
    assert_redirected_to :controller => 'issues', :action => 'show', :id => Issue.last.id
630
  end
631

632
  def test_post_create_with_required_custom_field_and_without_custom_fields_param
633 634 635 636
    field = IssueCustomField.find_by_name('Database')
    field.update_attribute(:is_required, true)

    @request.session[:user_id] = 2
637
    post :create, :project_id => 1,
638 639 640 641 642 643 644 645
               :issue => {:tracker_id => 1,
                          :subject => 'This is the test_new issue',
                          :description => 'This is the description',
                          :priority_id => 5}
    assert_response :success
    assert_template 'new'
    issue = assigns(:issue)
    assert_not_nil issue
646
    assert_equal I18n.translate('activerecord.errors.messages.invalid'), issue.errors.on(:custom_values)
647
  end
648

649
  def test_post_create_with_watchers
650 651
    @request.session[:user_id] = 2
    ActionMailer::Base.deliveries.clear
652

653
    assert_difference 'Watcher.count', 2 do
654
      post :create, :project_id => 1,
655 656 657 658 659 660 661
                 :issue => {:tracker_id => 1,
                            :subject => 'This is a new issue with watchers',
                            :description => 'This is the description',
                            :priority_id => 5,
                            :watcher_user_ids => ['2', '3']}
    end
    issue = Issue.find_by_subject('This is a new issue with watchers')
jplang's avatar
jplang committed
662 663
    assert_not_nil issue
    assert_redirected_to :controller => 'issues', :action => 'show', :id => issue
664

665 666 667 668 669 670 671 672
    # Watchers added
    assert_equal [2, 3], issue.watcher_user_ids.sort
    assert issue.watched_by?(User.find(3))
    # Watchers notified
    mail = ActionMailer::Base.deliveries.last
    assert_kind_of TMail::Mail, mail
    assert [mail.bcc, mail.cc].flatten.include?(User.find(3).mail)
  end
673

674
  def test_post_create_subissue
jplang's avatar
jplang committed
675
    @request.session[:user_id] = 2
676

jplang's avatar
jplang committed
677
    assert_difference 'Issue.count' do
678
      post :create, :project_id => 1,
jplang's avatar
jplang committed
679 680 681 682 683 684 685 686
                 :issue => {:tracker_id => 1,
                            :subject => 'This is a child issue',
                            :parent_issue_id => 2}
    end
    issue = Issue.find_by_subject('This is a child issue')
    assert_not_nil issue
    assert_equal Issue.find(2), issue.parent
  end
687 688 689

  def test_post_create_subissue_with_non_numeric_parent_id
    @request.session[:user_id] = 2
690

691
    assert_difference 'Issue.count' do
692
      post :create, :project_id => 1,
693 694 695 696 697 698 699 700
                 :issue => {:tracker_id => 1,
                            :subject => 'This is a child issue',
                            :parent_issue_id => 'ABC'}
    end
    issue = Issue.find_by_subject('This is a child issue')
    assert_not_nil issue
    assert_nil issue.parent
  end
701

702 703 704 705 706 707 708 709 710 711 712 713
  def test_post_create_private
    @request.session[:user_id] = 2

    assert_difference 'Issue.count' do
      post :create, :project_id => 1,
                 :issue => {:tracker_id => 1,
                            :subject => 'This is a private issue',
                            :is_private => '1'}
    end
    issue = Issue.first(:order => 'id DESC')
    assert issue.is_private?
  end
714

715 716 717 718
  def test_post_create_private_with_set_own_issues_private_permission
    role = Role.find(1)
    role.remove_permission! :set_issues_private
    role.add_permission! :set_own_issues_private
719

720 721 722 723 724 725 726 727 728 729 730
    @request.session[:user_id] = 2

    assert_difference 'Issue.count' do
      post :create, :project_id => 1,
                 :issue => {:tracker_id => 1,
                            :subject => 'This is a private issue',
                            :is_private => '1'}
    end
    issue = Issue.first(:order => 'id DESC')
    assert issue.is_private?
  end
731

732
  def test_post_create_should_send_a_notification
733 734
    ActionMailer::Base.deliveries.clear
    @request.session[:user_id] = 2
edavis10's avatar
edavis10 committed
735
    assert_difference 'Issue.count' do
736
      post :create, :project_id => 1,
edavis10's avatar
edavis10 committed
737 738 739 740 741 742 743 744
                 :issue => {:tracker_id => 3,
                            :subject => 'This is the test_new issue',
                            :description => 'This is the description',
                            :priority_id => 5,
                            :estimated_hours => '',
                            :custom_field_values => {'2' => 'Value for field 2'}}
    end
    assert_redirected_to :controller => 'issues', :action => 'show', :id => Issue.last.id
745

746 747
    assert_equal 1, ActionMailer::Base.deliveries.size
  end
748

749
  def test_post_create_should_preserve_fields_values_on_validation_failure
750
    @request.session[:user_id] = 2
751
    post :create, :project_id => 1,
752
               :issue => {:tracker_id => 1,
753 754 755
                          # empty subject
                          :subject => '',
                          :description => 'This is a description',
756 757 758 759
                          :priority_id => 6,
                          :custom_field_values => {'1' => 'Oracle', '2' => 'Value for field 2'}}
    assert_response :success
    assert_template 'new'
760

761 762
    assert_tag :textarea, :attributes => { :name => 'issue[description]' },
                          :content => 'This is a description'
763 764 765
    assert_tag :select, :attributes => { :name => 'issue[priority_id]' },
                        :child => { :tag => 'option', :attributes => { :selected => 'selected',
                                                                       :value => '6' },
766
                                                      :content => 'High' }
767 768 769 770
    # Custom fields
    assert_tag :select, :attributes => { :name => 'issue[custom_field_values][1]' },
                        :child => { :tag => 'option', :attributes => { :selected => 'selected',
                                                                       :value => 'Oracle' },
771
                                                      :content => 'Oracle' }
772 773 774
    assert_tag :input, :attributes => { :name => 'issue[custom_field_values][2]',
                                        :value => 'Value for field 2'}
  end
775

776
  def test_post_create_should_ignore_non_safe_attributes
777 778
    @request.session[:user_id] = 2
    assert_nothing_raised do
779
      post :create, :project_id => 1, :issue => { :tracker => "A param can not be a Tracker" }
780 781
    end
  end
782

783 784 785
  def test_post_create_with_attachment
    set_tmp_attachments_directory
    @request.session[:user_id] = 2
786

787 788
    assert_difference 'Issue.count' do
      assert_difference 'Attachment.count' do
789
        post :create, :project_id => 1,
790 791 792 793
          :issue => { :tracker_id => '1', :subject => 'With attachment' },
          :attachments => {'1' => {'file' => uploaded_test_file('testfile.txt', 'text/plain'), 'description' => 'test file'}}
      end
    end
794

795 796
    issue = Issue.first(:order => 'id DESC')
    attachment = Attachment.first(:order => 'id DESC')
797

798 799 800 801 802 803 804 805 806 807
    assert_equal issue, attachment.container
    assert_equal 2, attachment.author_id
    assert_equal 'testfile.txt', attachment.filename
    assert_equal 'text/plain', attachment.content_type
    assert_equal 'test file', attachment.description
    assert_equal 59, attachment.filesize
    assert File.exists?(attachment.diskfile)
    assert_equal 59, File.size(attachment.diskfile)
  end

808 809 810
  context "without workflow privilege" do
    setup do
      Workflow.delete_all(["role_id = ?", Role.anonymous.id])
811
      Role.anonymous.add_permission! :add_issues, :add_issue_notes
812
    end
813

814 815 816 817 818 819 820 821 822 823
    context "#new" do
      should "propose default status only" do
        get :new, :project_id => 1
        assert_response :success
        assert_template 'new'
        assert_tag :tag => 'select',
          :attributes => {:name => 'issue[status_id]'},
          :children => {:count => 1},
          :child => {:tag => 'option', :attributes => {:value => IssueStatus.default.id.to_s}}
      end
824

825 826
      should "accept default status" do
        assert_difference 'Issue.count' do
827
          post :create, :project_id => 1,
828 829 830 831 832 833 834
                     :issue => {:tracker_id => 1,
                                :subject => 'This is an issue',
                                :status_id => 1}
        end
        i