|
1 | | -require 'timeout' |
2 | | -require 'socket' |
3 | | -require 'json' |
4 | | - |
5 | 1 | module Postal |
6 | 2 | class MessageInspection |
7 | 3 |
|
8 | | - SPAM_EXCLUSIONS = { |
9 | | - :outgoing => ['NO_RECEIVED', 'NO_RELAYS', 'ALL_TRUSTED', 'FREEMAIL_FORGED_REPLYTO', 'RDNS_DYNAMIC', 'CK_HELO_GENERIC', /^SPF\_/, /^HELO\_/, /DKIM_/, /^RCVD_IN_/], |
10 | | - :incoming => [] |
11 | | - } |
| 4 | + attr_reader :message |
| 5 | + attr_reader :scope |
| 6 | + attr_reader :spam_checks |
| 7 | + attr_accessor :threat |
| 8 | + attr_accessor :threat_message |
12 | 9 |
|
13 | | - def initialize(message, scope = :incoming) |
| 10 | + def initialize(message, scope) |
14 | 11 | @message = message |
15 | 12 | @scope = scope |
16 | | - @threat = false |
17 | | - @spam_score = 0.0 |
18 | 13 | @spam_checks = [] |
19 | | - |
20 | | - if Postal.config.spamd.enabled? |
21 | | - scan_for_spam |
22 | | - end |
23 | | - |
24 | | - if Postal.config.clamav.enabled? |
25 | | - scan_for_threats |
26 | | - end |
| 14 | + @threat = false |
27 | 15 | end |
28 | 16 |
|
29 | 17 | def spam_score |
30 | | - @spam_score |
31 | | - end |
32 | | - |
33 | | - def spam_checks |
34 | | - @spam_checks |
35 | | - end |
| 18 | + return 0 if @spam_checks.empty? |
36 | 19 |
|
37 | | - def filtered_spam_checks |
38 | | - @filtered_spam_checks ||= @spam_checks.reject do |check| |
39 | | - SPAM_EXCLUSIONS[@scope].any? do |item| |
40 | | - item == check.code || (item.is_a?(Regexp) && item =~ check.code) |
41 | | - end |
42 | | - end |
43 | | - end |
44 | | - |
45 | | - def filtered_spam_score |
46 | | - filtered_spam_checks.inject(0.0) do |total, check| |
47 | | - total += check.score || 0.0 |
48 | | - end.round(2) |
| 20 | + @spam_checks.sum(&:score) |
49 | 21 | end |
50 | 22 |
|
51 | 23 | def threat? |
52 | | - @threat |
53 | | - end |
54 | | - |
55 | | - def threat_message |
56 | | - @threat_message |
| 24 | + @threat == true |
57 | 25 | end |
58 | 26 |
|
59 | | - private |
60 | | - |
61 | | - def scan_for_spam |
62 | | - data = nil |
63 | | - Timeout.timeout(15) do |
64 | | - tcp_socket = TCPSocket.new(Postal.config.spamd.host, Postal.config.spamd.port) |
65 | | - tcp_socket.write("REPORT SPAMC/1.2\r\n") |
66 | | - tcp_socket.write("Content-length: #{@message.bytesize}\r\n") |
67 | | - tcp_socket.write("\r\n") |
68 | | - tcp_socket.write(@message) |
69 | | - tcp_socket.close_write |
70 | | - data = tcp_socket.read |
71 | | - end |
72 | | - |
73 | | - spam_checks = [] |
74 | | - total = 0.0 |
75 | | - rules = data ? data.split(/^---(.*)\r?\n/).last.split(/\r?\n/) : [] |
76 | | - while line = rules.shift |
77 | | - if line =~ /\A([\- ]?[\d\.]+)\s+(\w+)\s+(.*)/ |
78 | | - total += $1.to_f |
79 | | - spam_checks << SPAMCheck.new($2, $1.to_f, $3) |
80 | | - else |
81 | | - spam_checks.last.description << " " + line.strip |
82 | | - end |
| 27 | + def scan |
| 28 | + MessageInspector.inspectors.each do |inspector| |
| 29 | + inspector.inspect_message(self) |
83 | 30 | end |
84 | | - |
85 | | - @spam_score = total.round(1) |
86 | | - @spam_checks = spam_checks |
87 | | - |
88 | | - rescue Timeout::Error |
89 | | - @spam_checks = [SPAMCheck.new("TIMEOUT", 0, "Timed out when scanning for spam")] |
90 | | - rescue => e |
91 | | - logger.error "Error talking to spamd: #{e.class} (#{e.message})" |
92 | | - logger.error e.backtrace[0,5] |
93 | | - @spam_checks = [SPAMCheck.new("ERROR", 0, "Error when scanning for spam")] |
94 | | - ensure |
95 | | - tcp_socket.close rescue nil |
96 | 31 | end |
97 | 32 |
|
98 | | - def scan_for_threats |
99 | | - @threat = false |
100 | | - |
101 | | - data = nil |
102 | | - Timeout.timeout(10) do |
103 | | - tcp_socket = TCPSocket.new(Postal.config.clamav.host, Postal.config.clamav.port) |
104 | | - tcp_socket.write("zINSTREAM\0") |
105 | | - tcp_socket.write([@message.bytesize].pack("N")) |
106 | | - tcp_socket.write(@message) |
107 | | - tcp_socket.write([0].pack("N")) |
108 | | - tcp_socket.close_write |
109 | | - data = tcp_socket.read |
110 | | - end |
111 | | - |
112 | | - if data && data =~ /\Astream\:\s+(.*?)[\s\0]+?/ |
113 | | - if $1.upcase == 'OK' |
114 | | - @threat = false |
115 | | - @threat_message = "No threats found" |
116 | | - else |
117 | | - @threat = true |
118 | | - @threat_message = $1 |
119 | | - end |
120 | | - else |
121 | | - @threat = false |
122 | | - @threat_message = "Could not scan message" |
| 33 | + class << self |
| 34 | + def scan(message, scope) |
| 35 | + inspection = new(message, scope) |
| 36 | + inspection.scan |
| 37 | + inspection |
123 | 38 | end |
124 | | - rescue Timeout::Error |
125 | | - @threat = false |
126 | | - @threat_message = "Timed out scanning for threats" |
127 | | - rescue => e |
128 | | - logger.error "Error talking to clamav: #{e.class} (#{e.message})" |
129 | | - logger.error e.backtrace[0,5] |
130 | | - @threat = false |
131 | | - @threat_message = "Error when scanning for threats" |
132 | | - ensure |
133 | | - tcp_socket.close rescue nil |
134 | | - end |
135 | | - |
136 | | - def logger |
137 | | - Postal.logger_for(:message_inspection) |
138 | 39 | end |
139 | 40 |
|
140 | 41 | end |
|
0 commit comments