2016-08-23 00:16:32 -04:00
|
|
|
|
--[[
|
|
|
|
|
utilities.lua
|
|
|
|
|
Functions shared among otouto plugins.
|
|
|
|
|
|
|
|
|
|
Copyright 2016 topkecleon <drew@otou.to>
|
|
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify it
|
|
|
|
|
under the terms of the GNU Affero General Public License version 3 as
|
|
|
|
|
published by the Free Software Foundation.
|
|
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
|
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
|
FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License
|
|
|
|
|
for more details.
|
|
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Affero General Public License
|
|
|
|
|
along with this program; if not, write to the Free Software Foundation,
|
|
|
|
|
Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
|
]]--
|
2015-07-02 18:15:52 -04:00
|
|
|
|
|
2016-04-08 14:12:02 -07:00
|
|
|
|
local utilities = {}
|
|
|
|
|
|
2016-09-07 15:55:57 +02:00
|
|
|
|
utf8 = require('lua-utf8')
|
2016-07-31 21:29:44 +02:00
|
|
|
|
ltn12 = require('ltn12')
|
|
|
|
|
http = require('socket.http')
|
|
|
|
|
https = require('ssl.https')
|
2016-07-31 21:35:05 +02:00
|
|
|
|
socket = require('socket')
|
2016-07-31 21:29:44 +02:00
|
|
|
|
URL = require('socket.url')
|
2016-08-03 19:21:07 +02:00
|
|
|
|
json = require("dkjson")
|
|
|
|
|
pcall(json.use_lpeg)
|
2016-07-31 21:29:44 +02:00
|
|
|
|
serpent = require("serpent")
|
|
|
|
|
redis = (loadfile "./otouto/redis.lua")()
|
2016-08-15 14:25:20 +02:00
|
|
|
|
mime = (loadfile "./otouto/mimetype.lua")()
|
2016-07-31 21:29:44 +02:00
|
|
|
|
OAuth = require "OAuth"
|
|
|
|
|
helpers = require "OAuth.helpers"
|
|
|
|
|
|
|
|
|
|
http.timeout = 5
|
|
|
|
|
https.timeout = 5
|
2016-07-12 00:04:42 +02:00
|
|
|
|
|
2016-05-29 13:08:39 -04:00
|
|
|
|
-- For the sake of ease to new contributors and familiarity to old contributors,
|
|
|
|
|
-- we'll provide a couple of aliases to real bindings here.
|
2016-08-24 15:38:29 +02:00
|
|
|
|
function utilities.send_message(chat_id, text, disable_web_page_preview, reply_to_message_id, use_markdown, reply_markup)
|
2016-08-13 22:46:18 -04:00
|
|
|
|
local parse_mode
|
|
|
|
|
if type(use_markdown) == 'string' then
|
|
|
|
|
parse_mode = use_markdown
|
|
|
|
|
elseif use_markdown == true then
|
|
|
|
|
parse_mode = 'markdown'
|
|
|
|
|
end
|
2016-08-23 00:16:32 -04:00
|
|
|
|
return bindings.request(
|
|
|
|
|
'sendMessage',
|
|
|
|
|
{
|
2016-08-24 15:38:29 +02:00
|
|
|
|
chat_id = chat_id,
|
2016-08-23 00:16:32 -04:00
|
|
|
|
text = text,
|
|
|
|
|
disable_web_page_preview = disable_web_page_preview,
|
|
|
|
|
reply_to_message_id = reply_to_message_id,
|
2016-08-24 15:38:29 +02:00
|
|
|
|
parse_mode = parse_mode,
|
|
|
|
|
reply_markup = reply_markup
|
2016-08-23 00:16:32 -04:00
|
|
|
|
}
|
|
|
|
|
)
|
2016-05-29 13:08:39 -04:00
|
|
|
|
end
|
|
|
|
|
|
2016-07-02 00:13:17 +02:00
|
|
|
|
-- https://core.telegram.org/bots/api#editmessagetext
|
2016-08-24 15:38:29 +02:00
|
|
|
|
function utilities.edit_message(chat_id, message_id, text, disable_web_page_preview, use_markdown, reply_markup)
|
2016-08-14 16:30:06 +02:00
|
|
|
|
local parse_mode
|
|
|
|
|
if type(use_markdown) == 'string' then
|
|
|
|
|
parse_mode = use_markdown
|
|
|
|
|
elseif use_markdown == true then
|
|
|
|
|
parse_mode = 'Markdown'
|
2016-08-04 13:02:47 +02:00
|
|
|
|
end
|
2016-08-24 15:38:29 +02:00
|
|
|
|
return bindings.request(
|
|
|
|
|
'editMessageText',
|
|
|
|
|
{
|
|
|
|
|
chat_id = chat_id,
|
|
|
|
|
message_id = message_id,
|
|
|
|
|
text = text,
|
|
|
|
|
disable_web_page_preview = disable_web_page_preview,
|
|
|
|
|
parse_mode = parse_mode,
|
|
|
|
|
reply_markup = reply_markup
|
|
|
|
|
}
|
|
|
|
|
)
|
|
|
|
|
end
|
|
|
|
|
|
2016-08-24 21:38:45 +02:00
|
|
|
|
function utilities.delete_message(chat_id, message_id)
|
|
|
|
|
return utilities.edit_message(chat_id, message_id, '<i>Gelöschte Nachricht</i>', true, 'HTML')
|
|
|
|
|
end
|
|
|
|
|
|
2016-08-24 15:38:29 +02:00
|
|
|
|
function utilities.send_reply(msg, text, use_markdown, reply_markup)
|
2016-08-23 00:16:32 -04:00
|
|
|
|
local parse_mode
|
|
|
|
|
if type(use_markdown) == 'string' then
|
|
|
|
|
parse_mode = use_markdown
|
|
|
|
|
elseif use_markdown == true then
|
|
|
|
|
parse_mode = 'markdown'
|
|
|
|
|
end
|
|
|
|
|
return bindings.request(
|
|
|
|
|
'sendMessage',
|
|
|
|
|
{
|
|
|
|
|
chat_id = msg.chat.id,
|
|
|
|
|
text = text,
|
|
|
|
|
disable_web_page_preview = true,
|
|
|
|
|
reply_to_message_id = msg.message_id,
|
2016-08-24 15:38:29 +02:00
|
|
|
|
parse_mode = parse_mode,
|
|
|
|
|
reply_markup = reply_markup
|
2016-08-23 00:16:32 -04:00
|
|
|
|
}
|
|
|
|
|
)
|
2016-05-29 13:08:39 -04:00
|
|
|
|
end
|
2016-06-12 20:53:20 +02:00
|
|
|
|
|
|
|
|
|
-- NOTE: Telegram currently only allows file uploads up to 50 MB
|
|
|
|
|
-- https://core.telegram.org/bots/api#sendphoto
|
2016-08-24 15:38:29 +02:00
|
|
|
|
function utilities.send_photo(chat_id, file, text, reply_to_message_id, reply_markup)
|
2016-07-04 01:29:51 +02:00
|
|
|
|
if not file then return false end
|
2016-08-24 15:38:29 +02:00
|
|
|
|
local output = bindings.request(
|
|
|
|
|
'sendPhoto',
|
|
|
|
|
{
|
|
|
|
|
chat_id = chat_id,
|
|
|
|
|
caption = text or nil,
|
|
|
|
|
reply_to_message_id = reply_to_message_id,
|
|
|
|
|
reply_markup = reply_markup
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
photo = file
|
|
|
|
|
}
|
|
|
|
|
)
|
2016-07-03 01:20:04 +02:00
|
|
|
|
if string.match(file, '/tmp/') then
|
|
|
|
|
os.remove(file)
|
|
|
|
|
print("Deleted: "..file)
|
|
|
|
|
end
|
2016-06-12 20:53:20 +02:00
|
|
|
|
return output
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- https://core.telegram.org/bots/api#sendaudio
|
2016-08-24 15:38:29 +02:00
|
|
|
|
function utilities.send_audio(chat_id, file, reply_to_message_id, duration, performer, title)
|
2016-07-04 01:29:51 +02:00
|
|
|
|
if not file then return false end
|
2016-08-24 15:38:29 +02:00
|
|
|
|
local output = bindings.request(
|
|
|
|
|
'sendAudio',
|
|
|
|
|
{
|
|
|
|
|
chat_id = chat_id,
|
|
|
|
|
duration = duration or nil,
|
|
|
|
|
performer = performer or nil,
|
|
|
|
|
title = title or nil,
|
|
|
|
|
reply_to_message_id = reply_to_message_id
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
audio = file
|
|
|
|
|
}
|
|
|
|
|
)
|
2016-07-03 01:20:04 +02:00
|
|
|
|
if string.match(file, '/tmp/') then
|
|
|
|
|
os.remove(file)
|
|
|
|
|
print("Deleted: "..file)
|
|
|
|
|
end
|
2016-06-12 20:53:20 +02:00
|
|
|
|
return output
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- https://core.telegram.org/bots/api#senddocument
|
2016-08-24 15:38:29 +02:00
|
|
|
|
function utilities.send_document(chat_id, file, text, reply_to_message_id, reply_markup)
|
2016-07-04 01:29:51 +02:00
|
|
|
|
if not file then return false end
|
2016-08-24 15:38:29 +02:00
|
|
|
|
local output = bindings.request(
|
|
|
|
|
'sendDocument',
|
|
|
|
|
{
|
|
|
|
|
chat_id = chat_id,
|
|
|
|
|
caption = text or nil,
|
|
|
|
|
reply_to_message_id = reply_to_message_id,
|
|
|
|
|
reply_markup = reply_markup
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
document = file
|
|
|
|
|
}
|
|
|
|
|
)
|
2016-07-03 01:20:04 +02:00
|
|
|
|
if string.match(file, '/tmp/') then
|
|
|
|
|
os.remove(file)
|
|
|
|
|
print("Deleted: "..file)
|
|
|
|
|
end
|
2016-06-12 20:53:20 +02:00
|
|
|
|
return output
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- https://core.telegram.org/bots/api#sendvideo
|
2016-08-24 15:38:29 +02:00
|
|
|
|
function utilities.send_video(chat_id, file, text, reply_to_message_id, duration, width, height)
|
2016-07-04 01:29:51 +02:00
|
|
|
|
if not file then return false end
|
2016-08-24 15:38:29 +02:00
|
|
|
|
local output = bindings.request(
|
|
|
|
|
'sendVideo',
|
|
|
|
|
{
|
|
|
|
|
chat_id = chat_id,
|
|
|
|
|
caption = text or nil,
|
|
|
|
|
duration = duration or nil,
|
|
|
|
|
width = width or nil,
|
|
|
|
|
height = height or nil,
|
|
|
|
|
reply_to_message_id = reply_to_message_id
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
video = file
|
|
|
|
|
}
|
|
|
|
|
)
|
2016-07-03 01:20:04 +02:00
|
|
|
|
if string.match(file, '/tmp/') then
|
|
|
|
|
os.remove(file)
|
|
|
|
|
print("Deleted: "..file)
|
|
|
|
|
end
|
2016-06-12 20:53:20 +02:00
|
|
|
|
return output
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- NOTE: Voice messages are .ogg files encoded with OPUS
|
|
|
|
|
-- https://core.telegram.org/bots/api#sendvoice
|
2016-08-24 22:33:19 +02:00
|
|
|
|
function utilities.send_voice(chat_id, file, reply_to_message_id, duration)
|
2016-07-04 01:29:51 +02:00
|
|
|
|
if not file then return false end
|
2016-08-24 15:38:29 +02:00
|
|
|
|
local output = bindings.request(
|
|
|
|
|
'sendVoice',
|
|
|
|
|
{
|
|
|
|
|
chat_id = chat_id,
|
|
|
|
|
duration = duration or nil,
|
|
|
|
|
reply_to_message_id = reply_to_message_id
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
voice = file
|
|
|
|
|
}
|
|
|
|
|
)
|
2016-07-03 01:20:04 +02:00
|
|
|
|
if string.match(file, '/tmp/') then
|
|
|
|
|
os.remove(file)
|
|
|
|
|
print("Deleted: "..file)
|
|
|
|
|
end
|
2016-06-12 20:53:20 +02:00
|
|
|
|
return output
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- https://core.telegram.org/bots/api#sendlocation
|
2016-08-24 15:38:29 +02:00
|
|
|
|
function utilities.send_location(chat_id, latitude, longitude, reply_to_message_id)
|
|
|
|
|
return bindings.request(
|
|
|
|
|
'sendLocation',
|
|
|
|
|
{
|
|
|
|
|
chat_id = chat_id,
|
|
|
|
|
latitude = latitude,
|
|
|
|
|
longitude = longitude,
|
|
|
|
|
reply_to_message_id = reply_to_message_id
|
|
|
|
|
}
|
|
|
|
|
)
|
2016-06-12 20:53:20 +02:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- NOTE: Venue is different from location: it shows information, such as the street adress or
|
|
|
|
|
-- title of the location with it.
|
|
|
|
|
-- https://core.telegram.org/bots/api#sendvenue
|
2016-08-24 15:38:29 +02:00
|
|
|
|
function utilities.send_venue(chat_id, latitude, longitude, reply_to_message_id, title, address)
|
|
|
|
|
return bindings.request(
|
|
|
|
|
'sendVenue',
|
|
|
|
|
{
|
|
|
|
|
chat_id = chat_id,
|
|
|
|
|
latitude = latitude,
|
|
|
|
|
longitude = longitude,
|
|
|
|
|
title = title,
|
|
|
|
|
address = address,
|
|
|
|
|
reply_to_message_id = reply_to_message_id
|
|
|
|
|
}
|
|
|
|
|
)
|
2016-06-12 20:53:20 +02:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- https://core.telegram.org/bots/api#sendchataction
|
2016-08-24 15:38:29 +02:00
|
|
|
|
function utilities.send_typing(chat_id, action)
|
|
|
|
|
return bindings.request(
|
|
|
|
|
'sendChatAction',
|
|
|
|
|
{
|
|
|
|
|
chat_id = chat_id,
|
|
|
|
|
action = action
|
|
|
|
|
}
|
|
|
|
|
)
|
2016-06-12 20:53:20 +02:00
|
|
|
|
end
|
|
|
|
|
|
2016-07-02 12:35:14 +02:00
|
|
|
|
-- https://core.telegram.org/bots/api#answercallbackquery
|
2016-08-24 15:38:29 +02:00
|
|
|
|
function utilities.answer_callback_query(callback, text, show_alert)
|
|
|
|
|
return bindings.request(
|
|
|
|
|
'answerCallbackQuery',
|
|
|
|
|
{
|
|
|
|
|
callback_query_id = callback.id,
|
|
|
|
|
text = text,
|
|
|
|
|
show_alert = show_alert
|
|
|
|
|
}
|
|
|
|
|
)
|
2016-07-02 12:35:14 +02:00
|
|
|
|
end
|
|
|
|
|
|
2016-07-05 22:26:46 +02:00
|
|
|
|
-- https://core.telegram.org/bots/api#getchat
|
2016-08-24 15:38:29 +02:00
|
|
|
|
function utilities.get_chat_info(chat_id)
|
|
|
|
|
return bindings.request(
|
|
|
|
|
'getChat',
|
|
|
|
|
{
|
|
|
|
|
chat_id = chat_id
|
|
|
|
|
}
|
|
|
|
|
)
|
2016-07-05 22:26:46 +02:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- https://core.telegram.org/bots/api#getchatadministrators
|
2016-08-24 15:38:29 +02:00
|
|
|
|
function utilities.get_chat_administrators(chat_id)
|
|
|
|
|
return bindings.request(
|
|
|
|
|
'getChatAdministrators',
|
|
|
|
|
{
|
|
|
|
|
chat_id = chat_id
|
|
|
|
|
}
|
|
|
|
|
)
|
2016-07-05 22:26:46 +02:00
|
|
|
|
end
|
|
|
|
|
|
2016-07-13 01:00:32 +02:00
|
|
|
|
-- https://core.telegram.org/bots/api#answerinlinequery
|
2016-08-24 15:38:29 +02:00
|
|
|
|
function utilities.answer_inline_query(inline_query, results, cache_time, is_personal, next_offset, switch_pm_text, switch_pm_parameter)
|
|
|
|
|
return bindings.request(
|
|
|
|
|
'answerInlineQuery',
|
|
|
|
|
{
|
|
|
|
|
inline_query_id = inline_query.id,
|
|
|
|
|
results = results,
|
|
|
|
|
cache_time = cache_time,
|
|
|
|
|
is_personal = is_personal,
|
|
|
|
|
next_offset = next_offset,
|
|
|
|
|
switch_pm_text = switch_pm_text,
|
|
|
|
|
switch_pm_parameter = switch_pm_parameter
|
|
|
|
|
}
|
|
|
|
|
)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
function abort_inline_query(inline_query)
|
|
|
|
|
return bindings.request(
|
|
|
|
|
'answerInlineQuery',
|
|
|
|
|
{
|
|
|
|
|
inline_query_id = inline_query.id,
|
|
|
|
|
cache_time = 5,
|
|
|
|
|
is_personal = true
|
|
|
|
|
}
|
|
|
|
|
)
|
2016-07-13 01:00:32 +02:00
|
|
|
|
end
|
|
|
|
|
|
2016-08-14 23:04:59 +02:00
|
|
|
|
-- get the indexed word in a string
|
|
|
|
|
function utilities.get_word(s, i)
|
|
|
|
|
s = s or ''
|
|
|
|
|
i = i or 1
|
|
|
|
|
local n = 0
|
|
|
|
|
for w in s:gmatch('%g+') do
|
|
|
|
|
n = n + 1
|
|
|
|
|
if n == i then return w end
|
|
|
|
|
end
|
|
|
|
|
return false
|
|
|
|
|
end
|
|
|
|
|
|
2016-01-12 05:22:28 -05:00
|
|
|
|
-- Returns the string after the first space.
|
2016-04-08 14:12:02 -07:00
|
|
|
|
function utilities.input(s)
|
|
|
|
|
if not s:find(' ') then
|
2015-07-28 18:13:46 -04:00
|
|
|
|
return false
|
|
|
|
|
end
|
2016-04-08 14:12:02 -07:00
|
|
|
|
return s:sub(s:find(' ')+1)
|
2015-07-28 18:13:46 -04:00
|
|
|
|
end
|
|
|
|
|
|
2016-08-13 22:26:44 -04:00
|
|
|
|
function utilities.input_from_msg(msg)
|
2016-08-14 16:30:06 +02:00
|
|
|
|
return utilities.input(msg.text) or (msg.reply_to_message and #msg.reply_to_message.text > 0 and msg.reply_to_message.text) or false
|
2016-08-13 22:26:44 -04:00
|
|
|
|
end
|
|
|
|
|
|
2016-05-27 18:49:58 +03:00
|
|
|
|
-- Calculates the length of the given string as UTF-8 characters
|
|
|
|
|
function utilities.utf8_len(s)
|
|
|
|
|
local chars = 0
|
|
|
|
|
for i = 1, string.len(s) do
|
|
|
|
|
local b = string.byte(s, i)
|
|
|
|
|
if b < 128 or b >= 192 then
|
|
|
|
|
chars = chars + 1
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
return chars
|
|
|
|
|
end
|
|
|
|
|
|
2016-07-05 13:14:22 +02:00
|
|
|
|
-- Trims whitespace from a string.
|
otouto 3.11
"things occurred"
Added some utilities (id_from_username, id_from_message), removed some utilities (latcyr, others?).
Removed cycle-wasting "shortcuts" -- no more automatic id_str or name; text_lower remains.
Moved userdata (nicknames, lastfm, etc) to a different tree in the database (automatic migration will occur). /me now returns userdata.
Speaking of migration, database now stores the latest version run to make future automigration easy.
Database now saves hourly rather than minutely.
Changed readme and some plugins to reflect above changes.
Removed broken rockspec (Brayden, feel free to re-add once it's working).
Added option to automatically block people (via drua) when blacklisted.
Fixed about.lua trigger problems.
administration 1.11 - Removed /kickme and /broadcast. Users should leave manually, and announcements should be made via channel rather than spam. /setqotd now handles forwarded messages correctly. /kick, /ban, /hammer,
/mod, /admin now support multiple arguments. Added get_targets function. No migration is necessary.
2016-07-05 03:29:11 -04:00
|
|
|
|
function utilities.trim(str)
|
2016-04-08 14:12:02 -07:00
|
|
|
|
local s = str:gsub('^%s*(.-)%s*$', '%1')
|
2015-11-24 21:22:04 -05:00
|
|
|
|
return s
|
2015-07-02 18:15:52 -04:00
|
|
|
|
end
|
|
|
|
|
|
2016-08-02 01:01:06 +02:00
|
|
|
|
-- Returns true if the string is empty
|
2016-06-14 13:14:09 +02:00
|
|
|
|
function string:isempty()
|
|
|
|
|
return self == nil or self == ''
|
|
|
|
|
end
|
|
|
|
|
|
2016-08-03 19:21:07 +02:00
|
|
|
|
-- Returns true if the string is blank
|
2016-06-14 13:14:09 +02:00
|
|
|
|
function string:isblank()
|
|
|
|
|
self = self:trim()
|
|
|
|
|
return self:isempty()
|
|
|
|
|
end
|
|
|
|
|
|
2016-06-14 12:21:16 +02:00
|
|
|
|
function get_name(msg)
|
|
|
|
|
local name = msg.from.first_name
|
|
|
|
|
if name == nil then
|
|
|
|
|
name = msg.from.id
|
|
|
|
|
end
|
|
|
|
|
return name
|
|
|
|
|
end
|
|
|
|
|
|
2016-06-11 14:46:41 +02:00
|
|
|
|
-- 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
|
|
|
|
|
|
2016-06-15 01:16:27 +02:00
|
|
|
|
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
|
|
|
|
|
|
2016-06-11 14:46:41 +02:00
|
|
|
|
function string.starts(String, Start)
|
|
|
|
|
return Start == string.sub(String,1,string.len(Start))
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- Saves file to $HOME/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)
|
2016-06-19 21:53:24 +02:00
|
|
|
|
print('url to download: '..url)
|
|
|
|
|
if not file_name then
|
|
|
|
|
file_name = '/tmp/' .. url:match('.+/(.-)$') or '/tmp/' .. os.time()
|
|
|
|
|
else
|
|
|
|
|
file_name = '/tmp/' .. file_name
|
|
|
|
|
end
|
|
|
|
|
local body = {}
|
2016-07-31 21:29:44 +02:00
|
|
|
|
local doer = http
|
2016-06-19 21:53:24 +02:00
|
|
|
|
local do_redir = true
|
|
|
|
|
if url:match('^https') then
|
2016-07-31 21:29:44 +02:00
|
|
|
|
doer = https
|
2016-06-19 21:53:24 +02:00
|
|
|
|
do_redir = false
|
|
|
|
|
end
|
|
|
|
|
local _, res = doer.request{
|
|
|
|
|
url = url,
|
|
|
|
|
sink = ltn12.sink.table(body),
|
|
|
|
|
redirect = do_redir
|
|
|
|
|
}
|
|
|
|
|
if res ~= 200 then return false end
|
|
|
|
|
local file = io.open(file_name, 'w+')
|
|
|
|
|
file:write(table.concat(body))
|
|
|
|
|
file:close()
|
|
|
|
|
print('Saved to: '..file_name)
|
|
|
|
|
return file_name
|
2016-06-11 14:46:41 +02:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
function vardump(value)
|
|
|
|
|
print(serpent.block(value, {comment=false}))
|
|
|
|
|
end
|
|
|
|
|
|
2016-01-12 05:22:28 -05:00
|
|
|
|
-- Loads a JSON file as a table.
|
2016-04-08 14:12:02 -07:00
|
|
|
|
function utilities.load_data(filename)
|
2015-07-28 18:13:46 -04:00
|
|
|
|
local f = io.open(filename)
|
2016-08-14 16:30:06 +02:00
|
|
|
|
if f then
|
|
|
|
|
local s = f:read('*all')
|
|
|
|
|
f:close()
|
|
|
|
|
return json.decode(s)
|
|
|
|
|
else
|
2015-07-28 18:13:46 -04:00
|
|
|
|
return {}
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2016-01-12 05:22:28 -05:00
|
|
|
|
-- Saves a table to a JSON file.
|
2016-04-08 14:12:02 -07:00
|
|
|
|
function utilities.save_data(filename, data)
|
2016-07-31 21:29:44 +02:00
|
|
|
|
local s = json.encode(data)
|
2015-07-28 18:13:46 -04:00
|
|
|
|
local f = io.open(filename, 'w')
|
|
|
|
|
f:write(s)
|
|
|
|
|
f:close()
|
2015-11-24 21:22:04 -05:00
|
|
|
|
end
|
|
|
|
|
|
2016-08-27 23:31:53 +02:00
|
|
|
|
-- Returns file size as Integer
|
|
|
|
|
-- See: https://www.lua.org/pil/21.3.html
|
|
|
|
|
function get_file_size(file)
|
|
|
|
|
local current = file:seek() -- get current position
|
|
|
|
|
local size = file:seek("end") -- get file size
|
|
|
|
|
file:seek("set", current) -- restore position
|
|
|
|
|
return tonumber(size)
|
|
|
|
|
end
|
|
|
|
|
|
2015-11-24 21:22:04 -05:00
|
|
|
|
-- Gets coordinates for a location. Used by gMaps.lua, time.lua, weather.lua.
|
2016-05-26 17:59:45 -07:00
|
|
|
|
function utilities.get_coords(input, config)
|
2016-08-06 23:11:53 +02:00
|
|
|
|
local url = 'https://maps.googleapis.com/maps/api/geocode/json?address='..URL.escape(input)..'&language=de'
|
|
|
|
|
local jstr, res = https.request(url)
|
|
|
|
|
if res ~= 200 then
|
|
|
|
|
return config.errors.connection
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
local jdat = json.decode(jstr)
|
|
|
|
|
if jdat.status == 'ZERO_RESULTS' then
|
|
|
|
|
return config.errors.results
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
lat = jdat.results[1].geometry.location.lat,
|
|
|
|
|
lon = jdat.results[1].geometry.location.lng,
|
|
|
|
|
addr = jdat.results[1].formatted_address
|
|
|
|
|
}
|
2016-01-12 05:22:28 -05:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- Get the number of values in a key/value table.
|
2016-04-08 14:12:02 -07:00
|
|
|
|
function utilities.table_size(tab)
|
2016-01-12 05:22:28 -05:00
|
|
|
|
local i = 0
|
2016-04-08 14:12:02 -07:00
|
|
|
|
for _,_ in pairs(tab) do
|
2016-01-12 05:22:28 -05:00
|
|
|
|
i = i + 1
|
|
|
|
|
end
|
|
|
|
|
return i
|
|
|
|
|
end
|
|
|
|
|
|
2016-04-02 19:20:28 -04:00
|
|
|
|
-- Just an easy way to get a user's full name.
|
2016-05-13 13:22:10 -04:00
|
|
|
|
-- Alternatively, abuse it to concat two strings like I do.
|
2016-04-08 14:12:02 -07:00
|
|
|
|
function utilities.build_name(first, last)
|
2016-04-02 19:20:28 -04:00
|
|
|
|
if last then
|
|
|
|
|
return first .. ' ' .. last
|
|
|
|
|
else
|
|
|
|
|
return first
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2016-04-08 14:12:02 -07:00
|
|
|
|
function utilities:resolve_username(input)
|
2016-03-22 06:16:26 -04:00
|
|
|
|
input = input:gsub('^@', '')
|
otouto 3.11
"things occurred"
Added some utilities (id_from_username, id_from_message), removed some utilities (latcyr, others?).
Removed cycle-wasting "shortcuts" -- no more automatic id_str or name; text_lower remains.
Moved userdata (nicknames, lastfm, etc) to a different tree in the database (automatic migration will occur). /me now returns userdata.
Speaking of migration, database now stores the latest version run to make future automigration easy.
Database now saves hourly rather than minutely.
Changed readme and some plugins to reflect above changes.
Removed broken rockspec (Brayden, feel free to re-add once it's working).
Added option to automatically block people (via drua) when blacklisted.
Fixed about.lua trigger problems.
administration 1.11 - Removed /kickme and /broadcast. Users should leave manually, and announcements should be made via channel rather than spam. /setqotd now handles forwarded messages correctly. /kick, /ban, /hammer,
/mod, /admin now support multiple arguments. Added get_targets function. No migration is necessary.
2016-07-05 03:29:11 -04:00
|
|
|
|
for _, user in pairs(self.database.users) do
|
|
|
|
|
if user.username and user.username:lower() == input:lower() then
|
|
|
|
|
local t = {}
|
|
|
|
|
for key, val in pairs(user) do
|
|
|
|
|
t[key] = val
|
|
|
|
|
end
|
|
|
|
|
return t
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2016-08-23 00:16:32 -04:00
|
|
|
|
function utilities:handle_exception(err, message, log_chat)
|
2016-08-24 15:54:16 +02:00
|
|
|
|
local output = string.format(
|
|
|
|
|
'[%s]\n%s: %s\n%s\n',
|
|
|
|
|
os.date('%F %T'),
|
|
|
|
|
self.info.username,
|
|
|
|
|
err or '',
|
|
|
|
|
message
|
|
|
|
|
)
|
|
|
|
|
if log_chat then
|
|
|
|
|
output = '<code>' .. utilities.html_escape(output) .. '</code>'
|
|
|
|
|
return utilities.send_message(log_chat, output, true, nil, 'html')
|
|
|
|
|
else
|
|
|
|
|
print(output)
|
|
|
|
|
end
|
|
|
|
|
|
2016-01-14 22:39:24 -05:00
|
|
|
|
end
|
|
|
|
|
|
2016-06-19 21:53:24 +02:00
|
|
|
|
-- MOVED TO DOWNLOAD_TO_FILE
|
2016-04-08 14:12:02 -07:00
|
|
|
|
function utilities.download_file(url, filename)
|
2016-06-19 21:53:24 +02:00
|
|
|
|
return download_to_file(url, filename)
|
2016-01-13 13:00:17 -05:00
|
|
|
|
end
|
2016-03-22 06:16:26 -04:00
|
|
|
|
|
2016-08-13 22:26:44 -04:00
|
|
|
|
function utilities.md_escape(text)
|
2016-08-14 16:30:06 +02:00
|
|
|
|
return text:gsub('_', '\\_')
|
|
|
|
|
:gsub('%[', '\\['):gsub('%]', '\\]')
|
|
|
|
|
:gsub('%*', '\\*'):gsub('`', '\\`')
|
2016-07-16 22:30:51 +02:00
|
|
|
|
end
|
|
|
|
|
|
2016-08-14 16:30:06 +02:00
|
|
|
|
utilities.markdown_escape = utilities.md_escape
|
|
|
|
|
|
2016-08-13 22:26:44 -04:00
|
|
|
|
function utilities.html_escape(text)
|
2016-08-14 16:30:06 +02:00
|
|
|
|
return text:gsub('&', '&'):gsub('<', '<'):gsub('>', '>')
|
2016-08-13 22:26:44 -04:00
|
|
|
|
end
|
2016-04-08 14:12:02 -07:00
|
|
|
|
|
2016-04-13 20:48:20 -07:00
|
|
|
|
utilities.triggers_meta = {}
|
|
|
|
|
utilities.triggers_meta.__index = utilities.triggers_meta
|
|
|
|
|
function utilities.triggers_meta:t(pattern, has_args)
|
2016-04-15 19:07:23 +00:00
|
|
|
|
local username = self.username:lower()
|
2016-05-26 20:28:44 -07:00
|
|
|
|
table.insert(self.table, '^'..self.cmd_pat..pattern..'$')
|
|
|
|
|
table.insert(self.table, '^'..self.cmd_pat..pattern..'@'..username..'$')
|
2016-04-08 14:12:02 -07:00
|
|
|
|
if has_args then
|
2016-05-26 20:28:44 -07:00
|
|
|
|
table.insert(self.table, '^'..self.cmd_pat..pattern..'%s+[^%s]*')
|
|
|
|
|
table.insert(self.table, '^'..self.cmd_pat..pattern..'@'..username..'%s+[^%s]*')
|
2016-04-08 14:12:02 -07:00
|
|
|
|
end
|
|
|
|
|
return self
|
|
|
|
|
end
|
|
|
|
|
|
2016-05-26 20:28:44 -07:00
|
|
|
|
function utilities.triggers(username, cmd_pat, trigger_table)
|
2016-04-13 20:48:20 -07:00
|
|
|
|
local self = setmetatable({}, utilities.triggers_meta)
|
2016-04-08 14:12:02 -07:00
|
|
|
|
self.username = username
|
2016-05-26 20:28:44 -07:00
|
|
|
|
self.cmd_pat = cmd_pat
|
2016-04-10 21:04:47 -07:00
|
|
|
|
self.table = trigger_table or {}
|
2016-04-08 14:12:02 -07:00
|
|
|
|
return self
|
2016-04-01 13:29:00 -04:00
|
|
|
|
end
|
2016-04-10 21:04:47 -07:00
|
|
|
|
|
|
|
|
|
function utilities.with_http_timeout(timeout, fun)
|
2016-07-31 21:29:44 +02:00
|
|
|
|
local original = http.TIMEOUT
|
|
|
|
|
http.TIMEOUT = timeout
|
2016-04-10 21:04:47 -07:00
|
|
|
|
fun()
|
2016-07-31 21:29:44 +02:00
|
|
|
|
http.TIMEOUT = original
|
2016-04-10 21:04:47 -07:00
|
|
|
|
end
|
2016-04-12 05:24:56 -04:00
|
|
|
|
|
2016-04-12 06:47:30 -07:00
|
|
|
|
function utilities.enrich_user(user)
|
2016-04-12 05:24:56 -04:00
|
|
|
|
user.id_str = tostring(user.id)
|
2016-04-12 06:47:30 -07:00
|
|
|
|
user.name = utilities.build_name(user.first_name, user.last_name)
|
2016-04-12 05:24:56 -04:00
|
|
|
|
return user
|
|
|
|
|
end
|
|
|
|
|
|
2016-04-12 06:47:30 -07:00
|
|
|
|
function utilities.enrich_message(msg)
|
2016-04-12 05:24:56 -04:00
|
|
|
|
if not msg.text then msg.text = msg.caption or '' end
|
|
|
|
|
msg.text_lower = msg.text:lower()
|
2016-04-12 06:47:30 -07:00
|
|
|
|
msg.from = utilities.enrich_user(msg.from)
|
2016-04-12 05:24:56 -04:00
|
|
|
|
msg.chat.id_str = tostring(msg.chat.id)
|
|
|
|
|
if msg.reply_to_message then
|
|
|
|
|
if not msg.reply_to_message.text then
|
|
|
|
|
msg.reply_to_message.text = msg.reply_to_message.caption or ''
|
|
|
|
|
end
|
|
|
|
|
msg.reply_to_message.text_lower = msg.reply_to_message.text:lower()
|
2016-04-12 06:47:30 -07:00
|
|
|
|
msg.reply_to_message.from = utilities.enrich_user(msg.reply_to_message.from)
|
2016-04-12 05:24:56 -04:00
|
|
|
|
msg.reply_to_message.chat.id_str = tostring(msg.reply_to_message.chat.id)
|
|
|
|
|
end
|
|
|
|
|
if msg.forward_from then
|
2016-04-12 06:47:30 -07:00
|
|
|
|
msg.forward_from = utilities.enrich_user(msg.forward_from)
|
2016-04-12 05:24:56 -04:00
|
|
|
|
end
|
|
|
|
|
if msg.new_chat_participant then
|
2016-04-12 06:47:30 -07:00
|
|
|
|
msg.new_chat_participant = utilities.enrich_user(msg.new_chat_participant)
|
2016-04-12 05:24:56 -04:00
|
|
|
|
end
|
|
|
|
|
if msg.left_chat_participant then
|
2016-04-12 06:47:30 -07:00
|
|
|
|
msg.left_chat_participant = utilities.enrich_user(msg.left_chat_participant)
|
2016-04-12 05:24:56 -04:00
|
|
|
|
end
|
|
|
|
|
return msg
|
|
|
|
|
end
|
2016-04-13 20:48:20 -07:00
|
|
|
|
|
2016-04-15 19:07:23 +00:00
|
|
|
|
function utilities.pretty_float(x)
|
|
|
|
|
if x % 1 == 0 then
|
|
|
|
|
return tostring(math.floor(x))
|
|
|
|
|
else
|
|
|
|
|
return tostring(x)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2016-05-20 03:38:43 -04:00
|
|
|
|
-- This table will store unsavory characters that are not properly displayed,
|
|
|
|
|
-- or are just not fun to type.
|
|
|
|
|
utilities.char = {
|
|
|
|
|
zwnj = '',
|
|
|
|
|
arabic = '[\216-\219][\128-\191]',
|
2016-05-22 16:08:45 -04:00
|
|
|
|
rtl_override = '',
|
2016-05-25 09:01:54 -04:00
|
|
|
|
rtl_mark = '',
|
2016-08-14 16:30:06 +02:00
|
|
|
|
em_dash = '—',
|
|
|
|
|
utf_8 = '[%z\1-\127\194-\244][\128-\191]',
|
2016-05-20 03:38:43 -04:00
|
|
|
|
}
|
|
|
|
|
|
2016-06-18 12:51:13 +02:00
|
|
|
|
-- 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
|
|
|
|
|
|
|
|
|
|
-- Returns at table of lua files inside plugins
|
|
|
|
|
function plugins_names()
|
|
|
|
|
local files = {}
|
|
|
|
|
for k, v in pairs(scandir("otouto/plugins")) do
|
|
|
|
|
-- Ends with .lua
|
|
|
|
|
if (v:match(".lua$")) then
|
2016-08-07 20:45:51 +02:00
|
|
|
|
files[#files+1] = v
|
2016-06-18 12:51:13 +02:00
|
|
|
|
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
|
|
|
|
|
|
2016-06-13 22:18:36 +02:00
|
|
|
|
-- Returns a table with matches or nil
|
|
|
|
|
function match_pattern(pattern, text)
|
|
|
|
|
if text then
|
|
|
|
|
local matches = { string.match(text, pattern) }
|
|
|
|
|
if next(matches) then
|
|
|
|
|
return matches
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
-- nil
|
|
|
|
|
end
|
|
|
|
|
|
2016-06-17 20:44:28 +02:00
|
|
|
|
function is_sudo(msg, config)
|
|
|
|
|
local var = false
|
|
|
|
|
-- Check if user id is sudoer
|
|
|
|
|
if config.admin == msg.from.id then
|
|
|
|
|
var = true
|
|
|
|
|
end
|
|
|
|
|
return var
|
|
|
|
|
end
|
|
|
|
|
|
2016-08-01 21:51:37 +02:00
|
|
|
|
function service_modify_msg(msg)
|
|
|
|
|
if msg.new_chat_member then
|
|
|
|
|
msg.text = '//tgservice new_chat_member'
|
|
|
|
|
msg.text_lower = msg.text
|
|
|
|
|
elseif msg.left_chat_member then
|
|
|
|
|
msg.text = '//tgservice left_chat_member'
|
|
|
|
|
msg.text_lower = msg.text
|
|
|
|
|
elseif msg.new_chat_title then
|
|
|
|
|
msg.text = '//tgservice new_chat_title'
|
|
|
|
|
msg.text_lower = msg.text
|
|
|
|
|
elseif msg.new_chat_photo then
|
|
|
|
|
msg.text = '//tgservice new_chat_photo'
|
|
|
|
|
msg.text_lower = msg.text
|
|
|
|
|
elseif msg.group_chat_created then
|
|
|
|
|
msg.text = '//tgservice group_chat_created'
|
|
|
|
|
msg.text_lower = msg.text
|
|
|
|
|
elseif msg.supergroup_chat_created then
|
|
|
|
|
msg.text = '//tgservice supergroup_chat_created'
|
|
|
|
|
msg.text_lower = msg.text
|
|
|
|
|
elseif msg.channel_chat_created then
|
|
|
|
|
msg.text = '//tgservice channel_chat_created'
|
|
|
|
|
msg.text_lower = msg.text
|
|
|
|
|
elseif msg.migrate_to_chat_id then
|
|
|
|
|
msg.text = '//tgservice migrate_to_chat_id'
|
|
|
|
|
msg.text_lower = msg.text
|
|
|
|
|
elseif msg.migrate_from_chat_id then
|
|
|
|
|
msg.text = '//tgservice migrate_from_chat_id'
|
|
|
|
|
msg.text_lower = msg.text
|
|
|
|
|
end
|
|
|
|
|
return msg
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
function is_service_msg(msg)
|
|
|
|
|
local var = false
|
|
|
|
|
if msg.new_chat_member then
|
|
|
|
|
var = true
|
|
|
|
|
elseif msg.left_chat_member then
|
|
|
|
|
var = true
|
|
|
|
|
elseif msg.new_chat_title then
|
|
|
|
|
var = true
|
|
|
|
|
elseif msg.new_chat_photo then
|
|
|
|
|
var = true
|
|
|
|
|
elseif msg.group_chat_created then
|
|
|
|
|
var = true
|
|
|
|
|
elseif msg.supergroup_chat_created then
|
|
|
|
|
var = true
|
|
|
|
|
elseif msg.channel_chat_created then
|
|
|
|
|
var = true
|
|
|
|
|
elseif msg.migrate_to_chat_id then
|
|
|
|
|
var = true
|
|
|
|
|
elseif msg.migrate_from_chat_id then
|
|
|
|
|
var = true
|
|
|
|
|
end
|
|
|
|
|
return var
|
|
|
|
|
end
|
|
|
|
|
|
2016-08-27 23:31:53 +02:00
|
|
|
|
-- Make a POST request
|
|
|
|
|
-- URL = obvious
|
|
|
|
|
-- Arguments = Things, that go into the body. If sending a file, use 'io.open('path/to/file', "r")'
|
|
|
|
|
-- Headers = Header table. If not set, we will set a few!
|
2016-06-14 13:14:09 +02:00
|
|
|
|
function post_petition(url, arguments, headers)
|
|
|
|
|
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 = headers or {},
|
|
|
|
|
redirect = false
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
local source = arguments
|
|
|
|
|
if type(arguments) == "table" then
|
2016-07-26 16:39:18 +02:00
|
|
|
|
source = helpers.url_encode_arguments(arguments)
|
2016-06-14 13:14:09 +02:00
|
|
|
|
end
|
2016-07-26 16:39:18 +02:00
|
|
|
|
|
2016-06-14 13:14:09 +02:00
|
|
|
|
if not headers then
|
|
|
|
|
request_constructor.headers["Content-Type"] = "application/x-www-form-urlencoded; charset=UTF8"
|
|
|
|
|
request_constructor.headers["X-Accept"] = "application/json"
|
|
|
|
|
request_constructor.headers["Accept"] = "application/json"
|
|
|
|
|
end
|
2016-08-27 23:31:53 +02:00
|
|
|
|
if type(arguments) == 'userdata' then
|
|
|
|
|
request_constructor.headers["Content-Length"] = get_file_size(source)
|
|
|
|
|
request_constructor.source = ltn12.source.file(source)
|
|
|
|
|
else
|
|
|
|
|
request_constructor.headers["Content-Length"] = tostring(#source)
|
|
|
|
|
request_constructor.source = ltn12.source.string(source)
|
|
|
|
|
end
|
2016-06-14 13:14:09 +02:00
|
|
|
|
|
|
|
|
|
if post_prot == "http" then
|
|
|
|
|
ok, response_code, response_headers, response_status_line = http.request(request_constructor)
|
|
|
|
|
else
|
2016-07-31 21:29:44 +02:00
|
|
|
|
ok, response_code, response_headers, response_status_line = https.request(request_constructor)
|
2016-06-14 13:14:09 +02:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
if not ok then
|
|
|
|
|
return nil
|
|
|
|
|
end
|
|
|
|
|
|
2016-07-31 21:29:44 +02:00
|
|
|
|
response_body = json.decode(table.concat(response_body))
|
2016-06-14 13:14:09 +02:00
|
|
|
|
|
|
|
|
|
return response_body, response_headers
|
|
|
|
|
end
|
|
|
|
|
|
2016-06-11 14:46:41 +02:00
|
|
|
|
function get_redis_hash(msg, var)
|
|
|
|
|
if msg.chat.type == 'group' or msg.chat.type == 'supergroup' then
|
|
|
|
|
return 'chat:'..msg.chat.id..':'..var
|
|
|
|
|
end
|
|
|
|
|
if msg.chat.type == 'private' then
|
|
|
|
|
return 'user:'..msg.from.id..':'..var
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- remove whitespace
|
|
|
|
|
function all_trim(s)
|
|
|
|
|
return s:match( "^%s*(.-)%s*$" )
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
function tablelength(T)
|
|
|
|
|
local count = 0
|
|
|
|
|
for _ in pairs(T) do count = count + 1 end
|
|
|
|
|
return count
|
|
|
|
|
end
|
|
|
|
|
|
2016-06-15 01:16:27 +02:00
|
|
|
|
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
|
|
|
|
|
|
2016-06-11 14:46:41 +02:00
|
|
|
|
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
|
|
|
|
|
|
2016-06-14 16:17:13 +02:00
|
|
|
|
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
|
|
|
|
|
|
2016-08-09 02:13:13 +02:00
|
|
|
|
function cache_data(plugin, query, data, timeout, typ, hash_field)
|
|
|
|
|
-- How to: cache_data(pluginname, query_name, data_to_cache, expire_in_seconds, type, hash_field (if hash))
|
2016-06-11 14:46:41 +02:00
|
|
|
|
local hash = 'telegram:cache:'..plugin..':'..query
|
|
|
|
|
if timeout then
|
|
|
|
|
print('Caching "'..query..'" from plugin '..plugin..' (expires in '..timeout..' seconds)')
|
|
|
|
|
else
|
|
|
|
|
print('Caching "'..query..'" from plugin '..plugin..' (expires never)')
|
|
|
|
|
end
|
|
|
|
|
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
|
2016-08-19 14:05:16 +02:00
|
|
|
|
redis:hmset(hash, data)
|
2016-06-11 14:46:41 +02:00
|
|
|
|
end
|
|
|
|
|
if timeout then
|
|
|
|
|
redis:expire(hash, timeout)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2016-07-05 19:21:37 +02:00
|
|
|
|
-- Caches file_id and last_modified
|
|
|
|
|
-- result = result of send_X() (see media.lua)
|
|
|
|
|
function cache_file(result, url, last_modified)
|
|
|
|
|
local hash = 'telegram:cache:sent_file'
|
|
|
|
|
if result.result.video then
|
|
|
|
|
file_id = result.result.video.file_id
|
|
|
|
|
elseif result.result.audio then
|
|
|
|
|
file_id = result.result.audio.file_id
|
|
|
|
|
elseif result.result.voice then
|
|
|
|
|
file_id = result.result.voice.file_id
|
|
|
|
|
elseif result.result.document then
|
|
|
|
|
file_id = result.result.document.file_id
|
|
|
|
|
elseif result.result.photo then
|
|
|
|
|
local lv = #result.result.photo
|
|
|
|
|
file_id = result.result.photo[lv].file_id
|
2016-08-13 16:51:32 +02:00
|
|
|
|
elseif result.result.sticker then
|
|
|
|
|
file_id = result.result.sticker.file_id
|
2016-07-05 19:21:37 +02:00
|
|
|
|
end
|
|
|
|
|
print('Caching File...')
|
|
|
|
|
redis:hset(hash..':'..url, 'file_id', file_id)
|
|
|
|
|
redis:hset(hash..':'..url, 'last_modified', last_modified)
|
|
|
|
|
-- Why do we set a TTL? Because Telegram recycles outgoing file_id's
|
|
|
|
|
-- See: https://core.telegram.org/bots/faq#can-i-count-on-file-ids-to-be-persistent
|
|
|
|
|
redis:expire(hash..':'..url, 5259600) -- 2 months
|
|
|
|
|
end
|
|
|
|
|
|
2016-07-23 13:58:15 +02:00
|
|
|
|
function get_http_header(url)
|
2016-08-01 01:20:45 +02:00
|
|
|
|
local doer = http
|
2016-07-05 19:21:37 +02:00
|
|
|
|
local do_redir = true
|
|
|
|
|
if url:match('^https') then
|
2016-07-31 21:29:44 +02:00
|
|
|
|
doer = https
|
2016-07-05 19:21:37 +02:00
|
|
|
|
do_redir = false
|
|
|
|
|
end
|
|
|
|
|
local _, code, header = doer.request {
|
|
|
|
|
method = "HEAD",
|
|
|
|
|
url = url,
|
|
|
|
|
redirect = do_redir
|
|
|
|
|
}
|
2016-07-07 00:31:19 +02:00
|
|
|
|
if not header then return end
|
2016-07-23 13:58:15 +02:00
|
|
|
|
return header, code
|
2016-07-05 19:21:37 +02:00
|
|
|
|
end
|
|
|
|
|
|
2016-08-03 18:35:58 +02:00
|
|
|
|
-- checks with If-Modified-Since header, if url has been changed
|
|
|
|
|
-- URL and Last-Modified heder are required
|
|
|
|
|
function was_modified_since(url, last_modified)
|
|
|
|
|
local doer = http
|
|
|
|
|
local do_redir = true
|
|
|
|
|
if url:match('^https') then
|
|
|
|
|
doer = https
|
|
|
|
|
do_redir = false
|
|
|
|
|
end
|
|
|
|
|
local _, code, header = doer.request {
|
|
|
|
|
url = url,
|
|
|
|
|
method = "HEAD",
|
|
|
|
|
redirect = do_redir,
|
|
|
|
|
headers = {
|
|
|
|
|
["If-Modified-Since"] = last_modified
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if code == 304 then
|
|
|
|
|
return false, nil, code
|
|
|
|
|
else
|
|
|
|
|
if header["last-modified"] then
|
|
|
|
|
new_last_modified = header["last-modified"]
|
|
|
|
|
elseif header["Last-Modified"] then
|
|
|
|
|
new_last_modified = header["Last-Modified"]
|
|
|
|
|
end
|
|
|
|
|
return true, new_last_modified, code
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2016-07-05 19:21:37 +02:00
|
|
|
|
-- only url is needed!
|
2016-08-24 15:38:29 +02:00
|
|
|
|
function get_cached_file(url, file_name, receiver, chat_action)
|
2016-07-05 19:21:37 +02:00
|
|
|
|
local hash = 'telegram:cache:sent_file'
|
|
|
|
|
local cached_file_id = redis:hget(hash..':'..url, 'file_id')
|
|
|
|
|
local cached_last_modified = redis:hget(hash..':'..url, 'last_modified')
|
|
|
|
|
|
2016-08-03 18:35:58 +02:00
|
|
|
|
if cached_last_modified then
|
|
|
|
|
was_modified, new_last_modified, code = was_modified_since(url, cached_last_modified)
|
|
|
|
|
if not was_modified then
|
|
|
|
|
print('File wasn\'t modified, skipping download...')
|
|
|
|
|
return cached_file_id, nil, true
|
|
|
|
|
else
|
|
|
|
|
if code ~= 200 then
|
|
|
|
|
redis:del(hash..':'..url)
|
|
|
|
|
return
|
|
|
|
|
end
|
|
|
|
|
print('File was modified, redownloading...')
|
2016-08-24 15:38:29 +02:00
|
|
|
|
if receiver and chat_action then
|
|
|
|
|
utilities.send_typing(receiver, chat_action)
|
2016-08-03 18:35:58 +02:00
|
|
|
|
end
|
|
|
|
|
file = download_to_file(url, file_name)
|
|
|
|
|
return file, new_last_modified, false
|
2016-07-05 19:21:37 +02:00
|
|
|
|
end
|
|
|
|
|
end
|
2016-07-23 13:58:15 +02:00
|
|
|
|
|
2016-08-03 18:35:58 +02:00
|
|
|
|
-- get last-modified and Content-Length header
|
|
|
|
|
local header, code = get_http_header(url)
|
|
|
|
|
|
2016-07-23 13:58:15 +02:00
|
|
|
|
-- file size limit is 50 MB
|
2016-08-27 14:46:47 +02:00
|
|
|
|
if header then
|
|
|
|
|
|
|
|
|
|
if header["Content-Length"] then
|
|
|
|
|
if tonumber(header["Content-Length"]) > 52420000 then
|
|
|
|
|
print('file is too large, won\'t send!')
|
|
|
|
|
return nil
|
|
|
|
|
end
|
|
|
|
|
elseif header["content-length"] then
|
|
|
|
|
if tonumber(header["content-length"]) > 52420000 then
|
|
|
|
|
print('file is too large, won\'t send!')
|
|
|
|
|
return nil
|
|
|
|
|
end
|
2016-07-23 13:58:15 +02:00
|
|
|
|
end
|
2016-08-27 14:46:47 +02:00
|
|
|
|
|
|
|
|
|
if header["last-modified"] then
|
|
|
|
|
last_modified = header["last-modified"]
|
|
|
|
|
elseif header["Last-Modified"] then
|
|
|
|
|
last_modified = header["Last-Modified"]
|
2016-07-23 13:58:15 +02:00
|
|
|
|
end
|
2016-08-03 18:35:58 +02:00
|
|
|
|
|
2016-08-27 14:46:47 +02:00
|
|
|
|
else
|
|
|
|
|
last_modified = nil
|
2016-07-23 13:58:15 +02:00
|
|
|
|
end
|
2016-07-05 19:21:37 +02:00
|
|
|
|
|
|
|
|
|
if not last_modified then
|
|
|
|
|
nocache = true
|
|
|
|
|
else
|
|
|
|
|
nocache = false
|
|
|
|
|
end
|
|
|
|
|
|
2016-08-24 15:38:29 +02:00
|
|
|
|
if receiver and chat_action then
|
|
|
|
|
utilities.send_typing(receiver, chat_action)
|
2016-07-05 19:21:37 +02:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
if not nocache then
|
2016-08-03 18:35:58 +02:00
|
|
|
|
file = download_to_file(url, file_name)
|
2016-07-05 19:21:37 +02:00
|
|
|
|
else
|
|
|
|
|
print('No Last-Modified header!')
|
|
|
|
|
file = download_to_file(url, file_name)
|
|
|
|
|
end
|
|
|
|
|
return file, last_modified, nocache
|
|
|
|
|
end
|
|
|
|
|
|
2016-06-11 14:46:41 +02:00
|
|
|
|
-- 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
|
2016-08-02 16:19:20 +02:00
|
|
|
|
if seconds == 1 then
|
|
|
|
|
return seconds..' Sekunde'
|
|
|
|
|
else
|
|
|
|
|
return seconds..' Sekunden'
|
|
|
|
|
end
|
2016-06-11 14:46:41 +02:00
|
|
|
|
elseif hours == 00 and minutes ~= 00 then
|
2016-08-02 16:19:20 +02:00
|
|
|
|
if minutes == 1 then
|
|
|
|
|
return string.format("%02d:%02d", minutes, seconds)..' Minute'
|
|
|
|
|
else
|
|
|
|
|
return string.format("%02d:%02d", minutes, seconds)..' Minuten'
|
|
|
|
|
end
|
2016-06-11 14:46:41 +02:00
|
|
|
|
elseif hours ~= 00 then
|
2016-08-02 16:19:20 +02:00
|
|
|
|
if hours == 1 then
|
|
|
|
|
return string.format("%02d:%02d:%02d", hours, minutes, seconds)..' Stunde'
|
|
|
|
|
else
|
|
|
|
|
return string.format("%02d:%02d:%02d", hours, minutes, seconds)..' Stunden'
|
|
|
|
|
end
|
2016-06-11 14:46:41 +02:00
|
|
|
|
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 unescape(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, "Uuml;", "Ü")
|
|
|
|
|
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
|
|
|
|
|
|
2016-06-15 18:00:59 +02:00
|
|
|
|
function url_encode(str)
|
|
|
|
|
if (str) then
|
|
|
|
|
str = string.gsub (str, "\n", "\r\n")
|
|
|
|
|
str = string.gsub (str, "([^%w %-%_%.%~])",
|
|
|
|
|
function (c) return string.format ("%%%02X", string.byte(c)) end)
|
|
|
|
|
str = string.gsub (str, " ", "+")
|
|
|
|
|
end
|
|
|
|
|
return str
|
|
|
|
|
end
|
|
|
|
|
|
2016-07-05 22:26:46 +02:00
|
|
|
|
function table.contains(table, element)
|
|
|
|
|
for _, value in pairs(table) do
|
|
|
|
|
if value == element then
|
|
|
|
|
return true
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
return false
|
|
|
|
|
end
|
|
|
|
|
|
2016-08-28 18:12:18 +02:00
|
|
|
|
-- Checks if bot was disabled on specific chat
|
|
|
|
|
function is_channel_disabled(msg)
|
|
|
|
|
local hash = 'chat:'..msg.chat.id..':disabled'
|
|
|
|
|
local disabled = redis:get(hash)
|
|
|
|
|
|
|
|
|
|
if not disabled or disabled == "false" then
|
|
|
|
|
return false
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
return disabled
|
|
|
|
|
end
|
|
|
|
|
|
2016-09-04 17:11:43 -04:00
|
|
|
|
-- Converts a gross string back into proper UTF-8.
|
|
|
|
|
-- Useful for fixing improper encoding caused by bad JSON escaping.
|
|
|
|
|
function utilities.fix_utf8(str)
|
2016-09-06 20:34:57 -04:00
|
|
|
|
return string.char(utf8.codepoint(str, 1, -1))
|
2016-09-04 21:04:45 +02:00
|
|
|
|
end
|
|
|
|
|
|
2016-09-07 13:07:36 +02:00
|
|
|
|
|
2016-09-07 15:55:57 +02:00
|
|
|
|
return utilities
|