824 lines
38 KiB
Ruby
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
|