aboutsummaryrefslogtreecommitdiffstats
path: root/src/plugins/lua/maps_stats.lua
blob: d4188103a4955d14fe985ded21251a252f814499 (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
120
121
122
123
124
125
126
127
128
129
130
131
132
133
--[[
Copyright (c) 2022, Vsevolod Stakhov <vsevolod@rspamd.com>

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

    http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
]]--

if confighelp then
  rspamd_config:add_example(nil, 'maps_stats',
      "Stores maps statistics in Redis", [[
maps_stats {
  # one iteration step per 2 minutes
  interval = 2m;
  # how many elements to store in Redis
  count = 1k;
  # common prefix for elements
  prefix = 'rm_';
}
]])
end

local redis_params
local lua_util = require "lua_util"
local rspamd_logger = require "rspamd_logger"
local lua_redis = require "lua_redis"
local N = "maps_stats"

local settings = {
  interval = 120, -- one iteration step per 2 minutes
  count = 1000, -- how many elements to store in Redis
  prefix = 'rm_', -- common prefix for elements
}

local function process_map(map, ev_base, _)
  if map:get_nelts() > 0 and map:get_uri() ~= 'static' then
    local key = settings.prefix .. map:get_uri()

    local function redis_zrange_cb(err, data)
      if err then
        rspamd_logger.errx(rspamd_config, 'cannot delete extra elements in %s: %s',
            key, err)
      elseif data then
        rspamd_logger.infox(rspamd_config, 'cleared %s elements from %s',
            data, key)
      end
    end
    local function redis_card_cb(err, data)
      if err then
        rspamd_logger.errx(rspamd_config, 'cannot get number of elements in %s: %s',
            key, err)
      elseif data then
        if settings.count > 0 and tonumber(data) > settings.count then
          lua_redis.rspamd_redis_make_request_taskless(ev_base,
              rspamd_config,
              redis_params, -- connect params
              key, -- hash key
              true, -- is write
              redis_zrange_cb, --callback
              'ZREMRANGEBYRANK', -- command
              { key, '0', tostring(-(settings.count) - 1) } -- arguments
          )
        end
      end
    end
    local ret, conn, _ = lua_redis.rspamd_redis_make_request_taskless(ev_base,
        rspamd_config,
        redis_params, -- connect params
        key, -- hash key
        true, -- is write
        redis_card_cb, --callback
        'ZCARD', -- command
        { key } -- arguments
    )

    if ret and conn then
      local stats = map:get_stats(true)
      for k, s in pairs(stats) do
        if s > 0 then
          conn:add_cmd('ZINCRBY', { key, tostring(s), k })
        end
      end
    end
  end
end

if not lua_util.check_experimental(N) then
  return
end

local opts = rspamd_config:get_all_opt(N)

if opts then
  for k, v in pairs(opts) do
    settings[k] = v
  end
end

redis_params = lua_redis.parse_redis_server(N, opts)
-- XXX, this is a poor approach as not all maps are defined here...
local tmaps = rspamd_config:get_maps()
for _, m in ipairs(tmaps) do
  if m:get_uri() ~= 'static' then
    lua_redis.register_prefix(settings.prefix .. m:get_uri(), N,
        'Maps stats data', {
          type = 'zlist',
          persistent = true,
        })
  end
end

if redis_params then
  rspamd_config:add_on_load(function(_, ev_base, worker)
    local maps = rspamd_config:get_maps()

    for _, m in ipairs(maps) do
      rspamd_config:add_periodic(ev_base,
          settings['interval'],
          function()
            process_map(m, ev_base, worker)
            return true
          end, true)
    end
  end)
end