Maki is a Discord bot that does things. Written in Python 3 and relies on Discord.py API implementation.
Nelze vybrat více než 25 témat Téma musí začínat písmenem nebo číslem, může obsahovat pomlčky („-“) a může být dlouhé až 35 znaků.

306 řádky
7.6KB

  1. # Maki
  2. # ----
  3. # Discord bot by MrDetonia
  4. #
  5. # Copyright 2017 Zac Herd
  6. # Licensed under BSD 3-clause License, see LICENSE.md for more info
  7. # IMPORTS
  8. import asyncio
  9. import os
  10. import sys
  11. import re
  12. import requests
  13. import random
  14. import subprocess
  15. # LOCAL IMPORTS
  16. from common import *
  17. from helpers import *
  18. from secret import lfmkey, steamkey
  19. import markov
  20. # COMMAND IMPLEMENTATIONS
  21. @asyncio.coroutine
  22. def cmd_help(client, msg):
  23. yield from discord_send(client, msg, helptext)
  24. @asyncio.coroutine
  25. def cmd_info(client, msg):
  26. pyver = "{}.{}.{}".format(sys.version_info[0], sys.version_info[1], sys.version_info[2])
  27. appinfo = yield from client.application_info()
  28. response = "I am **{}**, a Discord bot by **{}** | `{}` | Python `{}` | discord.py `{}`".format(appinfo.name, appinfo.owner.name, version, pyver, discord.__version__)
  29. yield from discord_send(client, msg, response)
  30. @asyncio.coroutine
  31. def cmd_upskirt(client, msg):
  32. response = "No, don\'t look at my pantsu, baka! <https://gitla.in/MrDetonia/maki>"
  33. yield from discord_send(client, msg, response)
  34. whoistring = "**{}#{}**: `{}`\n**Account Created:** `{}`"
  35. @asyncio.coroutine
  36. def cmd_whoami(client, msg):
  37. response = whoistring.format(msg.author.name, msg.author.discriminator, msg.author.id, strfromdt(msg.author.created_at))
  38. yield from discord_send(client, msg, response)
  39. @asyncio.coroutine
  40. def cmd_whois(client, msg):
  41. tmp = msg.content[7:]
  42. user = msg.server.get_member_named(tmp)
  43. if user == None:
  44. reponse = "I can't find `{}`".format(tmp)
  45. else:
  46. response = whoistring.format(user.name, user.discriminator, user.id, strfromdt(user.created_at))
  47. yield from discord_send(client, msg, response)
  48. @asyncio.coroutine
  49. def cmd_seen(client, msg):
  50. tmp = msg.content[6:]
  51. user = msg.server.get_member_named(tmp)
  52. if user == None:
  53. reponse = "I can't find `{}`".format(tmp)
  54. elif user.name == "Maki":
  55. reponse = "I'm right here!"
  56. else:
  57. target = msg.server.id + user.id
  58. if target in history and history[target][0] == msg.server.id:
  59. response = "**{}** was last seen saying the following at {}:\n{}".format(user.name, strfromdt(dtfromts(history[target][1])), history[target][2])
  60. else:
  61. response = "I haven't seen **{}** speak yet!".format(tmp)
  62. yield from discord_send(client, msg, response)
  63. @asyncio.coroutine
  64. def cmd_say(client, msg):
  65. print("IN CMD_SAY")
  66. response = msg.content[5:]
  67. yield from client.delete_message(msg)
  68. print("CALLING SEND")
  69. yield from discord_send(client, msg, response)
  70. @asyncio.coroutine
  71. def cmd_sayy(client, msg):
  72. response = " ".join(msg.content[6:])
  73. yield from client.delete_message(msg)
  74. yield from discord_send(client, msg, response)
  75. @asyncio.coroutine
  76. def cmd_markov(client, msg):
  77. discord_typing(client, msg)
  78. tmp = msg.content[8:]
  79. target = ""
  80. if tmp == "Maki":
  81. response = "My markovs always say the same thing"
  82. else:
  83. if tmp == "":
  84. target = "{}-{}".format(msg.server.id, msg.author.id)
  85. else:
  86. try:
  87. target = "{}-{}".format(msg.server.id, msg.server.get_member_named(tmp).id)
  88. except AttributeError:
  89. reponse = "I can't find `{}`".format(tmp)
  90. if target != "":
  91. mfile = "./markovs/" + target
  92. if os.path.isfile(mfile):
  93. mc = markov.Markov(open(mfile))
  94. response = mc.generate_text(random.randint(20,40))
  95. else:
  96. response = "I haven't seen `{}` speak yet.".format(tmp)
  97. yield from discord_send(client, msg, response);
  98. @asyncio.coroutine
  99. def cmd_roll(client, msg):
  100. tmp = msg.content[6:]
  101. pattern = re.compile("^([0-9]+)d([0-9]+)$")
  102. if pattern.match(tmp):
  103. # extract numbers
  104. nums = [int(s) for s in re.findall(r"\d+", tmp)]
  105. # limit ranges
  106. nums[0] = clamp(nums[0], 1,100)
  107. nums[1] = clamp(nums[1], 1, 1000000)
  108. # roll and sum dice
  109. rollsum = 0
  110. for i in range(nums[0]):
  111. rollsum += random.randint(1, nums[1])
  112. response = "Using `{}d{}` you rolled: `{}`".format(nums[0], nums[1], rollsum)
  113. else:
  114. response = "Expected format: `<num>d<value>`"
  115. yield from discord_send(client, msg, response)
  116. @asyncio.coroutine
  117. def cmd_qr(client, msg):
  118. tmp = msg.content[4:]
  119. discord_typing(client, msg)
  120. # generate qr code
  121. qr = subprocess.Popen("qrencode -t png -o -".split(), stdin=subprocess.PIPE, stdout=subprocess.PIPE)
  122. qr.stdin.write(tmp.encode("utf-8"))
  123. qr.stdin.close()
  124. out = subprocess.check_output("curl -F upload=@- https://w1r3.net".split(), stdin=qr.stdout)
  125. response = out.decode("utf-8").strip()
  126. yield from discord_send(client, msg, response)
  127. @asyncio.coroutine
  128. def cmd_np(client, msg):
  129. tmp = msg.content[4:]
  130. if tmp == "":
  131. response = lastfm_np(msg.author.name)
  132. else:
  133. response = lastfm_np(tmp)
  134. print("CALLING SEND")
  135. yield from discord_send(client, msg, response)
  136. @asyncio.coroutine
  137. def cmd_steam(client, msg):
  138. tmp = msg.content[7:]
  139. if tmp == "":
  140. response = steamdata(msg.author.name)
  141. else:
  142. response = steamdata(tmp)
  143. yield from discord_send(client, msg, response)
  144. # HELPER FUNCTIONS
  145. # gets now playing information from last.fm
  146. def lastfm_np(username):
  147. # sanitise username
  148. cleanusername = re.sub(r'[^a-zA-Z0-9_-]', '', username, 0)
  149. # fetch JSON from last.fm
  150. payload = {'format': 'json', 'method': 'user.getRecentTracks', 'user': cleanusername, 'limit': '1', 'api_key': lfmkey}
  151. r = requests.get("http://ws.audioscrobbler.com/2.0/", params=payload)
  152. # read json data
  153. np = r.json()
  154. # check we got a valid response
  155. if 'error' in np:
  156. return "I couldn't get last.fm data for `{}`".format(username)
  157. # get fields
  158. try:
  159. username = np['recenttracks']['@attr']['user']
  160. track = np['recenttracks']['track'][0]
  161. album = track['album']['#text']
  162. artist = track['artist']['#text']
  163. song = track['name']
  164. nowplaying = '@attr' in track
  165. except IndexError:
  166. return "It looks like `{}` hasn't played anything recently.".format(username)
  167. # grammar
  168. if album != "":
  169. albumtext = "` from the album `{}`".format(album)
  170. else:
  171. albumtext = "`"
  172. if nowplaying == True:
  173. nowplaying = " is listening"
  174. else:
  175. nowplaying = " last listened"
  176. # construct string
  177. return "{}{} to `{}` by `{}{}".format(username, nowplaying, song, artist, albumtext)
  178. # gets general steam user info from a vanityurl name
  179. def steamdata(vanityname):
  180. # sanitise username
  181. cleanvanityname = re.sub(r'[^a-zA-Z0-9_-]', '', vanityname, 0)
  182. resolveurl = 'http://api.steampowered.com/ISteamUser/ResolveVanityURL/v0001/?key='
  183. dataurl = 'http://api.steampowered.com/ISteamUser/GetPlayerSummaries/v0002/?key='
  184. # fetch json from steam
  185. try:
  186. idresponse = requests.get(resolveurl + steamkey + '&vanityurl=' + vanityname).json()['response']
  187. except:
  188. return "I can't connect to Steam"
  189. # check if user was found and extract steamid
  190. if idresponse['success'] is not 1:
  191. return "I couldn't find `{}`".format(vanityname)
  192. else:
  193. steamid = idresponse['steamid']
  194. # fetch steam user info
  195. try:
  196. dataresponse = requests.get(dataurl + steamkey + '&steamids=' + steamid).json()['response']['players'][0]
  197. except:
  198. return "Can't find info on `{}`".format(vanityname)
  199. personastates = ['Offline', 'Online', 'Busy', 'Away', 'Snoozed', 'Looking to trade', 'Looking to play']
  200. if 'personaname' in dataresponse: namestr = dataresponse['personaname']
  201. else: namestr = ''
  202. if 'personastate' in dataresponse: statestr = '`' + personastates[dataresponse['personastate']] + '`'
  203. else: statestr = ''
  204. if 'gameextrainfo' in dataresponse: gamestr = ' playing `' + dataresponse['gameextrainfo'] + '`'
  205. else: gamestr = ''
  206. responsetext = [(namestr + ' is ' + statestr + gamestr).replace(' ', ' ')]
  207. return '\n'.join(responsetext)
  208. # COMMAND HANDLING
  209. prefix = "."
  210. commands = {
  211. "help": cmd_help,
  212. "info": cmd_info,
  213. "upskirt": cmd_upskirt,
  214. "whoami": cmd_whoami,
  215. "whois": cmd_whois,
  216. "seen": cmd_seen,
  217. "say": cmd_say,
  218. "sayy": cmd_sayy,
  219. "markov": cmd_markov,
  220. "roll": cmd_roll,
  221. "qr": cmd_qr,
  222. "np": cmd_np,
  223. "steam": cmd_steam,
  224. }