summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--app/controllers/projects_controller.rb25
-rw-r--r--app/jobs/destroy_projects_job.rb31
-rw-r--r--app/views/context_menus/projects.html.erb6
-rw-r--r--app/views/projects/bulk_destroy.html.erb27
-rw-r--r--config/locales/en.yml7
-rw-r--r--config/routes.rb1
-rw-r--r--test/functional/projects_controller_test.rb25
-rw-r--r--test/unit/jobs/destroy_projects_job_test.rb63
8 files changed, 181 insertions, 4 deletions
diff --git a/app/controllers/projects_controller.rb b/app/controllers/projects_controller.rb
index d8b2e8b9a..695ca1dec 100644
--- a/app/controllers/projects_controller.rb
+++ b/app/controllers/projects_controller.rb
@@ -23,16 +23,16 @@ class ProjectsController < ApplicationController
menu_item :projects, :only => [:index, :new, :copy, :create]
before_action :find_project,
- :except => [:index, :autocomplete, :list, :new, :create, :copy]
+ :except => [:index, :autocomplete, :list, :new, :create, :copy, :bulk_destroy]
before_action :authorize,
:except => [:index, :autocomplete, :list, :new, :create, :copy,
:archive, :unarchive,
- :destroy]
+ :destroy, :bulk_destroy]
before_action :authorize_global, :only => [:new, :create]
- before_action :require_admin, :only => [:copy, :archive, :unarchive]
+ before_action :require_admin, :only => [:copy, :archive, :unarchive, :bulk_destroy]
accept_atom_auth :index
accept_api_auth :index, :show, :create, :update, :destroy, :archive, :unarchive, :close, :reopen
- require_sudo_mode :destroy
+ require_sudo_mode :destroy, :bulk_destroy
helper :custom_fields
helper :issues
@@ -315,6 +315,23 @@ class ProjectsController < ApplicationController
@project = nil
end
+ # Delete selected projects
+ def bulk_destroy
+ @projects = Project.where(id: params[:ids]).
+ where.not(status: Project::STATUS_SCHEDULED_FOR_DELETION).to_a
+
+ if @projects.empty?
+ render_404
+ return
+ end
+
+ if params[:confirm] == I18n.t(:general_text_Yes)
+ DestroyProjectsJob.schedule @projects
+ flash[:notice] = l(:notice_successful_delete)
+ redirect_to admin_projects_path
+ end
+ end
+
private
# Returns the ProjectEntry scope for index
diff --git a/app/jobs/destroy_projects_job.rb b/app/jobs/destroy_projects_job.rb
new file mode 100644
index 000000000..dba1213dc
--- /dev/null
+++ b/app/jobs/destroy_projects_job.rb
@@ -0,0 +1,31 @@
+# frozen_string_literal: true
+
+class DestroyProjectsJob < ApplicationJob
+ include Redmine::I18n
+
+ def self.schedule(projects_to_delete, user: User.current)
+ # make the projects disappear immediately
+ projects_to_delete.each do |project|
+ project.self_and_descendants.update_all status: Project::STATUS_SCHEDULED_FOR_DELETION
+ end
+ perform_later(projects_to_delete.map(&:id), user.id, user.remote_ip)
+ end
+
+ def perform(project_ids, user_id, remote_ip)
+ user = User.active.find_by_id(user_id)
+ unless user&.admin?
+ info "[DestroyProjectsJob] --- User check failed: User #{user_id} triggering projects destroy does not exist anymore or isn't admin/active."
+ return
+ end
+
+ project_ids.each do |project_id|
+ DestroyProjectJob.perform_now(project_id, user_id, remote_ip)
+ end
+ end
+
+ private
+
+ def info(*msg)
+ Rails.logger.info(*msg)
+ end
+end
diff --git a/app/views/context_menus/projects.html.erb b/app/views/context_menus/projects.html.erb
index 444d84c58..c200311f2 100644
--- a/app/views/context_menus/projects.html.erb
+++ b/app/views/context_menus/projects.html.erb
@@ -11,5 +11,11 @@
<li>
<%= context_menu_link l(:button_delete), project_path(@project, back_url: @back), method: :delete, class: 'icon icon-del' %>
</li>
+ <% else %>
+ <li>
+ <%= context_menu_link l(:button_delete),
+ {controller: 'projects', action: 'bulk_destroy', ids: @projects.map(&:id), back_url: @back},
+ method: :delete, data: {confirm: l(:text_projects_bulk_destroy_confirmation)}, class: 'icon icon-del' %>
+ </li>
<% end %>
</ul>
diff --git a/app/views/projects/bulk_destroy.html.erb b/app/views/projects/bulk_destroy.html.erb
new file mode 100644
index 000000000..43c6dcf51
--- /dev/null
+++ b/app/views/projects/bulk_destroy.html.erb
@@ -0,0 +1,27 @@
+<%= title l(:label_confirmation) %>
+
+<%= form_tag(bulk_destroy_projects_path(ids: @projects.map(&:id)), method: :delete) do %>
+<div class="warning">
+
+<p><%= simple_format l :text_projects_bulk_destroy_head %></p>
+
+<% @projects.each do |project| %>
+ <p>Project: <strong><%= project.to_s %></strong>
+ <% if project.descendants.any? %>
+ <br />
+ <%= l :text_subprojects_bulk_destroy, project.descendants.map(&:to_s).join(', ') %>
+ <% end %>
+ </p>
+<% end %>
+
+<p><%= l :text_projects_bulk_destroy_confirm, yes: l(:general_text_Yes) %></p>
+<p><%= text_field_tag 'confirm' %></p>
+
+</div>
+
+<p>
+ <%= submit_tag l(:button_delete), class: 'btn-alert btn-small' %>
+ <%= link_to l(:button_cancel), admin_projects_path %>
+</p>
+<% end %>
+
diff --git a/config/locales/en.yml b/config/locales/en.yml
index 6ca11855c..49023dd1c 100644
--- a/config/locales/en.yml
+++ b/config/locales/en.yml
@@ -1212,7 +1212,14 @@ en:
text_select_mail_notifications: Select actions for which email notifications should be sent.
text_regexp_info: eg. ^[A-Z0-9]+$
text_project_destroy_confirmation: Are you sure you want to delete this project and related data?
+ text_projects_bulk_destroy_confirmation: Are you sure you want to delete the selected projects and related data?
+ text_projects_bulk_destroy_head: |
+ You are about to permanently delete the following projects, including possible subprojects and any related data.
+ Please review the information below and confirm that this is indeed what you want to do.
+ This action cannot be undone.
+ text_projects_bulk_destroy_confirm: To confirm, please enter "%{yes}" in the box below.
text_subprojects_destroy_warning: "Its subproject(s): %{value} will be also deleted."
+ text_subprojects_bulk_destroy: "including its subproject(s): %{value}"
text_project_close_confirmation: Are you sure you want to close the '%{value}' project to make it read-only?
text_project_reopen_confirmation: Are you sure you want to reopen the '%{value}' project?
text_project_archive_confirmation: Are you sure you want to archive the '%{value}' project?
diff --git a/config/routes.rb b/config/routes.rb
index 9d6c54132..1e4b20d94 100644
--- a/config/routes.rb
+++ b/config/routes.rb
@@ -128,6 +128,7 @@ Rails.application.routes.draw do
resources :projects do
collection do
get 'autocomplete'
+ delete 'bulk_destroy'
end
member do
diff --git a/test/functional/projects_controller_test.rb b/test/functional/projects_controller_test.rb
index 411496f0d..07f590365 100644
--- a/test/functional/projects_controller_test.rb
+++ b/test/functional/projects_controller_test.rb
@@ -1227,6 +1227,31 @@ class ProjectsControllerTest < Redmine::ControllerTest
assert Project.find(1)
end
+ def test_bulk_destroy_should_require_admin
+ @request.session[:user_id] = 2 # non-admin
+ delete :bulk_destroy, params: { ids: [1, 2], confirm: 'Yes' }
+ assert_response 403
+ end
+
+ def test_bulk_destroy_should_require_confirmation
+ @request.session[:user_id] = 1 # admin
+ assert_difference 'Project.count', 0 do
+ delete :bulk_destroy, params: { ids: [1, 2] }
+ end
+ assert Project.find(1)
+ assert Project.find(2)
+ assert_response 200
+ end
+
+ def test_bulk_destroy_should_delete_projects
+ @request.session[:user_id] = 1 # admin
+ assert_difference 'Project.count', -2 do
+ delete :bulk_destroy, params: { ids: [2, 6], confirm: 'Yes' }
+ end
+ assert_equal 0, Project.where(id: [2, 6]).count
+ assert_redirected_to '/admin/projects'
+ end
+
def test_archive
@request.session[:user_id] = 1 # admin
post(:archive, :params => {:id => 1})
diff --git a/test/unit/jobs/destroy_projects_job_test.rb b/test/unit/jobs/destroy_projects_job_test.rb
new file mode 100644
index 000000000..980f983e4
--- /dev/null
+++ b/test/unit/jobs/destroy_projects_job_test.rb
@@ -0,0 +1,63 @@
+# frozen_string_literal: true
+
+# Redmine - project management software
+# Copyright (C) 2006-2022 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__)
+
+class DestroyProjectsJobTest < ActiveJob::TestCase
+ fixtures :users, :projects, :email_addresses
+
+ setup do
+ @projects = Project.where(id: [1, 2]).to_a
+ @user = User.find_by_admin true
+ end
+
+ test "schedule should mark projects and children for deletion" do
+ DestroyProjectsJob.schedule @projects, user: @user
+ @projects.each do |project|
+ project.reload
+ assert_equal Project::STATUS_SCHEDULED_FOR_DELETION, project.status
+ project.descendants.each do |child|
+ assert_equal Project::STATUS_SCHEDULED_FOR_DELETION, child.status
+ end
+ end
+ end
+
+ test "schedule should enqueue job" do
+ assert_enqueued_with(
+ job: DestroyProjectsJob,
+ args: [[1, 2], @user.id, '127.0.0.1']
+ ) do
+ @user.remote_ip = '127.0.0.1'
+ DestroyProjectsJob.schedule @projects, user: @user
+ end
+ end
+
+ test "should destroy projects and send emails" do
+ assert_difference 'Project.count', -6 do
+ DestroyProjectsJob.perform_now @projects.map(&:id), @user.id, '127.0.0.1'
+ end
+ assert_enqueued_with(
+ job: ActionMailer::MailDeliveryJob,
+ args: ->(job_args){
+ job_args[1] == 'security_notification' &&
+ job_args[3].to_s.include?("mail_destroy_project_with_subprojects_successful")
+ }
+ )
+ end
+end