# frozen_string_literal: true require 'rails_helper' RSpec.describe Message, type: :model do context 'with validations' do it { is_expected.to validate_presence_of(:inbox_id) } it { is_expected.to validate_presence_of(:conversation_id) } it { is_expected.to validate_presence_of(:account_id) } end describe '#reopen_conversation' do let(:conversation) { create(:conversation) } let(:message) { build(:message, message_type: :incoming, conversation: conversation) } it 'reopens resolved conversation when the message is from a contact' do conversation.resolved! message.save! expect(message.conversation.open?).to be true end it 'reopens snoozed conversation when the message is from a contact' do conversation.snoozed! message.save! expect(message.conversation.open?).to be true end it 'will not reopen if the conversation is muted' do conversation.resolved! conversation.mute! message.save! expect(message.conversation.open?).to be false end it 'will mark the conversation as pending if the agent bot is active' do agent_bot = create(:agent_bot) inbox = conversation.inbox inbox.agent_bot = agent_bot inbox.save! conversation.resolved! message.save! expect(conversation.open?).to be false expect(conversation.pending?).to be true end end context 'with webhook_data' do it 'contains the message attachment when attachment is present' do message = create(:message) attachment = message.attachments.new(account_id: message.account_id, file_type: :image) attachment.file.attach(io: File.open(Rails.root.join('spec/assets/avatar.png')), filename: 'avatar.png', content_type: 'image/png') attachment.save! expect(message.webhook_data.key?(:attachments)).to be true end it 'does not contain the message attachment when attachment is not present' do message = create(:message) expect(message.webhook_data.key?(:attachments)).to be false end end context 'when message is created' do let(:message) { build(:message, account: create(:account)) } it 'updates conversation last_activity_at when created' do message.save! expect(message.created_at).to eq message.conversation.last_activity_at end it 'updates contact last_activity_at when created' do expect { message.save! }.to(change { message.sender.last_activity_at }) end it 'triggers ::MessageTemplates::HookExecutionService' do hook_execution_service = double allow(::MessageTemplates::HookExecutionService).to receive(:new).and_return(hook_execution_service) allow(hook_execution_service).to receive(:perform).and_return(true) message.save! expect(::MessageTemplates::HookExecutionService).to have_received(:new).with(message: message) expect(hook_execution_service).to have_received(:perform) end context 'with conversation continuity' do it 'calls notify email method on after save for outgoing messages in website channel' do allow(ConversationReplyEmailWorker).to receive(:perform_in).and_return(true) message.message_type = 'outgoing' message.save! expect(ConversationReplyEmailWorker).to have_received(:perform_in) end it 'does not call notify email for website channel if continuity is disabled' do message.inbox = create(:inbox, account: message.account, channel: build(:channel_widget, account: message.account, continuity_via_email: false)) allow(ConversationReplyEmailWorker).to receive(:perform_in).and_return(true) message.message_type = 'outgoing' message.save! expect(ConversationReplyEmailWorker).not_to have_received(:perform_in) end it 'wont call notify email method for private notes' do message.private = true allow(ConversationReplyEmailWorker).to receive(:perform_in).and_return(true) message.save! expect(ConversationReplyEmailWorker).not_to have_received(:perform_in) end it 'calls EmailReply worker if the channel is email' do message.inbox = create(:inbox, account: message.account, channel: build(:channel_email, account: message.account)) allow(EmailReplyWorker).to receive(:perform_in).and_return(true) message.message_type = 'outgoing' message.save! expect(EmailReplyWorker).to have_received(:perform_in).with(1.second, message.id) end it 'wont call notify email method unless its website or email channel' do message.inbox = create(:inbox, account: message.account, channel: build(:channel_api, account: message.account)) allow(ConversationReplyEmailWorker).to receive(:perform_in).and_return(true) message.save! expect(ConversationReplyEmailWorker).not_to have_received(:perform_in) end end end context 'when content_type is blank' do let(:message) { build(:message, content_type: nil, account: create(:account)) } it 'sets content_type as text' do message.save! expect(message.content_type).to eq 'text' end end context 'when attachments size maximum' do let(:message) { build(:message, content_type: nil, account: create(:account)) } it 'add errors to message for attachment size is more than allowed limit' do 16.times.each do attachment = message.attachments.new(account_id: message.account_id, file_type: :image) attachment.file.attach(io: File.open(Rails.root.join('spec/assets/avatar.png')), filename: 'avatar.png', content_type: 'image/png') end expect(message.errors.messages).to eq({ attachments: ['exceeded maximum allowed'] }) end end context 'when email notifiable message' do let(:message) { build(:message, content_type: nil, account: create(:account)) } it 'return false if private message' do message.private = true message.message_type = 'outgoing' expect(message.email_notifiable_message?).to be false end it 'return false if incoming message' do message.private = false message.message_type = 'incoming' expect(message.email_notifiable_message?).to be false end it 'return false if activity message' do message.private = false message.message_type = 'activity' expect(message.email_notifiable_message?).to be false end it 'return false if message type is template and content type is not input_csat or text' do message.private = false message.message_type = 'template' message.content_type = 'incoming_email' expect(message.email_notifiable_message?).to be false end it 'return true if not private and not incoming and message content type is input_csat or text' do message.private = false message.message_type = 'template' message.content_type = 'text' expect(message.email_notifiable_message?).to be true end end context 'when facebook channel with unavailable story link' do let(:instagram_message) { create(:message, :instagram_story_mention) } before do # stubbing the request to facebook api during the message creation stub_request(:get, %r{https://graph.facebook.com/.*}).to_return(status: 200, body: { story: { mention: { link: 'http://graph.facebook.com/test-story-mention', id: '17920786367196703' } }, from: { username: 'Sender-id-1', id: 'Sender-id-1' }, id: 'instagram-message-id-1234' }.to_json, headers: {}) end it 'deletes the attachment for deleted stories' do expect(instagram_message.attachments.count).to eq 1 stub_request(:get, %r{https://graph.facebook.com/.*}).to_return(status: 404) instagram_message.push_event_data expect(instagram_message.reload.attachments.count).to eq 0 end it 'deletes the attachment for expired stories' do expect(instagram_message.attachments.count).to eq 1 # for expired stories, the link will be empty stub_request(:get, %r{https://graph.facebook.com/.*}).to_return(status: 200, body: { story: { mention: { link: '', id: '17920786367196703' } } }.to_json, headers: {}) instagram_message.push_event_data expect(instagram_message.reload.attachments.count).to eq 0 end end end