mirror of
				https://github.com/lingble/chatwoot.git
				synced 2025-10-31 19:17:48 +00:00 
			
		
		
		
	 9527ff6269
			
		
	
	9527ff6269
	
	
	
		
			
			- Add support for using labels as an action event for automation - Fix duplicated conversation_updated event dispatch for labels Fixes https://github.com/chatwoot/chatwoot/issues/8539 and multiple issues around duplication related to label change events. --------- Co-authored-by: Muhsin Keloth <muhsinkeramam@gmail.com>
		
			
				
	
	
		
			319 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			319 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # == Schema Information
 | |
| #
 | |
| # Table name: conversations
 | |
| #
 | |
| #  id                     :integer          not null, primary key
 | |
| #  additional_attributes  :jsonb
 | |
| #  agent_last_seen_at     :datetime
 | |
| #  assignee_last_seen_at  :datetime
 | |
| #  cached_label_list      :text
 | |
| #  contact_last_seen_at   :datetime
 | |
| #  custom_attributes      :jsonb
 | |
| #  first_reply_created_at :datetime
 | |
| #  identifier             :string
 | |
| #  last_activity_at       :datetime         not null
 | |
| #  priority               :integer
 | |
| #  snoozed_until          :datetime
 | |
| #  status                 :integer          default("open"), not null
 | |
| #  uuid                   :uuid             not null
 | |
| #  waiting_since          :datetime
 | |
| #  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
 | |
| #  sla_policy_id          :bigint
 | |
| #  team_id                :bigint
 | |
| #
 | |
| # Indexes
 | |
| #
 | |
| #  conv_acid_inbid_stat_asgnid_idx                    (account_id,inbox_id,status,assignee_id)
 | |
| #  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_id_and_account_id           (account_id,id)
 | |
| #  index_conversations_on_inbox_id                    (inbox_id)
 | |
| #  index_conversations_on_priority                    (priority)
 | |
| #  index_conversations_on_status_and_account_id       (status,account_id)
 | |
| #  index_conversations_on_status_and_priority         (status,priority)
 | |
| #  index_conversations_on_team_id                     (team_id)
 | |
| #  index_conversations_on_uuid                        (uuid) UNIQUE
 | |
| #  index_conversations_on_waiting_since               (waiting_since)
 | |
| #
 | |
| 
 | |
| class Conversation < ApplicationRecord
 | |
|   include Labelable
 | |
|   include LlmFormattable
 | |
|   include AssignmentHandler
 | |
|   include AutoAssignmentHandler
 | |
|   include ActivityMessageHandler
 | |
|   include UrlHelper
 | |
|   include SortHandler
 | |
|   include PushDataHelper
 | |
|   include ConversationMuteHelpers
 | |
| 
 | |
|   validates :account_id, presence: true
 | |
|   validates :inbox_id, presence: true
 | |
|   validates :contact_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 }
 | |
|   enum priority: { low: 0, medium: 1, high: 2, urgent: 3 }
 | |
| 
 | |
|   scope :unassigned, -> { where(assignee_id: nil) }
 | |
|   scope :assigned, -> { where.not(assignee_id: nil) }
 | |
|   scope :assigned_to, ->(agent) { where(assignee_id: agent.id) }
 | |
|   scope :unattended, -> { where(first_reply_created_at: nil).or(where.not(waiting_since: nil)) }
 | |
|   scope :resolvable_not_waiting, lambda { |auto_resolve_after|
 | |
|     return none if auto_resolve_after.to_i.zero?
 | |
| 
 | |
|     open.where('last_activity_at < ? AND waiting_since IS NULL', Time.now.utc - auto_resolve_after.minutes)
 | |
|   }
 | |
|   scope :resolvable_all, lambda { |auto_resolve_after|
 | |
|     return none if auto_resolve_after.to_i.zero?
 | |
| 
 | |
|     open.where('last_activity_at < ?', Time.now.utc - auto_resolve_after.minutes)
 | |
|   }
 | |
| 
 | |
|   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, inverse_of: :assigned_conversations
 | |
|   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 :conversation_participants, dependent: :destroy_async
 | |
|   has_many :notifications, as: :primary_actor, dependent: :destroy_async
 | |
|   has_many :attachments, through: :messages
 | |
|   has_many :reporting_events, dependent: :destroy_async
 | |
| 
 | |
|   before_save :ensure_snooze_until_reset
 | |
|   before_create :determine_conversation_status
 | |
|   before_create :ensure_waiting_since
 | |
| 
 | |
|   after_update_commit :execute_after_update_commit_callbacks
 | |
|   after_create_commit :notify_conversation_creation
 | |
|   after_create_commit :load_attributes_created_by_db_triggers
 | |
| 
 | |
|   delegate :auto_resolve_after, to: :account
 | |
| 
 | |
|   def can_reply?
 | |
|     Conversations::MessageWindowService.new(self).can_reply?
 | |
|   end
 | |
| 
 | |
|   def language
 | |
|     additional_attributes&.dig('conversation_language')
 | |
|   end
 | |
| 
 | |
|   # Be aware: The precision of created_at and last_activity_at may differ from Ruby's Time precision.
 | |
|   # Our DB column (see schema) stores timestamps with second-level precision (no microseconds), so
 | |
|   # if you assign a Ruby Time with microseconds, the DB will truncate it. This may cause subtle differences
 | |
|   # if you compare or copy these values in Ruby, also in our specs
 | |
|   # So in specs rely on to be_with(1.second) instead of to eq()
 | |
|   # TODO: Migrate to use a timestamp with microsecond precision
 | |
|   def last_activity_at
 | |
|     self[:last_activity_at] || created_at
 | |
|   end
 | |
| 
 | |
|   def last_incoming_message
 | |
|     messages&.incoming&.last
 | |
|   end
 | |
| 
 | |
|   def toggle_status
 | |
|     # FIXME: implement state machine with aasm
 | |
|     self.status = open? ? :resolved : :open
 | |
|     self.status = :open if pending? || snoozed?
 | |
|     save
 | |
|   end
 | |
| 
 | |
|   def toggle_priority(priority = nil)
 | |
|     self.priority = priority.presence
 | |
|     save
 | |
|   end
 | |
| 
 | |
|   def bot_handoff!
 | |
|     open!
 | |
|     dispatcher_dispatch(CONVERSATION_BOT_HANDOFF)
 | |
|   end
 | |
| 
 | |
|   def unread_messages
 | |
|     agent_last_seen_at.present? ? messages.created_since(agent_last_seen_at) : messages
 | |
|   end
 | |
| 
 | |
|   def unread_incoming_messages
 | |
|     unread_messages.where(account_id: account_id).incoming.last(10)
 | |
|   end
 | |
| 
 | |
|   def cached_label_list_array
 | |
|     (cached_label_list || '').split(',').map(&:strip)
 | |
|   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
 | |
| 
 | |
|   def csat_survey_link
 | |
|     "#{ENV.fetch('FRONTEND_URL', nil)}/survey/responses/#{uuid}"
 | |
|   end
 | |
| 
 | |
|   def dispatch_conversation_updated_event(previous_changes = nil)
 | |
|     dispatcher_dispatch(CONVERSATION_UPDATED, previous_changes)
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def execute_after_update_commit_callbacks
 | |
|     handle_resolved_status_change
 | |
|     notify_status_change
 | |
|     create_activity
 | |
|     notify_conversation_updation
 | |
|   end
 | |
| 
 | |
|   def handle_resolved_status_change
 | |
|     # When conversation is resolved, clear waiting_since using update_column to avoid callbacks
 | |
|     return unless saved_change_to_status? && status == 'resolved'
 | |
| 
 | |
|     # rubocop:disable Rails/SkipsModelValidations
 | |
|     update_column(:waiting_since, nil)
 | |
|     # rubocop:enable Rails/SkipsModelValidations
 | |
|   end
 | |
| 
 | |
|   def ensure_snooze_until_reset
 | |
|     self.snoozed_until = nil unless snoozed?
 | |
|   end
 | |
| 
 | |
|   def ensure_waiting_since
 | |
|     self.waiting_since = created_at
 | |
|   end
 | |
| 
 | |
|   def validate_additional_attributes
 | |
|     self.additional_attributes = {} unless additional_attributes.is_a?(Hash)
 | |
|   end
 | |
| 
 | |
|   def determine_conversation_status
 | |
|     self.status = :resolved and return if contact.blocked?
 | |
| 
 | |
|     # Message template hooks aren't executed for conversations from campaigns
 | |
|     # So making these conversations open for agent visibility
 | |
|     return if campaign.present?
 | |
| 
 | |
|     # 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? && allowed_keys?
 | |
| 
 | |
|     dispatch_conversation_updated_event(previous_changes)
 | |
|   end
 | |
| 
 | |
|   def list_of_keys
 | |
|     %w[team_id assignee_id status snoozed_until custom_attributes label_list waiting_since first_reply_created_at
 | |
|        priority]
 | |
|   end
 | |
| 
 | |
|   def allowed_keys?
 | |
|     (
 | |
|       previous_changes.keys.intersect?(list_of_keys) ||
 | |
|       (previous_changes['additional_attributes'].present? && previous_changes['additional_attributes'][1].keys.intersect?(%w[conversation_language]))
 | |
|     )
 | |
|   end
 | |
| 
 | |
|   def load_attributes_created_by_db_triggers
 | |
|     # Display id is set via a trigger in the database
 | |
|     # So we need to specifically fetch it after the record is created
 | |
|     # We can't use reload because it will clear the previous changes, which we need for the dispatcher
 | |
|     obj_from_db = self.class.find(id)
 | |
|     self[:display_id] = obj_from_db[:display_id]
 | |
|     self[:uuid] = obj_from_db[:uuid]
 | |
|   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)
 | |
| 
 | |
|     create_label_added(user_name, current_labels - previous_labels)
 | |
|     create_label_removed(user_name, previous_labels - current_labels)
 | |
|   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
 | |
| 
 | |
| Conversation.include_mod_with('Audit::Conversation')
 | |
| Conversation.include_mod_with('Concerns::Conversation')
 | |
| Conversation.prepend_mod_with('Conversation')
 |