diff --git a/README.md b/README.md index 2ae0797..9966e91 100755 --- a/README.md +++ b/README.md @@ -1,8 +1,6 @@ # otouto The plugin-wielding, multipurpose Telegram bot. -**This readme is outdated as of 3.7. Proceed with caution.** - [Public Bot](http://telegram.me/mokubot) | [Official Channel](http://telegram.me/otouto) | [Development Group](http://telegram.me/BotDevelopment) otouto is an independently-developed Telegram API bot written in Lua. Originally conceived as a CLI script in February of 2015, otouto has since been open-sourced and migrated to the API, and is being developed to this day. @@ -19,7 +17,7 @@ otouto is an independently-developed Telegram API bot written in Lua. Originally | [Contributors](#contributors) | ## Setup -You _must_ have Lua (5.2+), lua-socket, lua-sec, and dkjson installed. It is recommended you install these with LuaRocks. To upload files, you must have curl installed. To use fortune.lua, you must have fortune installed. +You _must_ have Lua (5.2+), luasocket, luasec, and dkjson installed. You should also have lpeg, though it is not required. It is recommended you install these with LuaRocks. To upload photos and other files, you must have curl installed. To use fortune.lua, you must have fortune installed. Clone the repository and set the following values in `config.lua`: @@ -32,7 +30,7 @@ Optionally: Some plugins are not enabled by default. If you wish to enable them, add them to the `plugins` array. -When you are ready to start the bot, run `./launch.sh`. To stop the bot, send "/halt" through Telegram. If you terminate the bot manually, you risk data loss. If you do you not want the bot to restart automatically, run it with `lua bot.lua`. +When you are ready to start the bot, run `./launch.sh`. To stop the bot, send "/halt" through Telegram. If you terminate the bot manually, you risk data loss. If you do you not want the bot to restart automatically, run it with `lua main.lua`. Note that certain plugins, such as translate.lua and greetings.lua, will require privacy mode to be disabled. Additionally, some plugins may require or make use of various API keys: @@ -54,21 +52,22 @@ Most plugins are intended for public use, but a few are for other purposes, like A plugin can have five components, and two of them are required: -| Component | Description | Required? | -|:----------|:---------------------------------------------|:----------| -| action | Main function. Expects `msg` table as an argument. | Y | -| triggers | Table of triggers for the plugin. Uses Lua patterns. | Y | -| cron | Optional function to be called every minute. | N | -| command | Basic command and syntax. Listed in the help text. | N | -| doc | Usage for the plugin. Returned by "/help $command". | N | +| Component | Description | Required? | +|:----------------|:---------------------------------------------|:----------| +| plugin:action | Main function. Expects `msg` table as an argument. | Y | +| plugin.triggers | Table of triggers for the plugin. Uses Lua patterns. | Y | +| plugin:init | Optional function run when the plugin is loaded. | N | +| plugin:cron | Optional function to be called every minute. | N | +| plugin.command | Basic command and syntax. Listed in the help text. | N | +| plugin.doc | Usage for the plugin. Returned by "/help $command". | N | -The `on_msg_receive()` function adds a few variables to the `msg` table for your convenience. These are self-explanatory: `msg.from.id_str`, `msg.to.id_str`, `msg.chat.id_str`, `msg.text_lower`, `msg.from.name`. +The `bot:on_msg_receive` function adds a few variables to the `msg` table for your convenience. These are self-explanatory: `msg.from.id_str`, `msg.to.id_str`, `msg.chat.id_str`, `msg.text_lower`, `msg.from.name`. -Return values from `action()` are optional, but they do effect the flow. If it returns a table, that table will become `msg`, and `on_msg_receive` will continue with that. If it returns `true`, it will continue with the current `msg`. +Return values from `plugin:action` are optional, but they do effect the flow. If it returns a table, that table will become `msg`, and `on_msg_receive` will continue with that. If it returns `true`, it will continue with the current `msg`. -When an action or cron function fails, the exception is caught and passed to the `handle_exception()` utilty (in utilities.lua) and is either printed to the console or send to the chat/channel defined in `log_chat` in config.lua. +When an action or cron function fails, the exception is caught and passed to the `handle_exception` utilty and is either printed to the console or send to the chat/channel defined in `log_chat` in config.lua. -Interactions with the bot API are straightforward. Every binding function shares the name of the API method (eg `sendMessage()`). An additional function, `sendReply()`, accepts the `msg` table and a string as an argument, and sends the string as a reply to that message. +Interactions with the bot API are straightforward. Every binding function shares the name of the API method (eg `sendMessage`). An additional function, `sendReply`, accepts the `msg` table and a string as an argument, and sends the string as a reply to that message. Several functions used in multiple plugins are defined in utilities.lua. Refer to that file for usage and documentation. @@ -92,7 +91,7 @@ The administration plugin enables self-hosted, single-realm group administration To get started, run `./tg-install.sh`. Note that this script is written for Ubuntu/Debian. If you're running Arch (the only acceptable alternative), you'll have to do it yourself. If that is the case, note that otouto uses the "test" branch of tg, and the AUR package `telegram-cli-git` will not be sufficient, as it does not have support for supergroups yet. -Once the installation is finished, enable `administration.lua` in your config file. You may have reason to change the default TCP port (4567); if that is the case, remember to change it in `tg-launch.sh` as well. Run `./tg-launch.sh` in a separate screen/tmux window. You'll have to enter your phone number and go through the login process the first time. The script is set to restart tg after two seconds, so you'll need to Ctrl+C after exiting. +Once the installation is finished, enable `administration.lua` in your config file. **The administration plugin must be loaded before about.lua and blacklist.lua.** You may have reason to change the default TCP port (4567); if that is the case, remember to change it in `tg-launch.sh` as well. Run `./tg-launch.sh` in a separate screen/tmux window. You'll have to enter your phone number and go through the login process the first time. The script is set to restart tg after two seconds, so you'll need to Ctrl+C after exiting. While tg is running, you may start/reload otouto with administration.lua enabled, and have access to a wide variety of administrative commands and automata. The administration "database" is stored in `administration.json`. To start using otouto to administrate a group (note that you must be the owner (or an administrator)), send `/gadd` to that group. For a list of commands, use `/ahelp`. Below I'll describe various functions now available to you. @@ -175,23 +174,6 @@ Some plugins are only useful when the bot is used in a Liberbot group, like floo `/floodcontrol {"groupid":987654321,"duration":600}` The bot will accept these commands from both Liberbot and the configured administrator. -**moderation.lua** allows the owner to use the bot to moderate a Liberbot realm, or set of groups. This works by adding the bot to the realm's admin group and making it an administrator. -You must configure the plugin in the "moderation" section of config.lua, in the following way: -```lua -moderation = { - admins = { - ['123456789'] = 'Adam', - ['246813579'] = 'Eve' - }, - admin_group = -987654321, - realm_name = 'My Realm' -} -``` - -Where Adam and Eve are realm administrators, and their IDs are set as their keys in the form of strings. admin_group is the group ID of the admin group, as a negative number. realm_name is the name of your Libebot realm. - -Once this is set up, put your bot in the admin group and run `/modadd` and `/modhelp` to get started. - * * * ## List of plugins @@ -254,9 +236,9 @@ Contributions are appreciated in any form. Monetary contributions will go toward | Contributors | |:----------------------------------------------| +| [bb010g](http://github.com/bb010g) | | [Juan Potato](http://github.com/JuanPotato) | | [Tiago Danin](http://github.com/TiagoDanin) | -| [bb010g](http://github.com/bb010g) | | [Ender](http://github.com/luksireiku) | | [Iman Daneshi](http://github.com/Imandaneshi) | | [HeitorPB](http://github.com/heitorPB) | diff --git a/bot.lua b/bot.lua index a6bea28..c15cbfd 100755 --- a/bot.lua +++ b/bot.lua @@ -36,10 +36,6 @@ function bot:init() -- The function run when the bot is started or reloaded. print('@' .. self.info.username .. ', AKA ' .. self.info.first_name ..' ('..self.info.id..')') - -- Generate a random seed and "pop" the first random number. :) - math.randomseed(os.time()) - math.random() - self.last_update = self.last_update or 0 -- Set loop variables: Update offset, self.last_cron = self.last_cron or os.date('%M') -- the time of the last cron job, self.is_started = true -- and whether or not the bot should be running. @@ -50,16 +46,12 @@ end function bot:on_msg_receive(msg) -- The fn run whenever a message is received. - -- Create a user entry if it does not exist. - if not self.database.users[tostring(msg.from.id)] then - self.database.users[tostring(msg.from.id)] = {} - end - -- Clear things that no longer exist. - self.database.users[tostring(msg.from.id)].username = nil - self.database.users[tostring(msg.from.id)].last_name = nil - -- Wee. - for k,v in pairs(msg.from) do - self.database.users[tostring(msg.from.id)][k] = v + -- Cache user info for those involved. + utilities.create_user_entry(self, msg.from) + if msg.forward_from and msg.forward_from.id ~= msg.from.id then + utilities.create_user_entry(self, msg.forward_from) + elseif msg.reply_to_message and msg.reply_to_message.from.id ~= msg.from.id then + utilities.create_user_entry(self, msg.reply_to_message.from) end if msg.date < os.time() - 5 then return end -- Do not process old messages. @@ -82,10 +74,10 @@ function bot:on_msg_receive(msg) -- The fn run whenever a message is received. utilities.handle_exception(self, result, msg.from.id .. ': ' .. msg.text) return end - -- If the action returns a table, make that table msg. + -- If the action returns a table, make that table the new msg. if type(result) == 'table' then msg = result - -- If the action returns true, don't stop. + -- If the action returns true, continue. elseif result ~= true then return end diff --git a/plugins/me.lua b/plugins/me.lua index 7b93984..7db354e 100644 --- a/plugins/me.lua +++ b/plugins/me.lua @@ -12,7 +12,7 @@ function me:action(msg) local target = self.database.users[msg.from.id_str] if msg.from.id == self.config.admin and (msg.reply_to_message or utilities.input(msg.text)) then - target = utilities.user_from_message(self, msg) + target = utilities.user_from_message(self, msg, true) if target.err then bindings.sendReply(self, msg, target.err) return diff --git a/plugins/reactions.lua b/plugins/reactions.lua index a6bcb27..3d472dd 100755 --- a/plugins/reactions.lua +++ b/plugins/reactions.lua @@ -1,7 +1,10 @@ -- Never change this plugin. It was not meant to be changed. -- You may add reactions. You must never remove reactions. -- You must never restructure. You must never disable this plugin. - -- ~ Drew, creator, a year later. + -- - Drew, creator, a year later. + + -- Nevermind, Brayden changed it. + -- - Drew, just now. local reactions = {} diff --git a/utilities.lua b/utilities.lua index ed6c6fe..3a865d0 100755 --- a/utilities.lua +++ b/utilities.lua @@ -154,12 +154,14 @@ function utilities:resolve_username(input) end -function utilities:user_from_message(msg) +function utilities:user_from_message(msg, no_extra) local input = utilities.input(msg.text_lower) local target = {} if msg.reply_to_message then - target = msg.reply_to_message.from + for k,v in pairs(self.database.users[msg.reply_to_message.from.id_str]) do + target[k] = v + end elseif input and tonumber(input) then target.id = tonumber(input) if self.database.users[input] then @@ -183,14 +185,16 @@ function utilities:user_from_message(msg) target.err = 'Please specify a user via reply, ID, or username.' end - if target.id then - target.id_str = tostring(target.id) + if not no_extra then + if target.id then + target.id_str = tostring(target.id) + end + if not target.first_name then + target.first_name = 'User' + end + target.name = utilities.build_name(target.first_name, target.last_name) end - if not target.first_name then target.first_name = 'User' end - - target.name = utilities.build_name(target.first_name, target.last_name) - return target end @@ -313,4 +317,19 @@ function utilities.pretty_float(x) end end +function utilities:create_user_entry(user) + local id = tostring(user.id) + -- Clear things that may no longer exist, or create a user entry. + if self.database.users[id] then + self.database.users[id].username = nil + self.database.users[id].last_name = nil + else + self.database.users[id] = {} + end + -- Add all the user info to the entry. + for k,v in pairs(user) do + self.database.users[id][k] = v + end +end + return utilities