123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- require 'rails_helper'
- RSpec.describe 'Whatsapp::Webhook::Payload::Error', :aggregate_failures, current_user_id: 1 do
- let(:channel) { create(:whatsapp_channel) }
- let(:wa_id) { Faker::PhoneNumber.unique.cell_phone_in_e164.delete('+') }
- let(:errors) { {} }
- let(:json) do
- {
- object: 'whatsapp_business_account',
- entry: [
- {
- id: '222259550976437',
- changes: [
- {
- value: {
- messaging_product: 'whatsapp',
- metadata: {
- display_phone_number: channel.options[:phone_number],
- phone_number_id: channel.options[:phone_number_id]
- },
- contacts: [
- {
- profile: {
- name: Faker::Name.unique.name
- },
- wa_id: wa_id
- }
- ],
- messages: [
- {
- from: wa_id,
- id: 'wamid.NDkxNTE1NTU1NTU5ODNBNTU3NkYyQTJCM0FGMUE1RjZECg==',
- timestamp: '1733484661',
- errors: [
- error
- ],
- type: 'any'
- }
- ]
- },
- field: 'messages'
- }
- ]
- }
- ]
- }.to_json
- end
- let(:uuid) { channel.options[:callback_url_uuid] }
- let(:signature) do
- OpenSSL::HMAC.hexdigest(OpenSSL::Digest.new('sha256'), channel.options[:app_secret], json)
- end
- context 'when error is initialized by sender' do
- let(:error) do
- {
- code: 131_051,
- title: 'Message type unknown',
- message: 'Message type unknown',
- error_data: {
- details: 'Message type is currently not supported.'
- }
- }
- end
- let(:message) { instance_double(Whatsapp::Outgoing::Message::Text) }
- before do
- allow(Whatsapp::Outgoing::Message::Text)
- .to receive(:new)
- .and_return(message)
- allow(message)
- .to receive(:deliver)
- .with(body: "Apologies, we're unable to process this kind of message due to restrictions within WhatsApp Business.")
- .and_return(true)
- allow(Rails.logger).to receive(:error)
- begin
- Whatsapp::Webhook::Payload.new(json: json, signature: signature, uuid: uuid).process
- rescue Whatsapp::Webhook::Payload::ProcessableError
- # noop
- end
- end
- it 'does log error' do
- expect(Rails.logger).to have_received(:error).once
- end
- it 'does inform sender' do
- expect(message).to have_received(:deliver).once
- end
- it 'does not update channel status' do
- expect(channel.reload.status_out).to be_nil
- expect(channel.reload.last_log_out).to be_nil
- end
- end
- context 'when error is recoverable' do
- let(:error) do
- {
- code: 131_026,
- title: 'Message undeliverable',
- message: 'Message undeliverable',
- error_data: {
- details: 'Message could not be delivered.'
- }
- }
- end
- before do
- allow(Rails.logger).to receive(:error)
- begin
- Whatsapp::Webhook::Payload.new(json: json, signature: signature, uuid: uuid).process
- rescue Whatsapp::Webhook::Payload::ProcessableError
- # noop
- end
- end
- it 'does log error' do
- expect(Rails.logger).to have_received(:error).once
- end
- it 'does not update channel status' do
- expect(channel.reload.status_out).to be_nil
- expect(channel.reload.last_log_out).to be_nil
- end
- end
- context 'when error is not recoverable' do
- let(:error) do
- {
- code: 131_056,
- title: 'Pair rate limit hit',
- message: 'Pair rate limit hit',
- error_data: {
- details: 'Too many messages sent to this user.'
- }
- }
- end
- before do
- allow(Rails.logger).to receive(:error)
- begin
- Whatsapp::Webhook::Payload.new(json: json, signature: signature, uuid: uuid).process
- rescue Whatsapp::Webhook::Payload::ProcessableError
- # noop
- end
- end
- it 'does log error' do
- expect(Rails.logger).to have_received(:error).once
- end
- it 'does update channel status' do
- expect(channel.reload.status_out).to eq('error')
- expect(channel.reload.last_log_out).to eq('Pair rate limit hit (131056)')
- end
- end
- end
|