Refactor Conversation model (#134)
* Add Conversation factory with dependent factories * Include FactoryBot methods in rspec config * Add unit tests for public methods of Conversation model * Move Current model into a separate file in lib folder * Disable Metrics/BlockLength rule for db/migrate and spec folders * Get rid of global $dispatcher variable * Create Message#unread_since scope * Refactor callback methods in Conversation model * Create Conversations::EventDataPresenter * Add translation keys for activity messages * Add pry-rails gem * Refactor Conversation#notify_status_change * Add mock_redis for test env
This commit is contained in:
committed by
Sojan Jose
parent
43e54a7bfb
commit
4768aca484
@@ -1,7 +1,3 @@
|
||||
module Current
|
||||
thread_mattr_accessor :user
|
||||
end
|
||||
|
||||
class ApplicationController < ActionController::Base
|
||||
include DeviseTokenAuth::Concerns::SetUserByToken
|
||||
include Pundit
|
||||
|
||||
@@ -4,7 +4,7 @@ class Dispatcher
|
||||
attr_reader :async_dispatcher, :sync_dispatcher
|
||||
|
||||
def self.dispatch(event_name, timestamp, data, async = false)
|
||||
$dispatcher.dispatch(event_name, timestamp, data, async)
|
||||
Rails.configuration.dispatcher.dispatch(event_name, timestamp, data, async)
|
||||
end
|
||||
|
||||
def initialize
|
||||
|
||||
@@ -53,10 +53,10 @@ class Account < ApplicationRecord
|
||||
end
|
||||
|
||||
def notify_creation
|
||||
$dispatcher.dispatch(ACCOUNT_CREATED, Time.zone.now, account: self)
|
||||
Rails.configuration.dispatcher.dispatch(ACCOUNT_CREATED, Time.zone.now, account: self)
|
||||
end
|
||||
|
||||
def notify_deletion
|
||||
$dispatcher.dispatch(ACCOUNT_DESTROYED, Time.zone.now, account: self)
|
||||
Rails.configuration.dispatcher.dispatch(ACCOUNT_DESTROYED, Time.zone.now, account: self)
|
||||
end
|
||||
end
|
||||
|
||||
@@ -19,89 +19,73 @@ class Conversation < ApplicationRecord
|
||||
|
||||
before_create :set_display_id, unless: :display_id?
|
||||
|
||||
after_update :notify_status_change,
|
||||
:create_activity,
|
||||
:send_email_notification_to_assignee
|
||||
after_update :notify_status_change, :create_activity, :send_email_notification_to_assignee
|
||||
|
||||
after_create :send_events, :run_round_robin
|
||||
|
||||
acts_as_taggable_on :labels
|
||||
|
||||
def update_assignee(agent = nil)
|
||||
self.assignee = agent
|
||||
save!
|
||||
update!(assignee: agent)
|
||||
end
|
||||
|
||||
def update_labels(labels = nil)
|
||||
self.label_list = labels
|
||||
save!
|
||||
update!(label_list: labels)
|
||||
end
|
||||
|
||||
def toggle_status
|
||||
self.status = open? ? :resolved : :open
|
||||
save! ? true : false
|
||||
save
|
||||
end
|
||||
|
||||
def lock!
|
||||
self.locked = true
|
||||
save!
|
||||
update!(locked: true)
|
||||
end
|
||||
|
||||
def unlock!
|
||||
self.locked = false
|
||||
save!
|
||||
update!(locked: false)
|
||||
end
|
||||
|
||||
def unread_messages
|
||||
# +1 is a hack to avoid https://makandracards.com/makandra/1057-why-two-ruby-time-objects-are-not-equal-although-they-appear-to-be
|
||||
# ente budhiparamaya neekam kandit entu tonunu?
|
||||
messages.where('EXTRACT(EPOCH FROM created_at) > (?)', agent_last_seen_at.to_i + 1)
|
||||
messages.unread_since(agent_last_seen_at)
|
||||
end
|
||||
|
||||
def unread_incoming_messages
|
||||
messages.incoming.where('EXTRACT(EPOCH FROM created_at) > (?)', agent_last_seen_at.to_i + 1)
|
||||
messages.incoming.unread_since(agent_last_seen_at)
|
||||
end
|
||||
|
||||
def push_event_data
|
||||
{
|
||||
meta: { sender: sender.push_event_data, assignee: assignee }, id: display_id,
|
||||
messages: [messages.chat.last&.push_event_data], inbox_id: inbox_id, status: status_before_type_cast.to_i,
|
||||
timestamp: created_at.to_i, user_last_seen_at: user_last_seen_at.to_i, agent_last_seen_at: agent_last_seen_at.to_i,
|
||||
unread_count: unread_incoming_messages.count
|
||||
}
|
||||
Conversations::EventDataPresenter.new(self).push_data
|
||||
end
|
||||
|
||||
def lock_event_data
|
||||
{
|
||||
id: display_id,
|
||||
locked: locked?
|
||||
}
|
||||
Conversations::EventDataPresenter.new(self).lock_data
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def dispatch_events
|
||||
$dispatcher.dispatch(CONVERSATION_RESOLVED, Time.zone.now, conversation: self)
|
||||
dispatcher_dispatch(CONVERSATION_RESOLVED)
|
||||
end
|
||||
|
||||
def send_events
|
||||
$dispatcher.dispatch(CONVERSATION_CREATED, Time.zone.now, conversation: self)
|
||||
dispatcher_dispatch(CONVERSATION_CREATED)
|
||||
end
|
||||
|
||||
def send_email_notification_to_assignee
|
||||
AssignmentMailer.conversation_assigned(self, assignee).deliver if assignee_id_changed? && assignee_id.present? && !self_assign?(assignee_id)
|
||||
return if self_assign?(assignee_id)
|
||||
|
||||
AssignmentMailer.conversation_assigned(self, assignee).deliver if saved_change_to_assignee_id? && assignee_id.present?
|
||||
end
|
||||
|
||||
def self_assign?(assignee_id)
|
||||
return false unless Current.user
|
||||
|
||||
Current.user.id == assignee_id
|
||||
assignee_id.present? && Current.user&.id == assignee_id
|
||||
end
|
||||
|
||||
def set_display_id
|
||||
self.display_id = loop do
|
||||
disp_id = account.conversations.maximum('display_id').to_i + 1
|
||||
break disp_id unless account.conversations.exists?(display_id: disp_id)
|
||||
next_display_id = account.conversations.maximum('display_id').to_i + 1
|
||||
break next_display_id unless account.conversations.exists?(display_id: next_display_id)
|
||||
end
|
||||
end
|
||||
|
||||
@@ -110,78 +94,48 @@ class Conversation < ApplicationRecord
|
||||
|
||||
user_name = Current.user&.name
|
||||
|
||||
create_status_change_message(user_name) if status_changed?
|
||||
create_assignee_change(username) if assignee_id_changed?
|
||||
end
|
||||
|
||||
def status_changed_message
|
||||
return "Conversation was marked resolved by #{Current.user.try(:name)}" if resolved?
|
||||
|
||||
"Conversation was reopened by #{Current.user.try(:name)}"
|
||||
end
|
||||
|
||||
def assignee_changed_message
|
||||
return "Assigned to #{assignee.name} by #{Current.user.try(:name)}" if assignee_id
|
||||
|
||||
"Conversation unassigned by #{Current.user.try(:name)}"
|
||||
create_status_change_message(user_name) if saved_change_to_assignee_id?
|
||||
create_assignee_change(user_name) if saved_change_to_assignee_id?
|
||||
end
|
||||
|
||||
def activity_message_params(content)
|
||||
{
|
||||
account_id: account_id,
|
||||
inbox_id: inbox_id,
|
||||
message_type: :activity,
|
||||
content: content
|
||||
}
|
||||
{ account_id: account_id, inbox_id: inbox_id, message_type: :activity, content: content }
|
||||
end
|
||||
|
||||
def notify_status_change
|
||||
resolve_conversation if status_changed?
|
||||
dispatcher_dispatch(CONVERSATION_READ) if user_last_seen_at_changed?
|
||||
dispatcher_dispatch(CONVERSATION_LOCK_TOGGLE) if locked_changed?
|
||||
dispatcher_dispatch(ASSIGNEE_CHANGED) if assignee_id_changed?
|
||||
end
|
||||
|
||||
def resolve_conversation
|
||||
if resolved? && assignee.present?
|
||||
dispatcher_dispatch(CONVERSATION_RESOLVED)
|
||||
{
|
||||
CONVERSATION_RESOLVED => -> { saved_change_to_status? && resolved? && assignee.present? },
|
||||
CONVERSATION_READ => -> { saved_change_to_user_last_seen_at? },
|
||||
CONVERSATION_LOCK_TOGGLE => -> { saved_change_to_locked? },
|
||||
ASSIGNEE_CHANGED => -> { saved_change_to_assignee_id? }
|
||||
}.each do |event, condition|
|
||||
condition.call && dispatcher_dispatch(event)
|
||||
end
|
||||
end
|
||||
|
||||
def dispatcher_dispatch(event_name)
|
||||
$dispatcher.dispatch(event_name, Time.zone.now, conversation: self)
|
||||
Rails.configuration.dispatcher.dispatch(event_name, Time.zone.now, conversation: self)
|
||||
end
|
||||
|
||||
def run_round_robin
|
||||
return unless true # conversation.account.has_feature?(round_robin)
|
||||
return unless true # conversation.account.round_robin_enabled?
|
||||
# return unless conversation.account.has_feature?(round_robin)
|
||||
# return unless conversation.account.round_robin_enabled?
|
||||
return if assignee
|
||||
|
||||
new_assignee = inbox.next_available_agent
|
||||
update_assignee(new_assignee) if new_assignee
|
||||
inbox.next_available_agent.then { |new_assignee| update_assignee(new_assignee) }
|
||||
end
|
||||
|
||||
def create_status_change_message(user_name)
|
||||
content = if resolved?
|
||||
"Conversation was marked resolved by #{user_name}"
|
||||
else
|
||||
"Conversation was reopened by #{user_name}"
|
||||
end
|
||||
content = I18n.t("conversations.activity.status.#{status}", user_name: user_name)
|
||||
|
||||
messages.create(activity_message_params(content))
|
||||
end
|
||||
|
||||
def create_assignee_change(username)
|
||||
content = if assignee_id
|
||||
"Assigned to #{assignee.name} by #{username}"
|
||||
else
|
||||
"Conversation unassigned by #{username}"
|
||||
end
|
||||
def create_assignee_change(user_name)
|
||||
params = { assignee_name: assignee&.name, user_name: user_name }.compact
|
||||
key = assignee_id ? 'assigned' : 'removed'
|
||||
content = I18n.t("conversations.activity.assignee.#{key}", params)
|
||||
|
||||
messages.create(activity_message_params(content))
|
||||
end
|
||||
|
||||
def resolved_and_assignee?
|
||||
resolved? && assignee.present?
|
||||
end
|
||||
end
|
||||
|
||||
@@ -8,6 +8,8 @@ class Message < ApplicationRecord
|
||||
enum message_type: [ :incoming, :outgoing, :activity ]
|
||||
enum status: [ :sent, :delivered, :read, :failed ]
|
||||
|
||||
# .succ is a hack to avoid https://makandracards.com/makandra/1057-why-two-ruby-time-objects-are-not-equal-although-they-appear-to-be
|
||||
scope :unread_since, ->(datetime) { where('EXTRACT(EPOCH FROM created_at) > (?)', datetime.to_i.succ) }
|
||||
scope :chat, -> { where.not(message_type: :activity, private: true) }
|
||||
default_scope { order(created_at: :asc) }
|
||||
|
||||
@@ -42,10 +44,10 @@ class Message < ApplicationRecord
|
||||
private
|
||||
|
||||
def dispatch_event
|
||||
$dispatcher.dispatch(MESSAGE_CREATED, Time.zone.now, message: self) unless self.conversation.messages.count == 1
|
||||
Rails.configuration.dispatcher.dispatch(MESSAGE_CREATED, Time.zone.now, message: self) unless self.conversation.messages.count == 1
|
||||
|
||||
if outgoing? && self.conversation.messages.outgoing.count == 1
|
||||
$dispatcher.dispatch(FIRST_REPLY_CREATED, Time.zone.now, message: self)
|
||||
Rails.configuration.dispatcher.dispatch(FIRST_REPLY_CREATED, Time.zone.now, message: self)
|
||||
end
|
||||
end
|
||||
|
||||
@@ -56,7 +58,7 @@ class Message < ApplicationRecord
|
||||
def reopen_conversation
|
||||
if incoming? && self.conversation.resolved?
|
||||
self.conversation.toggle_status
|
||||
$dispatcher.dispatch(CONVERSATION_REOPENED, Time.zone.now, conversation: self.conversation)
|
||||
Rails.configuration.dispatcher.dispatch(CONVERSATION_REOPENED, Time.zone.now, conversation: self.conversation)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -36,6 +36,6 @@ class Subscription < ApplicationRecord
|
||||
end
|
||||
|
||||
def notify_creation
|
||||
$dispatcher.dispatch(SUBSCRIPTION_CREATED, Time.zone.now, subscription: self)
|
||||
Rails.configuration.dispatcher.dispatch(SUBSCRIPTION_CREATED, Time.zone.now, subscription: self)
|
||||
end
|
||||
end
|
||||
|
||||
@@ -43,11 +43,11 @@ class User < ApplicationRecord
|
||||
end
|
||||
|
||||
def notify_creation
|
||||
$dispatcher.dispatch(AGENT_ADDED, Time.zone.now, account: self.account)
|
||||
Rails.configuration.dispatcher.dispatch(AGENT_ADDED, Time.zone.now, account: self.account)
|
||||
end
|
||||
|
||||
def notify_deletion
|
||||
$dispatcher.dispatch(AGENT_REMOVED, Time.zone.now, account: self.account)
|
||||
Rails.configuration.dispatcher.dispatch(AGENT_REMOVED, Time.zone.now, account: self.account)
|
||||
end
|
||||
|
||||
def push_event_data
|
||||
|
||||
37
app/presenters/conversations/event_data_presenter.rb
Normal file
37
app/presenters/conversations/event_data_presenter.rb
Normal file
@@ -0,0 +1,37 @@
|
||||
module Conversations
|
||||
class EventDataPresenter < SimpleDelegator
|
||||
def lock_data
|
||||
{ id: display_id, locked: locked? }
|
||||
end
|
||||
|
||||
def push_data
|
||||
{
|
||||
id: display_id,
|
||||
inbox_id: inbox_id,
|
||||
messages: push_messages,
|
||||
meta: push_meta,
|
||||
status: status_before_type_cast.to_i,
|
||||
unread_count: unread_incoming_messages.count,
|
||||
**push_timestamps
|
||||
}
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def push_messages
|
||||
[messages.chat.last&.push_event_data].compact
|
||||
end
|
||||
|
||||
def push_meta
|
||||
{ sender: sender.push_event_data, assignee: assignee }
|
||||
end
|
||||
|
||||
def push_timestamps
|
||||
{
|
||||
agent_last_seen_at: agent_last_seen_at.to_i,
|
||||
user_last_seen_at: user_last_seen_at.to_i,
|
||||
timestamp: created_at.to_i
|
||||
}
|
||||
end
|
||||
end
|
||||
end
|
||||
Reference in New Issue
Block a user