""" Bot commands. """ import logging import os import json from random import choice from discord import Embed, Intents from discord.ext import commands from settings.config import IMAGE_TYPES, OW_API_CONFIG, PERMISSIONS, QUOTE_STACK from utils.database import (count_quotes, count_quotes_user, get_by_id, get_quote_contains, get_quotes, remove_quote, set_quote) from utils.machine_monitor import Monitor from utils.news_paper import News from utils.tools import datetime_to_string, kbytes_to_gbytes from utils.weather import displayweather, getweatherdata client = commands.Bot(command_prefix='--', intents=Intents.all()) logger = logging.getLogger(__name__) with open(QUOTE_STACK, mode='r') as f: quote_id_stack = json.load(f) @client.command(aliases=['q']) async def quote(bot: object, *quote: str) -> str: """ Saves a quote into the database. """ if not quote: return await bot.send('You\'re not my mute uncle, tell me something to remember.\n'\ '(You haven\'t provided a quote)') quote = ' '.join(quote) if 'http' in quote and 'discord' in quote and not quote[-4:] in IMAGE_TYPES: return await bot.send("- _Check your link, dumbass! You're trying to quote an image from a"\ " 'message, but you're quoting the message itself!_\n"\ "'(Make sure to copy the link for the image by clicking on it, right-clicking the "\ "image and then clicking on \"Save Link\")'") try: user = bot.author.name qtid = set_quote(user, quote) except Exception as ex: if ex.args[0].find("Duplicate") != -1: return await bot.send("There's already a quote from that same person, with that "\ "exact match!") return await bot.send(f'{ex.args}\n_What the fuck are you doing?_') else: return await bot.send(f"Done: `{quote}\n` ID: `{qtid}`") @client.command(aliases=['rq']) async def random_quote(bot: object) -> str: """ Get a random quote from the database. """ quotes = get_quotes(quote_id_stack) stack_limit = int((len(quotes) * .25)) stack_len = len(quote_id_stack) if not quotes and stack_len > 0: quote_id_stack.pop(0) quotes = get_quotes(quote_id_stack) elif not quotes: return await bot.send('You\'ve got no quotes saved yet.\n(Save quotes by using '\ '`--q = stack_limit: quote_id_stack.pop(0) # Writes to persistent stackfile with open(QUOTE_STACK, mode='w') as f: json.dump(quote_id_stack, f) try: # To image links. if 'http' in chosen_one.quote: return await bot.send(f'{chosen_one.quote}') return await bot.send(f'{chosen_one.quote}\n`By: {chosen_one.user}`') except Exception as ex: return await bot.send(ex) @client.command(aliases=['qid']) async def by_id(bot: object, _id: int=None) -> str: """ Gets one quote by ID. """ syntax = "`--qid `" if not _id: return await bot.send("_If you don't tell me the ID, how the fuck do you expect me to "\ f"quote it to you!?_\n(The correct syntax is {syntax})") quote = get_by_id(_id) if not quote: return await bot.send(f"_Wrong ID, sucker!_\n(There's no such quote with id {_id})") try: # To image links. if 'http' in quote.quote: return await bot.send(f'{quote.quote}') return await bot.send(f'{quote.quote}\n`By: {quote.user}`') except Exception as ex: return await bot.send(ex) @client.command(aliases=['dq']) async def delete_quote(bot, _id: int=None) -> str: """ Deletes one quote by ID. """ syntax = "`--dq `" roles = [r.name for r in bot.author.roles] PermStatus = False if len(PERMISSIONS['dq']) < 1 or not len(set(PERMISSIONS['dq']).intersection(roles)) < 1: PermStatus = True if not PermStatus: return await bot.send("_And who the fuck do **YOU** think you are!?_.\n"\ "(You don't have the necessary role for this command)") if not _id: return await bot.send("_If you don't tell me the ID, how the fuck do you expect me to "\ f"delete it to you!?_\n(The correct syntax is {syntax})") quote = get_by_id(_id) if not quote: return await bot.send(f"_Wrong ID, sucker!_\n(There's no such quote with id {_id})") try: if not remove_quote(_id): return await bot.send('_Something wrong happened, dude!_') return await bot.send('_Evidence deleted, fella!_') except Exception as ex: return await bot.send(ex) @client.command(aliases=['qstack']) async def queue_stack(bot: object) -> str: """ Displays the 5 quote history stack. """ return await bot.send('A list of the 5 latest message IDs follows:'\ f' `{",".join(str(q) for q in quote_id_stack[-5:])}`') @client.command(aliases=['qc', 'cquotes']) async def quote_count(bot: object) -> str: """ Outputs a quote count from the database. """ amount = count_quotes() msg = f"Quote count: `{amount}`" return await bot.send(msg) @client.command(aliases=['v', 'version']) async def info(bot: object) -> str: """ Displays the bot's information. """ roles = [r.name for r in bot.author.roles] PermStatus = False if len(PERMISSIONS['v']) < 1 or not len(set(PERMISSIONS['v']).intersection(roles)) < 1: PermStatus = True if not PermStatus: return await bot.send("_And who the fuck do **YOU** think you are!?_.\n"\ "(You don't have the necessary role for this command)") motd = open("./motd", mode='r') text = motd.readlines() fullbanner = "" for lines in text: fullbanner = fullbanner + lines msg = f'''```\n{fullbanner}\n```''' return await bot.send(msg) @client.command(aliases=['w']) async def weather(bot: object, *location: str) -> str: """ Displays the weather information for a given place. """ if OW_API_CONFIG['api_id'] == 'no': return await bot.send("You haven't set up an API key! Make an user and set up an API key in https://openweathermap.org/\n \ (The weather command hansn't been set up properly, make sure you have `OPENWEATHER_API_TOKEN` set up") if location: location = ' '.join(location) location = location.encode('utf-8').decode('utf-8') location = location.replace(" ", "+") else: location = "curitiba,paraná".encode('utf-8').decode('utf-8') weatherdata = getweatherdata(location) msg = displayweather(weatherdata) default_msg = 'No data!' embed = Embed(type='rich') embed.add_field( name='City', value=msg.name, ) embed.add_field( name='Description', value=msg.description+f' {msg.icon}' if msg.description and msg.icon else default_msg, ) embed.add_field( name='Temperature', value=f'{msg.temp} ºC' if msg.temp else default_msg, ) embed.add_field( name='Feels like', value=f'{msg.feels_like} ºC' if msg.feels_like else default_msg, ) embed.add_field( name='Humidity', value=f'{msg.humidity} %' if msg.humidity else default_msg, ) embed.add_field( name='Cloud coverage', value=f'{msg.cloud_coverage} %' if msg.cloud_coverage else default_msg, ) embed.add_field( name='Wind gusts', value=f'{msg.wind_gusts} m/s' if msg.wind_gusts else default_msg, ) embed.add_field( name='Wind speed', value=f'{msg.wind_speed} m/s' if msg.wind_speed else default_msg, ) return await bot.send('**`Weather`**', embed=embed) @client.command(aliases=['qcontains', 'qsearch']) async def quote_contains(bot: object, part: str) -> str: """ Filter quote by part of saved message. """ syntax = '--qcontains ' if not part: return await bot.send("_If you don't tell me the part, how the fuck do you expect me to "\ f"find it to you!?_\n(The correct syntax is {syntax})") quotes = get_quote_contains(part) if not quotes: return await bot.send(f"_Wrong text, sucker!_\n(There's no such quote with text `{part}`)") for quote in quotes: await bot.send(f'```\nID: {quote.id}\nMessage: {quote.quote[:10]} ... '\ f'{quote.quote[-10:]}\nUser: {quote.user}\n```') return @client.command(aliases=['macinfo', 'minfo']) async def machine_info(bot: object, *args: str) -> str: """ Return machine information. """ embed = Embed(type='rich') supported_args = [ 'network' ] roles = [r.name for r in bot.author.roles] if 'BotMan' not in roles: return await bot.send("_And who the fuck do **YOU** think you are!?_.\n"\ "(You don't have the necessary role for this command)") if not args: embed.add_field(name='CPU', value=f'{Monitor.cpu_percent} %') embed.add_field(name='RAM', value=f'{Monitor.memory.percent} %') embed.add_field(name='Swap', value=f'{Monitor.swap.percent} %') embed.add_field(name='Disk total', value=f'{kbytes_to_gbytes(Monitor.disk_usage.total)} Gb') embed.add_field(name='Disk used', value=f'{kbytes_to_gbytes(Monitor.disk_usage.used)} Gb') embed.add_field(name='Disk free', value=f'{kbytes_to_gbytes(Monitor.disk_usage.free)} Gb') return await bot.send('**`Monitor`**', embed=embed) if args[0] not in supported_args: return await bot.send('The argument is not supported!') if args[0] == 'network': ios = Monitor.net_io_counters for io in ios: embed.clear_fields() embed.add_field(name='Bytes received', value=ios[io].bytes_recv, inline=True) embed.add_field(name='Bytes sent', value=ios[io].bytes_sent, inline=True) embed.add_field(name='Packets received', value=ios[io].packets_recv, inline=True) embed.add_field(name='Packets sent', value=ios[io].packets_sent, inline=True) embed.add_field(name='Drop in', value=ios[io].dropin, inline=True) embed.add_field(name='Drop out', value=ios[io].dropout, inline=True) embed.add_field(name='Error in', value=ios[io].errin, inline=True) embed.add_field(name='Error out', value=ios[io].errout, inline=True) await bot.send(f'**`{io}`**', embed=embed) return @client.command(aliases=['nw']) async def news(bot: object, *options: str) -> None: """ Return some news from Google. options: quantity: int search: str """ embed = Embed(type='rich') filter = {} news = None if not options: _news = News(quantity=5) news = _news.news() else: # Validate option operation. if not all(['=' in op for op in options]): return await bot.send('Blabla') for op in options: key, value = op.split('=') filter[key] = value _news = News(quantity=filter.get('quantity', 5)) news = _news.filter(phrase=filter.get('search')) if not news: return for new in news: dt = datetime_to_string(new['publishedAt']) embed.add_field(name='Font', value=new['source']['name'], inline=False) embed.add_field(name='Published at', value=dt, inline=False) embed.add_field(name='Link', value=new['url'], inline=False) embed.add_field(name=new['title'], value=new['description'], inline=False) embed.add_field(name='---', value='---') return await bot.send(f'**`News`**', embed=embed) @client.command(aliases=['nf', 'nfetch']) async def neofetch(bot: object) -> str: """ Runs neofetch on the host. """ os.system('neofetch --stdout --disable gpu --disable shell --disable packages --disable resolution --cpu_temp C > /tmp/neofetch') nfetch = open('/tmp/neofetch', mode='r') nfetch = nfetch.read() return await bot.send("```\n" + nfetch + "\n```") @client.command(aliases=['qlb']) async def count_leaderboard(bot:object): """ Returns a list of quoters, sorted by amount of quotes. """ qlb = "```\nLista de quoteiros\n" lis = count_quotes_user() lis = sorted(lis, key=lambda lis: lis[1], reverse=True) for data in lis: qlb = qlb + data[0] + " - " + str(data[1]) + "\n" qlb = qlb + "```" return await bot.send(qlb)