b05d06a28a
Adds the ability to lock conversation to a single thread for Whatsapp and Sms Inboxes when using outbound messages. demo: https://www.loom.com/share/c9e1e563c8914837a4139dfdd2503fef fixes: #4975 Co-authored-by: Nithin David <1277421+nithindavid@users.noreply.github.com>
164 lines
5 KiB
Ruby
164 lines
5 KiB
Ruby
class Api::V1::Accounts::ConversationsController < Api::V1::Accounts::BaseController
|
|
include Events::Types
|
|
include DateRangeHelper
|
|
|
|
before_action :conversation, except: [:index, :meta, :search, :create, :filter]
|
|
before_action :inbox, :contact, :contact_inbox, only: [:create]
|
|
|
|
def index
|
|
result = conversation_finder.perform
|
|
@conversations = result[:conversations]
|
|
@conversations_count = result[:count]
|
|
end
|
|
|
|
def meta
|
|
result = conversation_finder.perform
|
|
@conversations_count = result[:count]
|
|
end
|
|
|
|
def search
|
|
result = conversation_finder.perform
|
|
@conversations = result[:conversations]
|
|
@conversations_count = result[:count]
|
|
end
|
|
|
|
def create
|
|
ActiveRecord::Base.transaction do
|
|
@conversation = ConversationBuilder.new(params: params, contact_inbox: @contact_inbox).perform
|
|
Messages::MessageBuilder.new(Current.user, @conversation, params[:message]).perform if params[:message].present?
|
|
end
|
|
end
|
|
|
|
def show; end
|
|
|
|
def filter
|
|
result = ::Conversations::FilterService.new(params.permit!, current_user).perform
|
|
@conversations = result[:conversations]
|
|
@conversations_count = result[:count]
|
|
end
|
|
|
|
def mute
|
|
@conversation.mute!
|
|
head :ok
|
|
end
|
|
|
|
def unmute
|
|
@conversation.unmute!
|
|
head :ok
|
|
end
|
|
|
|
def transcript
|
|
render json: { error: 'email param missing' }, status: :unprocessable_entity and return if params[:email].blank?
|
|
|
|
ConversationReplyMailer.with(account: @conversation.account).conversation_transcript(@conversation, params[:email])&.deliver_later
|
|
head :ok
|
|
end
|
|
|
|
def toggle_status
|
|
if params[:status]
|
|
set_conversation_status
|
|
@status = @conversation.save!
|
|
else
|
|
@status = @conversation.toggle_status
|
|
end
|
|
assign_conversation if @conversation.status == 'open' && Current.user.is_a?(User) && Current.user&.agent?
|
|
end
|
|
|
|
def toggle_typing_status
|
|
case params[:typing_status]
|
|
when 'on'
|
|
trigger_typing_event(CONVERSATION_TYPING_ON, params[:is_private])
|
|
when 'off'
|
|
trigger_typing_event(CONVERSATION_TYPING_OFF, params[:is_private])
|
|
end
|
|
head :ok
|
|
end
|
|
|
|
def update_last_seen
|
|
update_last_seen_on_conversation(DateTime.now.utc, assignee?)
|
|
end
|
|
|
|
def unread
|
|
last_incoming_message = @conversation.messages.incoming.last
|
|
last_seen_at = last_incoming_message.created_at - 1.second if last_incoming_message.present?
|
|
update_last_seen_on_conversation(last_seen_at, true)
|
|
end
|
|
|
|
def custom_attributes
|
|
@conversation.custom_attributes = params.permit(custom_attributes: {})[:custom_attributes]
|
|
@conversation.save!
|
|
end
|
|
|
|
private
|
|
|
|
def update_last_seen_on_conversation(last_seen_at, update_assignee)
|
|
# rubocop:disable Rails/SkipsModelValidations
|
|
@conversation.update_column(:agent_last_seen_at, last_seen_at)
|
|
@conversation.update_column(:assignee_last_seen_at, last_seen_at) if update_assignee.present?
|
|
# rubocop:enable Rails/SkipsModelValidations
|
|
end
|
|
|
|
def set_conversation_status
|
|
# TODO: temporary fallback for the old bot status in conversation, we will remove after couple of releases
|
|
# commenting this out to see if there are any errors, if not we can remove this in subsequent releases
|
|
# status = params[:status] == 'bot' ? 'pending' : params[:status]
|
|
@conversation.status = params[:status]
|
|
@conversation.snoozed_until = parse_date_time(params[:snoozed_until].to_s) if params[:snoozed_until]
|
|
end
|
|
|
|
def assign_conversation
|
|
@agent = Current.account.users.find(current_user.id)
|
|
@conversation.update_assignee(@agent)
|
|
end
|
|
|
|
def trigger_typing_event(event, is_private)
|
|
user = current_user.presence || @resource
|
|
Rails.configuration.dispatcher.dispatch(event, Time.zone.now, conversation: @conversation, user: user, is_private: is_private)
|
|
end
|
|
|
|
def conversation
|
|
@conversation ||= Current.account.conversations.find_by!(display_id: params[:id])
|
|
authorize @conversation.inbox, :show?
|
|
end
|
|
|
|
def inbox
|
|
return if params[:inbox_id].blank?
|
|
|
|
@inbox = Current.account.inboxes.find(params[:inbox_id])
|
|
authorize @inbox, :show?
|
|
end
|
|
|
|
def contact
|
|
return if params[:contact_id].blank?
|
|
|
|
@contact = Current.account.contacts.find(params[:contact_id])
|
|
end
|
|
|
|
def contact_inbox
|
|
@contact_inbox = build_contact_inbox
|
|
|
|
# fallback for the old case where we do look up only using source id
|
|
# In future we need to change this and make sure we do look up on combination of inbox_id and source_id
|
|
# and deprecate the support of passing only source_id as the param
|
|
@contact_inbox ||= ::ContactInbox.find_by!(source_id: params[:source_id])
|
|
authorize @contact_inbox.inbox, :show?
|
|
end
|
|
|
|
def build_contact_inbox
|
|
return if @inbox.blank? || @contact.blank?
|
|
|
|
ContactInboxBuilder.new(
|
|
contact: @contact,
|
|
inbox: @inbox,
|
|
source_id: params[:source_id]
|
|
).perform
|
|
end
|
|
|
|
def conversation_finder
|
|
@conversation_finder ||= ConversationFinder.new(Current.user, params)
|
|
end
|
|
|
|
def assignee?
|
|
@conversation.assignee_id? && Current.user == @conversation.assignee
|
|
end
|
|
end
|