summaryrefslogtreecommitdiffstats
path: root/src/plugins/lua/forged_recipients.lua
blob: d082b386d343929906329b5b64ed2ab6e043d6dd (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
-- Plugin for comparing smtp dialog recipients and sender with recipients and sender
-- in mime headers

local symbol_rcpt = 'FORGED_RECIPIENTS'
local symbol_sender = 'FORGED_SENDER'

function check_forged_headers(task)
	local msg = task:get_message()
	local smtp_rcpt = task:get_recipients()
	local res = false
	
	if smtp_rcpt then
		local mime_rcpt = msg:get_header('To')
		local mime_cc = msg:get_header('Cc')
		local count = 0
		if mime_rcpt then
			count = table.maxn(mime_rcpt)
		end
		if mime_cc then
			count = count + table.maxn(mime_cc)
		end
		-- Check recipients count
		if count < table.maxn(smtp_rcpt) then
			task:insert_result(symbol_rcpt, 1)
		else
			-- Find pair for each smtp recipient recipient in To or Cc headers
			for _,sr in ipairs(smtp_rcpt) do
				if sr:sub(1,1) == '<' then
					-- Trim brackets
					sr = string.sub(sr, 2, -2)
				end
				if mime_rcpt then
					for _,mr in ipairs(mime_rcpt) do
						local i = string.find(mr, '<', 1, true)
						if i then
						    local j = string.find(mr, '>', i, true)
						    if j then
							mr = string.sub(mr, i+1, j-1)
						    end
						end

						if string.lower(mr) == string.lower(sr) then
							res = true
							break
						end
					end
				end
				if mime_cc then
					for _,mr in ipairs(mime_cc) do
						local i = string.find(mr, '<', 1, true)
						if i then
						    local j = string.find(mr, '>', i, true)
						    if j then
							mr = string.sub(mr, i+1, j-1)
						    end
						end
						if string.lower(mr) == string.lower(sr) then
							res = true
							break
						end
					end
				end

				if not res then
					task:insert_result(symbol_rcpt, 1)
					break
				end
			end
		end
	end
	-- Check sender
	local smtp_from = task:get_from()
	if smtp_form then
		local mime_from = msg:get_header('From')
		local i = string.find(mime_from[0], '<', 1, true)
		if i then
		    local j = string.find(mime_from[0], '>', i, true)
		    if j then
			mime_from[0] = string.sub(mime_from[0], i+1, j-1)
		    end
		end
		if not mime_from or not (string.lower(mime_from[0]) == string.lower(smtp_from)) then
			task:insert_result(symbol_sender, 1)
		end
	end
end

-- Registration
if type(rspamd_config.get_api_version) ~= 'nil' then
	if rspamd_config:get_api_version() >= 1 then
		rspamd_config:register_module_option('forged_recipients', 'symbol_rcpt', 'string')
		rspamd_config:register_module_option('forged_recipients', 'symbol_sender', 'string')
	end
end

-- Configuration
local opts =  rspamd_config:get_all_opt('forged_recipients')
if opts then
	if opts['symbol_rcpt'] or opts['symbol_sender'] then
		if opts['symbol_rcpt'] then
			symbol_rcpt = opts['symbol_rcpt']
			if type(rspamd_config.get_api_version) ~= 'nil' then
				rspamd_config:register_virtual_symbol(symbol_rcpt, 1.0, 'check_forged_headers')
			end
		end
		if opts['symbol_sender'] then
			symbol_sender = opts['symbol_sender']
			if type(rspamd_config.get_api_version) ~= 'nil' then
				rspamd_config:register_virtual_symbol(symbol_sender, 1.0)
			end
		end
		if type(rspamd_config.get_api_version) ~= 'nil' then
			rspamd_config:register_callback_symbol('FORGED_RECIPIENTS', 1.0, 'check_forged_headers')
		else
			rspamd_config:register_symbol('FORGED_RECIPIENTS', 1.0, 'check_forged_headers')
		end
		
	end
end