2016-08-23 06:16:32 +02: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-03 00:15:52 +02:00
|
|
|
|
|
2016-04-08 23:12:02 +02:00
|
|
|
|
local utilities = {}
|
|
|
|
|
|
2016-09-07 15:55:57 +02:00
|
|
|
|
utf8 = require('lua-utf8')
|
2016-08-01 21:07:27 +02:00
|
|
|
|
ltn12 = require('ltn12')
|
|
|
|
|
http = require('socket.http')
|
|
|
|
|
https = require('ssl.https')
|
|
|
|
|
socket = require('socket')
|
|
|
|
|
URL = require('socket.url')
|
2016-08-03 19:25:24 +02:00
|
|
|
|
json = require('dkjson')
|
|
|
|
|
pcall(json.use_lpeg)
|
2016-07-31 21:29:44 +02:00
|
|
|
|
serpent = require("serpent")
|
2016-09-11 14:07:55 +02:00
|
|
|
|
redis = (loadfile "./miku/redis.lua")()
|
2016-07-31 21:29:44 +02:00
|
|
|
|
OAuth = require "OAuth"
|
|
|
|
|
helpers = require "OAuth.helpers"
|
2016-09-11 14:14:50 +02:00
|
|
|
|
require('/miku/encoding')
|
2016-08-01 21:07:27 +02:00
|
|
|
|
|
2016-09-09 12:46:07 +02:00
|
|
|
|
http.TIMEOUT = 10
|
|
|
|
|
https.TIMEOUT = 10
|
2016-01-15 04:39:24 +01:00
|
|
|
|
|
2016-05-29 19:08:39 +02: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-14 04:46:18 +02: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 06:16:32 +02:00
|
|
|
|
return bindings.request(
|
|
|
|
|
'sendMessage',
|
|
|
|
|
{
|
2016-08-24 15:38:29 +02:00
|
|
|
|
chat_id = chat_id,
|
2016-08-23 06:16:32 +02: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 06:16:32 +02:00
|
|
|
|
}
|
|
|
|
|
)
|
2016-05-29 19:08:39 +02:00
|
|
|
|
end
|
|
|
|
|
|
2016-07-17 13:22:27 +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-15 23:14:28 +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:44:56 +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 06:16:32 +02: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 06:16:32 +02:00
|
|
|
|
}
|
|
|
|
|
)
|
2016-05-29 19:08:39 +02: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-17 13:22:27 +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-17 13:22:27 +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-17 13:22:27 +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-17 13:22:27 +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-17 13:22:27 +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-17 13:22:27 +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-17 13:22:27 +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-17 13:22:27 +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-17 13:22:27 +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-17 13:22:27 +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-17 13:22:27 +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-17 13:22:27 +02:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- 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-17 13:22:27 +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-17 13:22:27 +02:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- 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-17 13:22:27 +02:00
|
|
|
|
end
|
|
|
|
|
|
2016-01-12 11:22:28 +01:00
|
|
|
|
-- get the indexed word in a string
|
2016-04-08 23:12:02 +02:00
|
|
|
|
function utilities.get_word(s, i)
|
2016-08-15 23:14:28 +02:00
|
|
|
|
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
|
2015-11-25 03:22:04 +01:00
|
|
|
|
end
|
2015-07-29 00:13:46 +02:00
|
|
|
|
|
2016-01-12 11:22:28 +01:00
|
|
|
|
-- Returns the string after the first space.
|
2016-04-08 23:12:02 +02:00
|
|
|
|
function utilities.input(s)
|
|
|
|
|
if not s:find(' ') then
|
2015-07-29 00:13:46 +02:00
|
|
|
|
return false
|
|
|
|
|
end
|
2016-04-08 23:12:02 +02:00
|
|
|
|
return s:sub(s:find(' ')+1)
|
2015-07-29 00:13:46 +02:00
|
|
|
|
end
|
|
|
|
|
|
2016-08-15 23:14:28 +02:00
|
|
|
|
function utilities.input_from_msg(msg)
|
|
|
|
|
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
|
|
|
|
|
end
|
|
|
|
|
|
2016-07-17 13:22:27 +02:00
|
|
|
|
-- Trims whitespace from a string.
|
|
|
|
|
function utilities.trim(str)
|
2016-04-08 23:12:02 +02:00
|
|
|
|
local s = str:gsub('^%s*(.-)%s*$', '%1')
|
2015-11-25 03:22:04 +01:00
|
|
|
|
return s
|
2015-07-03 00:15:52 +02:00
|
|
|
|
end
|
|
|
|
|
|
2016-09-08 01:35:10 +02:00
|
|
|
|
-- Returns true if the string is blank/empty
|
2016-06-14 13:14:09 +02:00
|
|
|
|
function string:isempty()
|
2016-09-08 16:13:47 +02:00
|
|
|
|
self = utilities.trim(self)
|
2016-06-14 13:14:09 +02:00
|
|
|
|
return self == nil or self == ''
|
|
|
|
|
end
|
|
|
|
|
|
2016-06-14 12:21:16 +02:00
|
|
|
|
function get_name(msg)
|
|
|
|
|
local name = msg.from.first_name
|
2016-09-08 01:35:10 +02:00
|
|
|
|
if not name then
|
2016-06-14 12:21:16 +02:00
|
|
|
|
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-09-08 00:28:20 +02:00
|
|
|
|
function convert_timestamp(timestamp, date_format)
|
|
|
|
|
return os.date(date_format, timestamp)
|
2016-06-15 01:16:27 +02:00
|
|
|
|
end
|
|
|
|
|
|
2016-06-11 14:46:41 +02:00
|
|
|
|
-- 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-07-17 13:22:27 +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-08-01 21:07:27 +02:00
|
|
|
|
local doer = http
|
2016-07-17 13:22:27 +02:00
|
|
|
|
local do_redir = true
|
|
|
|
|
if url:match('^https') then
|
2016-08-01 21:07:27 +02:00
|
|
|
|
doer = https
|
2016-07-17 13:22:27 +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-08-15 23:14:28 +02:00
|
|
|
|
-- Loads a JSON file as a table.
|
2016-04-08 23:12:02 +02:00
|
|
|
|
function utilities.load_data(filename)
|
2015-07-29 00:13:46 +02:00
|
|
|
|
local f = io.open(filename)
|
2016-08-15 23:14:28 +02:00
|
|
|
|
if f then
|
|
|
|
|
local s = f:read('*all')
|
|
|
|
|
f:close()
|
|
|
|
|
return json.decode(s)
|
|
|
|
|
else
|
2015-07-29 00:13:46 +02:00
|
|
|
|
return {}
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2016-08-15 23:14:28 +02:00
|
|
|
|
-- Saves a table to a JSON file.
|
2016-04-08 23:12:02 +02:00
|
|
|
|
function utilities.save_data(filename, data)
|
2016-08-01 21:07:27 +02:00
|
|
|
|
local s = json.encode(data)
|
2015-07-29 00:13:46 +02:00
|
|
|
|
local f = io.open(filename, 'w')
|
|
|
|
|
f:write(s)
|
|
|
|
|
f:close()
|
2015-11-25 03:22:04 +01: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-25 03:22:04 +01:00
|
|
|
|
-- Gets coordinates for a location. Used by gMaps.lua, time.lua, weather.lua.
|
2016-05-27 02:59:45 +02:00
|
|
|
|
function utilities.get_coords(input, config)
|
2016-08-15 23:14:28 +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
|
2015-11-25 03:22:04 +01:00
|
|
|
|
|
2016-08-15 23:14:28 +02:00
|
|
|
|
local jdat = json.decode(jstr)
|
|
|
|
|
if jdat.status == 'ZERO_RESULTS' then
|
|
|
|
|
return config.errors.results
|
|
|
|
|
end
|
2015-11-25 03:22:04 +01:00
|
|
|
|
|
2016-08-15 23:14:28 +02:00
|
|
|
|
return {
|
|
|
|
|
lat = jdat.results[1].geometry.location.lat,
|
|
|
|
|
lon = jdat.results[1].geometry.location.lng,
|
|
|
|
|
addr = jdat.results[1].formatted_address
|
|
|
|
|
}
|
2016-01-12 11:22:28 +01:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- Get the number of values in a key/value table.
|
2016-04-08 23:12:02 +02:00
|
|
|
|
function utilities.table_size(tab)
|
2016-01-12 11:22:28 +01:00
|
|
|
|
local i = 0
|
2016-04-08 23:12:02 +02:00
|
|
|
|
for _,_ in pairs(tab) do
|
2016-01-12 11:22:28 +01:00
|
|
|
|
i = i + 1
|
|
|
|
|
end
|
|
|
|
|
return i
|
|
|
|
|
end
|
|
|
|
|
|
2016-04-03 01:20:28 +02:00
|
|
|
|
-- Just an easy way to get a user's full name.
|
2016-05-13 19:22:10 +02:00
|
|
|
|
-- Alternatively, abuse it to concat two strings like I do.
|
2016-04-08 23:12:02 +02:00
|
|
|
|
function utilities.build_name(first, last)
|
2016-04-03 01:20:28 +02:00
|
|
|
|
if last then
|
|
|
|
|
return first .. ' ' .. last
|
|
|
|
|
else
|
|
|
|
|
return first
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2016-08-23 06:16:32 +02: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-15 04:39:24 +01:00
|
|
|
|
end
|
|
|
|
|
|
2016-08-15 23:14:28 +02:00
|
|
|
|
function utilities.md_escape(text)
|
|
|
|
|
return text:gsub('_', '\\_')
|
|
|
|
|
:gsub('%[', '\\['):gsub('%]', '\\]')
|
|
|
|
|
:gsub('%*', '\\*'):gsub('`', '\\`')
|
2016-07-17 13:22:27 +02:00
|
|
|
|
end
|
2016-04-01 19:29:00 +02:00
|
|
|
|
|
2016-08-15 23:14:28 +02:00
|
|
|
|
function utilities.html_escape(text)
|
|
|
|
|
return text:gsub('&', '&'):gsub('<', '<'):gsub('>', '>')
|
|
|
|
|
end
|
2016-04-08 23:12:02 +02:00
|
|
|
|
|
2016-04-14 05:48:20 +02:00
|
|
|
|
utilities.triggers_meta = {}
|
|
|
|
|
utilities.triggers_meta.__index = utilities.triggers_meta
|
|
|
|
|
function utilities.triggers_meta:t(pattern, has_args)
|
2016-04-15 21:07:23 +02:00
|
|
|
|
local username = self.username:lower()
|
2016-05-27 05:28:44 +02:00
|
|
|
|
table.insert(self.table, '^'..self.cmd_pat..pattern..'$')
|
|
|
|
|
table.insert(self.table, '^'..self.cmd_pat..pattern..'@'..username..'$')
|
2016-04-08 23:12:02 +02:00
|
|
|
|
if has_args then
|
2016-05-27 05:28:44 +02: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 23:12:02 +02:00
|
|
|
|
end
|
|
|
|
|
return self
|
|
|
|
|
end
|
|
|
|
|
|
2016-05-27 05:28:44 +02:00
|
|
|
|
function utilities.triggers(username, cmd_pat, trigger_table)
|
2016-04-14 05:48:20 +02:00
|
|
|
|
local self = setmetatable({}, utilities.triggers_meta)
|
2016-04-08 23:12:02 +02:00
|
|
|
|
self.username = username
|
2016-05-27 05:28:44 +02:00
|
|
|
|
self.cmd_pat = cmd_pat
|
2016-04-11 06:04:47 +02:00
|
|
|
|
self.table = trigger_table or {}
|
2016-04-08 23:12:02 +02:00
|
|
|
|
return self
|
2016-04-01 19:29:00 +02:00
|
|
|
|
end
|
2016-04-11 06:04:47 +02:00
|
|
|
|
|
2016-04-12 15:47:30 +02:00
|
|
|
|
function utilities.enrich_user(user)
|
2016-04-12 11:24:56 +02:00
|
|
|
|
user.id_str = tostring(user.id)
|
2016-04-12 15:47:30 +02:00
|
|
|
|
user.name = utilities.build_name(user.first_name, user.last_name)
|
2016-04-12 11:24:56 +02:00
|
|
|
|
return user
|
|
|
|
|
end
|
|
|
|
|
|
2016-04-12 15:47:30 +02:00
|
|
|
|
function utilities.enrich_message(msg)
|
2016-04-12 11:24:56 +02:00
|
|
|
|
if not msg.text then msg.text = msg.caption or '' end
|
|
|
|
|
msg.text_lower = msg.text:lower()
|
2016-04-12 15:47:30 +02:00
|
|
|
|
msg.from = utilities.enrich_user(msg.from)
|
2016-04-12 11:24:56 +02: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 15:47:30 +02:00
|
|
|
|
msg.reply_to_message.from = utilities.enrich_user(msg.reply_to_message.from)
|
2016-04-12 11:24:56 +02:00
|
|
|
|
msg.reply_to_message.chat.id_str = tostring(msg.reply_to_message.chat.id)
|
|
|
|
|
end
|
|
|
|
|
if msg.forward_from then
|
2016-04-12 15:47:30 +02:00
|
|
|
|
msg.forward_from = utilities.enrich_user(msg.forward_from)
|
2016-04-12 11:24:56 +02:00
|
|
|
|
end
|
2016-09-08 01:35:10 +02:00
|
|
|
|
if msg.new_chat_member then
|
|
|
|
|
msg.new_chat_member = utilities.enrich_user(msg.new_chat_member)
|
2016-04-12 11:24:56 +02:00
|
|
|
|
end
|
2016-09-08 01:35:10 +02:00
|
|
|
|
if msg.left_chat_member then
|
|
|
|
|
msg.left_chat_member = utilities.enrich_user(msg.left_chat_member)
|
2016-04-12 11:24:56 +02:00
|
|
|
|
end
|
|
|
|
|
return msg
|
|
|
|
|
end
|
2016-04-14 05:48:20 +02:00
|
|
|
|
|
2016-04-15 21:07:23 +02: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 09:38:43 +02: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 22:08:45 +02:00
|
|
|
|
rtl_override = '',
|
2016-05-25 15:01:54 +02:00
|
|
|
|
rtl_mark = '',
|
2016-08-15 23:14:28 +02:00
|
|
|
|
em_dash = '—',
|
|
|
|
|
utf_8 = '[%z\1-\127\194-\244][\128-\191]',
|
2016-05-20 09:38:43 +02:00
|
|
|
|
}
|
|
|
|
|
|
2016-07-17 13:22:27 +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
|
2016-09-12 00:55:53 +02:00
|
|
|
|
if filename ~= "." and filename ~= ".." then
|
|
|
|
|
i = i + 1
|
|
|
|
|
t[i] = filename
|
|
|
|
|
end
|
2016-07-17 13:22:27 +02:00
|
|
|
|
end
|
|
|
|
|
return t
|
|
|
|
|
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-07-17 13:22:27 +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-03 19:05:05 +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 19:06:16 +02:00
|
|
|
|
source = helpers.url_encode_arguments(arguments)
|
2016-06-14 13:14:09 +02:00
|
|
|
|
end
|
2016-08-15 23:14:28 +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-08-01 21:07:27 +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-08-01 21:07:27 +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
|
|
|
|
|
|
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
|
|
|
|
|
|
2016-09-08 01:35:10 +02:00
|
|
|
|
function string.starts(String,Start)
|
|
|
|
|
return string.sub(String,1,string.len(Start))==Start
|
|
|
|
|
end
|
2016-06-11 14:46:41 +02:00
|
|
|
|
|
|
|
|
|
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-15 23:14:28 +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-17 13:22:27 +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-15 23:14:28 +02:00
|
|
|
|
elseif result.result.sticker then
|
|
|
|
|
file_id = result.result.sticker.file_id
|
2016-07-17 13:22:27 +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 20:04:22 +02:00
|
|
|
|
function get_http_header(url)
|
2016-08-01 21:07:27 +02:00
|
|
|
|
local doer = http
|
2016-07-17 13:22:27 +02:00
|
|
|
|
local do_redir = true
|
|
|
|
|
if url:match('^https') then
|
2016-08-01 21:07:27 +02:00
|
|
|
|
doer = https
|
2016-07-17 13:22:27 +02:00
|
|
|
|
do_redir = false
|
|
|
|
|
end
|
|
|
|
|
local _, code, header = doer.request {
|
|
|
|
|
method = "HEAD",
|
|
|
|
|
url = url,
|
|
|
|
|
redirect = do_redir
|
|
|
|
|
}
|
|
|
|
|
if not header then return end
|
2016-07-23 20:04:22 +02:00
|
|
|
|
return header, code
|
2016-06-11 14:46:41 +02:00
|
|
|
|
end
|
|
|
|
|
|
2016-08-03 19:05:05 +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-17 13:22:27 +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-17 13:22:27 +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-06-11 14:46:41 +02:00
|
|
|
|
|
2016-08-03 19:05:05 +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 19:05:05 +02:00
|
|
|
|
end
|
|
|
|
|
file = download_to_file(url, file_name)
|
|
|
|
|
return file, new_last_modified, false
|
2016-07-17 13:22:27 +02:00
|
|
|
|
end
|
|
|
|
|
end
|
2016-07-23 20:04:22 +02:00
|
|
|
|
|
2016-08-03 19:05:05 +02:00
|
|
|
|
-- get last-modified and Content-Length header
|
|
|
|
|
local header, code = get_http_header(url)
|
|
|
|
|
|
2016-07-23 20:04:22 +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 20:04:22 +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 20:04:22 +02:00
|
|
|
|
end
|
2016-08-15 23:14:28 +02:00
|
|
|
|
|
2016-08-27 14:46:47 +02:00
|
|
|
|
else
|
|
|
|
|
last_modified = nil
|
2016-07-23 20:04:22 +02:00
|
|
|
|
end
|
2016-07-17 13:22:27 +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-17 13:22:27 +02:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
if not nocache then
|
2016-08-03 19:05:05 +02:00
|
|
|
|
file = download_to_file(url, file_name)
|
2016-07-17 13:22:27 +02:00
|
|
|
|
else
|
|
|
|
|
print('No Last-Modified header!')
|
|
|
|
|
file = download_to_file(url, file_name)
|
|
|
|
|
end
|
|
|
|
|
return file, last_modified, nocache
|
2016-06-11 14:46:41 +02:00
|
|
|
|
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
|
2016-08-03 19:05:05 +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-03 19:05:05 +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-03 19:05:05 +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
|
2016-07-17 13:22:27 +02:00
|
|
|
|
|
|
|
|
|
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
|
2016-06-11 14:46:41 +02:00
|
|
|
|
|
2016-07-17 13:22:27 +02:00
|
|
|
|
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")
|
2016-08-23 12:49:27 +02:00
|
|
|
|
str = string.gsub(str, "NC%-17", "FSK18")
|
2016-07-17 13:22:27 +02:00
|
|
|
|
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
|
|
|
|
|
|
|
|
|
|
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 23:11:43 +02: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-07 02:34:57 +02: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-12 00:55:53 +02:00
|
|
|
|
return utilities
|