OpenProject is the leading open source project management software.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 
openproject/app/workers/enqueue_work_package_notifi...

129 lines
4.4 KiB

#-- encoding: UTF-8
#-- copyright
# OpenProject is a project management system.
# Copyright (C) 2012-2018 the OpenProject Foundation (OPF)
#
# 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-2017 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 docs/COPYRIGHT.rdoc for more details.
#++
# Enqueues
class EnqueueWorkPackageNotificationJob < ApplicationJob
def initialize(journal_id, author_id)
@journal_id = journal_id
@author_id = author_id
end
def perform
# if the WP has been deleted the unaggregated journal will have been deleted too
# and our job here is done
return nil unless raw_journal
@journal = find_aggregated_journal
# If we can't find the aggregated journal, it was superseded by a journal that aggregated ours.
# In that case a job for the new journal will have been enqueued that is now responsible for
# sending the notification. Our job here is done.
return nil unless @journal
@author = User.find_by(id: @author_id) || DeletedUser.first
# Do not deliver notifications if a follow-up journal will already have sent a notification
# on behalf of this job.
unless Journal::AggregatedJournal.hides_notifications?(@journal.successor, @journal)
deliver_notifications_for(@journal)
end
end
private
def find_aggregated_journal
wp_journals = Journal::AggregatedJournal.aggregated_journals(journable: work_package)
wp_journals.detect { |journal| journal.version == raw_journal.version }
end
def deliver_notifications_for(journal)
notification_receivers(work_package).each do |recipient|
job = DeliverWorkPackageNotificationJob.new(journal.id, recipient.id, @author_id)
Delayed::Job.enqueue job, priority: ::ApplicationJob.priority_number(:notification)
end
OpenProject::Notifications.send(
OpenProject::Events::AGGREGATED_WORK_PACKAGE_JOURNAL_READY,
journal_id: journal.id,
initial: journal.initial?
)
end
def raw_journal
@raw_journal ||= Journal.find_by(id: @journal_id)
end
def work_package
@work_package ||= raw_journal.journable
end
def notification_receivers(work_package)
(work_package.recipients + work_package.watcher_recipients + mentioned).uniq
end
def mentioned
mentioned_ids
.where(id: User.allowed(:view_work_packages, @work_package.project))
.where.not(mail_notification: User::USER_MAIL_OPTION_NON.first)
end
def text_for_mentions
potential_text = ""
potential_text << @journal.notes if @journal.try(:notes)
%i[description subject].each do |field|
if @journal.details[field].try(:any?)
from = @journal.details[field].first
to = @journal.details[field].second
potential_text << "\n" + Redmine::Helpers::Diff.new(to, from).additions.join(' ')
end
end
potential_text
end
def mentioned_ids
text = text_for_mentions
user_ids, user_login_names, group_ids = text
.scan(/\b(?:user#([\d]+))|(?:user:"(.+?)")|(?:group#([\d]+))\b/)
.transpose
.each(&:compact!)
base_scope = User
.includes(:groups)
.references(:groups_users)
by_id = base_scope.where(id: user_ids || [])
by_login = base_scope.where(login: user_login_names || [])
by_group = base_scope.where(groups_users: { id: group_ids || [] })
by_id
.or(by_login)
.or(by_group)
end
end