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{
(?:.|\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{