123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341 |
- # Redmine - project management software
- # Copyright (C) 2006-2017 Jean-Philippe Lang
- #
- # 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.
- #
- # 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.
- #
- # 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.
-
- require File.expand_path('../../application_system_test_case', __FILE__)
-
- class IssuesTest < ApplicationSystemTestCase
- fixtures :projects, :users, :email_addresses, :roles, :members, :member_roles,
- :trackers, :projects_trackers, :enabled_modules, :issue_statuses, :issues,
- :enumerations, :custom_fields, :custom_values, :custom_fields_trackers,
- :watchers, :journals, :journal_details, :versions
-
- def test_create_issue
- log_user('jsmith', 'jsmith')
- visit '/projects/ecookbook/issues/new'
- within('form#issue-form') do
- select 'Bug', :from => 'Tracker'
- select 'Low', :from => 'Priority'
- fill_in 'Subject', :with => 'new test issue'
- fill_in 'Description', :with => 'new issue'
- select '0 %', :from => 'Done'
- fill_in 'Searchable field', :with => 'Value for field 2'
- # click_button 'Create' would match both 'Create' and 'Create and continue' buttons
- find('input[name=commit]').click
- end
-
- # find created issue
- issue = Issue.find_by_subject("new test issue")
- assert_kind_of Issue, issue
-
- # check redirection
- find 'div#flash_notice', :visible => true, :text => "Issue \##{issue.id} created."
- assert_equal issue_path(:id => issue), current_path
-
- # check issue attributes
- assert_equal 'jsmith', issue.author.login
- assert_equal 1, issue.project.id
- assert_equal IssueStatus.find_by_name('New'), issue.status
- assert_equal Tracker.find_by_name('Bug'), issue.tracker
- assert_equal IssuePriority.find_by_name('Low'), issue.priority
- assert_equal 'Value for field 2', issue.custom_field_value(CustomField.find_by_name('Searchable field'))
- end
-
- def test_create_issue_with_form_update
- field1 = IssueCustomField.create!(
- :field_format => 'string',
- :name => 'Field1',
- :is_for_all => true,
- :trackers => Tracker.where(:id => [1, 2])
- )
- field2 = IssueCustomField.create!(
- :field_format => 'string',
- :name => 'Field2',
- :is_for_all => true,
- :trackers => Tracker.where(:id => 2)
- )
-
- Role.non_member.add_permission! :add_issues
- Role.non_member.remove_permission! :edit_issues, :add_issue_notes
-
- log_user('someone', 'foo')
- visit '/projects/ecookbook/issues/new'
- assert page.has_no_content?(field2.name)
- assert page.has_content?(field1.name)
-
- fill_in 'Subject', :with => 'New test issue'
- fill_in 'Description', :with => 'New test issue description'
- fill_in field1.name, :with => 'CF1 value'
- select 'Low', :from => 'Priority'
-
- # field2 should show up when changing tracker
- select 'Feature request', :from => 'Tracker'
- assert page.has_content?(field2.name)
- assert page.has_content?(field1.name)
-
- fill_in field2.name, :with => 'CF2 value'
- assert_difference 'Issue.count' do
- page.first(:button, 'Create').click
- end
-
- issue = Issue.order('id desc').first
- assert_equal 'New test issue', issue.subject
- assert_equal 'New test issue description', issue.description
- assert_equal 'Low', issue.priority.name
- assert_equal 'CF1 value', issue.custom_field_value(field1)
- assert_equal 'CF2 value', issue.custom_field_value(field2)
- end
-
- def test_create_issue_with_watchers
- user = User.generate!(:firstname => 'Some', :lastname => 'Watcher')
- assert_equal 'Some Watcher', user.name
- log_user('jsmith', 'jsmith')
- visit '/projects/ecookbook/issues/new'
- fill_in 'Subject', :with => 'Issue with watchers'
- # Add a project member as watcher
- check 'Dave Lopper'
- # Search for another user
- assert page.has_no_css?('form#new-watcher-form')
- assert page.has_no_content?('Some Watcher')
- click_link 'Search for watchers to add'
- within('form#new-watcher-form') do
- fill_in 'user_search', :with => 'watch'
- assert page.has_content?('Some Watcher')
- check 'Some Watcher'
- click_button 'Add'
- end
- assert page.has_css?('form#issue-form')
- assert page.has_css?('p#watchers_form')
- using_wait_time(30) do
- within('span#watchers_inputs') do
- within("label#issue_watcher_user_ids_#{user.id}") do
- assert has_content?('Some Watcher'), "No watcher content"
- end
- end
- end
- assert_difference 'Issue.count' do
- find('input[name=commit]').click
- end
-
- issue = Issue.order('id desc').first
- assert_equal ['Dave Lopper', 'Some Watcher'], issue.watcher_users.map(&:name).sort
- end
-
- def test_create_issue_with_attachment
- set_tmp_attachments_directory
- log_user('jsmith', 'jsmith')
-
- issue = new_record(Issue) do
- visit '/projects/ecookbook/issues/new'
- fill_in 'Subject', :with => 'Issue with attachment'
- attach_file 'attachments[dummy][file]', Rails.root.join('test/fixtures/files/testfile.txt')
- fill_in 'attachments[1][description]', :with => 'Some description'
- click_on 'Create'
- end
- assert_equal 1, issue.attachments.count
- assert_equal 'Some description', issue.attachments.first.description
- ensure
- set_fixtures_attachments_directory
- end
-
- def test_create_issue_with_new_target_version
- log_user('jsmith', 'jsmith')
-
- assert_difference 'Issue.count' do
- assert_difference 'Version.count' do
- visit '/projects/ecookbook/issues/new'
- fill_in 'Subject', :with => 'With a new version'
- click_on 'New version'
- within '#ajax-modal' do
- fill_in 'Name', :with => '4.0'
- click_on 'Create'
- end
- click_on 'Create'
- end
- end
-
- issue = Issue.order('id desc').first
- assert_not_nil issue.fixed_version
- assert_equal '4.0', issue.fixed_version.name
- end
-
- def test_preview_issue_description
- log_user('jsmith', 'jsmith')
- visit '/projects/ecookbook/issues/new'
- within('form#issue-form') do
- fill_in 'Subject', :with => 'new issue subject'
- fill_in 'Description', :with => 'new issue description'
- click_link 'Preview'
- find 'div.wiki-preview', :visible => true, :text => 'new issue description'
- end
- assert_difference 'Issue.count' do
- click_button('Create')
- end
-
- issue = Issue.order('id desc').first
- assert_equal 'new issue description', issue.description
- end
-
- def test_update_issue_with_form_update
- field = IssueCustomField.create!(
- :field_format => 'string',
- :name => 'Form update CF',
- :is_for_all => true,
- :trackers => Tracker.where(:name => 'Feature request')
- )
-
- Role.non_member.add_permission! :edit_issues, :add_issues
- Role.non_member.remove_permission! :add_issue_notes
-
- log_user('someone', 'foo')
- visit '/issues/1'
- assert page.has_no_content?('Form update CF')
-
- page.first(:link, 'Edit').click
- # the custom field should show up when changing tracker
- select 'Feature request', :from => 'Tracker'
- assert page.has_content?('Form update CF')
-
- fill_in 'Form update', :with => 'CF value'
- assert_no_difference 'Issue.count' do
- page.first(:button, 'Submit').click
- end
-
- issue = Issue.find(1)
- assert_equal 'CF value', issue.custom_field_value(field)
- end
-
- def test_remove_issue_watcher_from_sidebar
- user = User.find(3)
- Watcher.create!(:watchable => Issue.find(1), :user => user)
-
- log_user('jsmith', 'jsmith')
- visit '/issues/1'
- assert page.first('#sidebar').has_content?('Watchers (1)')
- assert page.first('#sidebar').has_content?(user.name)
- assert_difference 'Watcher.count', -1 do
- page.first('ul.watchers .user-3 a.delete').click
- assert page.first('#sidebar').has_content?('Watchers (0)')
- end
- assert page.first('#sidebar').has_no_content?(user.name)
- end
-
- def test_watch_should_update_watchers_list
- user = User.find(2)
- log_user('jsmith', 'jsmith')
- visit '/issues/1'
- assert page.first('#sidebar').has_content?('Watchers (0)')
-
- page.first('a.issue-1-watcher').click
- assert page.first('#sidebar').has_content?('Watchers (1)')
- assert page.first('#sidebar').has_content?(user.name)
- end
-
- def test_watch_issue_via_context_menu
- log_user('jsmith', 'jsmith')
- visit '/issues'
- assert page.has_css?('tr#issue-1')
- find('tr#issue-1 td.updated_on').click
- page.execute_script "$('tr#issue-1 td.updated_on').trigger('contextmenu');"
- assert_difference 'Watcher.count' do
- within('#context-menu') do
- click_link 'Watch'
- end
- # wait for ajax response
- assert page.has_css?('#context-menu .issue-1-watcher.icon-fav')
- assert page.has_css?('tr#issue-1')
- end
- assert Issue.find(1).watched_by?(User.find_by_login('jsmith'))
- end
-
- def test_bulk_watch_issues_via_context_menu
- log_user('jsmith', 'jsmith')
- visit '/issues'
- assert page.has_css?('tr#issue-1')
- assert page.has_css?('tr#issue-4')
- find('tr#issue-1 input[type=checkbox]').click
- find('tr#issue-4 input[type=checkbox]').click
- page.execute_script "$('tr#issue-1 td.updated_on').trigger('contextmenu');"
- assert_difference 'Watcher.count', 2 do
- within('#context-menu') do
- click_link 'Watch'
- end
- # wait for ajax response
- assert page.has_css?('#context-menu .issue-bulk-watcher.icon-fav')
- assert page.has_css?('tr#issue-1')
- assert page.has_css?('tr#issue-4')
- end
- assert Issue.find(1).watched_by?(User.find_by_login('jsmith'))
- assert Issue.find(4).watched_by?(User.find_by_login('jsmith'))
- end
-
- def test_issue_list_with_default_totalable_columns
- log_user('admin', 'admin')
- with_settings :issue_list_default_totals => ['estimated_hours'] do
- visit '/projects/ecookbook/issues'
- # Check that the page shows the Estimated hours total
- assert page.has_css?('p.query-totals')
- assert page.has_css?('span.total-for-estimated-hours')
- # Open the Options of the form (necessary for having the totalable columns options clickable)
- page.all('legend')[1].click
- # Deselect the default totalable column (none should be left)
- page.first('input[name="t[]"][value="estimated_hours"]').click
- within('#query_form') do
- click_link 'Apply'
- end
- # Check that Totals are not present in the reloaded page
- assert !page.has_css?('p.query-totals')
- assert !page.has_css?('span.total-for-estimated-hours')
- end
- end
-
- def test_update_journal_notes_with_preview
- log_user('admin', 'admin')
-
- visit '/issues/1'
- # Click on the edit button
- page.first('#change-2 a.icon-edit').click
- # Check that the textarea is displayed
- assert page.has_css?('#change-2 textarea')
- assert page.first('#change-2 textarea').has_content?('Some notes with Redmine links')
- # Update the notes
- fill_in 'Notes', :with => 'Updated notes'
- # Preview the change
- page.first('#change-2 a.tab-preview').click
- assert page.has_css?('#preview_journal_2_notes')
- assert page.first('#preview_journal_2_notes').has_content?('Updated notes')
- # Save
- click_on 'Save'
-
- sleep 1
- assert_equal 'Updated notes', Journal.find(2).notes
- end
-
- def test_index_as_csv_should_reflect_sort
- log_user('admin', 'admin')
-
- visit '/issues'
- # Sort issues by subject
- click_on 'Subject'
- click_on 'CSV'
- click_on 'Export'
-
- csv = CSV.read(downloaded_file)
- subject_index = csv.shift.index('Subject')
- subjects = csv.map {|row| row[subject_index]}
- assert_equal subjects.sort, subjects
- end
- end
|