Redmine 3.4.4

This commit is contained in:
Manuel Cillero 2018-02-02 22:19:29 +01:00
commit 64924a6376
2112 changed files with 259028 additions and 0 deletions

74
test/ui/base.rb Normal file
View file

@ -0,0 +1,74 @@
# 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('../../test_helper', __FILE__)
require 'capybara/rails'
Capybara.default_driver = :selenium
Capybara.register_driver :selenium do |app|
# Use the following driver definition to test locally using Chrome
# (also requires chromedriver to be in PATH)
# Capybara::Selenium::Driver.new(app, :browser => :chrome)
# Add :switches => %w[--lang=en] to force default browser locale to English
# Default for Selenium remote driver is to connect to local host on port 4444
# This can be change using :url => 'http://localhost:9195' if necessary
# PhantomJS 1.8 now directly supports Webdriver Wire API,
# simply run it with `phantomjs --webdriver 4444`
# Add :desired_capabilities => Selenium::WebDriver::Remote::Capabilities.internet_explorer)
# to run on Selenium Grid Hub with IE
Capybara::Selenium::Driver.new(app, :browser => :remote)
end
# default: 2
Capybara.default_wait_time = 2
module Redmine
module UiTest
# Base class for UI tests
class Base < ActionDispatch::IntegrationTest
include Capybara::DSL
# Stop ActiveRecord from wrapping tests in transactions
# Transactional fixtures do not work with Selenium tests, because Capybara
# uses a separate server thread, which the transactions would be hidden
self.use_transactional_fixtures = false
# Should not depend on locale since Redmine displays login page
# using default browser locale which depend on system locale for "real" browsers drivers
def log_user(login, password)
visit '/my/page'
assert_equal '/login', current_path
within('#login-form form') do
fill_in 'username', :with => login
fill_in 'password', :with => password
find('input[name=login]').click
end
assert_equal '/my/page', current_path
end
setup do
# Set the page width higher than 900 to get the full layout with sidebar
page.driver.browser.manage.window.resize_to(1024, 900)
end
teardown do
Capybara.reset_sessions! # Forget the (simulated) browser state
Capybara.use_default_driver # Revert Capybara.current_driver to Capybara.default_driver
end
end
end
end

335
test/ui/issues_test_ui.rb Normal file
View file

@ -0,0 +1,335 @@
# 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('../base', __FILE__)
class Redmine::UiTest::IssuesTest < Redmine::UiTest::Base
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
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 'Due date', :with => ''
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_start_due_date
with_settings :default_issue_start_date_to_creation_date => 0 do
log_user('jsmith', 'jsmith')
visit '/projects/ecookbook/issues/new'
assert_equal "", page.find('input#issue_start_date').value
assert_equal "", page.find('input#issue_due_date').value
page.first('p#start_date_area img').click
page.first("td.ui-datepicker-days-cell-over a").click
assert_equal Date.today.to_s, page.find('input#issue_start_date').value
page.first('p#due_date_area img').click
page.first("td.ui-datepicker-days-cell-over a").click
assert_equal Date.today.to_s, page.find('input#issue_due_date').value
end
end
def test_default_due_date_proposed_in_date_picker
log_user('jsmith', 'jsmith')
visit '/projects/ecookbook/issues/new'
# Future start date: due date should default to start date
fill_in 'Start date', :with => '2027-04-01'
fill_in 'Due date', :with => ''
page.first('p#due_date_area img').click
page.first("td.ui-datepicker-days-cell-over a").click
assert_equal '2027-04-01', page.find('input#issue_due_date').value
# Passed start date: due date should default to today
fill_in 'Start date', :with => '2012-04-01'
fill_in 'Due date', :with => ''
page.first('p#due_date_area img').click
page.first("td.ui-datepicker-days-cell-over a").click
assert_equal Date.today.to_s, page.find('input#issue_due_date').value
end
def test_default_start_date_proposed_in_date_picker
log_user('jsmith', 'jsmith')
visit '/projects/ecookbook/issues/new'
# Passed due date: start date should default to due date
fill_in 'Start date', :with => ''
fill_in 'Due date', :with => '2012-04-01'
page.first('p#start_date_area img').click
page.first("td.ui-datepicker-days-cell-over a").click
assert_equal '2012-04-01', page.find('input#issue_start_date').value
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'
end
find 'div#preview fieldset', :visible => true, :text => 'new issue description'
assert_difference 'Issue.count' do
find('input[name=commit]').click
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
click_on 'Preview'
assert page.has_css?('#journal_2_preview')
assert page.first('#journal_2_preview').has_content?('Updated notes')
# Save
click_on 'Save'
sleep 1
assert_equal 'Updated notes', Journal.find(2).notes
end
end

View file

@ -0,0 +1,54 @@
# 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('../base', __FILE__)
class Redmine::UiTest::MyPageTest < Redmine::UiTest::Base
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
def test_sort_assigned_issues
preferences = User.find(2).pref
preferences.my_page_layout = {'top' => ['issuesassignedtome']}
preferences.my_page_settings = {'issuesassignedtome' => {:columns => ['tracker', 'subject', 'due_date'], :sort => 'id:desc'}}
preferences.save!
log_user('jsmith', 'jsmith')
visit '/my/page'
assert page.has_css?('table.issues.sort-by-id')
assert page.has_css?('table.issues.sort-desc')
within('#block-issuesassignedtome') do
# sort by tracker asc
click_link 'Tracker'
assert page.has_css?('table.issues.sort-by-tracker')
assert page.has_css?('table.issues.sort-asc')
# and desc
click_link 'Tracker'
assert page.has_css?('table.issues.sort-by-tracker')
assert page.has_css?('table.issues.sort-desc')
end
# reload the page, sort order should be preserved
visit '/my/page'
assert page.has_css?('table.issues.sort-by-tracker')
assert page.has_css?('table.issues.sort-desc')
end
end

View file

@ -0,0 +1,68 @@
# 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('../base', __FILE__)
class Redmine::UiTest::SudoModeTest < Redmine::UiTest::Base
fixtures :users, :email_addresses
def setup
Redmine::SudoMode.stubs(:enabled?).returns(true)
super
end
def teardown
travel_back
super
end
def test_add_user
log_user('admin', 'admin')
expire_sudo_mode!
visit '/users/new'
assert_difference 'User.count' do
within('form#new_user') do
fill_in 'Login', :with => 'johnpaul'
fill_in 'First name', :with => 'John'
fill_in 'Last name', :with => 'Paul'
fill_in 'Email', :with => 'john@example.net'
fill_in 'Password', :with => 'password'
fill_in 'Confirmation', :with => 'password'
# click_button 'Create' would match both 'Create' and 'Create and continue' buttons
find('input[name=commit]').click
end
assert_equal '/users', current_path
assert page.has_content?("Confirm your password to continue")
assert page.has_css?('form#sudo-form')
within('form#sudo-form') do
fill_in 'Password', :with => 'admin'
click_button 'Submit'
end
end
end
private
# sudo mode is active after sign, let it expire by advancing the time
def expire_sudo_mode!
travel_to 20.minutes.from_now
end
end

View file

@ -0,0 +1,72 @@
# 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('../base', __FILE__)
class Redmine::UiTest::TimelogTest < Redmine::UiTest::Base
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,
:time_entries
def test_changing_project_should_update_activities
project = Project.find(1)
TimeEntryActivity.create!(:name => 'Design', :project => project, :parent => TimeEntryActivity.find_by_name('Design'), :active => false)
log_user 'jsmith', 'jsmith'
visit '/time_entries/new'
within 'select#time_entry_activity_id' do
assert has_content?('Development')
assert has_content?('Design')
end
within 'form#new_time_entry' do
select 'eCookbook', :from => 'Project'
end
within 'select#time_entry_activity_id' do
assert has_content?('Development')
assert !has_content?('Design')
end
end
def test_bulk_edit
log_user 'jsmith', 'jsmith'
visit '/time_entries/bulk_edit?ids[]=1&ids[]=2&ids[]=3'
fill_in 'Hours', :with => '8.5'
select 'QA', :from => 'Activity'
page.first(:button, 'Submit').click
entries = TimeEntry.where(:id => [1,2,3]).to_a
assert entries.all? {|entry| entry.hours == 8.5}
assert entries.all? {|entry| entry.activity.name == 'QA'}
end
def test_bulk_edit_with_failure
log_user 'jsmith', 'jsmith'
visit '/time_entries/bulk_edit?ids[]=1&ids[]=2&ids[]=3'
fill_in 'Hours', :with => 'A'
page.first(:button, 'Submit').click
assert page.has_css?('#errorExplanation')
fill_in 'Hours', :with => '7'
page.first(:button, 'Submit').click
assert_equal "/projects/ecookbook/time_entries", current_path
entries = TimeEntry.where(:id => [1,2,3]).to_a
assert entries.all? {|entry| entry.hours == 7.0}
end
end