admin.py 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285
  1. from common import userstatus, do_everything_to
  2. from commands.common import CommandHelpers as CH
  3. from commands.common import AdminHelpers as AH
  4. bold = "\x02"
  5. italic = "\x1D"
  6. underline = "\x1F"
  7. reverse = "\x16" # swap background and foreground colors ("reverse video")
  8. reset = "\x0F"
  9. blue = "\x0302"
  10. green = "\x0303"
  11. red = "\x0304"
  12. grey = "\x0314"
  13. def do_command(self, connection, event):
  14. cmdtype, trigger, command, replyto = CH.disect_command(self, event)
  15. if not command:
  16. return # Do nothing if there is no command.
  17. # Ignore channel commands from users that do not have at least voice in homechannel or operator status in target channel.
  18. if event.type == "pubmsg": # It's a channel message.
  19. if not userstatus.atleast_oper(self, event.source.nick, self.homechannel) and not userstatus.atleast_oper(self, event.source.nick, event.target): # Does not have at least voiced status in homechannel or operator status in target channel.
  20. return
  21. if command == "cmd" or command == "cmds" or command == "commands":
  22. if cmdtype == "cmd":
  23. message = grey + "Admin: "
  24. if CH.ccc(self, "channelfunctions", {"homechan": "oper", "chan": "oper"}, event):
  25. message += CH.ccc(self, "channelfunctions", {"homechan": "oper", "chan": "oper"}, event)
  26. if CH.ccc(self, "join", {"homechan": "oper", "chan": None}, event):
  27. message += CH.ccc(self, "join", {"homechan": "oper", "chan": None}, event)
  28. if CH.ccc(self, "part", {"homechan": "oper", "chan": None}, event):
  29. message += CH.ccc(self, "part", {"homechan": "oper", "chan": None}, event)
  30. if CH.ccc(self, "quit", {"homechan": "admin", "chan": None}, event):
  31. message += CH.ccc(self, "quit", {"homechan": "admin", "chan": None}, event)
  32. if CH.ccc(self, "reconnect", {"homechan": "oper", "chan": None}, event):
  33. message += CH.ccc(self, "reconnect", {"homechan": "oper", "chan": None}, event)
  34. if CH.ccc(self, "recovernick", {"homechan": "oper", "chan": None}, event):
  35. message += CH.ccc(self, "recovernick", {"homechan": "oper", "chan": None}, event)
  36. if CH.ccc(self, "msg", {"homechan": "oper", "chan": "oper"}, event):
  37. message += CH.ccc(self, "msg", {"homechan": "oper", "chan": "oper"}, event)
  38. if CH.ccc(self, "act", {"homechan": "oper", "chan": "oper"}, event):
  39. message += CH.ccc(self, "act", {"homechan": "oper", "chan": "oper"}, event)
  40. if message == grey + "Admin: ": # No commands to display.
  41. return
  42. connection.privmsg(replyto, message[:-2] + ".")
  43. elif command.split()[0] == "quit":
  44. if not userstatus.atleast_admin(self, event.source.nick, self.homechannel): #Insufficient rights.
  45. connection.privmsg(replyto, "Denied, you need to have admin (super operator) status or higher in " + red + self.homechannel + reset + ".")
  46. return
  47. if cmdtype == "help": # Display help text.
  48. if len(command.split()) is not 1:
  49. return
  50. connection.privmsg(replyto, "Disconnect and terminate " + connection.get_nickname() + ". Optionally with reason.")
  51. connection.privmsg(replyto, grey + "Usage: " + blue + "!quit " + reset + italic + "reason")
  52. elif cmdtype == "cmd":
  53. if len(command.split()) == 1:
  54. self.die(msg = "Killed by " + event.source.nick)
  55. else:
  56. self.die(msg = "[" + event.source.nick + "] " + command.split(maxsplit=1)[1])
  57. elif command.split()[0] == "reconnect":
  58. if not userstatus.atleast_oper(self, event.source.nick, self.homechannel):
  59. connection.privmsg(replyto, "Denied, you need to have operator status or higher in " + red + self.homechannel + reset + ".")
  60. return
  61. if cmdtype == "help": # Display help text.
  62. if len(command.split()) is not 1:
  63. return
  64. connection.privmsg(replyto, "Reconnect " + connection.get_nickname() + ". Reason optional.")
  65. connection.privmsg(replyto, grey + "Usage: " + blue + "!reconnect " + reset + italic + "reason")
  66. elif cmdtype == "cmd":
  67. if len(command.split()) == 1:
  68. self.disconnect(msg = "Reconnect requested by " + event.source.nick)
  69. else:
  70. self.disconnect(msg = "[" + event.source.nick + "] " + command.split(maxsplit=1)[1])
  71. elif command.split()[0] == "recovernick":
  72. if not userstatus.atleast_voiced(self, event.source.nick, self.homechannel):
  73. connection.privmsg(replyto, "Denied, you need to have voiced status or higher in " + red + self.homechannel + reset + ".")
  74. return
  75. if cmdtype == "help": # Display help text.
  76. if len(command.split()) is not 1:
  77. return
  78. connection.privmsg(replyto, "Let " + connection.get_nickname() + " try to recover " + connection.nickname + " as nickname.")
  79. elif cmdtype == "cmd":
  80. if connection.get_nickname() == connection.nickname:
  81. connection.action(replyto, "is already named " + red + connection.nickname + reset + ".")
  82. return
  83. from common.networkservices import NickServ
  84. NickServ.recover_nick(connection, self.password)
  85. elif command.split()[0] == "join":
  86. if not userstatus.atleast_oper(self, event.source.nick, self.homechannel):
  87. connection.privmsg(replyto, "Denied, you need to have operator status or higher in " + red + self.homechannel + reset + ".")
  88. return
  89. if cmdtype == "help": #Display help text.
  90. if len(command.split()) is not 1:
  91. return
  92. connection.privmsg(replyto, "Make " + connection.get_nickname() + " join a channel. Password optional.")
  93. connection.privmsg(replyto, grey + "Usage: " + blue + self.cmdchar + "join " + red + italic + "channel " + reset + italic + "password")
  94. elif cmdtype == "cmd":
  95. try:
  96. channel = command.split()[1]
  97. except IndexError:
  98. connection.privmsg(replyto, "Specify channel. For help type: " + blue + self.helpchar + "join")
  99. return
  100. try:
  101. key = command.split(maxsplit=2)[2]
  102. except IndexError:
  103. do_everything_to.join(self, connection, channel)
  104. return
  105. do_everything_to.join(self, connection, channel, key)
  106. elif command.split()[0] == "part":
  107. if cmdtype == "help": #Display help text.
  108. if len(command.split()) is not 1:
  109. return
  110. connection.privmsg(replyto, "Make " + connection.get_nickname() + " part a channel. Reason optional.")
  111. connection.privmsg(replyto, grey + "Usage: " + blue + self.cmdchar + "join " + red + italic + "channel " + reset + italic + "password")
  112. elif cmdtype == "cmd":
  113. if userstatus.atleast_oper(self, event.source.nick, self.homechannel): # Is at least operator in home channel.
  114. homeadmin = True
  115. if userstatus.atleast_oper(self, event.source.nick, event.target):
  116. targetadmin = True
  117. if len(command.split()) == 1: # No arguments.
  118. if event.target in self.channels: # It's a channel message.
  119. if not homeadmin and not targetadmin: # Insufficient rights:
  120. connection.privmsg(replyto, "Denied. You need to have at least operator status in " + red + self.homechan + reset + " or " + red + event.target + reset + ".")
  121. return
  122. if event.target == self.homechannel:
  123. connection.action("shall not abandon it's home channel!")
  124. return
  125. connection.part(event.target, event.source.nick)
  126. else: # It's a PM.
  127. connection.privmsg(replyto, "Specify a channel to part. For help type " + blue + self.helpchar + "part" + reset + ".")
  128. elif len(command.split()) > 1: # Arguments
  129. if command.split()[1] not in self.channels: # First argument is not a channel the bot inhabits.
  130. connection.action(replyto, "does not inhabit " + red + command.split()[1] + reset + ". For help type " + blue + self.helpchar + "part" + reset + ".")
  131. return
  132. if not homeadmin and not userstatus.atleast_oper(self, event.source.nick, command.split()[1]): # Insufficient rights.
  133. connection.privmsg(replyto, "Denied. You need to have at least operator status in " + red + self.homechan + reset + " or " + red + command.split()[1] + reset + ".")
  134. return
  135. if command.split()[1] == self.homechannel:
  136. connection.action("shall not abandon it's home channel!")
  137. return
  138. try:
  139. connection.part(command.split()[1], command.split(maxsplit=2)[2])
  140. except:
  141. connection.part(command.split()[1], event.source.nick)
  142. elif command.split()[0] == "msg" or command.split(maxsplit=1)[0] == "act":
  143. if cmdtype == "help": #Display help text.
  144. if len(command.split()) is not 1:
  145. return
  146. if command.split(maxsplit=1)[0] == "act":
  147. message = "Let " + connection.get_nickname() + " send an action to a channel."
  148. arguments = "action-message"
  149. else:
  150. message = "Let " + connection.get_nickname() + "send a message to a channel."
  151. arguments = "name message"
  152. connection.privmsg(replyto, message)
  153. connection.privmsg(replyto, grey + "Usage: " + blue + "!" + command.split(maxsplit=1)[0] + reset + italic + " target " + arguments)
  154. elif cmdtype == "cmd":
  155. # Parse user input.
  156. try:
  157. destination = trigger.split()[1]
  158. except IndexError: # User did not specify a destination.
  159. connection.privmsg(replyto, "Destination not specified. For help type: " + blue + self.helpchar + command.split(maxsplit=1)[0])
  160. return
  161. try:
  162. message = trigger.split(maxsplit=2)[2]
  163. except IndexError: # User did not specify a message.
  164. connection.privmsg(replyto, "Message not specified. For help type: " + blue + self.helpchar + command.split(maxsplit=1)[0])
  165. return
  166. # Send the message if user has owner status in the home channel.
  167. if self.channels[self.homechannel].is_owner(event.source.nick):
  168. if destination == connection.get_nickname():
  169. connection.privmsg(replyto, "To prevent dying in a loop I shall not message myself.")
  170. return
  171. if command.split(maxsplit=1)[0] == "act":
  172. connection.action(destination, message)
  173. else:
  174. connection.privmsg(destination, message)
  175. # Reply error when bot does not inhabit destination channel.
  176. elif destination not in self.channels:
  177. connection.action(replyto, "does not inhabit " + red + destination + reset + ".")
  178. # Send message if user has at least operator status the home channel or the channel the message is intended for.
  179. elif userstatus.atleast_oper(self, event.source.nick, self.homechannel) or userstatus.atleast_oper(self, event.source.nick, destination):
  180. if command.split(maxsplit=1)[0] == "act":
  181. connection.action(destination, message)
  182. else:
  183. connection.privmsg(destination, message)
  184. # Reply error if user is not operator of destination channel.
  185. else:
  186. connection.privmsg(replyto, "Denied, you need to be an operator of " + red + destination + reset +".")
  187. elif command.split()[0] == "channelfunctions":
  188. if cmdtype == "help": #Display help text. # Help code block first, as it is impossible to predict for what channel a later command is going to be issued. Rights filtering after help test.
  189. if len(command.split()) is not 1:
  190. return
  191. connection.privmsg(replyto, "Display or toggle the status channel functions. Channel, function and value optional. Get a description of a functio via the help argument.")
  192. connection.privmsg(replyto, grey + "Usage: " + blue + "!channelfunctions " + red + italic + "channel " + reset + italic + "function value")
  193. connection.privmsg(replyto, grey + "Usage: " + blue + "!channelfunctions describe " + reset + italic + "function")
  194. elif cmdtype == "cmd":
  195. if len(command.split()) == 1: # No arguments.
  196. if event.target == connection.get_nickname(): # Command issued via PM.
  197. connection.privmsg(replyto, "Nothing to display, Specify a channel.")
  198. else: # Command issued as channel message.
  199. message = AH.get_channelfunctions(self, event.target)
  200. connection.privmsg(replyto, message)
  201. elif len(command.split()) == 2: # One argument.
  202. if command.split()[1] in self.channels: # Info requested on specific channel.
  203. message = AH.get_channelfunctions(self, command.split()[1])
  204. connection.privmsg(replyto, message)
  205. else: # First argument is not a channel the bot inhabits.
  206. if not command.split()[1].lower() == "help": # Not a help request.
  207. connection.privmsg(replyto, command.split()[1] + " is not a channel I inhabit. For help type " + blue + self.helpchar + "channelfunctions" + reset + ".")
  208. else: # Help request.
  209. connection.privmsg(replyto, "Specify a channel function to get a description of. For help type " + blue + self.helpchar + "channelfunctions" + reset + ".")
  210. elif len(command.split()) == 3: # Two arguments.
  211. channel = event.target
  212. if event.target == connection.get_nickname(): # Command issued via PM.
  213. connection.privmsg(replyto, "One or three arguments required. For help type " + blue + self.helpchar + "channelfunctions" + reset + ".")
  214. else: # Command issued via channel.
  215. if not AH.is_channelfunction(command.split()[1]): # First argument is not a channelfunction.
  216. if not command.split()[1].lower() == "describe": # Not a help request.
  217. connection.privmsg(replyto, command.split()[1] + " is not a channel function. For help type " + blue + self.helpchar + "channelfunctions" + reset + ".")
  218. return
  219. if not AH.is_channelfunction(command.split()[2]): # Second argument not a channel function.
  220. connection.privmsg(replyto, command.split()[2] + " is not a channel function.")
  221. return
  222. connection.privmsg(replyto, AH.describe_channelfunction(command.split()[2]))
  223. return
  224. # Second argument unsupported.
  225. if not command.split()[2].lower() in ["on", "off"]:
  226. if command.split()[1].lower() == "aggressiveness":
  227. if not AH.is_aggressiveness(command.split()[2].lower()): # Is not an aggressiveness setting.
  228. connection.privmsg(replyto, command.split()[2] + " is not an aggressiveness setting. For help type " + blue + self.helpchar + "channelfunctions" + reset + ".")
  229. return
  230. else: # Channel function is not aggresiveness.
  231. connection.privmsg(replyto, "The value of this channel function can only be \"on\" or \"off\". For help type " + blue + self.helpchar + "channelfunctions" + reset + ".")
  232. return
  233. if not userstatus.atleast_oper(self, event.source.nick, self.homechannel) and not userstatus.atleast_oper(self, event.source.nick, event.target): # Does not have operator status or higher in target or home channel.
  234. connection.privmsg(replyto, "Denied. You need to have at least operator status in " + red + event.target + reset + " or " + red + self.homechannel + reset + ".")
  235. return
  236. if command.split()[1].lower() == "autojoin" and event.target == self.homechannel: # Chaning autojoin of homechannel.
  237. connection.action(replyto, "will always join it's homechannel " + red + self.homechannel + reset + ", regardless of the autojoin function.")
  238. #self.db.run("UPDATE channels SET " + command.split()[1].lower() + "='" + command.split()[2].lower() + "' WHERE name='" + event.target + "' AND network='" + self.network + "'")
  239. self.db.run("UPDATE channels SET " + command.split()[1].lower() + "=%s WHERE name='" + event.target + "' AND network='" + self.network + "'", (command.split()[2].lower(), ))
  240. elif len(command.split()) == 4: # Three arguments.
  241. if not command.split()[1] in self.channels: # Bot does not inhabit channel to be altered.
  242. connection.privmsg(replyto, command.split()[1] + " is not a channel I inhabit. For help type " + blue + self.helpchar + "channelfunctions" + reset + ".")
  243. return
  244. if not AH.is_channelfunction(command.split()[2]): # Function does not exist.
  245. connection.privmsg(replyto, command.split()[2] + " is not a valid channel function. For a list help type: " + blue + self.cmdchar + "channelfunctions" + red + italic + "channel")
  246. return
  247. if not command.split()[3].lower() in ["on", "off"] and not command.split()[2].lower() == "aggressiveness": # Third argument unsupported.
  248. connection.privmsg(replyto, "The value of this channel function can only be \"on\" or \"off\". For help type " + blue + self.helpchar + "channelfunctions" + reset + ".")
  249. return
  250. if not userstatus.atleast_oper(self, event.source.nick, self.homechannel) and not userstatus.atleast_oper(self, event.source.nick, command.split()[1]): # Does not have operator status or higher in target or home channel.
  251. connection.privmsg(replyto, "Denied. You need to have at least operator status in " + red + event.target + reset + " or " + red + self.homechannel + reset + ".")
  252. return
  253. if command.split()[2].lower() == "autojoin" and command.split()[1] == self.homechannel: # Chaning autojoin of homechannel.
  254. connection.action(replyto, "will always join it's homechannel " + red + self.homechannel + reset + ", regardless of the autojoin function.")
  255. try:
  256. self.db.run("UPDATE channels SET " + command.split()[2].lower() + "='" + command.split()[3].lower() + "' WHERE LOWER(name)=LOWER('" + command.split()[1] + "') AND network='" + self.network + "'")
  257. except:
  258. connection.privmsg(replyto, "Error, database record not updated.")
  259. return
  260. else: # Too many arguments.
  261. connection.privmsg(replyto, "Too many arguments. For help type " + blue + self.helpchar + "channelfunctions" + reset + ".")