filter/init.lua

149 lines
4.0 KiB
Lua

--[[
Copyright 2017 Auke Kok <sofar@foo-projects.org>
Permission is hereby granted, free of charge, to any person obtaining
a copy of this software and associated documentation files (the
"Software"), to deal in the Software without restriction, including
without limitation the rights to use, copy, modify, merge, publish,
distribute, sublicense, and/or sell copies of the Software, and to
permit persons to whom the Software is furnished to do so, subject
to the following conditions:
The above copyright notice and this permission notice shall be included
in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
]]--
filter = {}
local words = {}
local muted = {}
local violations = {}
local s = minetest.get_mod_storage()
function filter.init()
local sw = s:get_string("words")
if sw and sw ~= "" then
words = minetest.parse_json(sw)
end
if #words == 0 then
filter.import_file(minetest.get_modpath("filter") .. "/words.txt")
end
end
function filter.import_file(filepath)
local file = io.open(filepath, "r")
if file then
for line in file:lines() do
line = line:trim()
if line ~= "" then
words[#words + 1] = line:trim()
end
end
return true
else
return false
end
end
function filter.check_message(name, message)
for _, w in ipairs(words) do
if string.find(message:lower(), "%f[%a]" .. w .. "%f[%A]") then
return false
end
end
return true
end
function filter.on_violation(name, message)
violations[name] = (violations[name] or 0) + 1
local resolution
if violations[name] >= 3 then
resolution = "kicked"
minetest.kick_player(name, "Please mind your language!")
else
resolution = "muted"
local privs = minetest.get_player_privs(name)
privs.shout = nil
minetest.set_player_privs(name, privs)
minetest.chat_send_player(name, "Watch your language! You have been temporarily muted")
muted[name] = true
minetest.after(60, function()
muted[name] = nil
minetest.chat_send_player(name, "Chat privilege reinstated. Please do not abuse chat.")
privs.shout = true
minetest.set_player_privs(name, privs)
end)
end
minetest.log("action", "VIOLATION (" .. resolution .. "): <" .. name .. "> ".. message)
end
table.insert(minetest.registered_on_chat_messages, 1, function(name, message)
local privs = minetest.get_player_privs(name)
if not privs.shout and muted[name] then
minetest.chat_send_player(name, "You are temporarily muted.")
return true
end
if not filter.check_message(name, message) then
filter.on_violation(name, message)
return true
end
end)
local function step()
for name, v in pairs(violations) do
violations[name] = math.floor(v * 0.5)
if violations[name] < 1 then
violations[name] = nil
end
end
minetest.after(2*60, step)
end
minetest.after(2*60, step)
minetest.register_chatcommand("filter", {
params = "filter server",
description = "manage swear word filter",
privs = {server = true},
func = function(name, param)
local cmd, val = param:match("(%w+) (.+)")
if param == "list" then
return true, #words .. " words: " .. table.concat(words, ", ")
elseif cmd == "add" then
table.insert(words, val)
s:set_string("words", minetest.write_json(words))
return true, "Added \"" .. val .. "\"."
elseif cmd == "remove" then
for i, w in ipairs(words) do
if w == val then
table.remove(words, i)
s:set_string("words", minetest.write_json(words))
return true, "Removed \"" .. val .. "\"."
end
end
return true, "\"" .. val .. "\" not found in list."
else
return true, "I know " .. #words .. " words.\nUsage: /filter <add|remove|list> [<word>]"
end
end,
})
filter.init()