Files

155 lines
4.8 KiB
Ruby

# frozen_string_literal: true
#-- copyright
# OpenProject is an open source project management software.
# Copyright (C) the OpenProject GmbH
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License version 3.
#
# OpenProject is a fork of ChiliProject, which is a fork of Redmine. The copyright follows:
# Copyright (C) 2006-2013 Jean-Philippe Lang
# Copyright (C) 2010-2013 the ChiliProject Team
#
# 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.
#
# See COPYRIGHT and LICENSE files for more details.
#++
class WorkPackages::CopyService < BaseServices::BaseCallable
include ::Shared::ServiceContext
include Contracted
include ::Copy::Concerns::CopyAttachments
attr_accessor :user,
:work_package,
:contract_class
def initialize(user:, work_package:, contract_class: WorkPackages::CopyContract)
super()
self.user = user
self.work_package = work_package
self.contract_class = contract_class
end
def perform
attributes = params.except(:send_notifications, :copy_attachments, :copy_share_members)
in_context(work_package, send_notifications: params[:send_notifications]) do
copy(attributes, params[:send_notifications])
end
end
protected
def copy(attribute_override, send_notifications)
copied = create(work_package,
attribute_override,
send_notifications)
.on_success do |copy_call|
remove_author_watcher(copy_call.result)
copy_watchers(copy_call.result)
copy_work_package_attachments(copy_call.result) if copy_attachments?
copy_share_members(copy_call.result, send_notifications) if copy_share_members?
end
copied.state.copied_from_work_package_id = work_package&.id
copied
end
def create(work_package, attribute_overrides, send_notifications)
WorkPackages::CreateService
.new(user:,
contract_class:)
.with_state(state)
.call(**copied_attributes(work_package, attribute_overrides).merge(send_notifications:).symbolize_keys)
end
def copied_attributes(work_package, override)
overwritten_attributes = override.stringify_keys
attributes = work_package
.attributes
.slice(*writable_work_package_attributes(work_package))
.merge("custom_field_values" => work_package.custom_value_attributes)
.merge(overwritten_attributes)
if overwritten_attributes.has_key?("start_date") &&
overwritten_attributes.has_key?("due_date") &&
!overwritten_attributes.has_key?("duration")
attributes.delete("duration")
end
attributes
end
def writable_work_package_attributes(work_package)
instantiate_contract(work_package, user).writable_attributes
end
def remove_author_watcher(copied)
copied.remove_watcher(copied.author)
end
def copy_watchers(copied)
work_package.watcher_users.each do |user|
copied.add_watcher(user) if user.active?
end
end
def copy_work_package_attachments(copy)
copy_attachments(
"WorkPackage",
from: work_package,
to: copy,
references: %i[description]
)
end
def copy_share_members(copy, send_notifications)
work_package.members.each do |member|
create_share_membership(member, copy, send_notifications)
end
end
private
def create_share_membership(member, target, send_notifications)
role_ids = member.member_roles.map(&:role_id)
return if role_ids.empty?
attributes = member
.attributes.dup.except("id", "project_id", "entity_id", "created_at", "updated_at")
.merge(role_ids:,
project: target.project,
entity: target,
send_notifications:)
Shares::CreateService
.new(user: User.current, contract_class: EmptyContract)
.call(attributes)
end
def copy_attachments?
params[:copy_attachments] || params[:copy_attachments].nil?
end
def copy_share_members?
params[:copy_share_members] || params[:copy_share_members].nil?
end
end