2018-07-11 22:47:44 +00:00
from datetime import datetime
import re
from urllib . parse import urlparse
import discord
2018-07-12 00:14:32 +00:00
import humanize
2018-07-11 22:47:44 +00:00
from pluralkit import db
from pluralkit . bot import client , logger
2018-07-12 13:03:34 +00:00
from pluralkit . utils import command , generate_hid , generate_member_info_card , generate_system_info_card , member_command , parse_mention , text_input , get_system_fuzzy , get_member_fuzzy , command_map , make_default_embed , parse_channel_mention
2018-07-11 22:49:02 +00:00
2018-07-13 21:03:35 +00:00
@command ( cmd = " system new " , usage = " [name] " , description = " Registers a new system to this account. " )
2018-07-11 22:47:44 +00:00
async def new_system ( conn , message , args ) :
system = await db . get_system_by_account ( conn , message . author . id )
if system is not None :
return False , " You already have a system registered. To remove your system, use `pk;system remove`, or to unlink your system from this account, use `pk;system unlink`. "
system_name = None
2018-07-12 01:21:25 +00:00
if len ( args ) > 0 :
system_name = " " . join ( args )
2018-07-11 22:47:44 +00:00
async with conn . transaction ( ) :
# TODO: figure out what to do if this errors out on collision on generate_hid
hid = generate_hid ( )
system = await db . create_system ( conn , system_name = system_name , system_hid = hid )
# Link account
await db . link_account ( conn , system_id = system [ " id " ] , account_id = message . author . id )
return True , " System registered! To begin adding members, use `pk;member new <name>`. "
2018-07-11 22:49:02 +00:00
2018-07-13 21:03:35 +00:00
@command ( cmd = " system " , usage = " [system] " , description = " Shows information about a system. " )
2018-07-11 22:47:44 +00:00
async def system_info ( conn , message , args ) :
if len ( args ) == 0 :
# Use sender's system
system = await db . get_system_by_account ( conn , message . author . id )
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
if system is None :
return False , " No system is registered to this account. "
else :
# Look one up
system = await get_system_fuzzy ( conn , args [ 0 ] )
if system is None :
return False , " Unable to find system \" {} \" . " . format ( args [ 0 ] )
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
await client . send_message ( message . channel , embed = await generate_system_info_card ( conn , system ) )
return True
2018-07-11 22:49:02 +00:00
2018-07-13 21:03:35 +00:00
@command ( cmd = " system name " , usage = " [name] " , description = " Renames your system. Leave blank to clear. " )
2018-07-11 22:47:44 +00:00
async def system_name ( conn , message , args ) :
system = await db . get_system_by_account ( conn , message . author . id )
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
if system is None :
return False , " No system is registered to this account. "
if len ( args ) == 0 :
new_name = None
else :
new_name = " " . join ( args )
async with conn . transaction ( ) :
await db . update_system_field ( conn , system_id = system [ " id " ] , field = " name " , value = new_name )
return True , " Name updated to {} . " . format ( new_name ) if new_name else " Name cleared. "
2018-07-11 22:49:02 +00:00
2018-07-13 21:03:35 +00:00
@command ( cmd = " system description " , usage = " [clear] " , description = " Updates your system description. Add \" clear \" to clear. " )
2018-07-11 22:47:44 +00:00
async def system_description ( conn , message , args ) :
system = await db . get_system_by_account ( conn , message . author . id )
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
if system is None :
return False , " No system is registered to this account. "
# If "clear" in args, clear
if len ( args ) > 0 and args [ 0 ] == " clear " :
new_description = None
else :
new_description = await text_input ( message , " your system " )
if not new_description :
return True , " Description update cancelled. "
async with conn . transaction ( ) :
await db . update_system_field ( conn , system_id = system [ " id " ] , field = " description " , value = new_description )
return True , " Description set. " if new_description else " Description cleared. "
2018-07-11 22:49:02 +00:00
2018-07-13 21:03:35 +00:00
@command ( cmd = " system tag " , usage = " [tag] " , description = " Updates your system tag. Leave blank to clear. " )
2018-07-11 22:47:44 +00:00
async def system_tag ( conn , message , args ) :
system = await db . get_system_by_account ( conn , message . author . id )
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
if system is None :
return False , " No system is registered to this account. "
if len ( args ) == 0 :
tag = None
else :
tag = " " . join ( args )
max_length = 32
# Make sure there are no members which would make the combined length exceed 32
members_exceeding = await db . get_members_exceeding ( conn , system_id = system [ " id " ] , length = max_length - len ( tag ) )
if len ( members_exceeding ) > 0 :
# If so, error out and warn
2018-07-11 22:49:02 +00:00
member_names = " , " . join ( [ member [ " name " ]
for member in members_exceeding ] )
logger . debug ( " Members exceeding combined length with tag ' {} ' : {} " . format (
tag , member_names ) )
2018-07-11 22:47:44 +00:00
return False , " The maximum length of a name plus the system tag is 32 characters. The following members would exceed the limit: {} . Please reduce the length of the tag, or rename the members. " . format ( member_names )
async with conn . transaction ( ) :
await db . update_system_field ( conn , system_id = system [ " id " ] , field = " tag " , value = tag )
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
return True , " Tag updated to {} . " . format ( tag ) if tag else " Tag cleared. "
2018-07-11 22:49:02 +00:00
2018-07-13 21:03:35 +00:00
@command ( cmd = " system delete " , description = " Deletes your system from the database ***permanently***. " )
async def system_delete ( conn , message , args ) :
2018-07-11 22:47:44 +00:00
system = await db . get_system_by_account ( conn , message . author . id )
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
if system is None :
return False , " No system is registered to this account. "
2018-07-13 21:03:35 +00:00
await client . send_message ( message . channel , " Are you sure you want to delete your system? If so, reply to this message with the system ' s ID (` {} `). " . format ( system [ " hid " ] ) )
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
msg = await client . wait_for_message ( author = message . author , channel = message . channel )
if msg . content == system [ " hid " ] :
await db . remove_system ( conn , system_id = system [ " id " ] )
2018-07-13 21:03:35 +00:00
return True , " System deleted. "
2018-07-11 22:47:44 +00:00
else :
2018-07-13 21:03:35 +00:00
return True , " System deletion cancelled. "
2018-07-11 22:47:44 +00:00
2018-07-13 21:03:35 +00:00
@command ( cmd = " system link " , usage = " <account> " , description = " Links another account to your system. " )
2018-07-11 22:47:44 +00:00
async def system_link ( conn , message , args ) :
system = await db . get_system_by_account ( conn , message . author . id )
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
if system is None :
return False , " No system is registered to this account. "
if len ( args ) == 0 :
return False
# Find account to link
linkee = await parse_mention ( args [ 0 ] )
if not linkee :
return False , " Account not found. "
# Make sure account doesn't already have a system
account_system = await db . get_system_by_account ( conn , linkee . id )
if account_system :
return False , " Account is already linked to a system (` {} `) " . format ( account_system [ " hid " ] )
# Send confirmation message
msg = await client . send_message ( message . channel , " {} , please confirm the link by clicking the ✅ reaction on this message. " . format ( linkee . mention ) )
await client . add_reaction ( msg , " ✅ " )
await client . add_reaction ( msg , " ❌ " )
reaction = await client . wait_for_reaction ( emoji = [ " ✅ " , " ❌ " ] , message = msg , user = linkee )
# If account to be linked confirms...
if reaction . reaction . emoji == " ✅ " :
async with conn . transaction ( ) :
# Execute the link
await db . link_account ( conn , system_id = system [ " id " ] , account_id = linkee . id )
return True , " Account linked to system. "
else :
await client . clear_reactions ( msg )
return False , " Account link cancelled. "
2018-07-13 21:03:35 +00:00
@command ( cmd = " system unlink " , description = " Unlinks your system from this account. There must be at least one other account linked. " )
2018-07-11 22:47:44 +00:00
async def system_unlink ( conn , message , args ) :
system = await db . get_system_by_account ( conn , message . author . id )
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
if system is None :
return False , " No system is registered to this account. "
# Make sure you can't unlink every account
linked_accounts = await db . get_linked_accounts ( conn , system_id = system [ " id " ] )
if len ( linked_accounts ) == 1 :
return False , " This is the only account on your system, so you can ' t unlink it. "
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
async with conn . transaction ( ) :
await db . unlink_account ( conn , system_id = system [ " id " ] , account_id = message . author . id )
return True , " Account unlinked. "
2018-07-13 21:03:35 +00:00
@command ( cmd = " system fronter " , usage = " [system] " , description = " Gets the current fronter in the system. " )
2018-07-12 00:14:32 +00:00
async def system_fronter ( conn , message , args ) :
if len ( args ) == 0 :
system = await db . get_system_by_account ( conn , message . author . id )
if system is None :
return False , " No system is registered to this account. "
else :
system = await get_system_fuzzy ( conn , args [ 0 ] )
if system is None :
return False , " Can ' t find system \" {} \" . " . format ( args [ 0 ] )
current_fronter = await db . current_fronter ( conn , system_id = system [ " id " ] )
if not current_fronter :
return True , make_default_embed ( None ) . add_field ( name = " Current fronter " , value = " *(nobody)* " )
fronter_name = " *(nobody)* "
if current_fronter [ " member " ] :
member = await db . get_member ( conn , member_id = current_fronter [ " member " ] )
fronter_name = member [ " name " ]
if current_fronter [ " member_del " ] :
fronter_name = " *(deleted member)* "
since = current_fronter [ " timestamp " ]
embed = make_default_embed ( None )
embed . add_field ( name = " Current fronter " , value = fronter_name )
embed . add_field ( name = " Since " , value = " {} ( {} ) " . format ( since . isoformat ( sep = " " , timespec = " seconds " ) , humanize . naturaltime ( since ) ) )
return True , embed
2018-07-13 21:03:35 +00:00
@command ( cmd = " system fronthistory " , usage = " [system] " , description = " Shows the past 10 switches in the system. " )
2018-07-12 00:14:32 +00:00
async def system_fronthistory ( conn , message , args ) :
if len ( args ) == 0 :
system = await db . get_system_by_account ( conn , message . author . id )
if system is None :
return False , " No system is registered to this account. "
else :
system = await get_system_fuzzy ( conn , args [ 0 ] )
if system is None :
return False , " Can ' t find system \" {} \" . " . format ( args [ 0 ] )
switches = await db . past_fronters ( conn , system_id = system [ " id " ] , amount = 10 )
lines = [ ]
for switch in switches :
since = switch [ " timestamp " ]
time_text = since . isoformat ( sep = " " , timespec = " seconds " )
rel_text = humanize . naturaltime ( since )
lines . append ( " ** {} ** ( {} , at {} ) " . format ( switch [ " name " ] , time_text , rel_text ) )
embed = make_default_embed ( " \n " . join ( lines ) )
embed . title = " Past switches "
return True , embed
2018-07-11 22:49:02 +00:00
2018-07-13 21:03:35 +00:00
@command ( cmd = " member new " , usage = " <name> " , description = " Adds a new member to your system. " )
2018-07-11 22:47:44 +00:00
async def new_member ( conn , message , args ) :
system = await db . get_system_by_account ( conn , message . author . id )
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
if system is None :
return False , " No system is registered to this account. "
if len ( args ) == 0 :
return False
name = " " . join ( args )
async with conn . transaction ( ) :
# TODO: figure out what to do if this errors out on collision on generate_hid
hid = generate_hid ( )
# Insert member row
await db . create_member ( conn , system_id = system [ " id " ] , member_name = name , member_hid = hid )
return True , " Member \" {} \" (` {} `) registered! " . format ( name , hid )
2018-07-11 22:49:02 +00:00
2018-07-13 21:03:35 +00:00
@member_command ( cmd = " member info " , description = " Shows information about a system member. " , system_only = False )
2018-07-11 22:47:44 +00:00
async def member_info ( conn , message , member , args ) :
await client . send_message ( message . channel , embed = await generate_member_info_card ( conn , member ) )
return True
2018-07-11 22:49:02 +00:00
2018-07-13 21:03:35 +00:00
@member_command ( cmd = " member color " , usage = " [color] " , description = " Updates a member ' s associated color. Leave blank to clear. " )
2018-07-11 22:47:44 +00:00
async def member_color ( conn , message , member , args ) :
if len ( args ) == 0 :
color = None
else :
match = re . fullmatch ( " #?([0-9a-f] {6} ) " , args [ 0 ] )
if not match :
return False , " Color must be a valid hex color (eg. #ff0000) "
color = match . group ( 1 )
async with conn . transaction ( ) :
await db . update_member_field ( conn , member_id = member [ " id " ] , field = " color " , value = color )
return True , " Color updated to # {} . " . format ( color ) if color else " Color cleared. "
2018-07-11 22:49:02 +00:00
2018-07-13 21:03:35 +00:00
@member_command ( cmd = " member pronouns " , usage = " [pronouns] " , description = " Updates a member ' s pronouns. Leave blank to clear. " )
2018-07-11 22:47:44 +00:00
async def member_pronouns ( conn , message , member , args ) :
if len ( args ) == 0 :
pronouns = None
else :
pronouns = " " . join ( args )
async with conn . transaction ( ) :
await db . update_member_field ( conn , member_id = member [ " id " ] , field = " pronouns " , value = pronouns )
return True , " Pronouns set to {} " . format ( pronouns ) if pronouns else " Pronouns cleared. "
2018-07-11 22:49:02 +00:00
2018-07-13 21:03:35 +00:00
@member_command ( cmd = " member birthdate " , usage = " [birthdate] " , description = " Updates a member ' s birthdate. Must be in ISO-8601 format (eg. 1999-07-25). Leave blank to clear. " )
2018-07-11 22:47:44 +00:00
async def member_birthday ( conn , message , member , args ) :
if len ( args ) == 0 :
new_date = None
else :
2018-07-11 22:49:02 +00:00
# Parse date
2018-07-11 22:47:44 +00:00
try :
new_date = datetime . strptime ( args [ 0 ] , " % Y- % m- %d " ) . date ( )
except ValueError :
return False , " Invalid date. Date must be in ISO-8601 format (eg. 1999-07-25). "
async with conn . transaction ( ) :
await db . update_member_field ( conn , member_id = member [ " id " ] , field = " birthday " , value = new_date )
return True , " Birthdate set to {} " . format ( new_date ) if new_date else " Birthdate cleared. "
2018-07-11 22:49:02 +00:00
2018-07-13 21:03:35 +00:00
@member_command ( cmd = " member description " , description = " Updates a member ' s description. Add \" clear \" to clear. " )
2018-07-11 22:47:44 +00:00
async def member_description ( conn , message , member , args ) :
if len ( args ) > 0 and args [ 0 ] == " clear " :
new_description = None
else :
new_description = await text_input ( message , member [ " name " ] )
if not new_description :
return True , " Description update cancelled. "
async with conn . transaction ( ) :
await db . update_member_field ( conn , member_id = member [ " id " ] , field = " description " , value = new_description )
return True , " Description set. " if new_description else " Description cleared. "
2018-07-11 22:49:02 +00:00
2018-07-13 21:03:35 +00:00
@member_command ( cmd = " member remove " , description = " Removes a member from your system. " )
2018-07-11 22:47:44 +00:00
async def member_remove ( conn , message , member , args ) :
await client . send_message ( message . channel , " Are you sure you want to remove {} ? If so, reply to this message with the member ' s name. " . format ( member [ " name " ] ) )
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
msg = await client . wait_for_message ( author = message . author , channel = message . channel )
if msg . content == member [ " name " ] :
await db . delete_member ( conn , member_id = member [ " id " ] )
return True , " Member removed. "
else :
return True , " Member removal cancelled. "
2018-07-11 22:49:02 +00:00
2018-07-13 21:03:35 +00:00
@member_command ( cmd = " member avatar " , usage = " [user|url] " , description = " Updates a member ' s avatar. Can be an account mention (which will use that account ' s avatar), or a link to an image. Leave blank to clear. " )
2018-07-11 22:47:44 +00:00
async def member_avatar ( conn , message , member , args ) :
if len ( args ) == 0 :
avatar_url = None
else :
user = await parse_mention ( args [ 0 ] )
if user :
# Set the avatar to the mentioned user's avatar
# Discord doesn't like webp, but also hosts png alternatives
avatar_url = user . avatar_url . replace ( " .webp " , " .png " )
else :
# Validate URL
u = urlparse ( args [ 0 ] )
if u . scheme in [ " http " , " https " ] and u . netloc and u . path :
avatar_url = args [ 0 ]
else :
return False , " Invalid URL. "
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
async with conn . transaction ( ) :
await db . update_member_field ( conn , member_id = member [ " id " ] , field = " avatar_url " , value = avatar_url )
2018-07-11 23:16:07 +00:00
# Add the avatar you just set into the success embed
if not avatar_url :
return True , " Avatar cleared. "
else :
return True , make_default_embed ( " Avatar set. " ) . set_image ( url = avatar_url )
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
2018-07-13 21:03:35 +00:00
@member_command ( cmd = " member proxy " , usage = " [example] " , description = " Updates a member ' s proxy settings. Needs an \" example \" proxied message containing the string \" text \" (eg. [text], |text|, etc). " )
2018-07-11 22:47:44 +00:00
async def member_proxy ( conn , message , member , args ) :
if len ( args ) == 0 :
prefix , suffix = None , None
else :
# Sanity checking
example = " " . join ( args )
if " text " not in example :
return False , " Example proxy message must contain the string ' text ' . "
if example . count ( " text " ) != 1 :
return False , " Example proxy message must contain the string ' text ' exactly once. "
# Extract prefix and suffix
prefix = example [ : example . index ( " text " ) ] . strip ( )
suffix = example [ example . index ( " text " ) + 4 : ] . strip ( )
2018-07-11 22:49:02 +00:00
logger . debug (
" Matched prefix ' {} ' and suffix ' {} ' " . format ( prefix , suffix ) )
2018-07-11 22:47:44 +00:00
# DB stores empty strings as None, make that work
if not prefix :
prefix = None
if not suffix :
suffix = None
async with conn . transaction ( ) :
await db . update_member_field ( conn , member_id = member [ " id " ] , field = " prefix " , value = prefix )
await db . update_member_field ( conn , member_id = member [ " id " ] , field = " suffix " , value = suffix )
return True , " Proxy settings updated. " if prefix or suffix else " Proxy settings cleared. "
2018-07-11 22:49:02 +00:00
2018-07-13 21:03:35 +00:00
@command ( cmd = " message " , usage = " <id> " , description = " Shows information about a proxied message. Requires the message ID. " )
2018-07-11 22:47:44 +00:00
async def message_info ( conn , message , args ) :
try :
mid = int ( args [ 0 ] )
except ValueError :
return False
# Find the message in the DB
message_row = await db . get_message ( conn , mid )
if not message_row :
return False , " Message not found. "
# Find the actual message object
channel = client . get_channel ( str ( message_row [ " channel " ] ) )
message = await client . get_message ( channel , str ( message_row [ " mid " ] ) )
# Get the original sender of the message
original_sender = await client . get_user_info ( str ( message_row [ " sender " ] ) )
# Get sender member and system
member = await db . get_member ( conn , message_row [ " member " ] )
system = await db . get_system ( conn , member [ " system " ] )
embed = discord . Embed ( )
embed . timestamp = message . timestamp
embed . colour = discord . Colour . blue ( )
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
if system [ " name " ] :
system_value = " ` {} `: {} " . format ( system [ " hid " ] , system [ " name " ] )
else :
system_value = " ` {} ` " . format ( system [ " hid " ] )
embed . add_field ( name = " System " , value = system_value )
2018-07-11 22:49:02 +00:00
embed . add_field ( name = " Member " , value = " ` {} `: {} " . format (
member [ " hid " ] , member [ " name " ] ) )
embed . add_field ( name = " Sent by " , value = " {} # {} " . format (
original_sender . name , original_sender . discriminator ) )
2018-07-11 22:47:44 +00:00
embed . add_field ( name = " Content " , value = message . clean_content , inline = False )
embed . set_author ( name = member [ " name " ] , url = member [ " avatar_url " ] )
await client . send_message ( message . channel , embed = embed )
return True
2018-07-13 21:03:35 +00:00
@command ( cmd = " switch " , usage = " <name|id> " , description = " Registers a switch and changes the current fronter. " )
2018-07-12 00:14:32 +00:00
async def switch_member ( conn , message , args ) :
if len ( args ) == 0 :
return False
system = await db . get_system_by_account ( conn , message . author . id )
if system is None :
return False , " No system is registered to this account. "
# Find the member
member = await get_member_fuzzy ( conn , system [ " id " ] , " " . join ( args ) )
if not member :
return False , " Couldn ' t find member \" {} \" . " . format ( args [ 0 ] )
# Get current fronter
current_fronter = await db . current_fronter ( conn , system_id = system [ " id " ] )
if current_fronter and current_fronter [ " member " ] == member [ " id " ] :
return False , " Member \" {} \" is already fronting. " . format ( member [ " name " ] )
# Log the switch
await db . add_switch ( conn , system_id = system [ " id " ] , member_id = member [ " id " ] )
return True , " Switch registered. Current fronter is now {} . " . format ( member [ " name " ] )
2018-07-13 21:03:35 +00:00
@command ( cmd = " switch out " , description = " Registers a switch out, and leaves current fronter blank. " )
2018-07-12 00:14:32 +00:00
async def switch_out ( conn , message , args ) :
system = await db . get_system_by_account ( conn , message . author . id )
if system is None :
return False , " No system is registered to this account. "
# Get current fronter
current_fronter = await db . current_fronter ( conn , system_id = system [ " id " ] )
if not current_fronter or not current_fronter [ " member " ] :
return False , " There ' s already no one in front. "
# Log it
await db . add_switch ( conn , system_id = system [ " id " ] , member_id = None )
return True , " Switch-out registered. "
2018-07-11 22:49:02 +00:00
2018-07-13 21:03:35 +00:00
@command ( cmd = " mod log " , usage = " [channel] " , description = " Sets the bot to log events to a specified channel. Leave blank to disable. " )
2018-07-12 13:03:34 +00:00
async def set_log ( conn , message , args ) :
if not message . author . server_permissions . administrator :
return False , " You must be a server administrator to use this command. "
server = message . server
if len ( args ) == 0 :
channel_id = None
else :
channel = parse_channel_mention ( args [ 0 ] , server = server )
if not channel :
return False , " Channel not found. "
channel_id = channel . id
await db . update_server ( conn , server . id , logging_channel_id = channel_id )
return True , " Updated logging channel. " if channel_id else " Cleared logging channel. "
2018-07-11 23:16:07 +00:00
def make_help ( cmds ) :
2018-07-11 22:47:44 +00:00
embed = discord . Embed ( )
embed . colour = discord . Colour . blue ( )
embed . title = " PluralKit Help "
2018-07-11 22:49:02 +00:00
embed . set_footer (
text = " <> denotes mandatory arguments, [] denotes optional arguments " )
2018-07-11 22:47:44 +00:00
for cmd , subcommands in cmds :
for subcmd , ( _ , usage , description ) in subcommands . items ( ) :
2018-07-11 22:49:02 +00:00
embed . add_field ( name = " {} {} {} " . format (
cmd , subcmd or " " , usage or " " ) , value = description , inline = False )
2018-07-11 23:16:07 +00:00
return embed
2018-07-11 22:49:02 +00:00
2018-07-13 21:03:35 +00:00
@command ( cmd = " help " , usage = " [category] " , description = " Shows this help message. " )
2018-07-11 23:16:07 +00:00
async def show_help ( conn , message , args ) :
2018-07-12 01:21:25 +00:00
embed = make_default_embed ( None )
embed . title = " PluralKit Help "
if len ( args ) == 0 :
basics = [ ]
for ( cmd , sub ) , ( _ , usage , description , basic ) in command_map . items ( ) :
if basic :
basics . append ( " ** {} {} {} ** - {} " . format ( cmd , sub or " " , usage or " " , description or " " ) )
embed . add_field ( name = " Basic commands " , value = " \n " . join ( basics ) )
categories = [ ( " system " , " System commands " ) , ( " member " , " Member commands " ) , ( " switch " , " Switching commands " ) ]
categories_lines = [ " **pk;help {} ** - {} " . format ( key , desc ) for key , desc in categories ]
embed . add_field ( name = " More commands " , value = " \n " . join ( categories_lines ) )
2018-07-11 23:16:07 +00:00
else :
2018-07-12 01:21:25 +00:00
if args [ 0 ] not in [ " system " , " member " , " switch " ] :
return False , " Unknown help category. "
2018-07-11 23:16:07 +00:00
2018-07-12 01:21:25 +00:00
cmds = [ ( k , v ) for k , v in command_map . items ( ) if k [ 0 ] == " pk; " + args [ 0 ] ]
lines = [ ]
for ( cmd , sub ) , ( _ , usage , description , _ ) in cmds :
lines . append ( " ** {} {} {} ** - {} " . format ( cmd , sub or " " , usage or " " , description or " " ) )
embed . add_field ( name = " Commands " , value = " \n " . join ( lines ) )
return True , embed