Chore: Add a real-time event for contact resolution/update (#696)
* Chore: Add a real-time event for contact resolution/update * Chore: Ensure Events are sent to administrators Addresses: #419
This commit is contained in:
parent
818c769bb7
commit
ecccb103a0
7 changed files with 96 additions and 26 deletions
|
@ -3,23 +3,22 @@ class ActionCableListener < BaseListener
|
||||||
|
|
||||||
def conversation_created(event)
|
def conversation_created(event)
|
||||||
conversation, account, timestamp = extract_conversation_and_account(event)
|
conversation, account, timestamp = extract_conversation_and_account(event)
|
||||||
members = conversation.inbox.members.pluck(:pubsub_token)
|
send_to_administrators(account.administrators, CONVERSATION_CREATED, conversation.push_event_data)
|
||||||
send_to_members(members, CONVERSATION_CREATED, conversation.push_event_data)
|
send_to_agents(conversation.inbox.members, CONVERSATION_CREATED, conversation.push_event_data)
|
||||||
end
|
end
|
||||||
|
|
||||||
def conversation_read(event)
|
def conversation_read(event)
|
||||||
conversation, account, timestamp = extract_conversation_and_account(event)
|
conversation, account, timestamp = extract_conversation_and_account(event)
|
||||||
members = conversation.inbox.members.pluck(:pubsub_token)
|
send_to_administrators(account.administrators, CONVERSATION_READ, conversation.push_event_data)
|
||||||
send_to_members(members, CONVERSATION_READ, conversation.push_event_data)
|
send_to_agents(conversation.inbox.members, CONVERSATION_READ, conversation.push_event_data)
|
||||||
end
|
end
|
||||||
|
|
||||||
def message_created(event)
|
def message_created(event)
|
||||||
message, account, timestamp = extract_message_and_account(event)
|
message, account, timestamp = extract_message_and_account(event)
|
||||||
conversation = message.conversation
|
conversation = message.conversation
|
||||||
contact = conversation.contact
|
send_to_administrators(account.administrators, MESSAGE_CREATED, message.push_event_data)
|
||||||
members = conversation.inbox.members.pluck(:pubsub_token)
|
send_to_agents(conversation.inbox.members, MESSAGE_CREATED, message.push_event_data)
|
||||||
send_to_members(members, MESSAGE_CREATED, message.push_event_data)
|
send_to_contact(conversation.contact, MESSAGE_CREATED, message)
|
||||||
send_to_contact(contact, MESSAGE_CREATED, message)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def message_updated(event)
|
def message_updated(event)
|
||||||
|
@ -33,28 +32,48 @@ class ActionCableListener < BaseListener
|
||||||
|
|
||||||
def conversation_reopened(event)
|
def conversation_reopened(event)
|
||||||
conversation, account, timestamp = extract_conversation_and_account(event)
|
conversation, account, timestamp = extract_conversation_and_account(event)
|
||||||
members = conversation.inbox.members.pluck(:pubsub_token)
|
send_to_administrators(account.administrators, CONVERSATION_REOPENED, conversation.push_event_data)
|
||||||
send_to_members(members, CONVERSATION_REOPENED, conversation.push_event_data)
|
send_to_agents(conversation.inbox.members, CONVERSATION_REOPENED, conversation.push_event_data)
|
||||||
end
|
end
|
||||||
|
|
||||||
def conversation_lock_toggle(event)
|
def conversation_lock_toggle(event)
|
||||||
conversation, account, timestamp = extract_conversation_and_account(event)
|
conversation, account, timestamp = extract_conversation_and_account(event)
|
||||||
members = conversation.inbox.members.pluck(:pubsub_token)
|
send_to_administrators(account.administrators, CONVERSATION_LOCK_TOGGLE, conversation.lock_event_data)
|
||||||
send_to_members(members, CONVERSATION_LOCK_TOGGLE, conversation.lock_event_data)
|
send_to_agents(conversation.inbox.members, CONVERSATION_LOCK_TOGGLE, conversation.lock_event_data)
|
||||||
end
|
end
|
||||||
|
|
||||||
def assignee_changed(event)
|
def assignee_changed(event)
|
||||||
conversation, account, timestamp = extract_conversation_and_account(event)
|
conversation, account, timestamp = extract_conversation_and_account(event)
|
||||||
members = conversation.inbox.members.pluck(:pubsub_token)
|
send_to_administrators(account.administrators, ASSIGNEE_CHANGED, conversation.push_event_data)
|
||||||
send_to_members(members, ASSIGNEE_CHANGED, conversation.push_event_data)
|
send_to_agents(conversation.inbox.members, ASSIGNEE_CHANGED, conversation.push_event_data)
|
||||||
|
end
|
||||||
|
|
||||||
|
def contact_created(event)
|
||||||
|
contact, account, timestamp = extract_contact_and_account(event)
|
||||||
|
send_to_administrators(account.administrators, CONTACT_CREATED, contact.push_event_data)
|
||||||
|
send_to_agents(account.agents, CONTACT_CREATED, contact.push_event_data)
|
||||||
|
end
|
||||||
|
|
||||||
|
def contact_updated(event)
|
||||||
|
contact, account, timestamp = extract_contact_and_account(event)
|
||||||
|
send_to_administrators(account.administrators, CONTACT_UPDATED, contact.push_event_data)
|
||||||
|
send_to_agents(account.agents, CONTACT_UPDATED, contact.push_event_data)
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def send_to_members(members, event_name, data)
|
def send_to_administrators(admins, event_name, data)
|
||||||
return if members.blank?
|
admin_tokens = admins.pluck(:pubsub_token)
|
||||||
|
return if admin_tokens.blank?
|
||||||
|
|
||||||
::ActionCableBroadcastJob.perform_later(members, event_name, data)
|
::ActionCableBroadcastJob.perform_later(admin_tokens, event_name, data)
|
||||||
|
end
|
||||||
|
|
||||||
|
def send_to_agents(agents, event_name, data)
|
||||||
|
agent_tokens = agents.pluck(:pubsub_token)
|
||||||
|
return if agent_tokens.blank?
|
||||||
|
|
||||||
|
::ActionCableBroadcastJob.perform_later(agent_tokens, event_name, data)
|
||||||
end
|
end
|
||||||
|
|
||||||
def send_to_contact(contact, event_name, message)
|
def send_to_contact(contact, event_name, message)
|
||||||
|
@ -64,8 +83,4 @@ class ActionCableListener < BaseListener
|
||||||
|
|
||||||
::ActionCableBroadcastJob.perform_later([contact.pubsub_token], event_name, message.push_event_data)
|
::ActionCableBroadcastJob.perform_later([contact.pubsub_token], event_name, message.push_event_data)
|
||||||
end
|
end
|
||||||
|
|
||||||
def push(pubsub_token, data)
|
|
||||||
# Enqueue sidekiq job to push event to corresponding channel
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
|
@ -10,4 +10,9 @@ class BaseListener
|
||||||
message = event.data[:message]
|
message = event.data[:message]
|
||||||
[message, message.account, event.timestamp]
|
[message, message.account, event.timestamp]
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def extract_contact_and_account(event)
|
||||||
|
contact = event.data[:contact]
|
||||||
|
[contact, contact.account, event.timestamp]
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -38,9 +38,12 @@ class Account < ApplicationRecord
|
||||||
after_create :notify_creation
|
after_create :notify_creation
|
||||||
after_destroy :notify_deletion
|
after_destroy :notify_deletion
|
||||||
|
|
||||||
def channel
|
def agents
|
||||||
# This should be unique for account
|
users.where(account_users: { role: :agent })
|
||||||
'test_channel'
|
end
|
||||||
|
|
||||||
|
def administrators
|
||||||
|
users.where(account_users: { role: :administrator })
|
||||||
end
|
end
|
||||||
|
|
||||||
def all_conversation_tags
|
def all_conversation_tags
|
||||||
|
|
|
@ -25,6 +25,8 @@ class Contact < ApplicationRecord
|
||||||
include Pubsubable
|
include Pubsubable
|
||||||
include Avatarable
|
include Avatarable
|
||||||
include AvailabilityStatusable
|
include AvailabilityStatusable
|
||||||
|
include Events::Types
|
||||||
|
|
||||||
validates :account_id, presence: true
|
validates :account_id, presence: true
|
||||||
validates :email, allow_blank: true, uniqueness: { scope: [:account_id], case_sensitive: false }
|
validates :email, allow_blank: true, uniqueness: { scope: [:account_id], case_sensitive: false }
|
||||||
validates :identifier, allow_blank: true, uniqueness: { scope: [:account_id] }
|
validates :identifier, allow_blank: true, uniqueness: { scope: [:account_id] }
|
||||||
|
@ -36,6 +38,8 @@ class Contact < ApplicationRecord
|
||||||
has_many :messages, dependent: :destroy
|
has_many :messages, dependent: :destroy
|
||||||
|
|
||||||
before_validation :downcase_email
|
before_validation :downcase_email
|
||||||
|
after_create :dispatch_create_event
|
||||||
|
after_update :dispatch_update_event
|
||||||
|
|
||||||
def get_source_id(inbox_id)
|
def get_source_id(inbox_id)
|
||||||
contact_inboxes.find_by!(inbox_id: inbox_id).source_id
|
contact_inboxes.find_by!(inbox_id: inbox_id).source_id
|
||||||
|
@ -53,11 +57,22 @@ class Contact < ApplicationRecord
|
||||||
def webhook_data
|
def webhook_data
|
||||||
{
|
{
|
||||||
id: id,
|
id: id,
|
||||||
name: name
|
name: name,
|
||||||
|
avatar: avatar_url
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
|
|
||||||
def downcase_email
|
def downcase_email
|
||||||
email.downcase! if email.present?
|
email.downcase! if email.present?
|
||||||
end
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def dispatch_create_event
|
||||||
|
Rails.configuration.dispatcher.dispatch(CONTACT_CREATED, Time.zone.now, contact: self)
|
||||||
|
end
|
||||||
|
|
||||||
|
def dispatch_update_event
|
||||||
|
Rails.configuration.dispatcher.dispatch(CONTACT_UPDATED, Time.zone.now, contact: self)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -11,7 +11,7 @@ LANGUAGES_CONFIG = {
|
||||||
6 => { name: 'Italian', iso_639_3_code: 'ita', iso_639_1_code: 'it' },
|
6 => { name: 'Italian', iso_639_3_code: 'ita', iso_639_1_code: 'it' },
|
||||||
7 => { name: 'Japanese', iso_639_3_code: 'jpn', iso_639_1_code: 'ja' },
|
7 => { name: 'Japanese', iso_639_3_code: 'jpn', iso_639_1_code: 'ja' },
|
||||||
8 => { name: 'Korean', iso_639_3_code: 'kor', iso_639_1_code: 'ko' },
|
8 => { name: 'Korean', iso_639_3_code: 'kor', iso_639_1_code: 'ko' },
|
||||||
9 => { name: 'Portugues', iso_639_3_code: 'por', iso_639_1_code: 'pt' },
|
9 => { name: 'Portuguese', iso_639_3_code: 'por', iso_639_1_code: 'pt' },
|
||||||
10 => { name: 'Russian', iso_639_3_code: 'rus', iso_639_1_code: 'ru' },
|
10 => { name: 'Russian', iso_639_3_code: 'rus', iso_639_1_code: 'ru' },
|
||||||
11 => { name: 'Chinese', iso_639_3_code: 'zho', iso_639_1_code: 'zh' },
|
11 => { name: 'Chinese', iso_639_3_code: 'zho', iso_639_1_code: 'zh' },
|
||||||
12 => { name: 'Spanish', iso_639_3_code: 'spa', iso_639_1_code: 'es' },
|
12 => { name: 'Spanish', iso_639_3_code: 'spa', iso_639_1_code: 'es' },
|
||||||
|
|
|
@ -13,6 +13,9 @@ module Events::Types
|
||||||
CONVERSATION_LOCK_TOGGLE = 'conversation.lock_toggle'
|
CONVERSATION_LOCK_TOGGLE = 'conversation.lock_toggle'
|
||||||
ASSIGNEE_CHANGED = 'assignee.changed'
|
ASSIGNEE_CHANGED = 'assignee.changed'
|
||||||
|
|
||||||
|
CONTACT_CREATED = 'contact.created'
|
||||||
|
CONTACT_UPDATED = 'contact.updated'
|
||||||
|
|
||||||
ACCOUNT_CREATED = 'account.created'
|
ACCOUNT_CREATED = 'account.created'
|
||||||
ACCOUNT_DESTROYED = 'account.destroyed'
|
ACCOUNT_DESTROYED = 'account.destroyed'
|
||||||
|
|
||||||
|
|
29
spec/listeners/action_cable_listener_spec.rb
Normal file
29
spec/listeners/action_cable_listener_spec.rb
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
require 'rails_helper'
|
||||||
|
describe ActionCableListener do
|
||||||
|
let(:listener) { described_class.instance }
|
||||||
|
let!(:account) { create(:account) }
|
||||||
|
let!(:admin) { create(:user, account: account, role: :administrator) }
|
||||||
|
let!(:inbox) { create(:inbox, account: account) }
|
||||||
|
let!(:agent) { create(:user, account: account, role: :agent) }
|
||||||
|
let!(:conversation) { create(:conversation, account: account, inbox: inbox, assignee: agent) }
|
||||||
|
let!(:message) do
|
||||||
|
create(:message, message_type: 'outgoing',
|
||||||
|
account: account, inbox: inbox, conversation: conversation)
|
||||||
|
end
|
||||||
|
let!(:event) { Events::Base.new(event_name, Time.zone.now, message: message) }
|
||||||
|
|
||||||
|
before do
|
||||||
|
create(:inbox_member, inbox: inbox, user: agent)
|
||||||
|
end
|
||||||
|
|
||||||
|
describe '#message_created' do
|
||||||
|
let(:event_name) { :'message.created' }
|
||||||
|
|
||||||
|
it 'sends message to account admins, inbox agents and the contact' do
|
||||||
|
expect(ActionCableBroadcastJob).to receive(:perform_later).with([admin.pubsub_token], 'message.created', message.push_event_data)
|
||||||
|
expect(ActionCableBroadcastJob).to receive(:perform_later).with([agent.pubsub_token], 'message.created', message.push_event_data)
|
||||||
|
expect(ActionCableBroadcastJob).to receive(:perform_later).with([conversation.contact.pubsub_token], 'message.created', message.push_event_data)
|
||||||
|
listener.message_created(event)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
Loading…
Reference in a new issue