-- This module contains helper functions to modify mime parts
--]]
+local logger = require "rspamd_logger"
local rspamd_util = require "rspamd_util"
local rspamd_text = require "rspamd_text"
local ucl = require "ucl"
local add = hdr_alterations.add or {}
local remove = hdr_alterations.remove or {}
+ local add_headers = {} -- For Milter reply
local hdr_flattened = {} -- For C API
local function flatten_add_header(hname, hdr)
+ if not add_headers[hname] then
+ add_headers[hname] = {}
+ end
if not hdr_flattened[hname] then
hdr_flattened[hname] = {add = {}}
end
local add_tbl = hdr_flattened[hname].add
if hdr.value then
+ table.insert(add_headers[hname], {
+ order = (tonumber(hdr.order) or -1),
+ value = hdr.value,
+ })
table.insert(add_tbl, {tonumber(hdr.order) or -1, hdr.value})
elseif type(hdr) == 'table' then
for _,v in ipairs(hdr) do
- table.insert(add_tbl, {-1, v})
+ flatten_add_header(hname, v)
end
elseif type(hdr) == 'string' then
+ table.insert(add_headers[hname], {
+ order = -1,
+ value = hdr,
+ })
table.insert(add_tbl, {-1, hdr})
+ else
+ logger.errx(task, 'invalid modification of header: %s', hdr)
end
end
if hdr_alterations.order then
end
task:set_milter_reply({
- add_headers = hdr_alterations.add,
+ add_headers = add_headers,
remove_headers = hdr_alterations.remove
})
hname = g_string_new (key);
hvalue = g_string_new ("");
- if (nhdr >= 1) {
- rspamd_milter_send_action (session,
- RSPAMD_MILTER_CHGHEADER,
- nhdr, hname, hvalue);
+ if (nhdr > 0) {
+ if (ar->len >= nhdr) {
+ rspamd_milter_send_action (session,
+ RSPAMD_MILTER_CHGHEADER,
+ nhdr, hname, hvalue);
+ priv->cur_hdr --;
+ }
}
- else if (nhdr == 0 && ar->len > 0) {
+ else if (nhdr == 0) {
/* We need to clear all headers */
for (i = ar->len; i > 0; i --) {
rspamd_milter_send_action (session,
RSPAMD_MILTER_CHGHEADER,
i, hname, hvalue);
+ priv->cur_hdr --;
}
}
else {
rspamd_milter_send_action (session,
RSPAMD_MILTER_CHGHEADER,
ar->len + nhdr + 1, hname, hvalue);
+ priv->cur_hdr --;
}
}
g_string_free (hname, TRUE);
g_string_free (hvalue, TRUE);
+
+ if (priv->cur_hdr < 0) {
+ msg_err_milter("negative header count after removing %s", key);
+ priv->cur_hdr = 0;
+ }
}
}
else {
/* Calculate negative offset */
- if (-idx <= priv->cur_hdr) {
+ if (idx == -1) {
+ rspamd_milter_send_action (session,
+ RSPAMD_MILTER_ADDHEADER,
+ hname, hvalue);
+ }
+ else if (-idx <= priv->cur_hdr) {
+ /*
+ * Note: We should account MTA's own "Received:" field
+ * which wasn't passed by Milter's header command.
+ */
rspamd_milter_send_action (session,
RSPAMD_MILTER_INSHEADER,
- priv->cur_hdr + idx + 1,
+ priv->cur_hdr + idx + 2,
hname, hvalue);
}
else {
hname, hvalue);
}
+ priv->cur_hdr ++;
+
g_string_free (hname, TRUE);
g_string_free (hvalue, TRUE);
}
local function add_header(name, value, stop_chars, order)
local hname = settings.routines[name].header
- if order then
- if not add[hname] then
- add[hname] = {
- order = order,
- value = lua_util.fold_header(task, hname, value, stop_chars)
- }
- else
- if not add[hname][1] then
- -- Convert to a table
- add[hname] = {
- [1] = add[hname]
- }
- end
-
- table.insert(add[hname], {
- order = order,
- value = lua_util.fold_header(task, hname, value, stop_chars)
- })
- end
- else
- if not add[hname] then
- add[hname] = lua_util.fold_header(task, hname, value, stop_chars)
- else
- if not add[hname][1] then
- -- Convert to a table
- add[hname] = {
- [1] = add[hname]
- }
- end
-
- if settings.default_headers_order then
- table.insert(add[hname], {
- order = settings.default_headers_order,
- value = lua_util.fold_header(task, hname, value, stop_chars)
- })
- else
- table.insert(add[hname],
- lua_util.fold_header(task, hname, value, stop_chars))
- end
-
- end
+ if not add[hname] then
+ add[hname] = {}
end
+ table.insert(add[hname], {
+ order = (order or settings.default_headers_order or -1),
+ value = lua_util.fold_header(task, hname, value, stop_chars)
+ })
end
routines['x-spamd-result'] = function()