# Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
require 'rails_helper'
RSpec.describe Channel::EmailBuild, type: :model do
describe '#build' do
let(:html_body) do
<<~MSG_HTML.chomp
> Welcome!
>
> Thank you for installing Zammad. äöüß
>
MSG_HTML
end
let(:plain_text_body) do
<<~MSG_TEXT.chomp
> Welcome!
>
> Thank you for installing Zammad. äöüß
>
MSG_TEXT
end
let(:parser) { Channel::EmailParser.new }
let(:parsed_data) { parser.parse(mail.to_s) }
let(:html_part_attachment) do
parsed_data[:attachments]
.find { |attachment| attachment[:filename] == 'message.html' }
end
let(:file_attachment) do
parsed_data[:attachments]
.find { |attachment| attachment[:filename] == filename }
end
shared_examples 'adding the email html part as an attachment' do
it 'adds the html part as an attachment' do
expect(html_part_attachment).to be_a Hash
end
it 'adds the html part as an attachment' do
expect(html_part_attachment).to include(
'filename' => 'message.html',
'preferences' => include('content-alternative' => true, 'Charset' => 'UTF-8',
'Mime-Type' => 'text/html', 'original-format' => true)
)
end
it 'does not include content-id property in attachment preferences' do
expect(html_part_attachment).not_to include(
'preferences' => include('Content-ID')
)
end
end
shared_examples 'adding a text file as an attachment' do
it 'adds the text file as an attachment' do
expect(file_attachment).to include(
'filename' => filename,
'preferences' => include('Charset' => 'UTF-8', 'Mime-Type' => mime_type,
'Content-Type' => "text/plain; charset=UTF-8; filename=#{filename}")
)
end
it 'does not include content* properties in attachment preferences' do
expect(file_attachment).not_to include(
'preferences' => include('Content-ID', 'content-alternative')
)
end
end
shared_examples 'adding a file as an attachment' do |file_type|
it "adds a #{file_type} as an attachment'" do
expect(file_attachment).to include(
'data' => content, 'filename' => filename,
'preferences' => include('Charset' => 'UTF-8', 'Mime-Type' => mime_type,
'Content-Type' => preferences_content_type)
)
end
it 'does not include content* properties in attachment preferences' do
expect(file_attachment).not_to include(
'preferences' => include('content-alternative', 'Content-ID')
)
end
end
shared_examples 'not adding email content as attachment' do
it 'does not add email content as an attachment' do
expect(html_part_attachment).to be_nil
end
end
context 'with email only' do
let(:mail) do
described_class.build(
from: 'sender@example.com',
to: 'recipient@example.com',
body: mail_body,
content_type: content_type
)
end
let(:expected_text) do
<<~MSG_TEXT.chomp
> Welcome!\r
>\r
> Thank you for installing Zammad. äöüß\r
>\r
MSG_TEXT
end
context 'when email contains only html' do
let(:mail_body) { html_body }
let(:content_type) { 'text/html' }
let(:expected_html) do
<<~MSG_HTML.chomp
\r
\r
\r
\r
\r
\r
> Welcome!
>
> Thank you for installing Zammad. äöüß
>
\r
\r
MSG_HTML
end
let(:expected_body) { '> Welcome!
>
> Thank you for installing Zammad. äöüß
>
' }
it 'builds a mail with a text part' do
expect(mail.text_part.body.to_s).to eq expected_text
end
it 'builds a mail with a html part' do
expect(mail.html_part.body.to_s).to eq expected_html
end
it 'builds a mail that is parsed correctly' do
expect(parsed_data).to include(body: expected_body, content_type: 'text/html')
end
it_behaves_like 'adding the email html part as an attachment'
end
context 'when email contains only plain text' do
let(:mail_body) { plain_text_body }
let(:content_type) { 'text/plain' }
it 'builds a mail with a text part' do
expect(mail.body.to_s).to eq expected_text
end
it 'does not build a html part' do
expect(mail.html_part).to be_nil
end
it 'builds a mail that is parsed correctly' do
expect(parsed_data).to include(body: plain_text_body, content_type: 'text/plain')
end
it 'does not have an attachment' do
expect(parsed_data[:attachments].first).to be_nil
end
it_behaves_like 'not adding email content as attachment'
end
end
context 'with email and attachment' do
let(:mail) do
described_class.build(
from: 'sender@example.com',
to: 'recipient@example.com',
body: mail_body,
content_type: content_type,
attachments: attachments
)
end
let(:filename) { 'somename.txt' }
let(:mime_type) { 'text/plain' }
let(:content) { 'Some text' }
let(:direct_attachment) do
[{
'Mime-Type' => mime_type,
:content => content,
:filename => filename
}]
end
let(:ticket) { create(:ticket, title: 'some article text attachment test', group: group) }
let(:group) { Group.lookup(name: 'Users') }
let(:article) do
create(:ticket_article,
ticket: ticket,
body: 'some message article helper test1 asdasd
')
end
let(:store_attributes) do
{
object: 'Ticket::Article',
o_id: article.id,
data: content,
filename: filename,
preferences: {
'Mime-Type' => mime_type
}
}
end
let(:store) { create(:store, **store_attributes) }
shared_context 'with attachment checks' do
context 'when attachment is a text file' do
it_behaves_like 'adding a text file as an attachment'
end
context 'when attachment is a image file' do
let(:filename) { 'somename.png' }
let(:mime_type) { 'image/png' }
let(:preferences_content_type) { "#{mime_type}; filename=#{filename}" }
let(:content) { 'xxxxxxx' }
it_behaves_like 'adding a file as an attachment', 'image'
end
context 'when attachment is a calendar file' do
let(:filename) { 'schedule.ics' }
let(:mime_type) { 'text/calendar' }
let(:preferences_content_type) { "#{mime_type}; charset=UTF-8; filename=#{filename}" }
let(:content) { 'xxxxxxx' }
it_behaves_like 'adding a file as an attachment', 'calendar'
end
end
context 'with html email' do
let(:mail_body) { html_body }
let(:content_type) { 'text/html' }
context 'with direct attachment' do
let(:attachments) { direct_attachment }
it 'has two attachments' do
expect(parsed_data[:attachments].size).to eq 2
end
it_behaves_like 'adding the email html part as an attachment'
include_context 'with attachment checks'
end
context 'with attachement from store' do
let(:attachments) { [ store ] }
let(:filename) { 'text_file.txt' }
let(:mime_type) { 'text/plain' }
it 'has two attachments' do
expect(parsed_data[:attachments].size).to eq 2
end
it_behaves_like 'adding the email html part as an attachment'
include_context 'with attachment checks'
end
end
context 'with plain text email' do
let(:mail_body) { plain_text_body }
let(:content_type) { 'text/plain' }
context 'with direct attachment' do
let(:attachments) { direct_attachment }
let(:filename) { 'somename.txt' }
let(:mime_type) { 'text/plain' }
it 'has only one attachment' do
expect(parsed_data[:attachments].size).to eq 1
end
it_behaves_like 'not adding email content as attachment'
include_context 'with attachment checks'
end
context 'with attachement from store' do
let(:attachments) { [ store ] }
let(:filename) { 'text_file.txt' }
let(:mime_type) { 'text/plain' }
let(:mail_body) do
<<~MSG_TEXT.chomp
> Welcome!
>
> Email Content
MSG_TEXT
end
let(:content) { 'Text Content' }
it 'has only one attachment' do
expect(parsed_data[:attachments].size).to eq 1
end
# #2362 - Attached text files get prepended on e-mail reply instead of appended
it 'Email Content should appear before the Text Content within the raw email' do
expect(mail.to_s).to match(%r{Email Content[\s\S]*Text Content})
end
it_behaves_like 'not adding email content as attachment'
include_context 'with attachment checks'
end
end
end
end
describe '#recipient_line' do
let(:email) { 'some.body@example.com' }
let(:generated_recipient_line) { described_class.recipient_line(realname, email) }
context 'with quote in the realname' do
let(:realname) { 'Somebody @ "Company"' }
it 'escapes the quotes' do
expected_recipient_line = '"Somebody @ \"Company\""
'
expect(generated_recipient_line).to eq expected_recipient_line
end
end
context 'with a simple realname with no special characters' do
let(:realname) { 'Somebody' }
it 'wraps the realname with quotes and wraps the email with <>' do
expected_recipient_line = 'Somebody '
expect(generated_recipient_line).to eq expected_recipient_line
end
end
context 'with special characters (|) in the realname' do
let(:realname) { 'Somebody | Some Org' }
it 'wraps the email with <>' do
expected_recipient_line = 'Somebody | Some Org '
expect(generated_recipient_line).to eq expected_recipient_line
end
end
context 'with special characters (spaces) in the realname' do
let(:realname) { 'Test Admin Agent via Support' }
it 'wraps the email with <>' do
expected_recipient_line = 'Test Admin Agent via Support '
expect(generated_recipient_line).to eq expected_recipient_line
end
end
end
# https://github.com/zammad/zammad/issues/165
describe '#html_mail_client_fixes' do
let(:generated_html) { described_class.html_mail_client_fixes(html) }
shared_examples 'adding styles to the element' do
it 'adds style to the element' do
expect(generated_html).to eq expected_html
end
it { expect(generated_html).not_to eq html }
end
context 'when html element is a blockquote' do
let(:html) do
<<~HTML.chomp
some
text
123
some
text
HTML
end
let(:expected_html) do
<<~HTML.chomp
some
text
123
some
text
HTML
end
it_behaves_like 'adding styles to the element'
end
context 'when html element is a p' do
let(:html) do
<<~HTML.chomp
some
text
123
HTML
end
let(:expected_html) do
<<~HTML.chomp
some
text
123
HTML
end
it_behaves_like 'adding styles to the element'
end
context 'when html element is a hr' do
let(:html) do
<<~HTML.chomp
sometext
123
HTML
end
let(:expected_html) do
<<~HTML.chomp
sometext
123
HTML
end
it_behaves_like 'adding styles to the element'
context 'when hr is a closing tag' do
let(:html) do
<<~HTML.chomp
sometext
HTML
end
let(:expected_html) do
<<~HTML.chomp
sometext
HTML
end
it_behaves_like 'adding styles to the element'
end
end
context 'when html element does not contian p, hr or blockquote' do
let(:html) do
<<~HTML.chomp
HTML
end
it 'does not add style to the element' do
expect(generated_html).to eq html
end
end
end
describe '#html_complete_check' do
let(:generated_html) { described_class.html_complete_check(html) }
context 'when html element includes an html tag' do
let(:html) { 'test' }
it 'returns the html as it is' do
expect(generated_html).to eq html
end
end
context 'when html does not include an html tag' do
let(:html) { 'test' }
it 'adds DOCTYPE tag to the element' do
expect(generated_html).to start_with ''
end
it 'adds the original element' do
expect(generated_html).to match html
end
end
# Issue #1230, missing backslashes
# 'Test URL: \\storage\project\100242-Inc'
context 'when html includes a backslash' do
let(:html) { 'Test URL: \\\\storage\\project\\100242-Inc' }
it 'keeps the backslashes' do
expect(generated_html).to include html
end
end
context 'with a configured html_email_css_font setting' do
let(:html) { 'test' }
let(:css_font) { "font-family:'Helvetica Neue', sans-serif; font-size: 12px;" }
before { Setting.set('html_email_css_font', css_font) }
it 'includes the configured css font' do
expect(generated_html).to include("#{css_font}\n")
end
end
end
end