* feat: ignore bots in avg_first_response_time * feat: ignore bots in avg_first_response count * feat: add bot handoff event * feat: add handoff event listener and reporting event * fix: ignore agent bot in first response * refactor: calculate first_response with last handoff * refactor: method defn order * test: new reporting events * feat: Revert "feat: ignore bots in avg_first_response count" This reverts commit de1977c219a2e7a9180dd02272244fe3b3f7ce89. * feat: Revert "feat: ignore bots in avg_first_response_time" This reverts commit bb9171945d5e3b2f6015f4f96dd1b76b3efb6987. * fix: business hour calculation for first_reply * fix: event_start_time for first_response * feat: add migration to recompute first_responses * refactor: separate mute helpers for conversation * refactor: rename migration * refactor: migration script * fix: migration typo * fix: typo in query * feat: update schema.rb * Revert "feat: update schema.rb" This reverts commit 353ef355f2d956dd219907bb66982dc90ca5d896. * feat: update schema * refactor: update events as a batch job * fix: ignore the event if value is negative * feat: don't create a new hand-off if it's already present * refactor: break the action into smaller chunks * refactor: update reporting listener spec Handle the case to ensure extra bot handoffs are not created for a give conversation * fix: import error --------- Co-authored-by: Vishnu Narayanan <vishnu@chatwoot.com>
64 lines
1.7 KiB
Ruby
64 lines
1.7 KiB
Ruby
class Integrations::BotProcessorService
|
|
pattr_initialize [:event_name!, :hook!, :event_data!]
|
|
|
|
def perform
|
|
message = event_data[:message]
|
|
return unless should_run_processor?(message)
|
|
|
|
process_content(message)
|
|
rescue StandardError => e
|
|
ChatwootExceptionTracker.new(e, account: agent_bot).capture_exception
|
|
end
|
|
|
|
private
|
|
|
|
def should_run_processor?(message)
|
|
return if message.private?
|
|
return unless processable_message?(message)
|
|
return unless conversation.pending?
|
|
|
|
true
|
|
end
|
|
|
|
def conversation
|
|
message = event_data[:message]
|
|
@conversation ||= message.conversation
|
|
end
|
|
|
|
def process_content(message)
|
|
content = message_content(message)
|
|
response = get_response(conversation.contact_inbox.source_id, content) if content.present?
|
|
process_response(message, response) if response.present?
|
|
end
|
|
|
|
def message_content(message)
|
|
# TODO: might needs to change this to a way that we fetch the updated value from event data instead
|
|
# cause the message.updated event could be that that the message was deleted
|
|
|
|
return message.content_attributes['submitted_values']&.first&.dig('value') if event_name == 'message.updated'
|
|
|
|
message.content
|
|
end
|
|
|
|
def processable_message?(message)
|
|
# TODO: change from reportable and create a dedicated method for this?
|
|
return unless message.reportable?
|
|
return if message.outgoing? && !processable_outgoing_message?(message)
|
|
|
|
true
|
|
end
|
|
|
|
def processable_outgoing_message?(message)
|
|
event_name == 'message.updated' && ['input_select'].include?(message.content_type)
|
|
end
|
|
|
|
def process_action(message, action)
|
|
case action
|
|
when 'handoff'
|
|
message.conversation.bot_handoff!
|
|
when 'resolve'
|
|
message.conversation.resolved!
|
|
end
|
|
end
|
|
end
|