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:
Sojan Jose 2020-04-18 13:47:51 +05:30 committed by GitHub
parent 818c769bb7
commit ecccb103a0
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
7 changed files with 96 additions and 26 deletions

View file

@ -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

View file

@ -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

View file

@ -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

View file

@ -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

View file

@ -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' },

View file

@ -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'

View 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