mirror of
				https://github.com/minetest-mods/irc.git
				synced 2025-10-25 19:55:24 +02:00 
			
		
		
		
	Compare commits
	
		
			53 Commits
		
	
	
		
			split_mess
			...
			ee81559532
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| ee81559532 | |||
|  | 7fbbfd6cdb | ||
|  | cc78f12a4c | ||
| 0f46251d00 | |||
|  | ceb40e2bb3 | ||
| 612a94bbf1 | |||
|  | 5e3659761b | ||
|  | ac0387786c | ||
|  | 05ab5e9fbd | ||
| 2a88c47de4 | |||
| 0f00fa8c7e | |||
| 351a86cbf2 | |||
| 1b71c304c5 | |||
| 47f889c178 | |||
|  | 1c23f8987b | ||
|  | 66bb7fc47d | ||
|  | 6a7053582d | ||
|  | 5e877956a5 | ||
|  | 60bf32e157 | ||
|  | b4fbccd64a | ||
|  | c9c57a6f93 | ||
|  | f57bdba5e9 | ||
|  | 070eb51236 | ||
|  | e80bbe3a62 | ||
|  | 1d4fd5a162 | ||
|  | 6bf4e111c2 | ||
|  | 5f8850bc15 | ||
|  | 2f55e0f9a0 | ||
|  | 78fbe26a2c | ||
|  | 33542b07fe | ||
|  | 6bbb26f9f9 | ||
|  | ae0cb08e3e | ||
|  | bb5f549193 | ||
|  | 4c334e9967 | ||
|  | a18820356a | ||
|  | 0c8538bfb8 | ||
|  | 9f8b4fd4af | ||
|  | 73cdb58c99 | ||
|  | 3e2d98f9ef | ||
|  | 12248cc847 | ||
|  | 160088c232 | ||
|  | 03070e41c4 | ||
|  | b5786979ab | ||
|  | d5ad8ffca4 | ||
|  | 0f6e1dacc4 | ||
|  | ecc1d450ba | ||
|  | 18f286f4b5 | ||
|  | 0c47e10a90 | ||
|  | 43493eb2ea | ||
|  | f98282818d | ||
|  | 03edbd29ed | ||
|  | 464f2febee | ||
|  | 0c900dbac2 | 
							
								
								
									
										11
									
								
								.github/workflows/check-release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								.github/workflows/check-release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | on: [push, pull_request] | ||||||
|  | name: Check & Release | ||||||
|  | jobs: | ||||||
|  |   lint: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - uses: actions/checkout@master | ||||||
|  |       - name: lint | ||||||
|  |         uses: Roang-zero1/factorio-mod-luacheck@master | ||||||
|  |         with: | ||||||
|  |           luacheckrc_url: https://raw.githubusercontent.com/minetest-mods/irc/master/.luacheckrc | ||||||
							
								
								
									
										4
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							| @@ -1,3 +1,3 @@ | |||||||
| [submodule "src/LuaIRC"] | [submodule "irc"] | ||||||
| 	path = irc | 	path = irc | ||||||
| 	url = https://github.com/ShadowNinja/LuaIRC.git | 	url = https://sys4.fr/gitea/mtcontrib/LuaIRC.git | ||||||
|   | |||||||
							
								
								
									
										14
									
								
								.luacheckrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								.luacheckrc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | |||||||
|  |  | ||||||
|  | allow_defined_top = true | ||||||
|  |  | ||||||
|  | read_globals = { | ||||||
|  | 	"minetest" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | exclude_files = { | ||||||
|  | 	"irc/*", | ||||||
|  | } | ||||||
|  |  | ||||||
|  | globals = { | ||||||
|  | 	"irc", | ||||||
|  | } | ||||||
							
								
								
									
										16
									
								
								API.md
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								API.md
									
									
									
									
									
								
							| @@ -13,14 +13,14 @@ to your mod's `depends.txt` file. | |||||||
| Reference | Reference | ||||||
| --------- | --------- | ||||||
|  |  | ||||||
| irc:say([name,] message) | irc.say([name,] message) | ||||||
| Sends <message> to either the channel (if <name> is nil or not specified), | Sends <message> to either the channel (if <name> is nil or not specified), | ||||||
| or to the given user (if <name> is specified). | or to the given user (if <name> is specified). | ||||||
| Example: | Example: | ||||||
| 	irc:say("Hello, Channel!") | 	irc.say("Hello, Channel!") | ||||||
| 	irc:say("john1234", "How are you?") | 	irc.say("john1234", "How are you?") | ||||||
|  |  | ||||||
| irc:register_bot_command(name, cmdDef) | irc.register_bot_command(name, cmdDef) | ||||||
| 	Registers a new bot command named <name>. | 	Registers a new bot command named <name>. | ||||||
| 	When an user sends a private message to the bot with the command name, the | 	When an user sends a private message to the bot with the command name, the | ||||||
| 	command's function is called. | 	command's function is called. | ||||||
| @@ -38,7 +38,7 @@ irc:register_bot_command(name, cmdDef) | |||||||
| 			end, | 			end, | ||||||
| 		}; | 		}; | ||||||
| 	Example: | 	Example: | ||||||
| 		irc:register_bot_command("hello", { | 		irc.register_bot_command("hello", { | ||||||
| 			params = "", | 			params = "", | ||||||
| 			description = "Greet user", | 			description = "Greet user", | ||||||
| 			func = function(user, param) | 			func = function(user, param) | ||||||
| @@ -55,12 +55,12 @@ irc.joined_players[name] | |||||||
| 		-- Joe is talking on IRC | 		-- Joe is talking on IRC | ||||||
| 	end | 	end | ||||||
|  |  | ||||||
| irc:register_hook(name, func) | irc.register_hook(name, func) | ||||||
| 	Registers a function to be called when an event happens. <name> is the name | 	Registers a function to be called when an event happens. <name> is the name | ||||||
| 	of the event, and <func> is the function to be called. See HOOKS below | 	of the event, and <func> is the function to be called. See HOOKS below | ||||||
| 	for more information | 	for more information | ||||||
| 	Example: | 	Example: | ||||||
| 	irc:register_hook("OnSend", function(line) | 	irc.register_hook("OnSend", function(line) | ||||||
| 		print("SEND: "..line) | 		print("SEND: "..line) | ||||||
| 	end) | 	end) | ||||||
|  |  | ||||||
| @@ -83,7 +83,7 @@ not modify these settings at runtime or you might crash the server! | |||||||
| Hooks | Hooks | ||||||
| ----- | ----- | ||||||
|  |  | ||||||
| The `irc:register_hook` function can register functions to be called | The `irc.register_hook` function can register functions to be called | ||||||
| when some events happen. The events supported are the same as the LuaIRC | when some events happen. The events supported are the same as the LuaIRC | ||||||
| ones with a few added (mostly for internal use). | ones with a few added (mostly for internal use). | ||||||
| See src/LuaIRC/doc/irc.luadoc for more information. | See src/LuaIRC/doc/irc.luadoc for more information. | ||||||
|   | |||||||
							
								
								
									
										166
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										166
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,12 +1,17 @@ | |||||||
|  | [](https://github.com/minetest-mods/irc/actions) | ||||||
|  |  | ||||||
| IRC Mod for Minetest | IRC Mod for Minetest | ||||||
| ==================== | ==================== | ||||||
|  |  | ||||||
| Introduction | Introduction | ||||||
| ------------ | ------------ | ||||||
|  |  | ||||||
| This mod is just a glue between IRC and Minetest. It provides two-way | This mod is just a glue between IRC and Minetest. It provides two-way | ||||||
|  communication between the in-game chat, and an arbitrary IRC channel. |  communication between the in-game chat, and an arbitrary IRC channel. | ||||||
|  |  | ||||||
| The forum topic is at https://forum.minetest.net/viewtopic.php?f=11&t=3905 | The forum topic is [here][forum]. | ||||||
|  |  | ||||||
|  | [forum]: https://forum.minetest.net/viewtopic.php?f=11&t=3905 | ||||||
|  |  | ||||||
|  |  | ||||||
| Installing | Installing | ||||||
| @@ -14,109 +19,121 @@ Installing | |||||||
|  |  | ||||||
| Quick one line install for Linux: | Quick one line install for Linux: | ||||||
|  |  | ||||||
| 	cd <Mods directory> && git clone --recursive https://github.com/kaeza/minetest-irc.git irc | 	cd <Mods directory> && git clone --recursive https://github.com/minetest-mods/irc.git | ||||||
|  |  | ||||||
| Please change `<Mods directory>` to fit your installation of Minetest. | Please change `<Mods directory>` to fit your installation of Minetest. | ||||||
| For more information, see [the wiki](http://wiki.minetest.net/Installing_mods). | For more information, see [the wiki][wiki]. | ||||||
|  |  | ||||||
| The IRC mod's git repository uses submodules, therefore you will have to run | The IRC mod's git repository uses submodules, therefore you will have to run | ||||||
| `git submodule init` when first installing the mod (unless you used | `git submodule init` when first installing the mod (unless you used | ||||||
| `--recursive` as above), and `git submodule update` every time that a submodule | `--recursive` as above), and `git submodule update` every time that a submodule | ||||||
| is updated.  These steps can be combined into `git submodule update --init`. | is updated. These steps can be combined into `git submodule update --init`. | ||||||
|  |  | ||||||
| You'll need to install LuaSocket.  You can do so with your package manager on | You'll need to install LuaSocket. You can do so with your package manager on | ||||||
| many distributions, for example: | many distributions, for example: | ||||||
|  |  | ||||||
| 	# # On Arch Linux: | 	# # On Arch Linux: | ||||||
| 	# pacman -S lua51-socket | 	# pacman -S lua51-socket | ||||||
| 	# # On Debian/Ubuntu: | 	# # On Debian/Ubuntu: | ||||||
| 	# apt-get install lua-socket | 	# # Debian/Ubuntu's LuaSocket packages are broken, so use LuaRocks. | ||||||
|  | 	# apt-get install luarocks | ||||||
|  | 	# luarocks install luasocket | ||||||
|  |  | ||||||
|  | You will also need to add IRC to your trusted mods if you haven't disabled mod | ||||||
|  | security. Here's an example configuration line: | ||||||
|  |  | ||||||
|  | 	secure.trusted_mods = irc | ||||||
|  |  | ||||||
|  | [wiki]: https://wiki.minetest.net/Installing_mods | ||||||
|  |  | ||||||
|  |  | ||||||
| Settings | Settings | ||||||
| -------- | -------- | ||||||
|  |  | ||||||
| All settings are changed in `minetest.conf`. If any of these settings | All settings are changed in `minetest.conf`. If any of these settings | ||||||
| are not set, the default value is used. | are not set, the default value is used. | ||||||
|  |  | ||||||
|   * `irc.server` (string) | * `irc.server` (string): | ||||||
| 	The address of the IRC server to connect to. |   The address of the IRC server to connect to. | ||||||
|  |  | ||||||
|   * `irc.channel` (string) | * `irc.channel` (string): | ||||||
| 	The IRC channel to join. |   The IRC channel to join. | ||||||
|  |  | ||||||
|   * `irc.interval` (number, default 2.0) | * `irc.interval` (number, default 2.0): | ||||||
| 	This prevents the server from flooding. It should be at |   This prevents the server from flooding. It should be at | ||||||
| 	least 2.0 but can be higher. After four messages this much |   least 2.0 but can be higher. After four messages this much | ||||||
| 	time must pass between folowing messages. |   time must pass between folowing messages. | ||||||
|  |  | ||||||
|   * `irc.nick` (string) | * `irc.nick` (string): | ||||||
| 	Nickname the server uses when it connects to IRC. |   Nickname the server uses when it connects to IRC. | ||||||
|  |  | ||||||
|   * `irc.password` (string, default nil) | * `irc.password` (string, default nil): | ||||||
| 	Password to use when connecting to the server. |   Password to use when connecting to the server. | ||||||
|  |  | ||||||
|   * `irc.NSPass` (string, default nil) | * `irc.NSPass` (string, default nil): | ||||||
| 	NickServ password.  Don't set this if you use SASL authentication. |   NickServ password. Don't set this if you use SASL authentication. | ||||||
|  |  | ||||||
|   * `irc.sasl.pass` (string, default nil) | * `irc.sasl.pass` (string, default nil): | ||||||
| 	SASL password, same as nickserv password. |   SASL password, same as nickserv password. | ||||||
| 	You should use this instead of NickServ authentication |   You should use this instead of NickServ authentication | ||||||
| 	if the server supports it. |   if the server supports it. | ||||||
|  |  | ||||||
|   * `irc.sasl.user` (string, default `irc.nick`) | * `irc.sasl.user` (string, default `irc.nick`): | ||||||
| 	The SASL username.  This should normaly be set to your NickServ account name. |   The SASL username. This should normaly be set to your | ||||||
|  |   NickServ account name. | ||||||
|  |  | ||||||
|   * `irc.debug` (boolean, default false) | * `irc.debug` (boolean, default false): | ||||||
| 	Whether to output debug information. |   Whether to output debug information. | ||||||
|  |  | ||||||
|   * `irc.disable_auto_connect` (boolean, default false) | * `irc.disable_auto_connect` (boolean, default false): | ||||||
| 	If false, the bot is connected by default. If true, a player with |   If false, the bot is connected by default. If true, a player with | ||||||
| 	the 'irc_admin' privilege has to use the /irc_connect command to |   the 'irc_admin' privilege has to use the `/irc_connect` command to | ||||||
| 	connect to the server. |   connect to the server. | ||||||
|  |  | ||||||
|   * `irc.disable_auto_join` (boolean, default false) | * `irc.disable_auto_join` (boolean, default false): | ||||||
| 	If false, players join the channel automatically upon entering the |   If false, players join the channel automatically upon entering the | ||||||
| 	game. If true, each user must manually use the /join command to |   game. If true, each user must manually use the `/join` command to | ||||||
| 	join the channel. In any case, the players may use the /part |   join the channel. In any case, the players may use the `/part` | ||||||
| 	command to opt-out of being in the channel. |   command to opt-out of being in the channel. | ||||||
|  |  | ||||||
|  | * `irc.send_join_part` (boolean, default true): | ||||||
|  |   Determines whether to send player join and part messages to the channel. | ||||||
|  |  | ||||||
|   * `irc.send_join_part` (boolean, default true) |  | ||||||
| 	Determines whether to send player join and part messages to the channel. |  | ||||||
|  |  | ||||||
| Usage | Usage | ||||||
| ----- | ----- | ||||||
|  |  | ||||||
| Once the game is connected to the IRC channel, chatting in-game will send | Once the game is connected to the IRC channel, chatting in-game will send | ||||||
| messages to the channel, and will be visible by anyone.  Also, messages sent | messages to the channel, and will be visible by anyone. Also, messages sent | ||||||
| to the channel will be visible in-game. | to the channel will be visible in-game. | ||||||
|  |  | ||||||
| Messages that begin with `[off]` from in-game or IRC are not sent to the other side. | Messages that begin with `[off]` from in-game or IRC are not sent to the | ||||||
|  | other side. | ||||||
|  |  | ||||||
| This mod also adds a few chat commands: | This mod also adds a few chat commands: | ||||||
|  |  | ||||||
|   * `/irc_msg <nick> <message>` | * `/irc_msg <nick> <message>`: | ||||||
| 	Sends a private message to a IRC user. |   Send a private message to a IRC user. | ||||||
|  |  | ||||||
|   * `/join` | * `/join`: | ||||||
| 	Join the IRC chat. |   Join the IRC chat. | ||||||
|  |  | ||||||
|   * `/part` | * `/part`: | ||||||
| 	Part the IRC chat. |   Part the IRC chat. | ||||||
|  |  | ||||||
|   * `/irc_connect` | * `/irc_connect`: | ||||||
| 	Connect the bot manually to the IRC network. |   Connect the bot manually to the IRC network. | ||||||
|  |  | ||||||
|   * `/irc_disconnect` | * `/irc_disconnect`: | ||||||
| 	Disconnect the bot manually from the IRC network (this does not |   Disconnect the bot manually from the IRC network (this does not | ||||||
| 	shutdown the game). |   shutdown the game). | ||||||
|  |  | ||||||
|   * `/irc_reconnect` | * `/irc_reconnect`: | ||||||
| 	Equivilant to `/irc_disconnect` followed by `/irc_connect`. |   Equivalent to `/irc_disconnect` followed by `/irc_connect`. | ||||||
|  |  | ||||||
| You can also send private messages from IRC to in-game players. | You can also send private messages from IRC to in-game players | ||||||
|  | by sending a private message to the bot (set with the `irc.nick` | ||||||
| To do so, you must send a private message to the bot (set with | option above), in the following format: | ||||||
| the `irc.nick` option above), in the following format: |  | ||||||
|  |  | ||||||
| 	@playername message | 	@playername message | ||||||
|  |  | ||||||
| @@ -125,21 +142,29 @@ a private message from IRC with: | |||||||
|  |  | ||||||
| 	/msg server_nick @mtuser Hello! | 	/msg server_nick @mtuser Hello! | ||||||
|  |  | ||||||
| To avoid possible misunderstandings (since all in-game players use the | The bot also supports some basic commands, which are invoked by saying | ||||||
| same IRC user to converse with you), the "proxy" user will reject any | the bot name followed by either a colon or a comma and the command, or | ||||||
| private messages that are not in that format, and will send back a | sending a private message to it. For example: `ServerBot: help whereis`. | ||||||
| nice reminder as a private message. |  | ||||||
|  |  | ||||||
| The bot also supports some basic commands, which are invoked by sending | * `help [<command>]`: | ||||||
| a private message to it. Use `!list` to get a list of commands, and |   Prints help about a command, or a list of supported commands if no | ||||||
| `!help <command>` to get help about a specific command. |   command is given. | ||||||
|  |  | ||||||
|  | * `uptime`: | ||||||
|  |   Prints the server's running time. | ||||||
|  |  | ||||||
|  | * `whereis <player>`: | ||||||
|  |   Prints the coordinates of the given player. | ||||||
|  |  | ||||||
|  | * `players`: | ||||||
|  |   Lists players currently in the server. | ||||||
|  |  | ||||||
|  |  | ||||||
| Thanks | Thanks | ||||||
| ------ | ------ | ||||||
|  |  | ||||||
| I'd like to thank the users who supported this mod both on the Minetest | I'd like to thank the users who supported this mod both on the Minetest | ||||||
| Forums and on the #minetest channel. In no particular order: | Forums and on the `#minetest` channel. In no particular order: | ||||||
|  |  | ||||||
| 0gb.us, ShadowNinja, Shaun/kizeren, RAPHAEL, DARGON, Calinou, Exio, | 0gb.us, ShadowNinja, Shaun/kizeren, RAPHAEL, DARGON, Calinou, Exio, | ||||||
| vortexlabs/mrtux, marveidemanis, marktraceur, jmf/john\_minetest, | vortexlabs/mrtux, marveidemanis, marktraceur, jmf/john\_minetest, | ||||||
| @@ -151,10 +176,7 @@ forum topic. Thanks to you all! | |||||||
| License | License | ||||||
| ------- | ------- | ||||||
|  |  | ||||||
| Copyright © 2012-2013 Diego Martínez <kaeza@users.sf.net> | See `LICENSE.txt` for details. | ||||||
|  |  | ||||||
| See LICENSE.txt for licensing information. |  | ||||||
|  |  | ||||||
| The files in the irc directory are part of the LuaIRC project. |  | ||||||
| See irc/LICENSE.txt for licensing information. |  | ||||||
|  |  | ||||||
|  | The files in the `irc` directory are part of the LuaIRC project. | ||||||
|  | See `irc/LICENSE.txt` for details. | ||||||
|   | |||||||
							
								
								
									
										108
									
								
								botcmds.lua
									
									
									
									
									
								
							
							
						
						
									
										108
									
								
								botcmds.lua
									
									
									
									
									
								
							| @@ -1,39 +1,59 @@ | |||||||
|  |  | ||||||
| irc.bot_commands = {} | irc.bot_commands = {} | ||||||
|  |  | ||||||
| function irc:check_botcmd(msg) | -- From RFC1459: | ||||||
|  | -- "Because of IRC’s scandanavian origin, the characters {}| are | ||||||
|  | --  considered to be the lower case equivalents of the characters | ||||||
|  | --  []\, respectively." | ||||||
|  | local irctolower = { ["["]="{", ["\\"]="|", ["]"]="}" } | ||||||
|  |  | ||||||
|  | local function irclower(s) | ||||||
|  | 	return (s:lower():gsub("[%[%]\\]", irctolower)) | ||||||
|  | end | ||||||
|  |  | ||||||
|  | local function nickequals(nick1, nick2) | ||||||
|  | 	return irclower(nick1) == irclower(nick2) | ||||||
|  | end | ||||||
|  |  | ||||||
|  | function irc.check_botcmd(msg) | ||||||
| 	local prefix = irc.config.command_prefix | 	local prefix = irc.config.command_prefix | ||||||
| 	local nick = irc.conn.nick:lower() | 	local nick = irc.conn.nick | ||||||
| 	local text = msg.args[2] | 	local text = msg.args[2] | ||||||
| 	local nickpart = text:sub(1, #nick + 2):lower() | 	local nickpart = text:sub(1, #nick) | ||||||
|  | 	local suffix = text:sub(#nick+1, #nick+2) | ||||||
|  |  | ||||||
| 	-- First check for a nick prefix | 	-- First check for a nick prefix | ||||||
| 	if nickpart == nick..": " or | 	if nickequals(nickpart, nick) | ||||||
| 	   nickpart == nick..", " then | 			and (suffix == ": " or suffix == ", ") then | ||||||
| 		self:bot_command(msg, text:sub(#nick + 3)) | 		irc.bot_command(msg, text:sub(#nick + 3)) | ||||||
| 		return true | 		return true | ||||||
| 	-- Then check for the configured prefix | 	-- Then check for the configured prefix | ||||||
| 	elseif prefix and text:sub(1, #prefix):lower() == prefix:lower() then | 	elseif prefix and text:sub(1, #prefix):lower() == prefix:lower() then | ||||||
| 		self:bot_command(msg, text:sub(#prefix + 1)) | 		irc.bot_command(msg, text:sub(#prefix + 1)) | ||||||
| 		return true | 		return true | ||||||
| 	end | 	end | ||||||
| 	return false | 	return false | ||||||
| end | end | ||||||
|  |  | ||||||
|  |  | ||||||
| function irc:bot_command(msg, text) | function irc.bot_command(msg, text) | ||||||
|  | 	-- Remove leading whitespace | ||||||
|  | 	text = text:match("^%s*(.*)") | ||||||
| 	if text:sub(1, 1) == "@" then | 	if text:sub(1, 1) == "@" then | ||||||
| 		local found, _, player_to, message = text:find("^.([^%s]+)%s(.+)$") | 		local _, _, player_to, message = text:find("^.([^%s]+)%s(.+)$") | ||||||
| 		if not minetest.get_player_by_name(player_to) then | 		if not player_to then | ||||||
| 			irc:reply("User '"..player_to.."' is not in the game.") | 			return | ||||||
|  | 		elseif not minetest.get_player_by_name(player_to) then | ||||||
|  | 			irc.reply("User '"..player_to.."' is not in the game.") | ||||||
| 			return | 			return | ||||||
| 		elseif not irc.joined_players[player_to] then | 		elseif not irc.joined_players[player_to] then | ||||||
| 			irc:reply("User '"..player_to.."' is not using IRC.") | 			irc.reply("User '"..player_to.."' is not using IRC.") | ||||||
| 			return | 			return | ||||||
| 		end | 		end | ||||||
| 		minetest.chat_send_player(player_to, | 		minetest.chat_send_player(player_to, | ||||||
| 				"PM from "..msg.user.nick.."@IRC: "..message, false) | 				minetest.colorize(irc.config.pm_color, | ||||||
| 		irc:reply("Message sent!") | 				"PM from "..msg.user.nick.."@IRC: "..message, false)) | ||||||
|  | 		irc.reply("Message sent!") | ||||||
| 		return | 		return | ||||||
| 	end | 	end | ||||||
| 	local pos = text:find(" ", 1, true) | 	local pos = text:find(" ", 1, true) | ||||||
| @@ -45,45 +65,51 @@ function irc:bot_command(msg, text) | |||||||
| 		cmd = text | 		cmd = text | ||||||
| 		args = "" | 		args = "" | ||||||
| 	end | 	end | ||||||
|   |  | ||||||
| 	if not self.bot_commands[cmd] then | 	if not irc.bot_commands[cmd] then | ||||||
| 		self:reply("Unknown command '"..cmd.."'. Try 'list'." | 		irc.reply("Unknown command '"..cmd.."'. Try 'help'." | ||||||
| 			.." Or use @playername <message> to send a private message") | 			.." Or use @playername <message> to send a private message") | ||||||
| 		return | 		return | ||||||
| 	end | 	end | ||||||
|   |  | ||||||
| 	local success, message = self.bot_commands[cmd].func(msg.user, args) | 	local _, message = irc.bot_commands[cmd].func(msg.user, args) | ||||||
| 	if message then | 	if message then | ||||||
| 		self:reply(message) | 		irc.reply(message) | ||||||
| 	end | 	end | ||||||
| end | end | ||||||
|  |  | ||||||
|  |  | ||||||
| function irc:register_bot_command(name, def) | function irc.register_bot_command(name, def) | ||||||
| 	if (not def.func) or (type(def.func) ~= "function") then | 	if (not def.func) or (type(def.func) ~= "function") then | ||||||
| 		error("Erroneous bot command definition. def.func missing.", 2) | 		error("Erroneous bot command definition. def.func missing.", 2) | ||||||
| 	elseif name:sub(1, 1) == "@" then | 	elseif name:sub(1, 1) == "@" then | ||||||
| 		error("Erroneous bot command name. Command name begins with '@'.", 2) | 		error("Erroneous bot command name. Command name begins with '@'.", 2) | ||||||
| 	end | 	end | ||||||
| 	self.bot_commands[name] = def | 	irc.bot_commands[name] = def | ||||||
| end | end | ||||||
|  |  | ||||||
|  |  | ||||||
| irc:register_bot_command("help", { | irc.register_bot_command("help", { | ||||||
| 	params = "<command>", | 	params = "<command>", | ||||||
| 	description = "Get help about a command", | 	description = "Get help about a command", | ||||||
| 	func = function(user, args) | 	func = function(_, args) | ||||||
| 		if args == "" then | 		if args == "" then | ||||||
| 			return false, "No command name specified. Use 'list' for a list of commands." | 			local cmdlist = { } | ||||||
|  | 			for name in pairs(irc.bot_commands) do | ||||||
|  | 				cmdlist[#cmdlist+1] = name | ||||||
|  | 			end | ||||||
|  | 			return true, "Available commands: "..table.concat(cmdlist, ", ") | ||||||
|  | 					.." -- Use 'help <command name>' to get" | ||||||
|  | 					.." help about a specific command." | ||||||
| 		end | 		end | ||||||
|  |  | ||||||
| 		local cmd = irc.bot_commands[args] | 		local cmd = irc.bot_commands[args] | ||||||
| 		if not cmd then | 		if not cmd then | ||||||
| 			return false, "Unknown command '"..cmdname.."'." | 			return false, "Unknown command '"..args.."'." | ||||||
| 		end | 		end | ||||||
|  |  | ||||||
| 		return true, ("Usage: %c%s %s -- %s"):format( | 		return true, ("Usage: %s%s %s -- %s"):format( | ||||||
| 				irc.config.command_prefix, | 				irc.config.command_prefix or "", | ||||||
| 				args, | 				args, | ||||||
| 				cmd.params or "<no parameters>", | 				cmd.params or "<no parameters>", | ||||||
| 				cmd.description or "<no description>") | 				cmd.description or "<no description>") | ||||||
| @@ -91,24 +117,20 @@ irc:register_bot_command("help", { | |||||||
| }) | }) | ||||||
|  |  | ||||||
|  |  | ||||||
| irc:register_bot_command("list", { | irc.register_bot_command("list", { | ||||||
| 	params = "", | 	params = "", | ||||||
| 	description = "List available commands.", | 	description = "List available commands.", | ||||||
| 	func = function(user, args) | 	func = function() | ||||||
| 		local cmdlist = "Available commands: " | 		return false, "The `list` command has been merged into `help`." | ||||||
| 		for name, cmd in pairs(irc.bot_commands) do | 				.." Use `help` with no arguments to get a list." | ||||||
| 			cmdlist = cmdlist..name..", " |  | ||||||
| 		end |  | ||||||
| 		return true, cmdlist.." -- Use 'help <command name>' to get" |  | ||||||
| 			.." help about a specific command." |  | ||||||
| 	end | 	end | ||||||
| }) | }) | ||||||
|  |  | ||||||
|  |  | ||||||
| irc:register_bot_command("whereis", { | irc.register_bot_command("whereis", { | ||||||
| 	params = "<player>", | 	params = "<player>", | ||||||
| 	description = "Tell the location of <player>", | 	description = "Tell the location of <player>", | ||||||
| 	func = function(user, args) | 	func = function(_, args) | ||||||
| 		if args == "" then | 		if args == "" then | ||||||
| 			return false, "Player name required." | 			return false, "Player name required." | ||||||
| 		end | 		end | ||||||
| @@ -117,16 +139,16 @@ irc:register_bot_command("whereis", { | |||||||
| 			return false, "There is no player named '"..args.."'" | 			return false, "There is no player named '"..args.."'" | ||||||
| 		end | 		end | ||||||
| 		local fmt = "Player %s is at (%.2f,%.2f,%.2f)" | 		local fmt = "Player %s is at (%.2f,%.2f,%.2f)" | ||||||
| 		local pos = player:getpos() | 		local pos = player:get_pos() | ||||||
| 		return true, fmt:format(args, pos.x, pos.y, pos.z) | 		return true, fmt:format(args, pos.x, pos.y, pos.z) | ||||||
| 	end | 	end | ||||||
| }) | }) | ||||||
|  |  | ||||||
|  |  | ||||||
| local starttime = os.time() | local starttime = os.time() | ||||||
| irc:register_bot_command("uptime", { | irc.register_bot_command("uptime", { | ||||||
| 	description = "Tell how much time the server has been up", | 	description = "Tell how much time the server has been up", | ||||||
| 	func = function(user, args) | 	func = function() | ||||||
| 		local cur_time = os.time() | 		local cur_time = os.time() | ||||||
| 		local diff = os.difftime(cur_time, starttime) | 		local diff = os.difftime(cur_time, starttime) | ||||||
| 		local fmt = "Server has been running for %d:%02d:%02d" | 		local fmt = "Server has been running for %d:%02d:%02d" | ||||||
| @@ -139,9 +161,9 @@ irc:register_bot_command("uptime", { | |||||||
| }) | }) | ||||||
|  |  | ||||||
|  |  | ||||||
| irc:register_bot_command("players", { | irc.register_bot_command("players", { | ||||||
| 	description = "List the players on the server", | 	description = "List the players on the server", | ||||||
| 	func = function(user, args) | 	func = function() | ||||||
| 		local players = minetest.get_connected_players() | 		local players = minetest.get_connected_players() | ||||||
| 		local names = {} | 		local names = {} | ||||||
| 		for _, player in pairs(players) do | 		for _, player in pairs(players) do | ||||||
|   | |||||||
							
								
								
									
										11
									
								
								callback.lua
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								callback.lua
									
									
									
									
									
								
							| @@ -5,15 +5,16 @@ | |||||||
| minetest.register_on_joinplayer(function(player) | minetest.register_on_joinplayer(function(player) | ||||||
| 	local name = player:get_player_name() | 	local name = player:get_player_name() | ||||||
| 	if irc.connected and irc.config.send_join_part then | 	if irc.connected and irc.config.send_join_part then | ||||||
| 		irc:say("*** "..name.." joined the game") | 		irc.say("*** "..name.." joined the game") | ||||||
| 	end | 	end | ||||||
| end) | end) | ||||||
|  |  | ||||||
|  |  | ||||||
| minetest.register_on_leaveplayer(function(player) | minetest.register_on_leaveplayer(function(player, timed_out) | ||||||
| 	local name = player:get_player_name() | 	local name = player:get_player_name() | ||||||
| 	if irc.connected and irc.config.send_join_part then | 	if irc.connected and irc.config.send_join_part then | ||||||
| 		irc:say("*** "..name.." left the game") | 		irc.say("*** "..name.." left the game".. | ||||||
|  | 				(timed_out and " (Timed out)" or "")) | ||||||
| 	end | 	end | ||||||
| end) | end) | ||||||
|  |  | ||||||
| @@ -30,11 +31,11 @@ minetest.register_on_chat_message(function(name, message) | |||||||
| 	if nl then | 	if nl then | ||||||
| 		message = message:sub(1, nl - 1) | 		message = message:sub(1, nl - 1) | ||||||
| 	end | 	end | ||||||
| 	irc:say(irc:playerMessage(name, message)) | 	irc.say(irc.playerMessage(name, minetest.strip_colors(message))) | ||||||
| end) | end) | ||||||
|  |  | ||||||
|  |  | ||||||
| minetest.register_on_shutdown(function() | minetest.register_on_shutdown(function() | ||||||
| 	irc:disconnect("Game shutting down.") | 	irc.disconnect("Game shutting down.") | ||||||
| end) | end) | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										78
									
								
								chatcmds.lua
									
									
									
									
									
								
							
							
						
						
									
										78
									
								
								chatcmds.lua
									
									
									
									
									
								
							| @@ -11,32 +11,30 @@ minetest.register_chatcommand("irc_msg", { | |||||||
| 	privs = {shout=true}, | 	privs = {shout=true}, | ||||||
| 	func = function(name, param) | 	func = function(name, param) | ||||||
| 		if not irc.connected then | 		if not irc.connected then | ||||||
| 			minetest.chat_send_player(name, "Not connected to IRC. Use /irc_connect to connect.") | 			return false, "Not connected to IRC. Use /irc_connect to connect." | ||||||
| 			return |  | ||||||
| 		end | 		end | ||||||
| 		local found, _, toname, message = param:find("^([^%s]+)%s(.+)") | 		local found, _, toname, message = param:find("^([^%s]+)%s(.+)") | ||||||
| 		if not found then | 		if not found then | ||||||
| 			minetest.chat_send_player(name, "Invalid usage, see /help irc_msg.") | 			return false, "Invalid usage, see /help irc_msg." | ||||||
| 			return |  | ||||||
| 		end | 		end | ||||||
| 		local toname_l = toname:lower() | 		local toname_l = toname:lower() | ||||||
| 		local validNick = false | 		local validNick = false | ||||||
| 		for nick, user in pairs(irc.conn.channels[irc.config.channel].users) do | 		local hint = "They have to be in the channel" | ||||||
|  | 		for nick in pairs(irc.conn.channels[irc.config.channel].users) do | ||||||
| 			if nick:lower() == toname_l then | 			if nick:lower() == toname_l then | ||||||
| 				validNick = true | 				validNick = true | ||||||
| 				break | 				break | ||||||
| 			end | 			end | ||||||
| 		end | 		end | ||||||
| 		if toname_l:find("serv$") or toname_l:find("bot$") then | 		if toname_l:find("serv$") or toname_l:find("bot$") then | ||||||
|  | 			hint = "it looks like a bot or service" | ||||||
| 			validNick = false | 			validNick = false | ||||||
| 		end | 		end | ||||||
| 		if not validNick then | 		if not validNick then | ||||||
| 			minetest.chat_send_player(name, | 			return false, "You can not message that user. ("..hint..")" | ||||||
| 				"You can not message that user. (Hint: They have to be in the channel)") |  | ||||||
| 			return |  | ||||||
| 		end | 		end | ||||||
| 		irc:say(toname, irc:playerMessage(name, message)) | 		irc.say(toname, irc.playerMessage(name, message)) | ||||||
| 		minetest.chat_send_player(name, "Message sent!") | 		return true, "Message sent!" | ||||||
| 	end | 	end | ||||||
| }) | }) | ||||||
|  |  | ||||||
| @@ -44,16 +42,15 @@ minetest.register_chatcommand("irc_msg", { | |||||||
| minetest.register_chatcommand("irc_names", { | minetest.register_chatcommand("irc_names", { | ||||||
| 	params = "", | 	params = "", | ||||||
| 	description = "List the users in IRC.", | 	description = "List the users in IRC.", | ||||||
| 	func = function(name, params) | 	func = function() | ||||||
| 		if not irc.connected then | 		if not irc.connected then | ||||||
| 			minetest.chat_send_player(name, "Not connected to IRC. Use /irc_connect to connect.") | 			return false, "Not connected to IRC. Use /irc_connect to connect." | ||||||
| 			return |  | ||||||
| 		end | 		end | ||||||
| 		local users = { } | 		local users = { } | ||||||
| 		for k, v in pairs(irc.conn.channels[irc.config.channel].users) do | 		for nick in pairs(irc.conn.channels[irc.config.channel].users) do | ||||||
| 			table.insert(users, k) | 			table.insert(users, nick) | ||||||
| 		end | 		end | ||||||
| 		minetest.chat_send_player(name, "Users in IRC: "..table.concat(users, ", ")) | 		return true, "Users in IRC: "..table.concat(users, ", ") | ||||||
| 	end | 	end | ||||||
| }) | }) | ||||||
|  |  | ||||||
| @@ -61,13 +58,12 @@ minetest.register_chatcommand("irc_names", { | |||||||
| minetest.register_chatcommand("irc_connect", { | minetest.register_chatcommand("irc_connect", { | ||||||
| 	description = "Connect to the IRC server.", | 	description = "Connect to the IRC server.", | ||||||
| 	privs = {irc_admin=true}, | 	privs = {irc_admin=true}, | ||||||
| 	func = function(name, param) | 	func = function(name) | ||||||
| 		if irc.connected then | 		if irc.connected then | ||||||
| 			minetest.chat_send_player(name, "You are already connected to IRC.") | 			return false, "You are already connected to IRC." | ||||||
| 			return |  | ||||||
| 		end | 		end | ||||||
| 		minetest.chat_send_player(name, "IRC: Connecting...") | 		minetest.chat_send_player(name, "IRC: Connecting...") | ||||||
| 		irc:connect() | 		irc.connect() | ||||||
| 	end | 	end | ||||||
| }) | }) | ||||||
|  |  | ||||||
| @@ -78,13 +74,12 @@ minetest.register_chatcommand("irc_disconnect", { | |||||||
| 	privs = {irc_admin=true}, | 	privs = {irc_admin=true}, | ||||||
| 	func = function(name, param) | 	func = function(name, param) | ||||||
| 		if not irc.connected then | 		if not irc.connected then | ||||||
| 			minetest.chat_send_player(name, "You are not connected to IRC.") | 			return false, "Not connected to IRC. Use /irc_connect to connect." | ||||||
| 			return |  | ||||||
| 		end | 		end | ||||||
| 		if params == "" then | 		if param == "" then | ||||||
| 			params = "Manual disconnect by "..name | 			param = "Manual disconnect by "..name | ||||||
| 		end | 		end | ||||||
| 		irc:disconnect(param) | 		irc.disconnect(param) | ||||||
| 	end | 	end | ||||||
| }) | }) | ||||||
|  |  | ||||||
| @@ -92,13 +87,13 @@ minetest.register_chatcommand("irc_disconnect", { | |||||||
| minetest.register_chatcommand("irc_reconnect", { | minetest.register_chatcommand("irc_reconnect", { | ||||||
| 	description = "Reconnect to the IRC server.", | 	description = "Reconnect to the IRC server.", | ||||||
| 	privs = {irc_admin=true}, | 	privs = {irc_admin=true}, | ||||||
| 	func = function(name, param) | 	func = function(name) | ||||||
| 		if not irc.connected then | 		if not irc.connected then | ||||||
| 			minetest.chat_send_player(name, "You are not connected to IRC.") | 			return false, "Not connected to IRC. Use /irc_connect to connect." | ||||||
| 			return |  | ||||||
| 		end | 		end | ||||||
| 		irc:disconnect("Reconnecting...") | 		minetest.chat_send_player(name, "IRC: Reconnecting...") | ||||||
| 		irc:connect() | 		irc.disconnect("Reconnecting...") | ||||||
|  | 		irc.connect() | ||||||
| 	end | 	end | ||||||
| }) | }) | ||||||
|  |  | ||||||
| @@ -109,18 +104,31 @@ minetest.register_chatcommand("irc_quote", { | |||||||
| 	privs = {irc_admin=true}, | 	privs = {irc_admin=true}, | ||||||
| 	func = function(name, param) | 	func = function(name, param) | ||||||
| 		if not irc.connected then | 		if not irc.connected then | ||||||
| 			minetest.chat_send_player(name, "You are not connected to IRC.") | 			return false, "Not connected to IRC. Use /irc_connect to connect." | ||||||
| 			return |  | ||||||
| 		end | 		end | ||||||
| 		irc:queue(param) | 		irc.queue(param) | ||||||
| 		minetest.chat_send_player(name, "Command sent!") | 		minetest.chat_send_player(name, "Command sent!") | ||||||
| 	end | 	end | ||||||
| }) | }) | ||||||
|  |  | ||||||
|  |  | ||||||
| local oldme = minetest.chatcommands["me"].func | local oldme = minetest.chatcommands["me"].func | ||||||
|  | -- luacheck: ignore | ||||||
| minetest.chatcommands["me"].func = function(name, param, ...) | minetest.chatcommands["me"].func = function(name, param, ...) | ||||||
| 	oldme(name, param, ...) | 	irc.say(("* %s %s"):format(name, param)) | ||||||
| 	irc:say(("* %s %s"):format(name, param)) | 	return oldme(name, param, ...) | ||||||
| end | end | ||||||
|  |  | ||||||
|  | if irc.config.send_kicks and minetest.chatcommands["kick"] then | ||||||
|  | 	local oldkick = minetest.chatcommands["kick"].func | ||||||
|  | 	-- luacheck: ignore | ||||||
|  | 	minetest.chatcommands["kick"].func = function(name, param, ...) | ||||||
|  | 		local plname, reason = param:match("^(%S+)%s*(.*)$") | ||||||
|  | 		if not plname then | ||||||
|  | 			return false, "Usage: /kick player [reason]" | ||||||
|  | 		end | ||||||
|  | 		irc.say(("*** Kicked %s.%s"):format(name, | ||||||
|  | 				reason~="" and " Reason: "..reason or "")) | ||||||
|  | 		return oldkick(name, param, ...) | ||||||
|  | 	end | ||||||
|  | end | ||||||
|   | |||||||
							
								
								
									
										21
									
								
								config.lua
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								config.lua
									
									
									
									
									
								
							| @@ -6,12 +6,14 @@ irc.config = {} | |||||||
|  |  | ||||||
| local function setting(stype, name, default, required) | local function setting(stype, name, default, required) | ||||||
| 	local value | 	local value | ||||||
| 	if stype == "bool" then | 	if minetest.settings and minetest.settings.get and minetest.settings.get_bool then | ||||||
| 		value = minetest.setting_getbool("irc."..name) | 		if stype == "bool" then | ||||||
| 	elseif stype == "string" then | 			value = minetest.settings:get_bool("irc."..name) | ||||||
| 		value = minetest.setting_get("irc."..name) | 		elseif stype == "string" then | ||||||
| 	elseif stype == "number" then | 			value = minetest.settings:get("irc."..name) | ||||||
| 		value = tonumber(minetest.setting_get("irc."..name)) | 		elseif stype == "number" then | ||||||
|  | 			value = tonumber(minetest.settings:get("irc."..name)) | ||||||
|  | 		end | ||||||
| 	end | 	end | ||||||
| 	if value == nil then | 	if value == nil then | ||||||
| 		if required then | 		if required then | ||||||
| @@ -32,10 +34,13 @@ setting("string", "server", nil, true) -- Server address to connect to | |||||||
| setting("number", "port", 6667) -- Server port to connect to | setting("number", "port", 6667) -- Server port to connect to | ||||||
| setting("string", "NSPass") -- NickServ password | setting("string", "NSPass") -- NickServ password | ||||||
| setting("string", "sasl.user", irc.config.nick) -- SASL username | setting("string", "sasl.user", irc.config.nick) -- SASL username | ||||||
|  | setting("string", "username", "Minetest") -- Username/ident | ||||||
|  | setting("string", "realname", "Minetest") -- Real name/GECOS | ||||||
| setting("string", "sasl.pass") -- SASL password | setting("string", "sasl.pass") -- SASL password | ||||||
| setting("string", "channel", nil, true) -- Channel to join | setting("string", "channel", nil, true) -- Channel to join | ||||||
| setting("string", "key") -- Key for the channel | setting("string", "key") -- Key for the channel | ||||||
| setting("bool",   "send_join_part", true) -- Whether to send player join and part messages to the channel | setting("bool",   "send_join_part", true) -- Whether to send player join and part messages to the channel | ||||||
|  | setting("bool",   "send_kicks", false) -- Whether to send player kicked messages to the channel | ||||||
|  |  | ||||||
| ----------------------- | ----------------------- | ||||||
| -- ADVANCED SETTINGS -- | -- ADVANCED SETTINGS -- | ||||||
| @@ -44,9 +49,11 @@ setting("bool",   "send_join_part", true) -- Whether to send player join and par | |||||||
| setting("string", "password") -- Server password | setting("string", "password") -- Server password | ||||||
| setting("bool",   "secure", false) -- Enable a TLS connection, requires LuaSEC | setting("bool",   "secure", false) -- Enable a TLS connection, requires LuaSEC | ||||||
| setting("number", "timeout", 60) -- Underlying socket timeout in seconds. | setting("number", "timeout", 60) -- Underlying socket timeout in seconds. | ||||||
|  | setting("number", "reconnect", 600) -- Time between reconnection attempts, in seconds. | ||||||
| setting("string", "command_prefix") -- Prefix to use for bot commands | setting("string", "command_prefix") -- Prefix to use for bot commands | ||||||
| setting("bool",   "debug", false) -- Enable debug output | setting("bool",   "debug", false) -- Enable debug output | ||||||
| setting("bool",   "enable_player_part", true) -- Whether to enable players joining and parting the channel | setting("bool",   "enable_player_part", true) -- Whether to enable players joining and parting the channel | ||||||
| setting("bool",   "auto_join", true) -- Whether to automatically show players in the channel when they join | setting("bool",   "auto_join", true) -- Whether to automatically show players in the channel when they join | ||||||
| setting("bool",   "auto_connect", true) -- Whether to automatically connect to the server on mod load | setting("bool",   "auto_connect", true) -- Whether to automatically connect to the server on mod load | ||||||
|  | setting("string", "chat_color", "#339933") -- Color of IRC chat in-game, green by default | ||||||
|  | setting("string", "pm_color", "#8800AA") -- Color of IRC PMs in-game, purple by default | ||||||
|   | |||||||
							
								
								
									
										95
									
								
								hooks.lua
									
									
									
									
									
								
							
							
						
						
									
										95
									
								
								hooks.lua
									
									
									
									
									
								
							| @@ -1,19 +1,16 @@ | |||||||
| -- This file is licensed under the terms of the BSD 2-clause license. | -- This file is licensed under the terms of the BSD 2-clause license. | ||||||
| -- See LICENSE.txt for details. | -- See LICENSE.txt for details. | ||||||
|  |  | ||||||
|  | local ie = ... | ||||||
|  |  | ||||||
| -- MIME is part of LuaSocket | -- MIME is part of LuaSocket | ||||||
| local b64e = require("mime").b64 | local b64e = ie.require("mime").b64 | ||||||
|  |  | ||||||
| irc.hooks = {} | irc.hooks = {} | ||||||
| irc.registered_hooks = {} | irc.registered_hooks = {} | ||||||
|  |  | ||||||
|  |  | ||||||
| -- TODO: Add proper conversion from CP1252 to UTF-8. | local stripped_chars = "[\2\31]" | ||||||
| local stripped_chars = {"\2", "\31"} |  | ||||||
| for c = 127, 255 do |  | ||||||
| 	table.insert(stripped_chars, string.char(c)) |  | ||||||
| end |  | ||||||
| stripped_chars = "["..table.concat(stripped_chars, "").."]" |  | ||||||
|  |  | ||||||
| local function normalize(text) | local function normalize(text) | ||||||
| 	-- Strip colors | 	-- Strip colors | ||||||
| @@ -23,8 +20,8 @@ local function normalize(text) | |||||||
| end | end | ||||||
|  |  | ||||||
|  |  | ||||||
| function irc:doHook(conn) | function irc.doHook(conn) | ||||||
| 	for name, hook in pairs(self.registered_hooks) do | 	for name, hook in pairs(irc.registered_hooks) do | ||||||
| 		for _, func in pairs(hook) do | 		for _, func in pairs(hook) do | ||||||
| 			conn:hook(name, func) | 			conn:hook(name, func) | ||||||
| 		end | 		end | ||||||
| @@ -32,9 +29,9 @@ function irc:doHook(conn) | |||||||
| end | end | ||||||
|  |  | ||||||
|  |  | ||||||
| function irc:register_hook(name, func) | function irc.register_hook(name, func) | ||||||
| 	self.registered_hooks[name] = self.registered_hooks[name] or {} | 	irc.registered_hooks[name] = irc.registered_hooks[name] or {} | ||||||
| 	table.insert(self.registered_hooks[name], func) | 	table.insert(irc.registered_hooks[name], func) | ||||||
| end | end | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -83,13 +80,13 @@ function irc.hooks.ctcp(msg) | |||||||
| 	local command = args[1]:upper() | 	local command = args[1]:upper() | ||||||
|  |  | ||||||
| 	local function reply(s) | 	local function reply(s) | ||||||
| 		irc:queue(irc.msgs.notice(msg.user.nick, | 		irc.queue(irc.msgs.notice(msg.user.nick, | ||||||
| 				("\1%s %s\1"):format(command, s))) | 				("\1%s %s\1"):format(command, s))) | ||||||
| 	end | 	end | ||||||
|  |  | ||||||
| 	if command == "ACTION" and msg.args[1] == irc.config.channel then | 	if command == "ACTION" and msg.args[1] == irc.config.channel then | ||||||
| 		local action = text:sub(8, -1) | 		local action = text:sub(8, -1) | ||||||
| 		irc:sendLocal(("* %s@IRC %s"):format(msg.user.nick, action)) | 		irc.sendLocal(("* %s@IRC %s"):format(msg.user.nick, action)) | ||||||
| 	elseif command == "VERSION" then | 	elseif command == "VERSION" then | ||||||
| 		reply(("Minetest version %s, IRC mod version %s.") | 		reply(("Minetest version %s, IRC mod version %s.") | ||||||
| 			:format(get_core_version(), irc.version)) | 			:format(get_core_version(), irc.version)) | ||||||
| @@ -104,11 +101,12 @@ end | |||||||
| function irc.hooks.channelChat(msg) | function irc.hooks.channelChat(msg) | ||||||
| 	local text = normalize(msg.args[2]) | 	local text = normalize(msg.args[2]) | ||||||
|  |  | ||||||
| 	irc:check_botcmd(msg) | 	irc.check_botcmd(msg) | ||||||
|  |  | ||||||
| 	-- Don't let a user impersonate someone else by using the nick "IRC" | 	-- Don't let a user impersonate someone else by using the nick "IRC" | ||||||
| 	if msg.user.nick == "IRC" then | 	local fake = msg.user.nick:lower():match("^[il|]rc$") | ||||||
| 		irc:sendLocal("<IRC@IRC> "..text) | 	if fake then | ||||||
|  | 		irc.sendLocal("<"..msg.user.nick.."@IRC> "..text) | ||||||
| 		return | 		return | ||||||
| 	end | 	end | ||||||
|  |  | ||||||
| @@ -122,25 +120,30 @@ function irc.hooks.channelChat(msg) | |||||||
| 		text:find("^%*%*%* ([^%s]+) joined the game$") | 		text:find("^%*%*%* ([^%s]+) joined the game$") | ||||||
| 	local foundleave, _, leavenick = | 	local foundleave, _, leavenick = | ||||||
| 		text:find("^%*%*%* ([^%s]+) left the game$") | 		text:find("^%*%*%* ([^%s]+) left the game$") | ||||||
|  | 	local foundtimedout, _, timedoutnick = | ||||||
|  | 		text:find("^%*%*%* ([^%s]+) left the game %(Timed out%)$") | ||||||
| 	local foundaction, _, actionnick, actionmessage = | 	local foundaction, _, actionnick, actionmessage = | ||||||
| 		text:find("^%* ([^%s]+) (.*)$") | 		text:find("^%* ([^%s]+) (.*)$") | ||||||
|  |  | ||||||
| 	if text:sub(1, 5) == "[off]" then | 	if text:sub(1, 5) == "[off]" then | ||||||
| 		return | 		return | ||||||
| 	elseif foundchat then | 	elseif foundchat then | ||||||
| 		irc:sendLocal(("<%s@%s> %s") | 		irc.sendLocal(("<%s@%s> %s") | ||||||
| 				:format(chatnick, msg.user.nick, chatmessage)) | 				:format(chatnick, msg.user.nick, chatmessage)) | ||||||
| 	elseif foundjoin then | 	elseif foundjoin then | ||||||
| 		irc:sendLocal(("*** %s joined %s") | 		irc.sendLocal(("*** %s joined %s") | ||||||
| 				:format(joinnick, msg.user.nick)) | 				:format(joinnick, msg.user.nick)) | ||||||
| 	elseif foundleave then | 	elseif foundleave then | ||||||
| 		irc:sendLocal(("*** %s left %s") | 		irc.sendLocal(("*** %s left %s") | ||||||
| 				:format(leavenick, msg.user.nick)) | 				:format(leavenick, msg.user.nick)) | ||||||
|  | 	elseif foundtimedout then | ||||||
|  | 		irc.sendLocal(("*** %s left %s (Timed out)") | ||||||
|  | 				:format(timedoutnick, msg.user.nick)) | ||||||
| 	elseif foundaction then | 	elseif foundaction then | ||||||
| 		irc:sendLocal(("* %s@%s %s") | 		irc.sendLocal(("* %s@%s %s") | ||||||
| 				:format(actionnick, msg.user.nick, actionmessage)) | 				:format(actionnick, msg.user.nick, actionmessage)) | ||||||
| 	else | 	else | ||||||
| 		irc:sendLocal(("<%s@IRC> %s"):format(msg.user.nick, text)) | 		irc.sendLocal(("<%s@IRC> %s"):format(msg.user.nick, text)) | ||||||
| 	end | 	end | ||||||
| end | end | ||||||
|  |  | ||||||
| @@ -152,16 +155,16 @@ function irc.hooks.pm(msg) | |||||||
| 	if prefix and text:sub(1, #prefix) == prefix then | 	if prefix and text:sub(1, #prefix) == prefix then | ||||||
| 		text = text:sub(#prefix + 1) | 		text = text:sub(#prefix + 1) | ||||||
| 	end | 	end | ||||||
| 	irc:bot_command(msg, text) | 	irc.bot_command(msg, text) | ||||||
| end | end | ||||||
|  |  | ||||||
|  |  | ||||||
| function irc.hooks.kick(channel, target, prefix, reason) | function irc.hooks.kick(channel, target, prefix, reason) | ||||||
| 	if target == irc.conn.nick then | 	if target == irc.conn.nick then | ||||||
| 		minetest.chat_send_all("IRC: kicked from "..channel.." by "..prefix.nick..".") | 		minetest.chat_send_all("IRC: kicked from "..channel.." by "..prefix.nick..".") | ||||||
| 		irc:disconnect("Kicked") | 		irc.disconnect("Kicked") | ||||||
| 	else | 	else | ||||||
| 		irc:sendLocal(("-!- %s was kicked from %s by %s [%s]") | 		irc.sendLocal(("-!- %s was kicked from %s by %s [%s]") | ||||||
| 				:format(target, channel, prefix.nick, reason)) | 				:format(target, channel, prefix.nick, reason)) | ||||||
| 	end | 	end | ||||||
| end | end | ||||||
| @@ -169,7 +172,7 @@ end | |||||||
|  |  | ||||||
| function irc.hooks.notice(user, target, message) | function irc.hooks.notice(user, target, message) | ||||||
| 	if user.nick and target == irc.config.channel then | 	if user.nick and target == irc.config.channel then | ||||||
| 		irc:sendLocal("-"..user.nick.."@IRC- "..message) | 		irc.sendLocal("-"..user.nick.."@IRC- "..message) | ||||||
| 	end | 	end | ||||||
| end | end | ||||||
|  |  | ||||||
| @@ -190,31 +193,31 @@ end | |||||||
|  |  | ||||||
|  |  | ||||||
| function irc.hooks.nick(user, newNick) | function irc.hooks.nick(user, newNick) | ||||||
| 	irc:sendLocal(("-!- %s is now known as %s") | 	irc.sendLocal(("-!- %s is now known as %s") | ||||||
| 			:format(user.nick, newNick)) | 			:format(user.nick, newNick)) | ||||||
| end | end | ||||||
|  |  | ||||||
|  |  | ||||||
| function irc.hooks.join(user, channel) | function irc.hooks.join(user, channel) | ||||||
| 	irc:sendLocal(("-!- %s joined %s") | 	irc.sendLocal(("-!- %s joined %s") | ||||||
| 			:format(user.nick, channel)) | 			:format(user.nick, channel)) | ||||||
| end | end | ||||||
|  |  | ||||||
|  |  | ||||||
| function irc.hooks.part(user, channel, reason) | function irc.hooks.part(user, channel, reason) | ||||||
| 	reason = reason or "" | 	reason = reason or "" | ||||||
| 	irc:sendLocal(("-!- %s has left %s [%s]") | 	irc.sendLocal(("-!- %s has left %s [%s]") | ||||||
| 			:format(user.nick, channel, reason)) | 			:format(user.nick, channel, reason)) | ||||||
| end | end | ||||||
|  |  | ||||||
|  |  | ||||||
| function irc.hooks.quit(user, reason) | function irc.hooks.quit(user, reason) | ||||||
| 	irc:sendLocal(("-!- %s has quit [%s]") | 	irc.sendLocal(("-!- %s has quit [%s]") | ||||||
| 			:format(user.nick, reason)) | 			:format(user.nick, reason)) | ||||||
| end | end | ||||||
|  |  | ||||||
|  |  | ||||||
| function irc.hooks.disconnect(message, isError) | function irc.hooks.disconnect(_, isError) | ||||||
| 	irc.connected = false | 	irc.connected = false | ||||||
| 	if isError then | 	if isError then | ||||||
| 		minetest.log("error",  "IRC: Error: Disconnected, reconnecting in one minute.") | 		minetest.log("error",  "IRC: Error: Disconnected, reconnecting in one minute.") | ||||||
| @@ -242,19 +245,19 @@ function irc.hooks.preregister(conn) | |||||||
| end | end | ||||||
|  |  | ||||||
|  |  | ||||||
| irc:register_hook("PreRegister",     irc.hooks.preregister) | irc.register_hook("PreRegister",     irc.hooks.preregister) | ||||||
| irc:register_hook("OnRaw",           irc.hooks.raw) | irc.register_hook("OnRaw",           irc.hooks.raw) | ||||||
| irc:register_hook("OnSend",          irc.hooks.send) | irc.register_hook("OnSend",          irc.hooks.send) | ||||||
| irc:register_hook("DoPrivmsg",       irc.hooks.chat) | irc.register_hook("DoPrivmsg",       irc.hooks.chat) | ||||||
| irc:register_hook("OnPart",          irc.hooks.part) | irc.register_hook("OnPart",          irc.hooks.part) | ||||||
| irc:register_hook("OnKick",          irc.hooks.kick) | irc.register_hook("OnKick",          irc.hooks.kick) | ||||||
| irc:register_hook("OnJoin",          irc.hooks.join) | irc.register_hook("OnJoin",          irc.hooks.join) | ||||||
| irc:register_hook("OnQuit",          irc.hooks.quit) | irc.register_hook("OnQuit",          irc.hooks.quit) | ||||||
| irc:register_hook("NickChange",      irc.hooks.nick) | irc.register_hook("NickChange",      irc.hooks.nick) | ||||||
| irc:register_hook("OnCTCP",          irc.hooks.ctcp) | irc.register_hook("OnCTCP",          irc.hooks.ctcp) | ||||||
| irc:register_hook("PrivateMessage",  irc.hooks.pm) | irc.register_hook("PrivateMessage",  irc.hooks.pm) | ||||||
| irc:register_hook("OnNotice",        irc.hooks.notice) | irc.register_hook("OnNotice",        irc.hooks.notice) | ||||||
| irc:register_hook("OnChannelChat",   irc.hooks.channelChat) | irc.register_hook("OnChannelChat",   irc.hooks.channelChat) | ||||||
| irc:register_hook("OnModeChange",    irc.hooks.mode) | irc.register_hook("OnModeChange",    irc.hooks.mode) | ||||||
| irc:register_hook("OnDisconnect",    irc.hooks.disconnect) | irc.register_hook("OnDisconnect",    irc.hooks.disconnect) | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										180
									
								
								init.lua
									
									
									
									
									
								
							
							
						
						
									
										180
									
								
								init.lua
									
									
									
									
									
								
							| @@ -3,24 +3,50 @@ | |||||||
|  |  | ||||||
| local modpath = minetest.get_modpath(minetest.get_current_modname()) | local modpath = minetest.get_modpath(minetest.get_current_modname()) | ||||||
|  |  | ||||||
| package.path = | -- Handle mod security if needed | ||||||
|  | local ie, req_ie = _G, minetest.request_insecure_environment | ||||||
|  | if req_ie then ie = req_ie() end | ||||||
|  | if not ie then | ||||||
|  | 	error("The IRC mod requires access to insecure functions in order ".. | ||||||
|  | 		"to work.  Please add the irc mod to your secure.trusted_mods ".. | ||||||
|  | 		"setting or disable the irc mod.") | ||||||
|  | end | ||||||
|  |  | ||||||
|  | ie.package.path = | ||||||
| 		-- To find LuaIRC's init.lua | 		-- To find LuaIRC's init.lua | ||||||
| 		modpath.."/?/init.lua;" | 		modpath.."/?/init.lua;" | ||||||
| 		-- For LuaIRC to find its files | 		-- For LuaIRC to find its files | ||||||
| 		..modpath.."/?.lua;" | 		..modpath.."/?.lua;" | ||||||
| 		..package.path | 		..ie.package.path | ||||||
|  |  | ||||||
| -- The build of Lua that Minetest comes with only looks for libraries under | -- The build of Lua that Minetest comes with only looks for libraries under | ||||||
| -- /usr/local/share and /usr/local/lib but LuaSocket is often installed under | -- /usr/local/share and /usr/local/lib but LuaSocket is often installed under | ||||||
| -- /usr/share and /usr/lib. | -- /usr/share and /usr/lib. | ||||||
| if not rawget(_G, "jit") and package.config:sub(1, 1) == "/" then | if not rawget(_G, "jit") and package.config:sub(1, 1) == "/" then | ||||||
| 	package.path = package.path.. |  | ||||||
|  | 	ie.package.path = ie.package.path.. | ||||||
| 			";/usr/share/lua/5.1/?.lua".. | 			";/usr/share/lua/5.1/?.lua".. | ||||||
| 			";/usr/share/lua/5.1/?/init.lua" | 			";/usr/share/lua/5.1/?/init.lua" | ||||||
| 	package.cpath = package.cpath.. |  | ||||||
| 			";/usr/lib/lua/5.1/?.so" | 	ie.package.cpath = ie.package.cpath.. | ||||||
|  | 			";/usr/lib/lua/5.1/?.so".. | ||||||
|  | 			";/usr/lib64/lua/5.1/?.so" | ||||||
|  |  | ||||||
|  | 	ie.package.cpath = "/usr/lib/x86_64-linux-gnu/lua/5.1/?.so;"..ie.package.cpath | ||||||
|  |  | ||||||
|  |  | ||||||
| end | end | ||||||
|  |  | ||||||
|  | -- Temporarily set require so that LuaIRC can access it | ||||||
|  | local old_require = require | ||||||
|  | require = ie.require | ||||||
|  |  | ||||||
|  | -- Silence warnings about `module` in `ltn12`. | ||||||
|  | local old_module = rawget(_G, "module") | ||||||
|  | rawset(_G, "module", ie.module) | ||||||
|  |  | ||||||
|  | local lib = ie.require("irc") | ||||||
|  |  | ||||||
| irc = { | irc = { | ||||||
| 	version = "0.2.0", | 	version = "0.2.0", | ||||||
| 	connected = false, | 	connected = false, | ||||||
| @@ -29,45 +55,82 @@ irc = { | |||||||
| 	recent_message_count = 0, | 	recent_message_count = 0, | ||||||
| 	joined_players = {}, | 	joined_players = {}, | ||||||
| 	modpath = modpath, | 	modpath = modpath, | ||||||
| 	lib = require("irc"), | 	lib = lib, | ||||||
| } | } | ||||||
|  |  | ||||||
| -- Compatibility | -- Compatibility | ||||||
| mt_irc = irc | rawset(_G, "mt_irc", irc) | ||||||
|  |  | ||||||
|  | local getinfo = debug.getinfo | ||||||
|  | local warned = { } | ||||||
|  |  | ||||||
|  | local function warn_deprecated(k) | ||||||
|  | 	local info = getinfo(3) | ||||||
|  | 	local loc = info.source..":"..info.currentline | ||||||
|  | 	if warned[loc] then return end | ||||||
|  | 	warned[loc] = true | ||||||
|  | 	print("COLON: "..tostring(k)) | ||||||
|  | 	minetest.log("warning", "Deprecated use of colon notation when calling" | ||||||
|  | 			.." method `"..tostring(k).."` at "..loc) | ||||||
|  | end | ||||||
|  |  | ||||||
|  | -- This is a hack. | ||||||
|  | setmetatable(irc, { | ||||||
|  | 	__newindex = function(t, k, v) | ||||||
|  | 		if type(v) == "function" then | ||||||
|  | 			local f = v | ||||||
|  | 			v = function(me, ...) | ||||||
|  | 				if rawequal(me, t) then | ||||||
|  | 					warn_deprecated(k) | ||||||
|  | 					return f(...) | ||||||
|  | 				else | ||||||
|  | 					return f(me, ...) | ||||||
|  | 				end | ||||||
|  | 			end | ||||||
|  | 		end | ||||||
|  | 		rawset(t, k, v) | ||||||
|  | 	end, | ||||||
|  | }) | ||||||
|  |  | ||||||
| dofile(modpath.."/config.lua") | dofile(modpath.."/config.lua") | ||||||
| dofile(modpath.."/messages.lua") | dofile(modpath.."/messages.lua") | ||||||
| dofile(modpath.."/hooks.lua") | loadfile(modpath.."/hooks.lua")(ie) | ||||||
| dofile(modpath.."/callback.lua") | dofile(modpath.."/callback.lua") | ||||||
| dofile(modpath.."/chatcmds.lua") | dofile(modpath.."/chatcmds.lua") | ||||||
| dofile(modpath.."/botcmds.lua") | dofile(modpath.."/botcmds.lua") | ||||||
|  |  | ||||||
|  | -- Restore old (safe) functions | ||||||
|  | require = old_require | ||||||
|  | rawset(_G, "module", old_module) | ||||||
|  |  | ||||||
| if irc.config.enable_player_part then | if irc.config.enable_player_part then | ||||||
| 	dofile(modpath.."/player_part.lua") | 	dofile(modpath.."/player_part.lua") | ||||||
| else | else | ||||||
| 	setmetatable(irc.joined_players, {__index = function(index) return true end}) | 	setmetatable(irc.joined_players, {__index = function() return true end}) | ||||||
| end | end | ||||||
|  |  | ||||||
| minetest.register_privilege("irc_admin", { | minetest.register_privilege("irc_admin", { | ||||||
| 	description = "Allow IRC administrative tasks to be performed.", | 	description = "Allow IRC administrative tasks to be performed.", | ||||||
| 	give_to_singleplayer = true | 	give_to_singleplayer = true, | ||||||
|  | 	give_to_admin = true, | ||||||
| }) | }) | ||||||
|  |  | ||||||
| local stepnum = 0 | local stepnum = 0 | ||||||
|  |  | ||||||
| minetest.register_globalstep(function(dtime) return irc:step(dtime) end) | minetest.register_globalstep(function(dtime) return irc.step(dtime) end) | ||||||
|  |  | ||||||
| function irc:step(dtime) | function irc.step() | ||||||
| 	if stepnum == 3 then | 	if stepnum == 3 then | ||||||
| 		if self.config.auto_connect then | 		if irc.config.auto_connect then | ||||||
| 			self:connect() | 			irc.connect() | ||||||
| 		end | 		end | ||||||
| 	end | 	end | ||||||
| 	stepnum = stepnum + 1 | 	stepnum = stepnum + 1 | ||||||
|  |  | ||||||
| 	if not self.connected then return end | 	if not irc.connected then return end | ||||||
|  |  | ||||||
| 	-- Hooks will manage incoming messages and errors | 	-- Hooks will manage incoming messages and errors | ||||||
| 	local good, err = xpcall(function() self.conn:think() end, debug.traceback) | 	local good, err = xpcall(function() irc.conn:think() end, debug.traceback) | ||||||
| 	if not good then | 	if not good then | ||||||
| 		print(err) | 		print(err) | ||||||
| 		return | 		return | ||||||
| @@ -75,79 +138,90 @@ function irc:step(dtime) | |||||||
| end | end | ||||||
|  |  | ||||||
|  |  | ||||||
| function irc:connect() | function irc.connect() | ||||||
| 	if self.connected then | 	if irc.connected then | ||||||
| 		minetest.log("error", "IRC: Ignoring attempt to connect when already connected.") | 		minetest.log("error", "IRC: Ignoring attempt to connect when already connected.") | ||||||
| 		return | 		return | ||||||
| 	end | 	end | ||||||
| 	self.conn = irc.lib.new({ | 	irc.conn = irc.lib.new({ | ||||||
| 		nick = self.config.nick, | 		nick = irc.config.nick, | ||||||
| 		username = "Minetest", | 		username = irc.config.username, | ||||||
| 		realname = "Minetest", | 		realname = irc.config.realname, | ||||||
| 	}) | 	}) | ||||||
| 	self:doHook(self.conn) | 	irc.doHook(irc.conn) | ||||||
|  |  | ||||||
|  | 	-- We need to swap the `require` function again since | ||||||
|  | 	-- LuaIRC `require`s `ssl` if `irc.secure` is true. | ||||||
|  | 	old_require = require | ||||||
|  | 	require = ie.require | ||||||
|  |  | ||||||
| 	local good, message = pcall(function() | 	local good, message = pcall(function() | ||||||
| 		self.conn:connect({ | 		irc.conn:connect({ | ||||||
| 			host = self.config.server, | 			host = irc.config.server, | ||||||
| 			port = self.config.port, | 			port = irc.config.port, | ||||||
| 			password = self.config.password, | 			password = irc.config.password, | ||||||
| 			timeout = self.config.timeout, | 			timeout = irc.config.timeout, | ||||||
| 			secure = self.config.secure | 			reconnect = irc.config.reconnect, | ||||||
|  | 			secure = irc.config.secure | ||||||
| 		}) | 		}) | ||||||
| 	end) | 	end) | ||||||
|  |  | ||||||
|  | 	require = old_require | ||||||
|  |  | ||||||
| 	if not good then | 	if not good then | ||||||
| 		minetest.log("error", ("IRC: Connection error: %s: %s -- Reconnecting in ten minutes...") | 		minetest.log("error", ("IRC: Connection error: %s: %s -- Reconnecting in %d seconds...") | ||||||
| 					:format(self.config.server, message)) | 					:format(irc.config.server, message, irc.config.reconnect)) | ||||||
| 		minetest.after(600, function() self:connect() end) | 		minetest.after(irc.config.reconnect, function() irc.connect() end) | ||||||
| 		return | 		return | ||||||
| 	end | 	end | ||||||
|  |  | ||||||
| 	if self.config.NSPass then | 	if irc.config.NSPass then | ||||||
| 		self:say("NickServ", "IDENTIFY "..self.config.NSPass) | 		irc.conn:queue(irc.msgs.privmsg( | ||||||
|  | 				"NickServ", "IDENTIFY "..irc.config.NSPass)) | ||||||
| 	end | 	end | ||||||
|  |  | ||||||
| 	self.conn:join(self.config.channel, self.config.key) | 	irc.conn:join(irc.config.channel, irc.config.key) | ||||||
| 	self.connected = true | 	irc.connected = true | ||||||
| 	minetest.log("action", "IRC: Connected!") | 	minetest.log("action", "IRC: Connected!") | ||||||
| 	minetest.chat_send_all("IRC: Connected!") | 	minetest.chat_send_all("IRC: Connected!") | ||||||
| end | end | ||||||
|  |  | ||||||
|  |  | ||||||
| function irc:disconnect(message) | function irc.disconnect(message) | ||||||
| 	if self.connected then | 	if irc.connected then | ||||||
| 		--The OnDisconnect hook will clear self.connected and print a disconnect message | 		--The OnDisconnect hook will clear irc.connected and print a disconnect message | ||||||
| 		self.conn:disconnect(message) | 		irc.conn:disconnect(message) | ||||||
| 	end | 	end | ||||||
| end | end | ||||||
|  |  | ||||||
|  |  | ||||||
| function irc:say(to, message) | function irc.say(to, message) | ||||||
| 	if not message then | 	if not message then | ||||||
| 		message = to | 		message = to | ||||||
| 		to = self.config.channel | 		to = irc.config.channel | ||||||
| 	end | 	end | ||||||
| 	to = to or self.config.channel | 	to = to or irc.config.channel | ||||||
|  |  | ||||||
| 	self:queue(irc.msgs.privmsg(to, message)) | 	irc.queue(irc.msgs.privmsg(to, message)) | ||||||
| end | end | ||||||
|  |  | ||||||
|  |  | ||||||
| function irc:reply(message) | function irc.reply(message) | ||||||
| 	if not self.last_from then | 	if not irc.last_from then | ||||||
| 		return | 		return | ||||||
| 	end | 	end | ||||||
| 	message = message:gsub("[\r\n%z]", " \\n ") | 	message = message:gsub("[\r\n%z]", " \\n ") | ||||||
| 	self:say(self.last_from, message) | 	irc.say(irc.last_from, message) | ||||||
| end | end | ||||||
|  |  | ||||||
| function irc:send(msg) | function irc.send(msg) | ||||||
| 	if not self.connected then return end | 	if not irc.connected then return end | ||||||
| 	self.conn:send(msg) | 	irc.conn:send(msg) | ||||||
| end | end | ||||||
|  |  | ||||||
| function irc:queue(msg) | function irc.queue(msg) | ||||||
| 	if not self.connected then return end | 	if not irc.connected then return end | ||||||
| 	self.conn:queue(msg) | 	irc.conn:queue(msg) | ||||||
| end | end | ||||||
|  |  | ||||||
|  | minetest.log("action", "[irc] loaded.") | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								irc
									
									
									
									
									
								
							
							
								
								
								
								
								
							
						
						
									
										2
									
								
								irc
									
									
									
									
									
								
							 Submodule irc updated: e49a52ede1...b8d594e651
									
								
							
							
								
								
									
										12
									
								
								messages.lua
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								messages.lua
									
									
									
									
									
								
							| @@ -3,11 +3,15 @@ | |||||||
|  |  | ||||||
| irc.msgs = irc.lib.msgs | irc.msgs = irc.lib.msgs | ||||||
|  |  | ||||||
| function irc:sendLocal(message) | function irc.logChat(message) | ||||||
| 	minetest.chat_send_all(message) | 	minetest.log("action", "IRC CHAT: "..message) | ||||||
| end | end | ||||||
|  |  | ||||||
| function irc:playerMessage(name, message) | function irc.sendLocal(message) | ||||||
|  | 	minetest.chat_send_all(minetest.colorize(irc.config.chat_color, message)) | ||||||
|  | 	irc.logChat(message) | ||||||
|  | end | ||||||
|  |  | ||||||
|  | function irc.playerMessage(name, message) | ||||||
| 	return ("<%s> %s"):format(name, message) | 	return ("<%s> %s"):format(name, message) | ||||||
| end | end | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								mod.conf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								mod.conf
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | name = irc | ||||||
|  | description = """ | ||||||
|  | This mod is just a glue between IRC and Minetest. | ||||||
|  | It provides two-way communication between the in-game chat, and an arbitrary IRC channel. | ||||||
|  | """ | ||||||
| @@ -2,68 +2,71 @@ | |||||||
| -- See LICENSE.txt for details. | -- See LICENSE.txt for details. | ||||||
|  |  | ||||||
|  |  | ||||||
| function irc:player_part(name) | function irc.player_part(name) | ||||||
| 	if not self.joined_players[name] then | 	if not irc.joined_players[name] then | ||||||
| 		minetest.chat_send_player(name, "IRC: You are not in the channel.") | 		return false, "You are not in the channel" | ||||||
| 		return |  | ||||||
| 	end | 	end | ||||||
| 	self.joined_players[name] = nil | 	irc.joined_players[name] = nil | ||||||
| 	minetest.chat_send_player(name, "IRC: You are now out of the channel.") | 	return true, "You left the channel" | ||||||
| end | end | ||||||
|   |  | ||||||
| function irc:player_join(name) | function irc.player_join(name) | ||||||
| 	if self.joined_players[name] then | 	if irc.joined_players[name] then | ||||||
| 		minetest.chat_send_player(name, "IRC: You are already in the channel.") | 		return false, "You are already in the channel" | ||||||
| 		return | 	elseif not minetest.get_player_by_name(name) then | ||||||
|  | 		return false, "You need to be in-game to join the channel" | ||||||
| 	end | 	end | ||||||
| 	self.joined_players[name] = true | 	irc.joined_players[name] = true | ||||||
| 	minetest.chat_send_player(name, "IRC: You are now in the channel.") | 	return true, "You joined the channel" | ||||||
| end | end | ||||||
|  |  | ||||||
|  |  | ||||||
| minetest.register_chatcommand("join", { | minetest.register_chatcommand("join", { | ||||||
| 	description = "Join the IRC channel", | 	description = "Join the IRC channel", | ||||||
| 	privs = {shout=true}, | 	privs = {shout=true}, | ||||||
| 	func = function(name, param) | 	func = function(name) | ||||||
| 		irc:player_join(name) | 		return irc.player_join(name) | ||||||
| 	end |  | ||||||
| }) |  | ||||||
|   |  | ||||||
| minetest.register_chatcommand("part", { |  | ||||||
| 	description = "Part the IRC channel", |  | ||||||
| 	privs = {shout=true}, |  | ||||||
| 	func = function(name, param) |  | ||||||
| 		irc:player_part(name) |  | ||||||
| 	end |  | ||||||
| }) |  | ||||||
|   |  | ||||||
| minetest.register_chatcommand("who", { |  | ||||||
| 	description = "Tell who is currently on the channel", |  | ||||||
| 	privs = {}, |  | ||||||
| 	func = function(name, param) |  | ||||||
| 		local s = "" |  | ||||||
| 		for name, _ in pairs(irc.joined_players) do |  | ||||||
| 			s = s..", "..name |  | ||||||
| 		end |  | ||||||
| 		minetest.chat_send_player(name, "Players On Channel:"..s) |  | ||||||
| 	end | 	end | ||||||
| }) | }) | ||||||
|  |  | ||||||
|   | minetest.register_chatcommand("part", { | ||||||
|  | 	description = "Part the IRC channel", | ||||||
|  | 	privs = {shout=true}, | ||||||
|  | 	func = function(name) | ||||||
|  | 		return irc.player_part(name) | ||||||
|  | 	end | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | minetest.register_chatcommand("who", { | ||||||
|  | 	description = "Tell who is currently on the channel", | ||||||
|  | 	privs = {}, | ||||||
|  | 	func = function() | ||||||
|  | 		local out, n = { }, 0 | ||||||
|  | 		for plname in pairs(irc.joined_players) do | ||||||
|  | 			n = n + 1 | ||||||
|  | 			out[n] = plname | ||||||
|  | 		end | ||||||
|  | 		table.sort(out) | ||||||
|  | 		return true, n.." player(s) in channel: "..table.concat(out, ", ") | ||||||
|  | 	end | ||||||
|  | }) | ||||||
|  |  | ||||||
|  |  | ||||||
| minetest.register_on_joinplayer(function(player) | minetest.register_on_joinplayer(function(player) | ||||||
| 	local name = player:get_player_name() | 	local name = player:get_player_name() | ||||||
| 	irc.joined_players[name] = irc.config.auto_join | 	irc.joined_players[name] = irc.config.auto_join | ||||||
| end) | end) | ||||||
|   |  | ||||||
|   |  | ||||||
| minetest.register_on_leaveplayer(function(player) | minetest.register_on_leaveplayer(function(player) | ||||||
| 	local name = player:get_player_name() | 	local name = player:get_player_name() | ||||||
| 	irc.joined_players[name] = nil | 	irc.joined_players[name] = nil | ||||||
| end) | end) | ||||||
|  |  | ||||||
| function irc:sendLocal(message) | function irc.sendLocal(message) | ||||||
|         for name, _ in pairs(self.joined_players) do | 	for name, _ in pairs(irc.joined_players) do | ||||||
| 		minetest.chat_send_player(name, message) | 		minetest.chat_send_player(name, | ||||||
|  | 					minetest.colorize(irc.config.chat_color, message)) | ||||||
| 	end | 	end | ||||||
|  | 	irc.logChat(message) | ||||||
| end | end | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										75
									
								
								settingtypes.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								settingtypes.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,75 @@ | |||||||
|  |  | ||||||
|  | [Basic] | ||||||
|  |  | ||||||
|  | # Whether to automatically connect to the server on mod load. | ||||||
|  | # If false, you must use /irc_connect to connect. | ||||||
|  | irc.auto_connect (Auto-connect on load) bool true | ||||||
|  |  | ||||||
|  | # Nickname for the bot. May only contain characters A-Z, 0-9 | ||||||
|  | # '{', '}', '[', ']', '|', '^', '-', or '_'. | ||||||
|  | irc.nick (Bot nickname) string Minetest | ||||||
|  |  | ||||||
|  | # Server to connect to. | ||||||
|  | irc.server (IRC server) string irc.freenode.net | ||||||
|  |  | ||||||
|  | # Server port. | ||||||
|  | # The standard IRC protocol port is 6667 for regular servers, | ||||||
|  | # or 6697 for SSL-enabled servers. | ||||||
|  | # If unsure, leave at 6667. | ||||||
|  | irc.port (IRC server port) int 6667 1 65535 | ||||||
|  |  | ||||||
|  | # Channel the bot joins after connecting. | ||||||
|  | irc.channel (Channel to join) string ##mt-irc-mod | ||||||
|  |  | ||||||
|  | [Authentication] | ||||||
|  |  | ||||||
|  | # Password for authenticating to NickServ. | ||||||
|  | # Leave empty to not authenticate with NickServ. | ||||||
|  | irc.NSPass (NickServ password) string | ||||||
|  |  | ||||||
|  | # IRC server password. | ||||||
|  | # Leave empty for no password. | ||||||
|  | irc.password (Server password) string | ||||||
|  |  | ||||||
|  | # Password for joining the channel. | ||||||
|  | # Leave empty if your channel is not protected. | ||||||
|  | irc.key (Channel key) string | ||||||
|  |  | ||||||
|  | # Enable TLS connection. | ||||||
|  | # Requires LuaSEC <https://github.com/brunoos/luasec>. | ||||||
|  | irc.secure (Use TLS) bool false | ||||||
|  |  | ||||||
|  | # Username for SASL authentication. | ||||||
|  | # Leave empty to use the nickname. | ||||||
|  | irc.sasl.user (SASL username) string | ||||||
|  |  | ||||||
|  | # Password for SASL authentication. | ||||||
|  | # Leave empty to not authenticate via SASL. | ||||||
|  | irc.sasl.pass (SASL password) string | ||||||
|  |  | ||||||
|  | [Advanced] | ||||||
|  |  | ||||||
|  | # Enable this to make the bot send messages when players join | ||||||
|  | # or leave the game server. | ||||||
|  | irc.send_join_part (Send join and part messages) bool true | ||||||
|  |  | ||||||
|  | # Enable this to make the bot send messages when players are kicked. | ||||||
|  | irc.send_kicks (Send kick messages) bool false | ||||||
|  |  | ||||||
|  | # Underlying socket timeout in seconds. | ||||||
|  | irc.timeout (Timeout) int 60 1 | ||||||
|  |  | ||||||
|  | # Time between reconnection attempts, in seconds. | ||||||
|  | irc.reconnect (Reconnect delay) int 600 1 | ||||||
|  |  | ||||||
|  | # Prefix to use for bot commands. | ||||||
|  | irc.command_prefix (Command prefix) string | ||||||
|  |  | ||||||
|  | # Enable debug output. | ||||||
|  | irc.debug (Debug mode) bool false | ||||||
|  |  | ||||||
|  | # Whether to enable players joining and parting the channel. | ||||||
|  | irc.enable_player_part (Allow player join/part) bool true | ||||||
|  |  | ||||||
|  | # Whether to automatically show players in the channel when they join. | ||||||
|  | irc.auto_join (Auto join players) bool true | ||||||
		Reference in New Issue
	
	Block a user