Files
redmine/plugins/redmine_contacts_helpdesk/app/models/helpdesk_mailer.rb
T
2026-04-24 22:01:18 +00:00

824 lines
38 KiB
Ruby

require "digest/md5"
class HelpdeskMailer < MailHandler
include HelpdeskMailerHelper
include AbstractController::Callbacks
after_filter :set_delivery_options
attr_reader :contact, :user, :email, :project
def self.default_url_options
{ :host => Setting.host_name, :protocol => Setting.protocol }
end
def self.with_activated_perform_deliveries
perform_delivery_state = ActionMailer::Base.perform_deliveries
ActionMailer::Base.perform_deliveries = true
yield
ensure
ActionMailer::Base.perform_deliveries = perform_delivery_state
end
def issue_response(contact, journal, options={})
@project = journal.issue.project
to_address = options[:to_address] || (journal.journal_message && journal.journal_message.to_address) || contact.primary_email
cc_address = options[:cc_address] || (journal.journal_message && journal.journal_message.cc_address)
bcc_address = options[:bcc_address] || (journal.journal_message && journal.journal_message.bcc_address)
from_address = options[:from_address] || (!HelpdeskSettings["helpdesk_answer_from", journal.issue.project].blank? && HelpdeskSettings["helpdesk_answer_from", journal.issue.project] )|| Setting.mail_from
in_reply_to = options[:in_reply_to] || ((journal.issue.helpdesk_ticket.blank? || journal.issue.helpdesk_ticket.message_id.blank?) ? '' : "<#{journal.issue.helpdesk_ticket.message_id}>")
headers['X-Redmine-Ticket-ID'] = journal.issue.id.to_s
@email_header = self.class.apply_macro(HelpdeskSettings["helpdesk_emails_header", journal.issue.project], contact, journal.issue, journal.user) unless HelpdeskSettings["helpdesk_emails_header", journal.issue.project].blank?
@email_footer = self.class.apply_macro(HelpdeskSettings["helpdesk_emails_footer", journal.issue.project], contact, journal.issue, journal.user) unless HelpdeskSettings["helpdesk_emails_footer", journal.issue.project].blank?
@email_body = self.class.apply_macro(journal.notes, contact, journal.issue, journal.user)
@email_body = attachment_macro(@email_body, journal.issue)
raise MissingInformation.new(l(:text_helpdesk_to_address_cant_be_blank)) if to_address.blank?
raise MissingInformation.new(l(:text_helpdesk_message_body_cant_be_blank)) if @email_body.blank?
raise MissingInformation.new(l(:text_helpdesk_from_address_cant_be_blank)) if from_address.blank?
subject_macro = self.class.apply_macro(HelpdeskSettings["helpdesk_answer_subject", journal.issue.project], contact, journal.issue)
# subject_macro += " - [##{journal.issue.id}]" if !subject_macro.blank? && !subject_macro.include?("##{journal.issue.id}]")
@email_stylesheet = HelpdeskSettings[:helpdesk_helpdesk_css, journal.issue.project].to_s.html_safe
extract_attachments(journal)
if journal.details.blank? && journal.private_notes? && journal.notes.present?
details_journal = Journal.where('id != ?', journal.id).where(:created_on => journal.created_on).first
extract_attachments(details_journal) if details_journal
end
mail :from => self.class.apply_from_macro(from_address.to_s, journal.user),
:to => to_address.to_s,
:cc => cc_address.to_s,
:bcc => bcc_address.to_s,
:in_reply_to => in_reply_to.to_s,
:subject => (subject_macro.blank? ? journal.issue.subject + " [#{journal.issue.tracker} ##{journal.issue.id}]" : subject_macro) do |format|
format.text { render 'email_layout' }
format.html { render 'email_layout' } unless RedmineHelpdesk.settings["plain_text_mail"]
end
end
def extract_attachments(journal)
journal.details.where(:property => 'attachment').each do |attachment_journal|
if attach = Attachment.where(:id => attachment_journal.prop_key).first
attachments[attach.filename] = File.open(attach.diskfile, 'rb') { |io| io.read }
end
end
end
def auto_answer(contact, issue)
@project = issue.project
headers['X-Redmine-Ticket-ID'] = issue.id.to_s
headers['X-Auto-Response-Suppress'] = 'oof'
confirmation_body = self.class.apply_macro(HelpdeskSettings["helpdesk_first_answer_template", issue.project_id], contact, issue)
@email_stylesheet = HelpdeskSettings[:helpdesk_helpdesk_css, issue.project_id].to_s.html_safe
@email_body = confirmation_body
from_address = HelpdeskSettings["helpdesk_answer_from", issue.project].blank? ? Setting.mail_from : HelpdeskSettings["helpdesk_answer_from", issue.project]
mail :from => self.class.apply_from_macro(from_address.to_s, nil),
:to => contact.primary_email,
:cc => issue.helpdesk_ticket.try(:cc_address),
:subject => self.class.apply_macro(HelpdeskSettings["helpdesk_first_answer_subject", issue.project_id], contact, issue) || "Helpdesk auto answer [Case ##{issue.id}]",
:in_reply_to => issue.helpdesk_ticket.try(:message_id) do |format|
format.text { render 'email_layout'}
format.html { render 'email_layout' } unless RedmineHelpdesk.settings["plain_text_mail"]
end
logger.info "##{issue.id}: Sending confirmation to #{contact.primary_email}" if logger
end
def initial_message(contact, issue, params)
@project = issue.project
headers['X-Redmine-Ticket-ID'] = issue.id.to_s
@email_header = self.class.apply_macro(HelpdeskSettings["helpdesk_emails_header", issue.project], contact, issue, issue.author) unless HelpdeskSettings["helpdesk_emails_header", issue.project].blank?
@email_footer = self.class.apply_macro(HelpdeskSettings["helpdesk_emails_footer", issue.project], contact, issue, issue.author) unless HelpdeskSettings["helpdesk_emails_footer", issue.project].blank?
@email_body = self.class.apply_macro(issue.description, contact, issue, issue.author)
@email_body = attachment_macro(@email_body, issue)
raise MissingInformation.new("Contact #{contact.name} should have mail") if contact.email.blank?
raise MissingInformation.new("Message shouldn't be blank") if @email_body.blank?
@email_stylesheet = HelpdeskSettings[:helpdesk_helpdesk_css, issue.project].to_s.html_safe
params[:attachments].each_value do |mail_attachment|
if file = mail_attachment['file']
file.rewind if file
attachments[file.original_filename] = file.read
file.rewind if file
elsif token = mail_attachment['token']
if token.to_s =~ /^(\d+)\.([0-9a-f]+)$/
attachment_id, attachment_digest = $1, $2
if a = Attachment.where(:id => attachment_id, :digest => attachment_digest).first
attachments[a.filename] = File.open(a.diskfile, 'rb'){|io| io.read}
end
end
end
end unless params[:attachments].blank?
to_address = (params[:helpdesk] && !params[:helpdesk][:to_address].blank?) ? params[:helpdesk][:to_address] : contact.primary_email
from_address = HelpdeskSettings["helpdesk_answer_from", issue.project].blank? ? Setting.mail_from : HelpdeskSettings["helpdesk_answer_from", issue.project]
logger.error "##{issue.id}: From address couldn't be black" if from_address.blank? && logger
mail :from => self.class.apply_from_macro(from_address.to_s, nil),
:to => to_address,
:subject => self.class.apply_macro(HelpdeskSettings["helpdesk_first_answer_subject", issue.project_id], contact, issue) || issue.subject do |format|
format.text { render 'email_layout' }
format.html { render 'email_layout' } unless RedmineHelpdesk.settings["plain_text_mail"]
end
end
# Receive email methods
def self.receive(raw_email, options={})
@@helpdesk_mailer_options = options.dup
raw_email.force_encoding('ASCII-8BIT') if raw_email.respond_to?(:force_encoding)
email = Mail.new(raw_email)
new.receive(email)
end
# Processes incoming emails
# Returns the created object (eg. an issue, a message) or false
def receive(email)
@email = email
if !target_project.module_enabled?(:contacts) || !target_project.module_enabled?(:issue_tracking)
logger.error "#{email && email.message_id}: Contacts and issues modules should be enable for #{target_project.name} project" if logger
return false
end
@@helpdesk_mailer_options = HelpdeskMailer.get_issue_options(@@helpdesk_mailer_options, target_project.id)
sender_email = message_sender(email)
# Ignore emails received from the application emission address to avoid hell cycles
if sender_email.downcase == Setting.mail_from.to_s.strip.downcase
logger.info "#{email && email.message_id}: Ignoring email from Redmine emission address [#{sender_email}]" if logger
return false
end
return false unless handle_ignored(email)
if !check_blacklist?(email)
logger.info "#{email && email.message_id}: Email #{sender_email} ignored because in blacklist" if logger
return false
end
@user = User.find_by_mail(sender_email) || User.anonymous
@contact = contact_from_email(email)
User.current = @user
if @contact
logger.info "#{email && email.message_id}: [#{@contact.name}] contact created/founded" if logger
else
logger.error "#{email && email.message_id}: could not create/found contact for [#{sender_email}]" if logger
return false
end
dispatch
end
def self.check_project(project_id)
msg_count = 0
unless Project.find_by_id(project_id).blank? || HelpdeskSettings[:helpdesk_protocol, project_id].blank?
mail_options, options = self.get_mail_options(project_id)
case mail_options[:protocol]
when "pop3" then
msg_count = RedmineContacts::Mailer.check_pop3(self, mail_options, options)
when "imap" then
msg_count = RedmineContacts::Mailer.check_imap(self, mail_options, options)
end
end
if project = Project.find_by_id(project_id)
HelpdeskTicket.autoclose(project)
end
msg_count
end
def self.get_mail_options(project_id)
case HelpdeskSettings[:helpdesk_protocol, project_id]
when "gmail"
protocol = "imap"
host = "imap.gmail.com"
port = "993"
ssl = "1"
when "yahoo"
protocol = "imap"
host = "imap.mail.yahoo.com"
port = "993"
ssl = "1"
when "yandex"
protocol = "imap"
host = "imap.yandex.ru"
port = "993"
ssl = "1"
else
protocol = HelpdeskSettings[:helpdesk_protocol, project_id]
host = HelpdeskSettings[:helpdesk_host, project_id]
port = HelpdeskSettings[:helpdesk_port, project_id]
ssl = HelpdeskSettings[:helpdesk_use_ssl, project_id] != "1" ? nil : "1"
end
mail_options = {:protocol => protocol,
:host => host,
:port => port,
:ssl => ssl,
:apop => HelpdeskSettings[:helpdesk_apop, project_id],
:username => HelpdeskSettings[:helpdesk_username, project_id],
:password => HelpdeskSettings[:helpdesk_password, project_id],
:folder => HelpdeskSettings[:helpdesk_imap_folder, project_id],
:move_on_success => HelpdeskSettings[:helpdesk_move_on_success, project_id],
:move_on_failure => HelpdeskSettings[:helpdesk_move_on_failure, project_id],
:delete_unprocessed => HelpdeskSettings[:helpdesk_delete_unprocessed, project_id].to_i > 0
}
options = get_issue_options({}, project_id)
[mail_options, options]
end
def self.get_issue_options(options, project_id)
options = { :issue => {} } unless options[:issue]
options[:issue][:project_id] = project_id
options[:issue][:status_id] = HelpdeskSettings[:helpdesk_new_status, project_id] unless options[:issue][:status_id]
options[:issue][:assigned_to_id] = HelpdeskSettings["helpdesk_assigned_to", project_id] unless options[:issue][:assigned_to_id]
options[:issue][:tracker_id] = HelpdeskSettings["helpdesk_tracker", project_id] unless options[:issue][:tracker_id]
options[:issue][:priority_id] = HelpdeskSettings[:helpdesk_issue_priority, project_id] unless options[:issue][:priority_id]
options[:issue][:due_date] = HelpdeskSettings[:helpdesk_issue_due_date, project_id] unless options[:issue][:due_date]
options[:issue][:reopen_status_id] = HelpdeskSettings["helpdesk_reopen_status", project_id] unless options[:issue][:reopen_status_id]
options
end
def attachment_macro(text, issue)
text.scan(/\{\{send_file\(([^%\}]+)\)\}\}/).flatten.each do |file_name|
attachment = file_name.match(/^(\d)+$/) ? Attachment.where(:id => file_name).first : issue.attachments.where(:filename => file_name).first
attachments[attachment.filename] = File.open(attachment.diskfile, 'rb'){|io| io.read} if attachment
end
text.gsub(/\{\{send_file\(([^%\}]+)\)\}\}/, '')
end
def self.apply_from_macro(text, journal_user = nil)
return text unless text =~ /\A\{%.+%\}.*?<.+@.+\..{2,}>\z/
text = text[/<.*>/] if journal_user.nil? || journal_user == User.anonymous
text = text.gsub(/\{%response.author%\}/, journal_user.name) if text =~ /\{%response.author%\}/
text = text.gsub(/\{%response.author.first_name%\}/, journal_user.firstname) if text =~ /\{%response.author.first_name%\}/
text
end
def self.apply_macro(text, contact, issue, journal_user=nil)
return '' if text.blank?
text = text.gsub(/%%NAME%%|\{%contact.first_name%\}/, contact.first_name)
text = text.gsub(/%%FULL_NAME%%|\{%contact.name%\}/, contact.name)
text = text.gsub(/%%COMPANY%%|\{%contact.company%\}/, contact.company) if contact.company
text = text.gsub(/%%LAST_NAME%%|\{%contact.last_name%\}/, contact.last_name.blank? ? "" : contact.last_name)
text = text.gsub(/%%MIDDLE_NAME%%|\{%contact.middle_name%\}/, contact.middle_name.blank? ? "" : contact.middle_name)
text = text.gsub(/\{%contact.email%\}/, contact.primary_email.to_s)
text = text.gsub(/%%DATE%%|\{%date%\}/, ApplicationHelper.format_date(Date.today))
text = text.gsub(/%%ASSIGNEE%%|\{%ticket.assigned_to%\}/, issue.assigned_to.blank? ? "" : issue.assigned_to.name)
text = text.gsub(/%%ISSUE_ID%%|\{%ticket.id%\}/, issue.id.to_s) if issue.id
text = text.gsub(/%%ISSUE_TRACKER%%|\{%ticket.tracker%\}/, issue.tracker.name) if issue.tracker
text = text.gsub(/%%QUOTED_ISSUE_DESCRIPTION%%|\{%ticket.quoted_description%\}/, issue.description.gsub(/^/, "> ")) if issue.description
text = text.gsub(/%%PROJECT%%|\{%ticket.project%\}/, issue.project.name) if issue.project
text = text.gsub(/%%SUBJECT%%|\{%ticket.subject%\}/, issue.subject) if issue.subject
text = text.gsub(/%%NOTE_AUTHOR%%|\{%response.author%\}/, journal_user.name) if journal_user
text = text.gsub(/%%NOTE_AUTHOR.FIRST_NAME%%|\{%response.author.first_name%\}/, journal_user.firstname) if journal_user
text = text.gsub(/%%NOTE_AUTHOR.LAST_NAME%%|\{%response.author.last_name%\}/, journal_user.lastname) if journal_user
text = text.gsub(/\{%ticket.status%\}/, issue.status.name) if issue.status
text = text.gsub(/\{%ticket.priority%\}/, issue.priority.name) if issue.priority
text = text.gsub(/\{%ticket.estimated_hours%\}/, issue.estimated_hours ? issue.estimated_hours.to_s : "")
text = text.gsub(/\{%ticket.done_ratio%\}/, issue.done_ratio.to_s) if issue.done_ratio
text = text.gsub(/\{%ticket.closed_on%\}/, issue.closed_on ? ApplicationHelper.format_date(issue.closed_on) : "") if issue.respond_to?(:closed_on)
text = text.gsub(/\{%ticket.due_date%\}/, issue.due_date ? ApplicationHelper.format_date(issue.due_date) : "")
text = text.gsub(/\{%ticket.start_date%\}/, issue.start_date ? ApplicationHelper.format_date(issue.start_date) : "")
text = text.gsub(/\{%ticket.public_url%\}/, Setting.protocol + '://' + Setting.host_name + Rails.application.routes.url_helpers.public_ticket_path(issue.helpdesk_ticket.id, issue.helpdesk_ticket.token) ) if text.match(/\{%ticket.public_url%\}/) && issue.helpdesk_ticket
if RedmineHelpdesk.vote_allow?
text = text.gsub(/\{%ticket.voting%\}/, Setting.protocol + '://' + Setting.host_name + Rails.application.routes.url_helpers.helpdesk_votes_show_path(issue.helpdesk_ticket.id, issue.helpdesk_ticket.token) ) if text.match(/\{%ticket.voting%\}/)
text = text.gsub(/\{%ticket.voting.good%\}/, Setting.protocol + '://' + Setting.host_name + Rails.application.routes.url_helpers.helpdesk_votes_fast_vote_path(issue.helpdesk_ticket.id, 2, issue.helpdesk_ticket.token) ) if text.match(/\{%ticket.voting.good%\}/)
text = text.gsub(/\{%ticket.voting.okay%\}/, Setting.protocol + '://' + Setting.host_name + Rails.application.routes.url_helpers.helpdesk_votes_fast_vote_path(issue.helpdesk_ticket.id, 1, issue.helpdesk_ticket.token) ) if text.match(/\{%ticket.voting.okay%\}/)
text = text.gsub(/\{%ticket.voting.bad%\}/, Setting.protocol + '://' + Setting.host_name + Rails.application.routes.url_helpers.helpdesk_votes_fast_vote_path(issue.helpdesk_ticket.id, 0, issue.helpdesk_ticket.token) ) if text.match(/\{%ticket.voting.bad%\}/)
end
if text =~ /\{%ticket.history%\}/
ticket_history = ''
issue.journals.eager_load(:journal_message).map(&:journal_message).compact.each do |journal_message|
message_author = "*#{l(:label_crm_added_by)} #{journal_message.is_incoming? ? journal_message.from_address : journal_message.journal.user.name}, #{format_time(journal_message.message_date)}*"
ticket_history = (message_author + "\n" + journal_message.journal.notes + "\n" + ticket_history).gsub(/^/, "> ")
end
text = text.gsub(/\{%ticket.history%\}/, ticket_history)
end
issue.custom_field_values.each do |value|
text = text.gsub(/%%#{value.custom_field.name}%%/, value.value.to_s)
end
contact.custom_field_values.each do |value|
text = text.gsub(/%%#{value.custom_field.name}%%/, value.value.to_s)
end if contact.respond_to?("custom_field_values")
journal_user.custom_field_values.each do |value|
text = text.gsub(/\{%response.author.custom_field: #{value.custom_field.name}%\}/, value.value.to_s)
end if journal_user
text
end
private
def dispatch
m = email.subject && email.subject.match(ISSUE_REPLY_SUBJECT_RE)
journal_message = !email.in_reply_to.blank? && JournalMessage.find_by_message_id(email.in_reply_to)
helpdesk_ticket = !email.in_reply_to.blank? && HelpdeskTicket.find_by_message_id(email.in_reply_to)
if journal_message && journal_message.journal && journal_message.journal.issue
receive_issue_reply(journal_message.journal.issue.id)
elsif helpdesk_ticket && helpdesk_ticket.issue
receive_issue_reply(helpdesk_ticket.issue.id)
elsif m && Issue.exists?(m[1].to_i)
receive_issue_reply(m[1].to_i)
else
dispatch_to_default
end
rescue MissingInformation => e
logger.error "#{email && email.message_id}: missing information from #{user}: #{e.message}" if logger
false
rescue UnauthorizedAction => e
logger.error "#{email && email.message_id}: unauthorized attempt from #{user}" if logger
false
rescue Exception => e
# TODO: send a email to the user
logger.error "#{email && email.message_id}: dispatch error #{e.message}" if logger
false
end
def dispatch_to_default
receive_issue
end
def target_project
@target_project ||= Project.find_by_identifier(get_keyword(:project) || get_keyword(:project_id))
@target_project ||= Project.find_by_id(get_keyword(:project_id)) if @target_project.nil?
raise MissingInformation.new('Unable to determine @target_project project') if @target_project.nil?
@target_project
end
def helpdesk_issue_attributes_from_keywords(issue)
# assigned_to = ((k = get_keyword(:assigned_to_id, :override => true)) && User.find_by_id(k)) || ((k = get_keyword(:assigned_to, :override => true)) && find_user_from_keyword(k))
assigned_to = ((k = get_keyword(:assigned_to_id, :override => true)) && (User.find_by_id(k) || Group.find_by_id(k))) || ((k = get_keyword(:assigned_to, :override => true)) && find_user_from_keyword(k))
attrs = {
'status_id' => ((k = get_keyword(:status)) && IssueStatus.named(k).first.try(:id) ) || ((k = get_keyword(:status_id)) && IssueStatus.find_by_id(k).try(:id)),
'priority_id' => ((k = get_keyword(:priority)) && IssuePriority.named(k).first.try(:id)) || ((k = get_keyword(:priority_id)) && IssuePriority.find_by_id(k).try(:id)),
'category_id' => (k = get_keyword(:category)) && issue.project.issue_categories.named(k).first.try(:id),
'assigned_to_id' => assigned_to.try(:id),
'fixed_version_id' => (k = get_keyword(:fixed_version, :override => true)) && issue.project.shared_versions.named(k).first.try(:id),
'start_date' => get_keyword(:start_date, :override => true, :format => '\d{4}-\d{2}-\d{2}'),
'due_date' => get_keyword(:due_date, :override => true, :format => '\d{4}-\d{2}-\d{2}'),
'estimated_hours' => get_keyword(:estimated_hours, :override => true),
'done_ratio' => get_keyword(:done_ratio, :override => true, :format => '(\d|10)?0')
}.delete_if {|k, v| v.blank? }
attrs
end
def calculated_tracker_id(issue)
issue_tracker_id = ((k = get_keyword(:tracker)) && issue.project.trackers.named(k).first.try(:id)) ||
((k = get_keyword(:tracker_id)) && issue.project.trackers.find_by_id(k).try(:id))
issue_tracker_id = issue.project.trackers.first.try(:id) unless issue_tracker_id
issue_tracker_id
end
# Creates a new issue
def receive_issue
project = target_project
issue = Issue.new
issue.author = user
issue.project = project
issue.safe_attributes = helpdesk_issue_attributes_from_keywords(issue)
issue.safe_attributes = {'custom_field_values' => custom_field_values_from_keywords(issue)}
issue.tracker_id = calculated_tracker_id(issue)
issue.subject = cleaned_up_subject(email)
issue.subject = '(no subject)' if issue.subject.blank?
issue.description = escaped_cleaned_up_text_body
issue.start_date ||= Date.today if Setting.default_issue_start_date_to_creation_date?
helpdesk_ticket = HelpdeskTicket.new(:from_address => message_sender(email).downcase.to_s.slice(0, 255),
:to_address => email.to_addrs.join(',').downcase.to_s.slice(0, 255),
:cc_address => email.cc_addrs.join(',').downcase.to_s.slice(0, 255),
:ticket_date => email.date || Time.now,
:message_id => email.message_id.to_s.slice(0, 255),
:is_incoming => true,
:customer => contact,
:issue => issue,
:source => HelpdeskTicket::HELPDESK_EMAIL_SOURCE)
issue.helpdesk_ticket = helpdesk_ticket
issue.contacts << cc_contacts if HelpdeskSettings["helpdesk_save_cc", target_project.id].to_i > 0
issue.assigned_to = @contact.find_assigned_user(project, issue.assigned_to_id)
save_email_as_attachment(helpdesk_ticket)
add_attachments(issue)
Redmine::Hook.call_hook(:helpdesk_mailer_receive_issue_before_save, { :issue => issue, :contact => contact, :helpdesk_ticket => helpdesk_ticket, :email => email})
ActiveRecord::Base.transaction do
issue.save!(:validate => false)
ContactNote.create(:content => "*#{issue.subject}* [#{issue.tracker.name} - ##{issue.id}]\n\n" + issue.description,
:type_id => Note.note_types[:email],
:source => contact,
:author_id => issue.author_id) if HelpdeskSettings["helpdesk_add_contact_notes", project]
begin
notification = HelpdeskMailer.auto_answer(contact, issue).deliver if HelpdeskSettings["helpdesk_send_notification", project].to_i > 0
logger.info "#{email && email.message_id}: notification was sent to #{notification.to_addrs.first}" if logger && notification
rescue Exception => e
logger.error "#{email && email.message_id}: notification was not sent #{e.message}" if logger
false
end
logger.info "#{email && email.message_id}: issue ##{issue.id} created by #{user} for #{contact.name}" if logger
issue
end #transaction
end
# Adds a note to an existing issue
def receive_issue_reply(issue_id)
issue = Issue.find_by_id(issue_id)
return unless issue
# if lifetime expaired create new issue
if (HelpdeskSettings["helpdesk_lifetime", target_project].to_i > 0) && issue.journals && issue.journals.last && ((Date.today) - issue.journals.last.created_on.to_date > HelpdeskSettings["helpdesk_lifetime", target_project].to_i)
email.subject = email.subject.to_s.gsub(ISSUE_REPLY_SUBJECT_RE, '')
return receive_issue
end
journal = issue.init_journal(user)
journal.notes = escaped_cleaned_up_text_body
journal_message = JournalMessage.create(:from_address => message_sender(email).downcase,
:to_address => email.to_addrs.join(',').downcase,
:bcc_address => email.bcc_addrs.join(',').downcase,
:cc_address => email.cc_addrs.join(',').downcase,
:message_id => email.message_id,
:is_incoming => true,
:message_date => email.date || Time.now,
:contact => contact,
:journal => journal)
issue.contacts << cc_contacts if HelpdeskSettings["helpdesk_save_cc", target_project.id].to_i > 0
add_attachments(issue)
save_email_as_attachment(journal_message, "reply-#{DateTime.now.strftime('%d.%m.%y-%H.%M.%S')}.eml")
if reopen_status_id = ((k = @@helpdesk_mailer_options[:reopen_status]) && IssueStatus.named(k).first.try(:id) ) || ((k = get_keyword(:reopen_status_id)) && IssueStatus.find_by_id(k).try(:id))
issue.status_id = reopen_status_id
end
issue.save!
logger.info "#{email && email.message_id}: issue ##{issue.id} updated by #{user}" if logger
journal
end
# Reply will be added to the issue
def receive_journal_reply(journal_id)
journal = Journal.find_by_id(journal_id)
if journal && journal.journalized_type == 'Issue'
receive_issue_reply(journal.journalized_id)
end
end
def add_attachments(obj)
fwd_attachments = email.parts.map { |p|
if p.content_type =~ /message\/rfc822/
Mail.new(p.body).attachments
elsif p.parts.empty?
p if p.attachment?
else
p.attachments
end
}.flatten.compact
email_attachments = fwd_attachments | email.attachments
unless email_attachments.blank?
email_attachments.each do |attachment|
if RUBY_VERSION < '1.9'
attachment_filename = (attachment[:content_type].filename rescue nil) ||
(attachment[:content_disposition].filename rescue nil) ||
(attachment[:content_location].location rescue nil) ||
"attachment"
attachment_filename = Mail::Encodings.unquote_and_convert_to(attachment_filename, 'utf-8') rescue 'unprocessable_filename'
attachment_filename = helpdesk_to_utf8(attachment_filename)
else
attachment_filename = helpdesk_to_utf8(attachment.filename, 'binary')
end
new_attachment = Attachment.new(:container => obj,
:file => (attachment.decoded rescue nil) || (attachment.decode_body rescue nil) || attachment.raw_source,
:filename => attachment_filename,
:author => user,
:content_type => attachment.mime_type)
if obj.attachments.where(:digest => attachment_digest(attachment.body.to_s)).empty? && accept_attachment?(new_attachment)
obj.attachments << new_attachment
logger.info "#{email && email.message_id}: attachment #{attachment_filename} added to ticket: '#{obj.subject}'" if logger
end
end
end
end
def get_keyword(attr, options = {})
@keywords ||= {}
if !@keywords.has_key?(attr)
if (options[:override] || attr_overridable?(attr)) &&
v = extract_keyword!(escaped_cleaned_up_text_body, attr, options[:format])
@keywords[attr] = v
elsif !@@helpdesk_mailer_options[:issue][attr].blank?
@keywords[attr] = @@helpdesk_mailer_options[:issue][attr]
end
end
@keywords[attr]
end
def attr_overridable?(attr)
@@helpdesk_mailer_options[:allow_override].present? &&
@@helpdesk_mailer_options[:allow_override].include?(attr.to_s)
end
def find_user_from_keyword(keyword)
user ||= User.find_by_mail(keyword)
user ||= User.find_by_login(keyword)
if user.nil? && keyword.match(/ /)
firstname, lastname = *(keyword.split) # "First Last Throwaway"
user ||= User.find_by_firstname_and_lastname(firstname, lastname)
end
user
end
def check_blacklist?(email)
return true if HelpdeskSettings["helpdesk_blacklist", target_project].blank?
addr = email.from_addrs.first.to_s.strip
from_addr = addr # (addr && !addr.spec.blank?) ? addr.spec : email.header["from"].inspect.match(/[-A-z0-9.]+@[-A-z0-9.]+/).to_s
cond = "(" + HelpdeskSettings["helpdesk_blacklist", target_project].split("\n").map{|u| u.strip unless u.blank?}.compact.join('|') + ")"
!from_addr.match(cond)
end
def new_contact_from_attributes(email_address, fullname=nil)
contact = Contact.new
# Truncating the email address would result in an invalid format
contact.email = email_address
names = fullname.blank? ? email_address.gsub(/@.*$/, '').split('.') : fullname.split
contact.first_name = names.shift.slice(0, 255)
contact.last_name = names.join(' ').slice(0, 255)
contact.company = email_address.downcase.slice(0, 255)
contact.last_name = '-' if contact.last_name.blank?
if contact.last_name =~ %r(\((.*)\))
contact.last_name, contact.company = $`, $1
end
if contact.first_name =~ /,$/
contact.first_name = contact.last_name
contact.last_name = $` # everything before the match
end
contact.projects << target_project
contact.tag_list = HelpdeskSettings["helpdesk_created_contact_tag", target_project] if HelpdeskSettings["helpdesk_created_contact_tag", target_project]
contact
end
def cc_contacts
email[:cc].to_s
email.cc_addrs.each_with_index.map do |cc_addr, index|
cc_name = email[:cc].display_names[index]
create_contact_from_address(cc_addr, cc_name)
end.compact
end
def create_contact_from_address(addr, name)
contacts = Contact.find_by_emails([addr])
unless contacts.blank?
contact = contacts.first
if contact.projects.blank? || HelpdeskSettings[:helpdesk_add_contact_to_project, target_project].to_i > 0
contact.projects << target_project
contact.save!
end
return contact
end
if HelpdeskSettings["helpdesk_is_not_create_contacts", target_project].to_i > 0
logger.error "#{email && email.message_id}: can't find contact with email: #{addr} in whitelist. Not create new contacts option enable" if logger
nil
else
contact = new_contact_from_attributes(addr, name)
if contact.save(:validate => false)
contact
else
logger.error "Helpdeks MailHandler: failed to create Contact: #{contact.errors.full_messages}" if logger
nil
end
end
end
# Get or create contact for the +email+ sender
def contact_from_email(email)
# from = email.header['from'].to_s
# debugger
from = cleaned_up_from_address
addr, name = from, nil
if m = from.match(/^"?(.+?)"?\s+<(.+@.+)>$/)
addr, name = m[2], m[1]
end
if addr.present?
create_contact_from_address(addr, name)
else
logger.error "#{email && email.message_id}: failed to create Contact: no FROM address found" if logger
nil
end
end
# Returns a Hash of issue custom field values extracted from keywords in the email body
def custom_field_values_from_keywords(customized)
customized.custom_field_values.inject({}) do |h, v|
if value = get_keyword(v.custom_field.name, :override => true)
h[v.custom_field.id.to_s] = value
end
h
end
end
def save_email_as_attachment(container, filename="message.eml")
Attachment.create(:container => container,
:file => email.raw_source.to_s,
:author => user,
:filename => filename,
:content_type => "message/rfc822")
end
def plain_text_body
return @plain_text_body unless @plain_text_body.nil?
part = email.text_part || email.html_part
unless part
return @plain_text_body = '' if email.parts.present? && email.parts.all? { |part| part.attachment? }
part = email
end
is_html = email.text_part.blank?
part_charset = Mail::RubyVer.pick_encoding(part.charset).to_s rescue part.charset
@plain_text_body = helpdesk_to_utf8(part.body.decoded, part_charset)
# strip html tags and remove doctype directive
@plain_text_body.gsub! %r{^[ ]+}, ''
if is_html && RedmineHelpdesk.strip_tags?
@plain_text_body.gsub! %r{<head>(?:.|\n|\r)+?<\/head>}, ""
@plain_text_body.gsub! %r{<\/(li|ol|ul|h1|h2|h3|h4)>}, "\r\n"
@plain_text_body.gsub! %r{<\/(p|div|pre)>}, "\r\n\r\n"
@plain_text_body.gsub! %r{<li>}, " - "
@plain_text_body.gsub! %r{<br[^>]*>}, "\r\n"
@plain_text_body = strip_tags(@plain_text_body.strip)
@plain_text_body.sub! %r{^<!DOCTYPE .*$}, ''
end
@plain_text_body.strip
rescue Exception => e
logger.error "#{email && email.message_id}: Message body processing error - #{e.message}" if logger
@plain_text_body = '(Unprocessable message body)'
end
def cleaned_up_subject(email)
return "" if email[:subject].blank?
subject = decode_subject(email[:subject].value)
subject = helpdesk_to_utf8(subject)
subject.strip[0,255]
rescue Exception => e
logger.error "#{email && email.message_id}: Message subject processing error - #{e.message}" if logger
'(Unprocessable subject)'
end
def cleaned_up_from_address
from = email.header['reply-to'].to_s.present? ? email.header['reply-to'] : email.header['from']
from.to_s.strip[0, 255]
end
def logger
HelpdeskLogger
end
def helpdesk_to_utf8(str, encoding="UTF-8")
return str if str.nil?
if str.respond_to?(:force_encoding)
begin
cleaned = str.force_encoding('UTF-8')
cleaned = cleaned.encode("UTF-8", encoding) if encoding.upcase == 'ISO-2022-JP'
unless cleaned.valid_encoding?
cleaned = str.encode('UTF-8', encoding, :invalid => :replace, :undef => :replace, :replace => '').chars.select{|i| i.valid_encoding?}.join
end
str = cleaned
rescue EncodingError
str.encode!( 'UTF-8', :invalid => :replace, :undef => :replace )
end
elsif RUBY_PLATFORM == 'java'
begin
ic = Iconv.new('UTF-8', encoding + '//IGNORE')
str = ic.iconv(str)
rescue
str = str.gsub(%r{[^\r\n\t\x20-\x7e]}, '?')
end
else
ic = Iconv.new('UTF-8', encoding + '//IGNORE')
txtar = ""
begin
txtar += ic.iconv(str)
rescue Iconv::IllegalSequence
txtar += $!.success
str = '?' + $!.failed[1,$!.failed.length]
retry
rescue
txtar += $!.success
end
str = txtar
end
str
end
def attachment_digest(file_source)
encoder = Redmine::VERSION.to_s >= '3.4' || RedmineContacts.unstable_branch? ? Digest::SHA256.new : Digest::MD5.new
encoder.update(file_source)
encoder.hexdigest
end
def set_delivery_options
return false if HelpdeskSettings[:helpdesk_smtp_use_default_settings, project.id].to_i == 0
message.delivery_method(:smtp)
settings = {:address => HelpdeskSettings[:helpdesk_smtp_server, project.id],
:port => HelpdeskSettings[:helpdesk_smtp_port, project.id] || 25,
:domain => HelpdeskSettings[:helpdesk_smtp_domain, project.id],
:enable_starttls_auto => true,
:ssl => HelpdeskSettings[:helpdesk_smtp_ssl, project.id].to_i > 0 &&
HelpdeskSettings[:helpdesk_smtp_tls, project.id].to_i == 0}
authentication = HelpdeskSettings[:helpdesk_smtp_authentication, project.id]
unless authentication.blank?
settings.merge!(:authentication => authentication,
:user_name => HelpdeskSettings[:helpdesk_smtp_username, project.id],
:password => HelpdeskSettings[:helpdesk_smtp_password, project.id])
end
message.delivery_method.settings.merge!(settings)
end
def decode_subject(str)
# Optimization: If there's no encoded-words in the string, just return it
return str unless str.index("=?")
str = str.gsub(/\?=(\s*)=\?/, '?=????=?') # Replace whitespaces between 'encoded-word's on special symbols
str.split('????').map do |text|
if text.index('=?') .nil?
text
else
text.gsub!(/[\r\n]/, '')
text.scan(/\=\?.+?\?[qQbB]\?.+?\?\=/).map do |part|
if part.index(/\=\?.+\?[Bb]\?.+\?\=/m)
part.gsub(/\=\?.+\?[Bb]\?.+\?=/m) { |substr| Mail::Encodings.b_value_decode(substr) }
elsif part.index(/\=\?.+\?[Qq]\?.+\?\=/m)
part.gsub(/\=\?.+\?[Qq]\?.+\?\=/m) { |substr| Mail::Encodings.q_value_decode(substr) }
end
end
end
end.join('')
end
def self.ignored_helpdesk_headers
helpdesk_headers = {
'X-Auto-Response-Suppress' => /\A(all|AutoReply|oof)/
}
ignored_emails_headers.merge(helpdesk_headers)
end
def handle_ignored(email)
self.class.ignored_helpdesk_headers.each do |key, ignored_value|
value = email.header[key]
if value
value = value.to_s.downcase
if (ignored_value.is_a?(Regexp) && value.match(ignored_value)) || value == ignored_value
if logger
logger.info "#{email && email.message_id}: ignoring email with #{key}:#{value} header"
end
return false
end
end
end
return true
end
def escaped_cleaned_up_text_body
text_body = cleaned_up_text_body
return text_body unless (ActiveRecord::Base.connection.adapter_name =~ /mysql/i).present?
text_body.gsub(/./) { |c| c.bytesize == 4 ? ' ' : c }
end
end