canvas-lms/app/models/delayed_notification.rb

89 lines
2.8 KiB
Ruby

#
# Copyright (C) 2011 - present Instructure, Inc.
#
# This file is part of Canvas.
#
# Canvas is free software: you can redistribute it and/or modify it under
# the terms of the GNU Affero General Public License as published by the Free
# Software Foundation, version 3 of the License.
#
# Canvas 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 Affero General Public License for more
# details.
#
# You should have received a copy of the GNU Affero General Public License along
# with this program. If not, see <http://www.gnu.org/licenses/>.
#
class DelayedNotification < ActiveRecord::Base
include Workflow
belongs_to :asset, polymorphic:
[:assessment_request, :attachment, :content_migration, :content_export, :collaborator, :submission,
:assignment, :communication_channel, :calendar_event, :conversation_message, :discussion_entry,
:submission_comment, { quiz_submission: 'Quizzes::QuizSubmission' }, :discussion_topic, :course, :enrollment,
:wiki_page, :group_membership, :web_conference], polymorphic_prefix: true, exhaustive: false
include NotificationPreloader
attr_accessor :data
validates_presence_of :notification_id, :asset_id, :asset_type, :workflow_state
serialize :recipient_keys
workflow do
state :to_be_processed do
event :do_process, :transitions_to => :processed
end
state :processed
state :errored
end
def self.process(asset, notification, recipient_keys, data)
DelayedNotification.new(
asset: asset,
notification: notification,
recipient_keys: recipient_keys,
data: data
).process
end
def process
tos = self.to_list
if self.asset && !tos.empty?
res = self.notification.create_message(self.asset, tos, data: self.data)
end
self.do_process unless self.new_record?
res
rescue => e
Canvas::Errors.capture(e, message: "Delayed Notification processing failed")
logger.error "delayed notification processing failed: #{e.message}\n#{e.backtrace.join "\n"}"
self.workflow_state = 'errored'
self.save
[]
end
def to_list
return @to_list if @to_list
lookups = {}
(recipient_keys || []).each do |key|
pieces = key.split('_')
id = pieces.pop
klass = pieces.join('_').classify.constantize
lookups[klass] ||= []
lookups[klass] << id
end
res = []
lookups.each do |klass, ids|
includes = []
includes = [:user] if klass == CommunicationChannel
res += klass.where(:id => ids).preload(includes).to_a rescue []
end
@to_list = res.uniq
end
scope :to_be_processed, lambda { |limit|
where(:workflow_state => 'to_be_processed').limit(limit).order("delayed_notifications.created_at")
}
end