849 lines
23 KiB
Lua
849 lines
23 KiB
Lua
http = require("socket.http")
|
||
https = require("ssl.https")
|
||
ltn12 = require "ltn12"
|
||
URL = require("socket.url")
|
||
feedparser = require ("feedparser")
|
||
json = (loadfile "./libs/JSON.lua")()
|
||
serpent = (loadfile "./libs/serpent.lua")()
|
||
mimetype = (loadfile "./libs/mimetype.lua")()
|
||
redis = (loadfile "./libs/redis.lua")()
|
||
|
||
http.TIMEOUT = 5
|
||
|
||
function get_receiver(msg)
|
||
if msg.to.type == 'user' then
|
||
return 'user#id'..msg.from.id
|
||
end
|
||
if msg.to.type == 'chat' then
|
||
return 'chat#id'..msg.to.id
|
||
end
|
||
if msg.to.type == 'encr_chat' then
|
||
return msg.to.print_name
|
||
end
|
||
end
|
||
|
||
function is_chat_msg( msg )
|
||
if msg.to.type == 'chat' then
|
||
return true
|
||
end
|
||
return false
|
||
end
|
||
|
||
function string.random(length)
|
||
local str = "";
|
||
for i = 1, length do
|
||
math.random(97, 122)
|
||
str = str..string.char(math.random(97, 122));
|
||
end
|
||
return str;
|
||
end
|
||
|
||
function string:split(sep)
|
||
local sep, fields = sep or ":", {}
|
||
local pattern = string.format("([^%s]+)", sep)
|
||
self:gsub(pattern, function(c) fields[#fields+1] = c end)
|
||
return fields
|
||
end
|
||
|
||
-- Removes spaces
|
||
function string.trim(s)
|
||
return s:gsub("^%s*(.-)%s*$", "%1")
|
||
end
|
||
|
||
function get_http_file_name(url, headers)
|
||
-- Eg: foo.var
|
||
local file_name = url:match("[^%w]+([%.%w]+)$")
|
||
-- Any delimited alphanumeric on the url
|
||
file_name = file_name or url:match("[^%w]+(%w+)[^%w]+$")
|
||
-- Random name, hope content-type works
|
||
file_name = file_name or str:random(5)
|
||
|
||
local content_type = headers["content-type"]
|
||
|
||
local extension = nil
|
||
if content_type then
|
||
extension = mimetype.get_mime_extension(content_type)
|
||
end
|
||
|
||
if extension then
|
||
file_name = file_name.."."..extension
|
||
end
|
||
|
||
local disposition = headers["content-disposition"]
|
||
if disposition then
|
||
-- attachment; filename=CodeCogsEqn.png
|
||
file_name = disposition:match('filename=([^;]+)') or file_name
|
||
file_name = string.gsub(file_name, "\"", "")
|
||
end
|
||
|
||
return file_name
|
||
end
|
||
|
||
-- Saves file to tmp/. If file_name isn't provided,
|
||
-- will get the text after the last "#" for filename
|
||
-- and content-type for extension
|
||
function download_to_file(url, file_name)
|
||
print("Download URL: "..url)
|
||
|
||
local respbody = {}
|
||
local options = {
|
||
url = url,
|
||
sink = ltn12.sink.table(respbody),
|
||
redirect = true
|
||
}
|
||
|
||
-- nil, code, headers, status
|
||
local response = nil
|
||
|
||
if url:starts('https') then
|
||
options.redirect = false
|
||
response = {https.request(options)}
|
||
else
|
||
response = {http.request(options)}
|
||
end
|
||
|
||
local code = response[2]
|
||
local headers = response[3]
|
||
local status = response[4]
|
||
|
||
if code ~= 200 then return nil end
|
||
|
||
file_name = file_name or get_http_file_name(url, headers)
|
||
|
||
local file_path = "tmp/"..file_name
|
||
print("Gespeichert in: "..file_path)
|
||
|
||
file = io.open(file_path, "w+")
|
||
file:write(table.concat(respbody))
|
||
file:close()
|
||
|
||
return file_path
|
||
end
|
||
|
||
function vardump(value)
|
||
print(serpent.block(value, {comment=false}))
|
||
end
|
||
|
||
-- taken from http://stackoverflow.com/a/11130774/3163199
|
||
function scandir(directory)
|
||
local i, t, popen = 0, {}, io.popen
|
||
for filename in popen('ls -a "'..directory..'"'):lines() do
|
||
i = i + 1
|
||
t[i] = filename
|
||
end
|
||
return t
|
||
end
|
||
|
||
-- http://www.lua.org/manual/5.2/manual.html#pdf-io.popen
|
||
function run_command(str)
|
||
local cmd = io.popen(str)
|
||
local result = cmd:read('*all')
|
||
cmd:close()
|
||
return result
|
||
end
|
||
|
||
function is_sudo(msg)
|
||
local var = false
|
||
-- Check if user id is in sudoers table
|
||
for v,user in pairs(sudo_users) do
|
||
if string.match(user, msg.from.id) then
|
||
var = true
|
||
end
|
||
end
|
||
return var
|
||
end
|
||
|
||
function can_use_bot(msg)
|
||
local var = false
|
||
-- Check users id in config
|
||
for v,user in pairs(_config.can_use_bot) do
|
||
if user == msg.from.id then
|
||
var = true
|
||
end
|
||
end
|
||
return var
|
||
end
|
||
|
||
-- Returns the name of the sender
|
||
function get_name(msg)
|
||
local name = msg.from.first_name
|
||
if name == nil then
|
||
name = msg.from.id
|
||
end
|
||
return name
|
||
end
|
||
|
||
-- Returns at table of lua files inside plugins
|
||
function plugins_names( )
|
||
local files = {}
|
||
for k, v in pairs(scandir("plugins")) do
|
||
-- Ends with .lua
|
||
if (v:match(".lua$")) then
|
||
table.insert(files, v)
|
||
end
|
||
end
|
||
return files
|
||
end
|
||
|
||
-- Function name explains what it does.
|
||
function file_exists(name)
|
||
local f = io.open(name,"r")
|
||
if f ~= nil then
|
||
io.close(f)
|
||
return true
|
||
else
|
||
return false
|
||
end
|
||
end
|
||
|
||
-- Save into file the data serialized for lua.
|
||
-- Set uglify true to minify the file.
|
||
function serialize_to_file(data, file, uglify)
|
||
file = io.open(file, 'w+')
|
||
local serialized
|
||
if not uglify then
|
||
serialized = serpent.block(data, {
|
||
comment = false,
|
||
name = '_'
|
||
})
|
||
else
|
||
serialized = serpent.dump(data)
|
||
end
|
||
file:write(serialized)
|
||
file:close()
|
||
end
|
||
|
||
-- Returns true if the string is empty
|
||
function string:isempty()
|
||
return self == nil or self == ''
|
||
end
|
||
|
||
-- Returns true if the string is blank
|
||
function string:isblank()
|
||
self = self:trim()
|
||
return self:isempty()
|
||
end
|
||
|
||
-- DEPRECATED!!!!!
|
||
function string.starts(String, Start)
|
||
print("string.starts(String, Start) is DEPRECATED use string:starts(text) instead")
|
||
return Start == string.sub(String,1,string.len(Start))
|
||
end
|
||
|
||
-- Returns true if String starts with Start
|
||
function string:starts(text)
|
||
return text == string.sub(self,1,string.len(text))
|
||
end
|
||
|
||
-- Send image to user and delete it when finished.
|
||
-- cb_function and cb_extra are optionals callback
|
||
function _send_photo(receiver, file_path, cb_function, cb_extra)
|
||
local cb_extra = {
|
||
file_path = file_path,
|
||
cb_function = cb_function,
|
||
cb_extra = cb_extra
|
||
}
|
||
-- Call to remove with optional callback
|
||
send_photo(receiver, file_path, cb_function, cb_extra)
|
||
end
|
||
|
||
-- Download the image and send to receiver, it will be deleted.
|
||
-- cb_function and cb_extra are optionals callback
|
||
function send_photo_from_url(receiver, url, cb_function, cb_extra, sendNotErrMsg)
|
||
-- If callback not provided
|
||
cb_function = cb_function or ok_cb
|
||
cb_extra = cb_extra or false
|
||
local file_path = download_to_file(url, false)
|
||
if not file_path then -- Error
|
||
if sendNotErrMsg then
|
||
return false
|
||
else
|
||
local text = 'Fehler beim Laden des Bildes'
|
||
send_msg(receiver, text, cb_function, cb_extra)
|
||
end
|
||
else
|
||
print("Datei Pfad: "..file_path)
|
||
_send_photo(receiver, file_path, cb_function, cb_extra)
|
||
return true
|
||
end
|
||
end
|
||
|
||
-- Same as send_photo_from_url but as callback function
|
||
function send_photo_from_url_callback(cb_extra, success, result, sendErrMsg)
|
||
local receiver = cb_extra.receiver
|
||
local url = cb_extra.url
|
||
|
||
local file_path = download_to_file(url, false)
|
||
if not file_path then -- Error
|
||
local text = 'Fehler beim Herunterladen des Bildes'
|
||
send_msg(receiver, text, ok_cb, false)
|
||
else
|
||
print("Datei Pfad: "..file_path)
|
||
_send_photo(receiver, file_path, ok_cb, false)
|
||
end
|
||
end
|
||
|
||
-- Same as above, but with send_as_document
|
||
function send_document_from_url_callback(cb_extra, success, result)
|
||
local receiver = cb_extra.receiver
|
||
local url = cb_extra.url
|
||
|
||
local file_path = download_to_file(url, false)
|
||
if not file_path then -- Error
|
||
local text = 'Fehler beim Herunterladen des Dokumentes'
|
||
send_msg(receiver, text, ok_cb, false)
|
||
else
|
||
print("File path: "..file_path)
|
||
_send_document(receiver, file_path, ok_cb, false)
|
||
end
|
||
end
|
||
|
||
-- Send multiple images asynchronous.
|
||
-- param urls must be a table.
|
||
function send_photos_from_url(receiver, urls)
|
||
local cb_extra = {
|
||
receiver = receiver,
|
||
urls = urls,
|
||
remove_path = nil
|
||
}
|
||
send_photos_from_url_callback(cb_extra)
|
||
end
|
||
|
||
-- Use send_photos_from_url.
|
||
-- This function might be difficult to understand.
|
||
function send_photos_from_url_callback(cb_extra, success, result)
|
||
-- cb_extra is a table containing receiver, urls and remove_path
|
||
local receiver = cb_extra.receiver
|
||
local urls = cb_extra.urls
|
||
local remove_path = cb_extra.remove_path
|
||
|
||
-- The previously image to remove
|
||
if remove_path ~= nil then
|
||
os.remove(remove_path)
|
||
print(remove_path.." gelöscht!")
|
||
end
|
||
|
||
-- Nil or empty, exit case (no more urls)
|
||
if urls == nil or #urls == 0 then
|
||
return false
|
||
end
|
||
|
||
-- Take the head and remove from urls table
|
||
local head = table.remove(urls, 1)
|
||
|
||
local file_path = download_to_file(head, false)
|
||
local cb_extra = {
|
||
receiver = receiver,
|
||
urls = urls,
|
||
remove_path = file_path
|
||
}
|
||
|
||
-- Send first and postpone the others as callback
|
||
send_photo(receiver, file_path, send_photos_from_url_callback, cb_extra)
|
||
end
|
||
|
||
-- Callback to remove a file
|
||
function rmtmp_cb(cb_extra, success, result)
|
||
local file_path = cb_extra.file_path
|
||
local cb_function = cb_extra.cb_function or ok_cb
|
||
local cb_extra = cb_extra.cb_extra
|
||
|
||
if file_path ~= nil then
|
||
os.remove(file_path)
|
||
print(file_path.." gelöscht!")
|
||
end
|
||
-- Finally call the callback
|
||
cb_function(cb_extra, success, result)
|
||
end
|
||
|
||
-- Send document to user and delete it when finished.
|
||
-- cb_function and cb_extra are optionals callback
|
||
function _send_document(receiver, file_path, cb_function, cb_extra)
|
||
local cb_extra = {
|
||
file_path = file_path,
|
||
cb_function = cb_function or ok_cb,
|
||
cb_extra = cb_extra or false
|
||
}
|
||
-- Call to remove with optional callback
|
||
send_document(receiver, file_path, rmtmp_cb, cb_extra)
|
||
end
|
||
|
||
-- Download the image and send to receiver, it will be deleted.
|
||
-- cb_function and cb_extra are optionals callback
|
||
function send_document_from_url(receiver, url, cb_function, cb_extra, sendNotErrMsg)
|
||
-- If callback not provided
|
||
cb_function = cb_function or ok_cb
|
||
cb_extra = cb_extra or false
|
||
|
||
local file_path = download_to_file(url, false)
|
||
if not file_path then -- Error
|
||
if sendNotErrMsg then
|
||
return false
|
||
else
|
||
local text = 'Fehler beim Herunterladen des Dokumentes'
|
||
send_msg(receiver, text, cb_function, cb_extra)
|
||
end
|
||
else
|
||
print("Datei Pfad: "..file_path)
|
||
_send_document(receiver, file_path, cb_function, cb_extra)
|
||
return true
|
||
end
|
||
end
|
||
|
||
-- Parameters in ?a=1&b=2 style
|
||
function format_http_params(params, is_get)
|
||
local str = ''
|
||
-- If is get add ? to the beginning
|
||
if is_get then str = '?' end
|
||
local first = true -- Frist param
|
||
for k,v in pairs (params) do
|
||
if v then -- nil value
|
||
if first then
|
||
first = false
|
||
str = str..k.. "="..v
|
||
else
|
||
str = str.."&"..k.. "="..v
|
||
end
|
||
end
|
||
end
|
||
return str
|
||
end
|
||
|
||
-- Check if user can use the plugin and warns user
|
||
-- Returns true if user was warned and false if not warned (is allowed)
|
||
function warns_user_not_allowed(plugin, msg)
|
||
if not user_allowed(plugin, msg) then
|
||
local text = 'Das darf nur mein Meister!'
|
||
local receiver = get_receiver(msg)
|
||
send_msg(receiver, text, ok_cb, false)
|
||
return true
|
||
else
|
||
return false
|
||
end
|
||
end
|
||
|
||
-- Check if user can use the plugin
|
||
function user_allowed(plugin, msg)
|
||
if plugin.privileged and not is_sudo(msg) then
|
||
return false
|
||
end
|
||
return true
|
||
end
|
||
|
||
function send_order_msg(destination, msgs)
|
||
local cb_extra = {
|
||
destination = destination,
|
||
msgs = msgs
|
||
}
|
||
send_order_msg_callback(cb_extra, true)
|
||
end
|
||
|
||
function send_large_msg(destination, text)
|
||
local cb_extra = {
|
||
destination = destination,
|
||
text = text
|
||
}
|
||
send_large_msg_callback(cb_extra, true)
|
||
end
|
||
|
||
-- If text is longer than 4096 chars, send multiple msg.
|
||
-- https://core.telegram.org/method/messages.sendMessage
|
||
function send_large_msg_callback(cb_extra, success, result)
|
||
local text_max = 4096
|
||
|
||
local destination = cb_extra.destination
|
||
local text = cb_extra.text
|
||
local text_len = string.len(text)
|
||
local num_msg = math.ceil(text_len / text_max)
|
||
|
||
if num_msg <= 1 then
|
||
send_msg(destination, text, ok_cb, false)
|
||
else
|
||
|
||
local my_text = string.sub(text, 1, 4096)
|
||
local rest = string.sub(text, 4096, text_len)
|
||
|
||
local cb_extra = {
|
||
destination = destination,
|
||
text = rest
|
||
}
|
||
|
||
send_msg(destination, my_text, send_large_msg_callback, cb_extra)
|
||
end
|
||
end
|
||
|
||
-- Returns a table with matches or nil
|
||
--function match_pattern(pattern, text, lower_case)
|
||
function match_pattern(pattern, text)
|
||
if text then
|
||
local matches = { string.match(text, pattern) }
|
||
if next(matches) then
|
||
return matches
|
||
end
|
||
end
|
||
-- nil
|
||
end
|
||
|
||
function sleep(n)
|
||
os.execute("sleep " .. tonumber(n))
|
||
end
|
||
|
||
-- Function to read data from files
|
||
function load_from_file(file, default_data)
|
||
local f = io.open(file, "r+")
|
||
-- If file doesn't exists
|
||
if f == nil then
|
||
-- Create a new empty table
|
||
default_data = default_data or {}
|
||
serialize_to_file(default_data, file)
|
||
print ('Erstelle Datei', file)
|
||
else
|
||
print ('Daten geladen von', file)
|
||
f:close()
|
||
end
|
||
return loadfile (file)()
|
||
end
|
||
|
||
function run_bash(str)
|
||
local cmd = io.popen(str)
|
||
local result = cmd:read('*all')
|
||
cmd:close()
|
||
return result
|
||
end
|
||
|
||
function run_sh(msg)
|
||
name = get_name(msg)
|
||
text = ''
|
||
bash = msg.text:sub(4,-1)
|
||
text = run_bash(bash)
|
||
return text
|
||
end
|
||
|
||
function round(num, idp)
|
||
if idp and idp>0 then
|
||
local mult = 10^idp
|
||
return math.floor(num * mult + 0.5) / mult
|
||
end
|
||
return math.floor(num + 0.5)
|
||
end
|
||
|
||
function unescape(str)
|
||
-- Character encoding
|
||
str = string.gsub(str, "´", "´")
|
||
str = string.gsub(str, "•", "•")
|
||
str = string.gsub(str, ">", ">")
|
||
str = string.gsub(str, "…", "…")
|
||
str = string.gsub(str, "<", "<")
|
||
str = string.gsub(str, "—", "—")
|
||
str = string.gsub(str, "∇", "∇")
|
||
str = string.gsub(str, " ", " ")
|
||
str = string.gsub(str, "–", "–")
|
||
str = string.gsub(str, "Ψ", "ψ")
|
||
str = string.gsub(str, "ψ", "ψ")
|
||
str = string.gsub(str, """, '"')
|
||
str = string.gsub(str, "»", "»")
|
||
str = string.gsub(str, "®", "®")
|
||
str = string.gsub(str, "ß", "ß")
|
||
str = string.gsub(str, "™", "™")
|
||
str = string.gsub(str, "&", "&")
|
||
str = string.gsub(str, "'", "'")
|
||
str = string.gsub(str, """, '"')
|
||
str = string.gsub(str, "'", "'")
|
||
str = string.gsub(str, "|", "|")
|
||
str = string.gsub(str, " ", " ")
|
||
str = string.gsub(str, "®", "®")
|
||
str = string.gsub(str, "»", "»")
|
||
str = string.gsub(str, "ß", "ß")
|
||
str = string.gsub(str, "–", "–")
|
||
str = string.gsub(str, "’", "'")
|
||
str = string.gsub(str, "“", "“")
|
||
str = string.gsub(str, "”", "”")
|
||
str = string.gsub(str, "„", "„")
|
||
str = string.gsub(str, "…", "…")
|
||
str = string.gsub(str, "‹", "‹")
|
||
str = string.gsub(str, "€", "€")
|
||
str = string.gsub(str, "♪", "♪")
|
||
str = string.gsub(str, "<EFBFBD>", "")
|
||
|
||
-- Ä Ö Ü
|
||
str = string.gsub(str, "ä", "ä")
|
||
str = string.gsub(str, "Ä", "Ä")
|
||
str = string.gsub(str, "ä", "ä")
|
||
str = string.gsub(str, "Ä", "Ä")
|
||
str = string.gsub(str, "ö", "ö")
|
||
str = string.gsub(str, "Ö", "Ö")
|
||
str = string.gsub(str, "ö", "ö")
|
||
str = string.gsub(str, "Ö", "Ö")
|
||
str = string.gsub(str, "ü", "ü")
|
||
str = string.gsub(str, "Ü", "Ü")
|
||
str = string.gsub(str, "ü", "ü")
|
||
str = string.gsub(str, "Ü", "Ü")
|
||
--str = string.gsub( str, '&#(%d+);', function(n) return string.char(n) end )
|
||
--str = string.gsub( str, '&#x(%d+);', function(n) return string.char(tonumber(n,16)) end )
|
||
str = string.gsub( str, '&', '&' ) -- Be sure to do this after all others
|
||
return str
|
||
end
|
||
|
||
function gerRating(str)
|
||
str = string.gsub(str, "de/0", "FSK0")
|
||
str = string.gsub(str, "TV%-G", "FSK0")
|
||
str = string.gsub(str, "TV%-Y$", "FSK0")
|
||
str = string.gsub(str, "G %- All Ages", "FSK0")
|
||
str = string.gsub(str, "de/6", "FSK6")
|
||
str = string.gsub(str, "TV%-Y7", "FSK6")
|
||
str = string.gsub(str, "TV%-PG", "FSK6")
|
||
str = string.gsub(str, "PG %- Children", "FSK6")
|
||
str = string.gsub(str, "de/12", "FSK12")
|
||
str = string.gsub(str, "de/16", "FSK16")
|
||
str = string.gsub(str, "TV%-14", "FSK16")
|
||
str = string.gsub(str, "PG%-13 %- Teens 13 or older", "FSK16")
|
||
str = string.gsub(str, "de/18", "FSK18")
|
||
str = string.gsub(str, "TV%-MA", "FSK18")
|
||
str = string.gsub(str, "R %- 17%+ %(violence & profanity%)", "FSK18")
|
||
str = string.gsub(str, "R%+ %- Mild Nudity", "FSK18")
|
||
str = string.gsub(str, "Rx %- Hentai", "FSK18")
|
||
return str
|
||
end
|
||
|
||
function convertNumbers(str)
|
||
str = string.gsub(str, "^1$", "01")
|
||
str = string.gsub(str, "^2$", "02")
|
||
str = string.gsub(str, "^3$", "03")
|
||
str = string.gsub(str, "^4$", "04")
|
||
str = string.gsub(str, "^5$", "05")
|
||
str = string.gsub(str, "^6$", "06")
|
||
str = string.gsub(str, "^7$", "07")
|
||
str = string.gsub(str, "^8$", "08")
|
||
str = string.gsub(str, "^9$", "09")
|
||
return str
|
||
end
|
||
|
||
-- See http://stackoverflow.com/a/14899740
|
||
function unescape_html(str)
|
||
local map = {
|
||
["lt"] = "<",
|
||
["gt"] = ">",
|
||
["amp"] = "&",
|
||
["quot"] = '"',
|
||
["apos"] = "'"
|
||
}
|
||
new = string.gsub(str, '(&(#?x?)([%d%a]+);)', function(orig, n, s)
|
||
var = map[s] or n == "#" and string.char(s)
|
||
var = var or n == "#x" and string.char(tonumber(s,16))
|
||
var = var or orig
|
||
return var
|
||
end)
|
||
return new
|
||
end
|
||
|
||
function post_petition(url, arguments)
|
||
local url, h = string.gsub(url, "http://", "")
|
||
local url, hs = string.gsub(url, "https://", "")
|
||
local post_prot = "http"
|
||
if hs == 1 then
|
||
post_prot = "https"
|
||
end
|
||
local response_body = {}
|
||
local request_constructor = {
|
||
url = post_prot..'://'..url,
|
||
method = "POST",
|
||
sink = ltn12.sink.table(response_body),
|
||
headers = {},
|
||
redirect = false
|
||
}
|
||
|
||
local source = arguments
|
||
if type(arguments) == "table" then
|
||
local source = helpers.url_encode_arguments(arguments)
|
||
end
|
||
request_constructor.headers["Content-Type"] = "application/x-www-form-urlencoded"
|
||
request_constructor.headers["Content-Length"] = tostring(#source)
|
||
request_constructor.source = ltn12.source.string(source)
|
||
if post_prot == "http" then
|
||
ok, response_code, response_headers, response_status_line = http.request(request_constructor)
|
||
else
|
||
ok, response_code, response_headers, response_status_line = https.request(request_constructor)
|
||
end
|
||
|
||
if not ok then
|
||
return nil
|
||
end
|
||
|
||
response_body = json:decode(table.concat(response_body))
|
||
|
||
return response_body
|
||
end
|
||
|
||
function get_redis_hash(msg, var)
|
||
if msg.to.type == 'chat' then
|
||
return 'chat:'..msg.to.id..':'..var
|
||
end
|
||
if msg.to.type == 'user' then
|
||
return 'user:'..msg.from.id..':'..var
|
||
end
|
||
end
|
||
|
||
function tablelength(T)
|
||
local count = 0
|
||
for _ in pairs(T) do count = count + 1 end
|
||
return count
|
||
end
|
||
|
||
function comma_value(amount)
|
||
local formatted = amount
|
||
while true do
|
||
formatted, k = string.gsub(formatted, "^(-?%d+)(%d%d%d)", '%1.%2')
|
||
if (k==0) then
|
||
break
|
||
end
|
||
end
|
||
return formatted
|
||
end
|
||
|
||
|
||
function string.ends(str, fin)
|
||
return fin=='' or string.sub(str,-string.len(fin)) == fin
|
||
end
|
||
|
||
function get_location(user_id)
|
||
local hash = 'user:'..user_id
|
||
local set_location = redis:hget(hash, 'location')
|
||
if set_location == 'false' or set_location == nil then
|
||
return false
|
||
else
|
||
return set_location
|
||
end
|
||
end
|
||
|
||
function cache_data(plugin, query, data, timeout, typ)
|
||
-- How to: cache_data(pluginname, query_name, data_to_cache, expire_in_seconds)
|
||
if not timeout then timeout = 86400 end
|
||
local hash = 'telegram:cache:'..plugin..':'..query
|
||
print('Caching "'..query..'" from plugin '..plugin..' (expires in '..timeout..' seconds)')
|
||
if typ == 'key' then
|
||
redis:set(hash, data)
|
||
elseif typ == 'set' then
|
||
-- make sure that you convert your data into a table:
|
||
-- {"foo", "bar", "baz"} instead of
|
||
-- {"bar" = "foo", "foo" = "bar", "bar" = "baz"}
|
||
-- because other formats are not supported by redis (or I haven't found a way to store them)
|
||
for _,str in pairs(data) do
|
||
redis:sadd(hash, str)
|
||
end
|
||
else
|
||
redis:hmset(hash, data)
|
||
end
|
||
redis:expire(hash, timeout)
|
||
end
|
||
|
||
--[[
|
||
Ordered table iterator, allow to iterate on the natural order of the keys of a
|
||
table.
|
||
-- http://lua-users.org/wiki/SortedIteration
|
||
]]
|
||
|
||
function __genOrderedIndex( t )
|
||
local orderedIndex = {}
|
||
for key in pairs(t) do
|
||
table.insert( orderedIndex, key )
|
||
end
|
||
table.sort( orderedIndex )
|
||
return orderedIndex
|
||
end
|
||
|
||
function orderedNext(t, state)
|
||
-- Equivalent of the next function, but returns the keys in the alphabetic
|
||
-- order. We use a temporary ordered key table that is stored in the
|
||
-- table being iterated.
|
||
|
||
key = nil
|
||
--print("orderedNext: state = "..tostring(state) )
|
||
if state == nil then
|
||
-- the first time, generate the index
|
||
t.__orderedIndex = __genOrderedIndex( t )
|
||
key = t.__orderedIndex[1]
|
||
else
|
||
-- fetch the next value
|
||
for i = 1,table.getn(t.__orderedIndex) do
|
||
if t.__orderedIndex[i] == state then
|
||
key = t.__orderedIndex[i+1]
|
||
end
|
||
end
|
||
end
|
||
|
||
if key then
|
||
return key, t[key]
|
||
end
|
||
|
||
-- no more value to return, cleanup
|
||
t.__orderedIndex = nil
|
||
return
|
||
end
|
||
|
||
function orderedPairs(t)
|
||
-- Equivalent of the pairs() function on tables. Allows to iterate
|
||
-- in order
|
||
return orderedNext, t, nil
|
||
end
|
||
|
||
-- converts total amount of seconds (e.g. 65 seconds) to human redable time (e.g. 1:05 minutes)
|
||
function makeHumanTime(totalseconds)
|
||
local seconds = totalseconds % 60
|
||
local minutes = math.floor(totalseconds / 60)
|
||
local minutes = minutes % 60
|
||
local hours = math.floor(totalseconds / 3600)
|
||
if minutes == 00 and hours == 00 then
|
||
return seconds..' Sekunden'
|
||
elseif hours == 00 and minutes ~= 00 then
|
||
return string.format("%02d:%02d", minutes, seconds)..' Minuten'
|
||
elseif hours ~= 00 then
|
||
return string.format("%02d:%02d:%02d", hours, minutes, seconds)..' Stunden'
|
||
end
|
||
end
|
||
|
||
function formatMilliseconds(milliseconds)
|
||
local totalseconds = math.floor( milliseconds / 1000 )
|
||
milliseconds = milliseconds % 1000
|
||
local seconds = totalseconds % 60
|
||
local minutes = math.floor( totalseconds / 60 )
|
||
local hours = math.floor( minutes / 60 )
|
||
local days = math.floor( hours / 24 )
|
||
minutes = minutes % 60
|
||
hours = hours % 24
|
||
--return string.format( "%03d:%02d:%02d:%02d:%03d", days, hours, minutes, seconds, milliseconds )
|
||
if minutes == 00 and hours == 00 then
|
||
return seconds..' Sekunden'
|
||
elseif hours == 00 and minutes ~= 00 then
|
||
return string.format("%02d:%02d", minutes, seconds)..' Minuten'
|
||
elseif hours ~= 00 then
|
||
return string.format("%02d:%02d:%02d", hours, minutes, seconds)..' Stunden'
|
||
end
|
||
end
|
||
|
||
function is_blacklisted(msg)
|
||
_blacklist = redis:smembers("telegram:img_blacklist")
|
||
local var = false
|
||
for v,word in pairs(_blacklist) do
|
||
if string.find(string.lower(msg), string.lower(word)) then
|
||
print("Wort steht auf der Blacklist!")
|
||
var = true
|
||
break
|
||
end
|
||
end
|
||
return var
|
||
end
|
||
|
||
function convert_timestamp(timestamp, format)
|
||
local converted_date = run_command('date -d @'..timestamp..' +'..format)
|
||
local converted_date = string.gsub(converted_date, '%\n', '')
|
||
return converted_date
|
||
end
|
||
|
||
function pretty_float(x)
|
||
if x % 1 == 0 then
|
||
return tostring(math.floor(x))
|
||
else
|
||
return tostring(x)
|
||
end
|
||
end
|