A Python robot that edits Wikipedia and interacts with people over IRC https://en.wikipedia.org/wiki/User:EarwigBot
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

103 lines
4.1 KiB

  1. # -*- coding: utf-8 -*-
  2. #
  3. # Copyright (C) 2009-2021 Ben Kurtovic <ben.kurtovic@gmail.com>
  4. #
  5. # Permission is hereby granted, free of charge, to any person obtaining a copy
  6. # of this software and associated documentation files (the "Software"), to deal
  7. # in the Software without restriction, including without limitation the rights
  8. # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  9. # copies of the Software, and to permit persons to whom the Software is
  10. # furnished to do so, subject to the following conditions:
  11. #
  12. # The above copyright notice and this permission notice shall be included in
  13. # all copies or substantial portions of the Software.
  14. #
  15. # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  16. # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  17. # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  18. # AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  19. # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  20. # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  21. # SOFTWARE.
  22. from earwigbot.commands import Command
  23. class ChanOps(Command):
  24. """Voice, devoice, op, or deop users in the channel, or join or part from
  25. other channels."""
  26. name = "chanops"
  27. commands = ["chanops", "voice", "devoice", "op", "deop", "join", "part", "listchans"]
  28. def process(self, data):
  29. if data.command == "chanops":
  30. msg = "Available commands are {0}."
  31. self.reply(data, msg.format(", ".join(
  32. "!" + cmd for cmd in self.commands if cmd != data.command)))
  33. return
  34. de_escalate = data.command in ["devoice", "deop"]
  35. if de_escalate and (not data.args or data.args[0] == data.nick):
  36. target = data.nick
  37. elif not self.config.irc["permissions"].is_admin(data):
  38. self.reply(data, "You must be a bot admin to use this command.")
  39. return
  40. if data.command == "join":
  41. self.do_join(data)
  42. elif data.command == "part":
  43. self.do_part(data)
  44. elif data.command == "listchans":
  45. self.do_list(data)
  46. else:
  47. # If it is just !op/!devoice/whatever without arguments, assume
  48. # they want to do this to themselves:
  49. if not data.args:
  50. target = data.nick
  51. else:
  52. target = data.args[0]
  53. command = data.command.upper()
  54. self.say("ChanServ", " ".join((command, data.chan, target)))
  55. log = "{0} requested {1} on {2} in {3}"
  56. self.logger.info(log.format(data.nick, command, target, data.chan))
  57. def do_join(self, data):
  58. if data.args:
  59. channel = data.args[0]
  60. if not channel.startswith("#"):
  61. channel = "#" + channel
  62. else:
  63. msg = "You must specify a channel to join or part from."
  64. self.reply(data, msg)
  65. return
  66. self.join(channel)
  67. log = "{0} requested JOIN to {1}".format(data.nick, channel)
  68. self.logger.info(log)
  69. def do_part(self, data):
  70. channel = data.chan
  71. reason = None
  72. if data.args:
  73. if data.args[0].startswith("#"):
  74. # "!part #channel reason for parting"
  75. channel = data.args[0]
  76. if data.args[1:]:
  77. reason = " ".join(data.args[1:])
  78. else: # "!part reason for parting"; assume current channel
  79. reason = " ".join(data.args)
  80. msg = "Requested by {0}".format(data.nick)
  81. log = "{0} requested PART from {1}".format(data.nick, channel)
  82. if reason:
  83. msg += ": {0}".format(reason)
  84. log += ' ("{0}")'.format(reason)
  85. self.part(channel, msg)
  86. self.logger.info(log)
  87. def do_list(self, data):
  88. chans = self.bot.frontend.channels
  89. if not chans:
  90. self.reply(data, "I am currently in no channels.")
  91. return
  92. self.reply(data, "I am currently in \x02{0}\x0F channel{1}: {2}.".format(
  93. len(chans), "" if len(chans) == 1 else "s", ", ".join(chans)))