2014-02-03 19:23:00 +00:00
|
|
|
# Copyright (C) 2012-2014 Zammad Foundation, http://zammad-foundation.org/
|
2013-06-12 15:59:58 +00:00
|
|
|
|
2012-06-15 11:10:23 +00:00
|
|
|
# encoding: utf-8
|
|
|
|
|
2012-04-13 16:42:25 +00:00
|
|
|
require 'mail'
|
2012-12-05 01:38:30 +00:00
|
|
|
require 'encode'
|
2012-12-05 00:28:04 +00:00
|
|
|
|
2012-12-05 01:27:56 +00:00
|
|
|
class Channel::EmailParser
|
2012-10-04 06:54:21 +00:00
|
|
|
|
|
|
|
=begin
|
|
|
|
|
|
|
|
mail = parse( msg_as_string )
|
|
|
|
|
|
|
|
mail = {
|
|
|
|
:from => 'Some Name <some@example.com>',
|
|
|
|
:from_email => 'some@example.com',
|
|
|
|
:from_local => 'some',
|
|
|
|
:from_domain => 'example.com',
|
|
|
|
:from_display_name => 'Some Name',
|
|
|
|
:message_id => 'some_message_id@example.com',
|
2012-10-04 07:09:27 +00:00
|
|
|
:to => 'Some System <system@example.com>',
|
|
|
|
:cc => 'Somebody <somebody@example.com>',
|
|
|
|
:subject => 'some message subject',
|
|
|
|
:body => 'some message body',
|
2012-10-04 06:54:21 +00:00
|
|
|
:attachments => [
|
2012-10-04 07:09:27 +00:00
|
|
|
{
|
|
|
|
:data => 'binary of attachment',
|
|
|
|
:filename => 'file_name_of_attachment.txt',
|
|
|
|
:preferences => {
|
|
|
|
:content-alternative => true,
|
|
|
|
:Mime-Type => 'text/plain',
|
|
|
|
:Charset => 'iso-8859-1',
|
|
|
|
},
|
|
|
|
},
|
2012-10-04 06:54:21 +00:00
|
|
|
],
|
2012-10-04 07:09:27 +00:00
|
|
|
|
|
|
|
# ignore email header
|
|
|
|
:x-zammad-ignore => 'false',
|
|
|
|
|
|
|
|
# customer headers
|
|
|
|
:x-zammad-customer-login => '',
|
|
|
|
:x-zammad-customer-email => '',
|
|
|
|
:x-zammad-customer-firstname => '',
|
|
|
|
:x-zammad-customer-lastname => '',
|
|
|
|
|
|
|
|
# ticket headers
|
|
|
|
:x-zammad-group => 'some_group',
|
|
|
|
:x-zammad-state => 'some_state',
|
|
|
|
:x-zammad-priority => 'some_priority',
|
|
|
|
:x-zammad-owner => 'some_owner_login',
|
|
|
|
|
|
|
|
# article headers
|
2013-08-01 09:11:55 +00:00
|
|
|
:x-zammad-article-visibility => 'internal',
|
2012-10-04 07:09:27 +00:00
|
|
|
:x-zammad-article-type => 'agent',
|
|
|
|
:x-zammad-article-sender => 'customer',
|
|
|
|
|
|
|
|
# all other email headers
|
|
|
|
:some-header => 'some_value',
|
2012-10-04 06:54:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
=end
|
|
|
|
|
2012-05-04 11:33:05 +00:00
|
|
|
def parse (msg)
|
|
|
|
data = {}
|
2012-04-13 16:42:25 +00:00
|
|
|
mail = Mail.new( msg )
|
2012-05-04 11:33:05 +00:00
|
|
|
|
2012-05-05 09:24:05 +00:00
|
|
|
# set all headers
|
|
|
|
mail.header.fields.each { |field|
|
2013-09-25 13:58:25 +00:00
|
|
|
data[field.name.to_s.downcase.to_sym] = Encode.conv( 'utf8', field.to_s )
|
2012-05-05 09:24:05 +00:00
|
|
|
}
|
|
|
|
|
2013-10-31 22:56:08 +00:00
|
|
|
# get sender
|
|
|
|
from = nil
|
|
|
|
['from', 'reply-to', 'return-path'].each { |item|
|
|
|
|
if !from
|
|
|
|
if mail[ item.to_sym ]
|
|
|
|
from = mail[ item.to_sym ].value
|
|
|
|
end
|
|
|
|
end
|
|
|
|
}
|
|
|
|
|
2012-05-05 09:24:05 +00:00
|
|
|
# set extra headers
|
2013-12-01 17:51:35 +00:00
|
|
|
begin
|
|
|
|
data[:from_email] = Mail::Address.new( from ).address
|
|
|
|
data[:from_local] = Mail::Address.new( from ).local
|
|
|
|
data[:from_domain] = Mail::Address.new( from ).domain
|
|
|
|
data[:from_display_name] = Mail::Address.new( from ).display_name ||
|
|
|
|
( Mail::Address.new( from ).comments && Mail::Address.new( from ).comments[0] )
|
|
|
|
rescue
|
|
|
|
data[:from_email] = from
|
|
|
|
data[:from_local] = from
|
|
|
|
data[:from_domain] = from
|
|
|
|
end
|
2012-05-04 11:33:05 +00:00
|
|
|
|
2012-05-05 09:24:05 +00:00
|
|
|
# do extra decoding because we needed to use field.value
|
|
|
|
data[:from_display_name] = Mail::Field.new( 'X-From', data[:from_display_name] ).to_s
|
|
|
|
|
|
|
|
# compat headers
|
|
|
|
data[:message_id] = data['message-id'.to_sym]
|
2012-05-04 11:33:05 +00:00
|
|
|
|
|
|
|
# body
|
2013-06-12 15:59:58 +00:00
|
|
|
# plain_part = mail.multipart? ? (mail.text_part ? mail.text_part.body.decoded : nil) : mail.body.decoded
|
|
|
|
# html_part = message.html_part ? message.html_part.body.decoded : nil
|
2012-05-07 22:37:07 +00:00
|
|
|
data[:attachments] = []
|
2012-10-05 06:42:12 +00:00
|
|
|
|
2012-07-02 18:52:27 +00:00
|
|
|
# multi part email
|
2012-05-04 19:30:22 +00:00
|
|
|
if mail.multipart?
|
2012-10-05 06:42:12 +00:00
|
|
|
|
2012-07-02 18:52:27 +00:00
|
|
|
# text attachment/body exists
|
2012-06-15 11:10:23 +00:00
|
|
|
if mail.text_part
|
2012-10-04 06:54:21 +00:00
|
|
|
data[:body] = mail.text_part.body.decoded
|
2012-12-05 01:27:56 +00:00
|
|
|
data[:body] = Encode.conv( mail.text_part.charset, data[:body] )
|
2012-10-05 06:42:12 +00:00
|
|
|
|
2013-06-12 15:59:58 +00:00
|
|
|
# html attachment/body may exists and will be converted to text
|
2012-06-15 11:10:23 +00:00
|
|
|
else
|
|
|
|
filename = '-no name-'
|
|
|
|
if mail.html_part.body
|
|
|
|
filename = 'html-email'
|
2012-10-04 06:54:21 +00:00
|
|
|
data[:body] = mail.html_part.body.to_s
|
2012-12-05 01:27:56 +00:00
|
|
|
data[:body] = Encode.conv( mail.html_part.charset.to_s, data[:body] )
|
2012-10-04 06:54:21 +00:00
|
|
|
data[:body] = html2ascii( data[:body] )
|
2012-06-15 11:10:23 +00:00
|
|
|
|
2013-06-12 15:59:58 +00:00
|
|
|
# any other attachments
|
2012-06-15 11:10:23 +00:00
|
|
|
else
|
2012-10-04 06:54:21 +00:00
|
|
|
data[:body] = 'no visible content'
|
2012-06-15 11:10:23 +00:00
|
|
|
end
|
2012-07-02 18:52:27 +00:00
|
|
|
end
|
2012-06-15 11:10:23 +00:00
|
|
|
|
2012-07-02 18:52:27 +00:00
|
|
|
# add html attachment/body as real attachment
|
|
|
|
if mail.html_part
|
|
|
|
filename = 'message.html'
|
|
|
|
headers_store = {
|
|
|
|
'content-alternative' => true,
|
|
|
|
}
|
2012-06-15 11:10:23 +00:00
|
|
|
if mail.mime_type
|
|
|
|
headers_store['Mime-Type'] = mail.html_part.mime_type
|
|
|
|
end
|
|
|
|
if mail.charset
|
|
|
|
headers_store['Charset'] = mail.html_part.charset
|
|
|
|
end
|
|
|
|
attachment = {
|
2012-06-15 11:27:52 +00:00
|
|
|
:data => mail.html_part.body.to_s,
|
2012-06-15 11:10:23 +00:00
|
|
|
:filename => mail.html_part.filename || filename,
|
2012-10-04 06:54:21 +00:00
|
|
|
:preferences => headers_store
|
2012-06-15 11:10:23 +00:00
|
|
|
}
|
|
|
|
data[:attachments].push attachment
|
|
|
|
end
|
2012-10-04 06:54:21 +00:00
|
|
|
|
2012-07-02 18:52:27 +00:00
|
|
|
# get attachments
|
2013-01-23 13:47:57 +00:00
|
|
|
if mail.parts
|
|
|
|
attachment_count_total = 0
|
|
|
|
mail.parts.each { |part|
|
|
|
|
attachment_count_total += 1
|
2013-07-22 08:13:38 +00:00
|
|
|
|
|
|
|
# protect process to work fine with spam emails, see test/fixtures/mail15.box
|
|
|
|
begin
|
|
|
|
if mail.text_part && mail.text_part == part
|
|
|
|
# ignore text/plain attachments - already shown in view
|
|
|
|
elsif mail.html_part && mail.html_part == part
|
|
|
|
# ignore text/html - html part, already shown in view
|
|
|
|
else
|
|
|
|
attachs = self._get_attachment( part, data[:attachments] )
|
|
|
|
data[:attachments].concat( attachs )
|
|
|
|
end
|
|
|
|
rescue
|
2013-01-23 13:47:57 +00:00
|
|
|
attachs = self._get_attachment( part, data[:attachments] )
|
|
|
|
data[:attachments].concat( attachs )
|
2012-07-02 18:52:27 +00:00
|
|
|
end
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2013-06-12 15:59:58 +00:00
|
|
|
# not multipart email
|
2012-05-04 19:30:22 +00:00
|
|
|
else
|
2012-05-07 22:37:07 +00:00
|
|
|
|
|
|
|
# text part
|
|
|
|
if !mail.mime_type || mail.mime_type.to_s == '' || mail.mime_type.to_s.downcase == 'text/plain'
|
2012-10-04 06:54:21 +00:00
|
|
|
data[:body] = mail.body.decoded
|
2012-12-05 01:27:56 +00:00
|
|
|
data[:body] = Encode.conv( mail.charset, data[:body] )
|
2012-05-07 22:37:07 +00:00
|
|
|
|
2013-06-12 15:59:58 +00:00
|
|
|
# html part
|
2012-07-02 18:52:27 +00:00
|
|
|
else
|
2012-05-07 22:37:07 +00:00
|
|
|
filename = '-no name-'
|
|
|
|
if mail.mime_type.to_s.downcase == 'text/html'
|
|
|
|
filename = 'html-email'
|
2012-10-04 06:54:21 +00:00
|
|
|
data[:body] = mail.body.decoded
|
2012-12-05 01:27:56 +00:00
|
|
|
data[:body] = Encode.conv( mail.charset, data[:body] )
|
2012-10-04 06:54:21 +00:00
|
|
|
data[:body] = html2ascii( data[:body] )
|
2012-05-07 22:37:07 +00:00
|
|
|
|
2013-06-12 15:59:58 +00:00
|
|
|
# any other attachments
|
2012-05-07 22:37:07 +00:00
|
|
|
else
|
2012-10-04 06:54:21 +00:00
|
|
|
data[:body] = 'no visible content'
|
2012-05-07 22:37:07 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
# add body as attachment
|
2012-07-02 18:52:27 +00:00
|
|
|
headers_store = {
|
|
|
|
'content-alternative' => true,
|
|
|
|
}
|
2012-05-07 22:37:07 +00:00
|
|
|
if mail.mime_type
|
|
|
|
headers_store['Mime-Type'] = mail.mime_type
|
|
|
|
end
|
|
|
|
if mail.charset
|
|
|
|
headers_store['Charset'] = mail.charset
|
|
|
|
end
|
|
|
|
attachment = {
|
|
|
|
:data => mail.body.decoded,
|
|
|
|
:filename => mail.filename || filename,
|
2012-10-24 18:23:04 +00:00
|
|
|
:preferences => headers_store
|
2012-05-07 22:37:07 +00:00
|
|
|
}
|
|
|
|
data[:attachments].push attachment
|
|
|
|
end
|
2012-05-04 19:30:22 +00:00
|
|
|
end
|
2012-05-04 11:33:05 +00:00
|
|
|
|
2012-07-02 18:52:27 +00:00
|
|
|
# strip not wanted chars
|
2012-10-04 06:54:21 +00:00
|
|
|
data[:body].gsub!( /\r\n/, "\n" )
|
|
|
|
data[:body].gsub!( /\r/, "\n" )
|
2012-07-02 18:52:27 +00:00
|
|
|
|
2012-05-04 11:33:05 +00:00
|
|
|
return data
|
|
|
|
end
|
|
|
|
|
2013-01-23 13:47:57 +00:00
|
|
|
def _get_attachment( file, attachments )
|
|
|
|
|
|
|
|
# check if sub parts are available
|
|
|
|
if !file.parts.empty?
|
|
|
|
a = []
|
|
|
|
file.parts.each {|p|
|
|
|
|
a.concat( self._get_attachment( p, attachments ) )
|
|
|
|
}
|
|
|
|
return a
|
|
|
|
end
|
|
|
|
|
|
|
|
# get file preferences
|
|
|
|
headers_store = {}
|
|
|
|
file.header.fields.each { |field|
|
|
|
|
headers_store[field.name.to_s] = field.value.to_s
|
|
|
|
}
|
|
|
|
|
|
|
|
# get filename from content-disposition
|
|
|
|
filename = nil
|
|
|
|
|
|
|
|
# workaround for: NoMethodError: undefined method `filename' for #<Mail::UnstructuredField:0x007ff109e80678>
|
|
|
|
begin
|
|
|
|
filename = file.header[:content_disposition].filename
|
|
|
|
rescue
|
|
|
|
result = file.header[:content_disposition].to_s.scan( /filename=("|)(.+?)("|);/i )
|
|
|
|
if result && result[0] && result[0][1]
|
|
|
|
filename = result[0][1]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# for some broken sm mail clients (X-MimeOLE: Produced By Microsoft Exchange V6.5)
|
|
|
|
if !filename
|
|
|
|
filename = file.header[:content_location].to_s
|
|
|
|
end
|
|
|
|
|
|
|
|
# generate file name
|
|
|
|
if !filename || filename.empty?
|
|
|
|
attachment_count = 0
|
|
|
|
(1..1000).each {|count|
|
|
|
|
filename_exists = false
|
|
|
|
filename = 'file-' + count.to_s
|
|
|
|
attachments.each {|attachment|
|
|
|
|
if attachment[:filename] == filename
|
|
|
|
filename_exists = true
|
|
|
|
end
|
|
|
|
}
|
|
|
|
break if filename_exists == false
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
# get mime type
|
|
|
|
if file.header[:content_type] && file.header[:content_type].string
|
|
|
|
headers_store['Mime-Type'] = file.header[:content_type].string
|
|
|
|
end
|
|
|
|
|
|
|
|
# get charset
|
|
|
|
if file.header && file.header.charset
|
|
|
|
headers_store['Charset'] = file.header.charset
|
|
|
|
end
|
|
|
|
|
|
|
|
# remove not needed header
|
|
|
|
headers_store.delete('Content-Transfer-Encoding')
|
|
|
|
headers_store.delete('Content-Disposition')
|
|
|
|
|
|
|
|
attach = {
|
|
|
|
:data => file.body.to_s,
|
|
|
|
:filename => filename,
|
|
|
|
:preferences => headers_store,
|
|
|
|
}
|
|
|
|
return [attach]
|
|
|
|
end
|
|
|
|
|
2012-05-04 11:33:05 +00:00
|
|
|
def process(channel, msg)
|
|
|
|
mail = parse( msg )
|
2012-04-13 16:42:25 +00:00
|
|
|
|
2012-10-05 06:42:12 +00:00
|
|
|
# run postmaster pre filter
|
|
|
|
filters = {
|
|
|
|
'0010' => Channel::Filter::Trusted,
|
|
|
|
'1000' => Channel::Filter::Database,
|
|
|
|
}
|
2012-10-04 07:09:27 +00:00
|
|
|
|
2012-10-05 06:42:12 +00:00
|
|
|
# filter( channel, mail )
|
|
|
|
filters.each {|prio, backend|
|
|
|
|
begin
|
|
|
|
backend.run( channel, mail )
|
|
|
|
rescue Exception => e
|
|
|
|
puts "can't run postmaster pre filter #{backend}"
|
|
|
|
puts e.inspect
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
}
|
2012-10-04 07:09:27 +00:00
|
|
|
|
2012-05-06 20:48:23 +00:00
|
|
|
# check ignore header
|
|
|
|
return true if mail[ 'x-zammad-ignore'.to_sym ] == 'true' || mail[ 'x-zammad-ignore'.to_sym ] == true
|
|
|
|
|
|
|
|
ticket = nil
|
|
|
|
article = nil
|
|
|
|
user = nil
|
|
|
|
|
2012-04-13 16:42:25 +00:00
|
|
|
# use transaction
|
|
|
|
ActiveRecord::Base.transaction do
|
|
|
|
|
2013-04-20 10:14:27 +00:00
|
|
|
# reset current_user
|
|
|
|
UserInfo.current_user_id = 1
|
|
|
|
|
|
|
|
|
2012-05-06 20:48:23 +00:00
|
|
|
if mail[ 'x-zammad-customer-login'.to_sym ]
|
|
|
|
user = User.where( :login => mail[ 'x-zammad-customer-login'.to_sym ] ).first
|
|
|
|
end
|
|
|
|
if !user
|
|
|
|
user = User.where( :email => mail[ 'x-zammad-customer-email'.to_sym ] || mail[:from_email] ).first
|
|
|
|
end
|
|
|
|
if !user
|
2012-04-13 16:42:25 +00:00
|
|
|
puts 'create user...'
|
|
|
|
roles = Role.where( :name => 'Customer' )
|
|
|
|
user = User.create(
|
2012-05-06 20:48:23 +00:00
|
|
|
:login => mail[ 'x-zammad-customer-login'.to_sym ] || mail[ 'x-zammad-customer-email'.to_sym ] || mail[:from_email],
|
|
|
|
:firstname => mail[ 'x-zammad-customer-firstname'.to_sym ] || mail[:from_display_name],
|
|
|
|
:lastname => mail[ 'x-zammad-customer-lastname'.to_sym ],
|
|
|
|
:email => mail[ 'x-zammad-customer-email'.to_sym ] || mail[:from_email],
|
2012-04-13 16:42:25 +00:00
|
|
|
:password => '',
|
|
|
|
:active => true,
|
|
|
|
:roles => roles,
|
2012-11-07 12:57:13 +00:00
|
|
|
:updated_by_id => 1,
|
|
|
|
:created_by_id => 1,
|
2012-04-13 16:42:25 +00:00
|
|
|
)
|
|
|
|
end
|
2012-10-04 07:09:27 +00:00
|
|
|
|
2012-04-13 16:42:25 +00:00
|
|
|
# set current user
|
|
|
|
UserInfo.current_user_id = user.id
|
2012-10-04 07:09:27 +00:00
|
|
|
|
2012-04-13 16:42:25 +00:00
|
|
|
# get ticket# from subject
|
2013-08-15 22:16:38 +00:00
|
|
|
ticket = Ticket::Number.check( mail[:subject] )
|
2012-05-04 11:33:05 +00:00
|
|
|
|
2012-04-13 16:42:25 +00:00
|
|
|
# set ticket state to open if not new
|
|
|
|
if ticket
|
|
|
|
ticket_state = Ticket::State.find( ticket.ticket_state_id )
|
2013-01-01 23:35:46 +00:00
|
|
|
ticket_state_type = Ticket::StateType.find( ticket_state.state_type_id )
|
2012-08-28 05:21:45 +00:00
|
|
|
|
|
|
|
# if tickte is merged, find linked ticket
|
|
|
|
if ticket_state_type.name == 'merged'
|
|
|
|
|
|
|
|
end
|
|
|
|
|
2012-04-13 16:42:25 +00:00
|
|
|
if ticket_state_type.name != 'new'
|
|
|
|
ticket.ticket_state = Ticket::State.where( :name => 'open' ).first
|
|
|
|
ticket.save
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# create new ticket
|
2012-05-06 20:48:23 +00:00
|
|
|
if !ticket
|
|
|
|
|
|
|
|
# set attributes
|
|
|
|
ticket_attributes = {
|
2012-05-04 19:30:22 +00:00
|
|
|
:group_id => channel[:group_id] || 1,
|
2012-04-13 16:42:25 +00:00
|
|
|
:customer_id => user.id,
|
2012-08-28 05:21:45 +00:00
|
|
|
:title => mail[:subject] || '',
|
2012-05-06 20:48:23 +00:00
|
|
|
:ticket_state_id => Ticket::State.where( :name => 'new' ).first.id,
|
|
|
|
:ticket_priority_id => Ticket::Priority.where( :name => '2 normal' ).first.id,
|
|
|
|
}
|
|
|
|
|
|
|
|
# x-headers lookup
|
|
|
|
map = [
|
|
|
|
[ 'x-zammad-group', Group, 'group_id', 'name' ],
|
|
|
|
[ 'x-zammad-state', Ticket::State, 'ticket_state_id', 'name' ],
|
|
|
|
[ 'x-zammad-priority', Ticket::Priority, 'ticket_priority_id', 'name' ],
|
|
|
|
[ 'x-zammad-owner', User, 'owner_id', 'login' ],
|
|
|
|
]
|
2012-10-05 06:42:12 +00:00
|
|
|
object_lookup( ticket_attributes, map, mail )
|
2012-05-06 20:48:23 +00:00
|
|
|
|
|
|
|
# create ticket
|
|
|
|
ticket = Ticket.create( ticket_attributes )
|
2012-04-13 16:42:25 +00:00
|
|
|
end
|
2012-10-04 06:54:21 +00:00
|
|
|
|
2012-04-13 16:42:25 +00:00
|
|
|
# import mail
|
2012-10-04 06:54:21 +00:00
|
|
|
|
2012-05-06 20:48:23 +00:00
|
|
|
# set attributes
|
|
|
|
internal = false
|
2013-08-01 09:11:55 +00:00
|
|
|
if mail[ 'X-Zammad-Article-Visibility'.to_sym ] && mail[ 'X-Zammad-Article-Visibility'.to_sym ] == 'internal'
|
2012-05-06 20:48:23 +00:00
|
|
|
internal = true
|
|
|
|
end
|
|
|
|
article_attributes = {
|
2013-06-12 15:59:58 +00:00
|
|
|
:ticket_id => ticket.id,
|
2012-05-06 20:48:23 +00:00
|
|
|
:ticket_article_type_id => Ticket::Article::Type.where( :name => 'email' ).first.id,
|
|
|
|
:ticket_article_sender_id => Ticket::Article::Sender.where( :name => 'Customer' ).first.id,
|
2013-06-12 15:59:58 +00:00
|
|
|
:body => mail[:body],
|
2012-05-04 11:33:05 +00:00
|
|
|
:from => mail[:from],
|
|
|
|
:to => mail[:to],
|
|
|
|
:cc => mail[:cc],
|
|
|
|
:subject => mail[:subject],
|
|
|
|
:message_id => mail[:message_id],
|
2012-05-06 20:48:23 +00:00
|
|
|
:internal => internal,
|
|
|
|
}
|
|
|
|
|
|
|
|
# x-headers lookup
|
|
|
|
map = [
|
|
|
|
[ 'x-zammad-article-type', Ticket::Article::Type, 'ticket_article_type_id', 'name' ],
|
|
|
|
[ 'x-zammad-article-sender', Ticket::Article::Sender, 'ticket_article_sender_id', 'name' ],
|
|
|
|
]
|
2012-10-05 06:42:12 +00:00
|
|
|
object_lookup( article_attributes, map, mail )
|
2012-05-06 20:48:23 +00:00
|
|
|
|
|
|
|
# create article
|
|
|
|
article = Ticket::Article.create(article_attributes)
|
2012-04-13 16:42:25 +00:00
|
|
|
|
|
|
|
# store mail plain
|
|
|
|
Store.add(
|
|
|
|
:object => 'Ticket::Article::Mail',
|
|
|
|
:o_id => article.id,
|
|
|
|
:data => msg,
|
2012-07-02 12:47:09 +00:00
|
|
|
:filename => "ticket-#{ticket.number}-#{article.id}.eml",
|
2012-04-13 16:42:25 +00:00
|
|
|
:preferences => {}
|
|
|
|
)
|
|
|
|
|
|
|
|
# store attachments
|
2012-05-04 11:33:05 +00:00
|
|
|
if mail[:attachments]
|
|
|
|
mail[:attachments].each do |attachment|
|
2012-04-13 16:42:25 +00:00
|
|
|
Store.add(
|
|
|
|
:object => 'Ticket::Article',
|
|
|
|
:o_id => article.id,
|
2012-05-04 11:33:05 +00:00
|
|
|
:data => attachment[:data],
|
|
|
|
:filename => attachment[:filename],
|
|
|
|
:preferences => attachment[:preferences]
|
2012-04-13 16:42:25 +00:00
|
|
|
)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2012-11-07 16:38:09 +00:00
|
|
|
# execute ticket events
|
2012-11-18 10:56:48 +00:00
|
|
|
Observer::Ticket::Notification.transaction
|
2012-05-06 20:48:23 +00:00
|
|
|
|
2012-10-05 06:42:12 +00:00
|
|
|
# run postmaster post filter
|
|
|
|
filters = {
|
2013-06-12 15:59:58 +00:00
|
|
|
# '0010' => Channel::Filter::Trusted,
|
2012-10-05 06:42:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
# filter( channel, mail )
|
|
|
|
filters.each {|prio, backend|
|
|
|
|
begin
|
|
|
|
backend.run( channel, mail, ticket, article, user )
|
|
|
|
rescue Exception => e
|
|
|
|
puts "can't run postmaster post filter #{backend}"
|
|
|
|
puts e.inspect
|
|
|
|
end
|
|
|
|
}
|
|
|
|
|
2012-05-06 20:48:23 +00:00
|
|
|
# return new objects
|
|
|
|
return ticket, article, user
|
2012-04-13 16:42:25 +00:00
|
|
|
end
|
2012-11-07 16:38:09 +00:00
|
|
|
|
2012-10-05 06:42:12 +00:00
|
|
|
def object_lookup( attributes, map, mail )
|
|
|
|
map.each { |item|
|
|
|
|
if mail[ item[0].to_sym ]
|
2012-11-07 16:38:09 +00:00
|
|
|
new_object = item[1].where( "lower(#{item[3]}) = ?", mail[ item[0].to_sym ].downcase ).first
|
|
|
|
if new_object
|
|
|
|
attributes[ item[2].to_sym ] = new_object.id
|
2012-10-05 06:42:12 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2012-05-07 22:37:07 +00:00
|
|
|
def html2ascii(string)
|
|
|
|
|
|
|
|
# find <a href=....> and replace it with [x]
|
|
|
|
link_list = ''
|
|
|
|
counter = 0
|
|
|
|
string.gsub!( /<a\s.*?href=("|')(.+?)("|').*?>/ix ) { |item|
|
|
|
|
link = $2
|
|
|
|
counter = counter + 1
|
|
|
|
link_list += "[#{counter}] #{link}\n"
|
|
|
|
"[#{counter}]"
|
|
|
|
}
|
|
|
|
|
|
|
|
# remove empty lines
|
|
|
|
string.gsub!( /^\s*/m, '' )
|
|
|
|
|
|
|
|
# fix some bad stuff from opera and others
|
2012-06-15 11:10:23 +00:00
|
|
|
string.gsub!( /(\n\r|\r\r\n|\r\n)/, "\n" )
|
|
|
|
|
|
|
|
# strip all other tags
|
|
|
|
string.gsub!( /\<(br|br\/|br\s\/)\>/, "\n" )
|
2012-05-07 22:37:07 +00:00
|
|
|
|
|
|
|
# strip all other tags
|
2012-06-15 11:10:23 +00:00
|
|
|
string.gsub!( /\<.+?\>/, '' )
|
2012-05-07 22:37:07 +00:00
|
|
|
|
2014-02-12 22:18:27 +00:00
|
|
|
# strip all & < > "
|
|
|
|
string.gsub!( '&', '&' )
|
|
|
|
string.gsub!( '<', '<' )
|
|
|
|
string.gsub!( '>', '>' )
|
|
|
|
string.gsub!( '"', '"' )
|
|
|
|
|
2012-05-07 22:37:07 +00:00
|
|
|
# encode html entities like "–"
|
|
|
|
string.gsub!( /(&\#(\d+);?)/x ) { |item|
|
|
|
|
$2.chr
|
|
|
|
}
|
|
|
|
|
|
|
|
# encode html entities like "d;"
|
|
|
|
string.gsub!( /(&\#[xX]([0-9a-fA-F]+);?)/x ) { |item|
|
|
|
|
chr_orig = $1
|
|
|
|
hex = $2.hex
|
|
|
|
if hex
|
|
|
|
chr = hex.chr
|
|
|
|
if chr
|
2014-02-12 22:18:27 +00:00
|
|
|
chr_orig = chr
|
2012-05-07 22:37:07 +00:00
|
|
|
else
|
|
|
|
chr_orig
|
|
|
|
end
|
|
|
|
else
|
|
|
|
chr_orig
|
|
|
|
end
|
2014-02-12 22:18:27 +00:00
|
|
|
|
|
|
|
# check valid encoding
|
|
|
|
begin
|
|
|
|
if !chr_orig.encode('UTF-8').valid_encoding?
|
|
|
|
chr_orig = '?'
|
|
|
|
end
|
|
|
|
rescue
|
|
|
|
chr_orig = '?'
|
|
|
|
end
|
|
|
|
chr_orig
|
2012-05-07 22:37:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
# remove empty lines
|
|
|
|
string.gsub!( /^\s*\n\s*\n/m, "\n" )
|
|
|
|
|
|
|
|
# add extracted links
|
|
|
|
if link_list
|
|
|
|
string += "\n\n" + link_list
|
|
|
|
end
|
|
|
|
|
|
|
|
return string
|
|
|
|
end
|
2012-07-25 13:58:08 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
# workaround to parse subjects with 2 different encodings correctly (e. g. quoted-printable see test/fixtures/mail9.box)
|
|
|
|
module Mail
|
|
|
|
module Encodings
|
|
|
|
def Encodings.value_decode(str)
|
|
|
|
# Optimization: If there's no encoded-words in the string, just return it
|
|
|
|
return str unless str.index("=?")
|
|
|
|
|
|
|
|
str = str.gsub(/\?=(\s*)=\?/, '?==?') # Remove whitespaces between 'encoded-word's
|
|
|
|
|
|
|
|
# Split on white-space boundaries with capture, so we capture the white-space as well
|
|
|
|
str.split(/([ \t])/).map do |text|
|
|
|
|
if text.index('=?') .nil?
|
|
|
|
text
|
|
|
|
else
|
|
|
|
# Join QP encoded-words that are adjacent to avoid decoding partial chars
|
2013-06-12 15:59:58 +00:00
|
|
|
# text.gsub!(/\?\=\=\?.+?\?[Qq]\?/m, '') if text =~ /\?==\?/
|
2012-07-25 13:58:08 +00:00
|
|
|
|
|
|
|
# Search for occurences of quoted strings or plain strings
|
|
|
|
text.scan(/( # Group around entire regex to include it in matches
|
2013-06-13 07:01:06 +00:00
|
|
|
\=\?[^?]+\?([QB])\?[^?]+?\?\= # Quoted String with subgroup for encoding method
|
|
|
|
| # or
|
|
|
|
.+?(?=\=\?|$) # Plain String
|
2013-06-12 15:59:58 +00:00
|
|
|
)/xmi).map do |matches|
|
2012-07-25 13:58:08 +00:00
|
|
|
string, method = *matches
|
|
|
|
if method == 'b' || method == 'B'
|
|
|
|
b_value_decode(string)
|
|
|
|
elsif method == 'q' || method == 'Q'
|
|
|
|
q_value_decode(string)
|
|
|
|
else
|
|
|
|
string
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end.join("")
|
|
|
|
end
|
|
|
|
end
|
2013-06-12 15:59:58 +00:00
|
|
|
end
|