# == Schema Information # # Table name: conversations # # id :integer not null, primary key # additional_attributes :jsonb # agent_last_seen_at :datetime # assignee_last_seen_at :datetime # contact_last_seen_at :datetime # custom_attributes :jsonb # first_reply_created_at :datetime # identifier :string # last_activity_at :datetime not null # snoozed_until :datetime # status :integer default("open"), not null # uuid :uuid not null # created_at :datetime not null # updated_at :datetime not null # account_id :integer not null # assignee_id :integer # campaign_id :bigint # contact_id :bigint # contact_inbox_id :bigint # display_id :integer not null # inbox_id :integer not null # team_id :bigint # # Indexes # # index_conversations_on_account_id (account_id) # index_conversations_on_account_id_and_display_id (account_id,display_id) UNIQUE # index_conversations_on_assignee_id_and_account_id (assignee_id,account_id) # index_conversations_on_campaign_id (campaign_id) # index_conversations_on_contact_id (contact_id) # index_conversations_on_contact_inbox_id (contact_inbox_id) # index_conversations_on_first_reply_created_at (first_reply_created_at) # index_conversations_on_inbox_id (inbox_id) # index_conversations_on_last_activity_at (last_activity_at) # index_conversations_on_status_and_account_id (status,account_id) # index_conversations_on_team_id (team_id) # index_conversations_on_uuid (uuid) UNIQUE # class Conversation < ApplicationRecord include Labelable include AssignmentHandler include AutoAssignmentHandler include ActivityMessageHandler include UrlHelper include SortHandler include PgSearch::Model include MultiSearchableHelpers multisearchable( against: [:display_id, :name, :email, :phone_number, :account_id], additional_attributes: lambda { |conversation| { conversation_id: conversation.id, account_id: conversation.account_id, inbox_id: conversation.inbox_id } } ) validates :account_id, presence: true validates :inbox_id, presence: true before_validation :validate_additional_attributes validates :additional_attributes, jsonb_attributes_length: true validates :custom_attributes, jsonb_attributes_length: true validates :uuid, uniqueness: true validate :validate_referer_url enum status: { open: 0, resolved: 1, pending: 2, snoozed: 3 } scope :unassigned, -> { where(assignee_id: nil) } scope :assigned, -> { where.not(assignee_id: nil) } scope :assigned_to, ->(agent) { where(assignee_id: agent.id) } scope :resolvable, lambda { |auto_resolve_duration| return [] if auto_resolve_duration.to_i.zero? open.where('last_activity_at < ? ', Time.now.utc - auto_resolve_duration.days) } scope :last_user_message_at, lambda { joins( "INNER JOIN (#{last_messaged_conversations.to_sql}) AS grouped_conversations ON grouped_conversations.conversation_id = conversations.id" ).sort_on_last_user_message_at } belongs_to :account belongs_to :inbox belongs_to :assignee, class_name: 'User', optional: true belongs_to :contact belongs_to :contact_inbox belongs_to :team, optional: true belongs_to :campaign, optional: true has_many :mentions, dependent: :destroy_async has_many :messages, dependent: :destroy_async, autosave: true has_one :csat_survey_response, dependent: :destroy_async has_many :notifications, as: :primary_actor, dependent: :destroy_async before_save :ensure_snooze_until_reset before_create :mark_conversation_pending_if_bot after_update_commit :execute_after_update_commit_callbacks after_create_commit :notify_conversation_creation after_create_commit :update_contact_search_document, if: :contact_id? after_commit :set_display_id, unless: :display_id? delegate :auto_resolve_duration, to: :account delegate :name, :email, :phone_number, to: :contact, allow_nil: true def can_reply? channel = inbox&.channel return can_reply_on_instagram? if additional_attributes['type'] == 'instagram_direct_message' return true unless channel&.messaging_window_enabled? messaging_window = inbox.api? ? channel.additional_attributes['agent_reply_time_window'].to_i : 24 last_message_in_messaging_window?(messaging_window) end def last_incoming_message messages&.incoming&.last end def last_message_in_messaging_window?(time) return false if last_incoming_message.nil? Time.current < last_incoming_message.created_at + time.hours end def can_reply_on_instagram? global_config = GlobalConfig.get('ENABLE_MESSENGER_CHANNEL_HUMAN_AGENT') return false if last_incoming_message.nil? if global_config['ENABLE_MESSENGER_CHANNEL_HUMAN_AGENT'] Time.current < last_incoming_message.created_at + 7.days else last_message_in_messaging_window?(24) end end def update_assignee(agent = nil) update!(assignee: agent) end def toggle_status # FIXME: implement state machine with aasm self.status = open? ? :resolved : :open self.status = :open if pending? || snoozed? save end def mute! resolved! Redis::Alfred.setex(mute_key, 1, mute_period) create_muted_message end def unmute! Redis::Alfred.delete(mute_key) create_unmuted_message end def muted? Redis::Alfred.get(mute_key).present? end def unread_messages messages.unread_since(agent_last_seen_at) end def unread_incoming_messages messages.incoming.unread_since(agent_last_seen_at) end def push_event_data Conversations::EventDataPresenter.new(self).push_data end def lock_event_data Conversations::EventDataPresenter.new(self).lock_data end def webhook_data Conversations::EventDataPresenter.new(self).push_data end def notifiable_assignee_change? return false unless saved_change_to_assignee_id? return false if assignee_id.blank? return false if self_assign?(assignee_id) true end def tweet? inbox.inbox_type == 'Twitter' && additional_attributes['type'] == 'tweet' end def recent_messages messages.chat.last(5) end # NOTE: To add multi search records with conversation_id associated to contacts for previously added records. # We can not find conversation_id from contacts directly so we added this joins here. def self.rebuild_pg_search_documents(account_id) return super unless name == 'Conversation' connection.execute <<~SQL.squish INSERT INTO pg_search_documents (searchable_type, searchable_id, content, account_id, conversation_id, inbox_id, created_at, updated_at) SELECT 'Conversation' AS searchable_type, conversations.id AS searchable_id, CONCAT_WS(' ', conversations.display_id, contacts.email, contacts.name, contacts.phone_number, conversations.account_id) AS content, conversations.account_id::int AS account_id, conversations.id::int AS conversation_id, conversations.inbox_id::int AS inbox_id, now() AS created_at, now() AS updated_at FROM conversations INNER JOIN contacts ON conversations.contact_id = contacts.id WHERE conversations.account_id = #{account_id} SQL end private def execute_after_update_commit_callbacks notify_status_change create_activity notify_conversation_updation end def ensure_snooze_until_reset self.snoozed_until = nil unless snoozed? end def validate_additional_attributes self.additional_attributes = {} unless additional_attributes.is_a?(Hash) end def mark_conversation_pending_if_bot # TODO: make this an inbox config instead of assuming bot conversations should start as pending self.status = :pending if inbox.active_bot? end def notify_conversation_creation dispatcher_dispatch(CONVERSATION_CREATED) end def notify_conversation_updation return unless previous_changes.keys.present? && (previous_changes.keys & %w[team_id assignee_id status snoozed_until custom_attributes label_list first_reply_created_at]).present? dispatcher_dispatch(CONVERSATION_UPDATED, previous_changes) end def self_assign?(assignee_id) assignee_id.present? && Current.user&.id == assignee_id end def set_display_id reload end def notify_status_change { CONVERSATION_OPENED => -> { saved_change_to_status? && open? }, CONVERSATION_RESOLVED => -> { saved_change_to_status? && resolved? }, CONVERSATION_STATUS_CHANGED => -> { saved_change_to_status? }, CONVERSATION_READ => -> { saved_change_to_contact_last_seen_at? }, CONVERSATION_CONTACT_CHANGED => -> { saved_change_to_contact_id? } }.each do |event, condition| condition.call && dispatcher_dispatch(event, status_change) end end def dispatcher_dispatch(event_name, changed_attributes = nil) Rails.configuration.dispatcher.dispatch(event_name, Time.zone.now, conversation: self, notifiable_assignee_change: notifiable_assignee_change?, changed_attributes: changed_attributes, performed_by: Current.executed_by) end def conversation_status_changed_to_open? return false unless open? # saved_change_to_status? method only works in case of update return true if previous_changes.key?(:id) || saved_change_to_status? end def create_label_change(user_name) return unless user_name previous_labels, current_labels = previous_changes[:label_list] return unless (previous_labels.is_a? Array) && (current_labels.is_a? Array) dispatcher_dispatch(CONVERSATION_UPDATED, previous_changes) create_label_added(user_name, current_labels - previous_labels) create_label_removed(user_name, previous_labels - current_labels) end def mute_key format(Redis::RedisKeys::CONVERSATION_MUTE_KEY, id: id) end def mute_period 6.hours end def validate_referer_url return unless additional_attributes['referer'] self['additional_attributes']['referer'] = nil unless url_valid?(additional_attributes['referer']) end # creating db triggers trigger.before(:insert).for_each(:row) do "NEW.display_id := nextval('conv_dpid_seq_' || NEW.account_id);" end end