123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- require 'rails_helper'
- RSpec.describe Channel::Driver::Smtp, integration: true, required_envs: %w[MAIL_SERVER MAIL_ADDRESS MAIL_PASS] do
- let(:server_host) { ENV['MAIL_SERVER'] }
- let(:server_login) { ENV['MAIL_ADDRESS'] }
- let(:server_password) { ENV['MAIL_PASS'] }
- let(:email_address) { create(:email_address, name: 'me Helpdesk', email: "some-zammad-#{server_login}") }
- let(:group) { create(:group, name: 'DeliverTest', email_address: email_address) }
- let(:channel) do
- create(:email_channel,
- group: group,
- outbound: outbound,
- inbound: {
- adapter: 'imap',
- options: {
- host: 'mx1.example.com',
- user: 'example',
- password: 'some_pw',
- ssl: true,
- }
- })
- end
- let(:state_name) { 'new' }
- let(:ticket) { create(:ticket, title: 'some delivery test', group: group, state_name: state_name) }
- let(:article) { create(:ticket_article, :outbound_email, ticket: ticket, to: Faker::Internet.unique.email, subject: 'some subject', message_id: 'some@id', body: 'some message delivery test') }
- before do
- ENV['ZAMMAD_MAIL_TO_FILE'] = '1'
- freeze_time
- email_address.update!(channel_id: channel.id)
- ticket && article
- end
- context 'when modifying channel options', :aggregate_failures do
- let(:outbound) { { adapter: 'sendmail' } }
- it 'updates article delivery preferences' do
- expect(article.preferences).not_to include(:delivery_retry,
- :delivery_status,
- :delivery_status_date,
- :delivery_status_message)
- TicketArticleCommunicateEmailJob.new.perform(article.id)
- expect(article.reload.preferences).to include(delivery_retry: 1,
- delivery_status: 'success',
- delivery_status_date: be_present,
- delivery_status_message: be_nil)
- # Send with invalid smtp settings.
- channel.options.tap do |options|
- options['outbound'] = {
- adapter: 'smtp',
- options: {
- host: 'mx1.example.com',
- port: 25,
- start_tls: true,
- user: 'not_existing',
- password: 'not_existing',
- },
- }
- end
- channel.save!
- expect { TicketArticleCommunicateEmailJob.new.perform(article.id) }.to raise_error(RuntimeError)
- expect(article.reload.preferences).to include(delivery_retry: 2,
- delivery_status: 'fail',
- delivery_status_date: be_present,
- delivery_status_message: be_present)
- # Send with valid smtp settings.
- channel.options.tap do |options|
- options['outbound'] = {
- adapter: 'smtp',
- options: {
- host: server_host,
- port: 25,
- start_tls: true,
- user: server_login,
- password: server_password,
- ssl_verify: false,
- },
- }
- end
- channel.save!
- TicketArticleCommunicateEmailJob.new.perform(article.id)
- expect(article.reload.preferences).to include(delivery_retry: 3,
- delivery_status: 'success',
- delivery_status_date: be_present,
- delivery_status_message: be_nil)
- end
- end
- context 'when encounters sending errors', :aggregate_failures, performs_jobs: true do
- let(:state_name) { 'closed' }
- let(:outbound) do
- {
- adapter: 'smtp',
- options: {
- host: 'mx1.example.com',
- port: 25,
- start_tls: true,
- user: 'not_existing',
- password: 'not_existing',
- },
- }
- end
- it 'retries delivery in expected intervals' do
- expect do
- perform_enqueued_jobs
- end.to have_performed_job(TicketArticleCommunicateEmailJob)
- expect(ticket.reload.articles.count).to eq(1)
- expect(ticket.state.name).to eq('closed')
- expect(article.reload.preferences).to include(delivery_retry: 1,
- delivery_status: 'fail',
- delivery_status_date: be_present,
- delivery_status_message: be_present)
- expect do
- perform_enqueued_jobs
- end.to have_performed_job(TicketArticleCommunicateEmailJob).at(25.seconds.from_now)
- expect(article.reload.preferences).to include(delivery_retry: 2,
- delivery_status: 'fail',
- delivery_status_date: be_present,
- delivery_status_message: be_present)
- expect(ticket.reload.articles.count).to eq(1)
- expect(ticket.state.name).to eq('closed')
- expect do
- perform_enqueued_jobs
- end.to have_performed_job(TicketArticleCommunicateEmailJob).at(50.seconds.from_now)
- expect(article.reload.preferences).to include(delivery_retry: 3,
- delivery_status: 'fail',
- delivery_status_date: be_present,
- delivery_status_message: be_present)
- expect(ticket.reload.articles.count).to eq(1)
- expect(ticket.state.name).to eq('closed')
- expect do
- perform_enqueued_jobs
- end.to raise_error(RuntimeError).and have_performed_job(TicketArticleCommunicateEmailJob).at(75.seconds.from_now)
- expect(article.reload.preferences).to include(delivery_retry: 4,
- delivery_status: 'fail',
- delivery_status_date: be_present,
- delivery_status_message: be_present)
- expect(ticket.reload.articles.count).to eq(2)
- expect(ticket.state).to eq(Ticket::State.find_by(default_follow_up: true))
- expect(ticket.articles.last).to have_attributes(sender: Ticket::Article::Sender.lookup(name: 'System'),
- preferences: include(delivery_message: true,
- delivery_article_id_related: article.id,
- notification: true))
- end
- end
- end
|