From 0d59a5220ff911990c831ed2a65c503f184c8932 Mon Sep 17 00:00:00 2001 From: bobloy Date: Fri, 5 Oct 2018 13:06:17 -0400 Subject: [PATCH 01/73] ctx.me instead of ctx.guild.me to work in dm's --- werewolf/builder.py | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/werewolf/builder.py b/werewolf/builder.py index da0bf1e..4364cb1 100644 --- a/werewolf/builder.py +++ b/werewolf/builder.py @@ -190,7 +190,7 @@ async def encode(roles, rand_roles): async def next_group(ctx: commands.Context, pages: list, controls: dict, message: discord.Message, page: int, timeout: float, emoji: str): - perms = message.channel.permissions_for(ctx.guild.me) + perms = message.channel.permissions_for(ctx.me) if perms.manage_messages: # Can manage messages, so remove react try: await message.remove_reaction(emoji, ctx.author) @@ -210,7 +210,7 @@ async def next_group(ctx: commands.Context, pages: list, async def prev_group(ctx: commands.Context, pages: list, controls: dict, message: discord.Message, page: int, timeout: float, emoji: str): - perms = message.channel.permissions_for(ctx.guild.me) + perms = message.channel.permissions_for(ctx.me) if perms.manage_messages: # Can manage messages, so remove react try: await message.remove_reaction(emoji, ctx.author) @@ -295,7 +295,7 @@ class GameBuilder: async def list_roles(self, ctx: commands.Context, pages: list, controls: dict, message: discord.Message, page: int, timeout: float, emoji: str): - perms = message.channel.permissions_for(ctx.guild.me) + perms = message.channel.permissions_for(ctx.me) if perms.manage_messages: # Can manage messages, so remove react try: await message.remove_reaction(emoji, ctx.author) @@ -310,7 +310,7 @@ class GameBuilder: async def select_page(self, ctx: commands.Context, pages: list, controls: dict, message: discord.Message, page: int, timeout: float, emoji: str): - perms = message.channel.permissions_for(ctx.guild.me) + perms = message.channel.permissions_for(ctx.me) if perms.manage_messages: # Can manage messages, so remove react try: await message.remove_reaction(emoji, ctx.author) From 0ee0199b11f852bbdc13e33cbacf66aafd184576 Mon Sep 17 00:00:00 2001 From: bobloy Date: Fri, 5 Oct 2018 13:50:55 -0400 Subject: [PATCH 02/73] role `__init__.py` experiment --- werewolf/builder.py | 4 +--- werewolf/roles/__init__.py | 5 +++++ 2 files changed, 6 insertions(+), 3 deletions(-) create mode 100644 werewolf/roles/__init__.py diff --git a/werewolf/builder.py b/werewolf/builder.py index 4364cb1..ce00b28 100644 --- a/werewolf/builder.py +++ b/werewolf/builder.py @@ -8,9 +8,7 @@ import discord # Import all roles here from redbot.core import commands -from werewolf.roles.seer import Seer -from werewolf.roles.vanillawerewolf import VanillaWerewolf -from werewolf.roles.villager import Villager +from werewolf.roles import Seer, VanillaWerewolf, Villager from redbot.core.utils.menus import menu, prev_page, next_page, close_menu # All roles in this list for iterating diff --git a/werewolf/roles/__init__.py b/werewolf/roles/__init__.py new file mode 100644 index 0000000..ba929e5 --- /dev/null +++ b/werewolf/roles/__init__.py @@ -0,0 +1,5 @@ +from .seer import Seer +from .shifter import Shifter +from .vanillawerewolf import VanillaWerewolf +from .villager import Villager + From 849262969c2c1936a45a0ec5c3cd73d9740dc9aa Mon Sep 17 00:00:00 2001 From: bobloy Date: Wed, 5 Dec 2018 14:59:10 -0500 Subject: [PATCH 03/73] forgot some await's --- werewolf/game.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/werewolf/game.py b/werewolf/game.py index 181c198..78bd6f6 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -210,7 +210,7 @@ class Game: print("Pre-cycle") await asyncio.sleep(1) - asyncio.ensure_future(self._cycle()) # Start the loop + await asyncio.ensure_future(self._cycle()) # Start the loop ############START Notify structure############ async def _cycle(self): @@ -276,7 +276,7 @@ class Game: # Need a loop here to wait for trial to end (can_vote?) while self.ongoing_vote: - asyncio.sleep(5) + await asyncio.sleep(5) if check(): return From 339492d6d923f421b187cc33d0d65837bc022b1f Mon Sep 17 00:00:00 2001 From: bobloy Date: Fri, 18 Sep 2020 14:10:04 -0400 Subject: [PATCH 04/73] Black formatting --- werewolf/builder.py | 153 ++++++++++++++++++-------- werewolf/game.py | 250 ++++++++++++++++++++++++++++-------------- werewolf/role.py | 18 +-- werewolf/votegroup.py | 2 +- 4 files changed, 281 insertions(+), 142 deletions(-) diff --git a/werewolf/builder.py b/werewolf/builder.py index b6a52df..1648b45 100644 --- a/werewolf/builder.py +++ b/werewolf/builder.py @@ -17,7 +17,7 @@ from redbot.core.utils.menus import menu, prev_page, next_page, close_menu ROLE_LIST = sorted([Villager, Seer, VanillaWerewolf], key=lambda x: x.alignment) -ALIGNMENT_COLORS = [0x008000, 0xff0000, 0xc0c0c0] +ALIGNMENT_COLORS = [0x008000, 0xFF0000, 0xC0C0C0] TOWN_ROLES = [(idx, role) for idx, role in enumerate(ROLE_LIST) if role.alignment == 1] WW_ROLES = [(idx, role) for idx, role in enumerate(ROLE_LIST) if role.alignment == 2] OTHER_ROLES = [(idx, role) for idx, role in enumerate(ROLE_LIST) if role.alignment not in [0, 1]] @@ -26,21 +26,38 @@ ROLE_PAGES = [] PAGE_GROUPS = [0] ROLE_CATEGORIES = { - 1: "Random", 2: "Investigative", 3: "Protective", 4: "Government", - 5: "Killing", 6: "Power (Special night action)", - 11: "Random", 12: "Deception", 15: "Killing", 16: "Support", - 21: "Benign", 22: "Evil", 23: "Killing"} + 1: "Random", + 2: "Investigative", + 3: "Protective", + 4: "Government", + 5: "Killing", + 6: "Power (Special night action)", + 11: "Random", + 12: "Deception", + 15: "Killing", + 16: "Support", + 21: "Benign", + 22: "Evil", + 23: "Killing", +} CATEGORY_COUNT = [] def role_embed(idx, role, color): - embed = discord.Embed(title="**{}** - {}".format(idx, str(role.__name__)), description=role.game_start_message, - color=color) - embed.add_field(name='Alignment', value=['Town', 'Werewolf', 'Neutral'][role.alignment - 1], inline=True) - embed.add_field(name='Multiples Allowed', value=str(not role.unique), inline=True) - embed.add_field(name='Role Type', value=", ".join(ROLE_CATEGORIES[x] for x in role.category), inline=True) - embed.add_field(name='Random Option', value=str(role.rand_choice), inline=True) + embed = discord.Embed( + title="**{}** - {}".format(idx, str(role.__name__)), + description=role.game_start_message, + color=color, + ) + embed.add_field( + name="Alignment", value=["Town", "Werewolf", "Neutral"][role.alignment - 1], inline=True + ) + embed.add_field(name="Multiples Allowed", value=str(not role.unique), inline=True) + embed.add_field( + name="Role Type", value=", ".join(ROLE_CATEGORIES[x] for x in role.category), inline=True + ) + embed.add_field(name="Random Option", value=str(role.rand_choice), inline=True) return embed @@ -60,7 +77,11 @@ def setup(): PAGE_GROUPS.append(len(ROLE_PAGES) - 1) for k, v in ROLE_CATEGORIES.items(): if 0 < k <= 6: - ROLE_PAGES.append(discord.Embed(title="RANDOM:Town Role", description="Town {}".format(v), color=0x008000)) + ROLE_PAGES.append( + discord.Embed( + title="RANDOM:Town Role", description="Town {}".format(v), color=0x008000 + ) + ) CATEGORY_COUNT.append(k) # Random WW Roles @@ -69,7 +90,12 @@ def setup(): for k, v in ROLE_CATEGORIES.items(): if 10 < k <= 16: ROLE_PAGES.append( - discord.Embed(title="RANDOM:Werewolf Role", description="Werewolf {}".format(v), color=0xff0000)) + discord.Embed( + title="RANDOM:Werewolf Role", + description="Werewolf {}".format(v), + color=0xFF0000, + ) + ) CATEGORY_COUNT.append(k) # Random Neutral Roles if len(ROLE_PAGES) - 1 not in PAGE_GROUPS: @@ -77,7 +103,10 @@ def setup(): for k, v in ROLE_CATEGORIES.items(): if 20 < k <= 26: ROLE_PAGES.append( - discord.Embed(title="RANDOM:Neutral Role", description="Neutral {}".format(v), color=0xc0c0c0)) + discord.Embed( + title="RANDOM:Neutral Role", description="Neutral {}".format(v), color=0xC0C0C0 + ) + ) CATEGORY_COUNT.append(k) @@ -187,9 +216,15 @@ async def encode(roles, rand_roles): return out_code -async def next_group(ctx: commands.Context, pages: list, - controls: dict, message: discord.Message, page: int, - timeout: float, emoji: str): +async def next_group( + ctx: commands.Context, + pages: list, + controls: dict, + message: discord.Message, + page: int, + timeout: float, + emoji: str, +): perms = message.channel.permissions_for(ctx.me) if perms.manage_messages: # Can manage messages, so remove react try: @@ -203,13 +238,18 @@ async def next_group(ctx: commands.Context, pages: list, else: page = PAGE_GROUPS[page] - return await menu(ctx, pages, controls, message=message, - page=page, timeout=timeout) + return await menu(ctx, pages, controls, message=message, page=page, timeout=timeout) -async def prev_group(ctx: commands.Context, pages: list, - controls: dict, message: discord.Message, page: int, - timeout: float, emoji: str): +async def prev_group( + ctx: commands.Context, + pages: list, + controls: dict, + message: discord.Message, + page: int, + timeout: float, + emoji: str, +): perms = message.channel.permissions_for(ctx.me) if perms.manage_messages: # Can manage messages, so remove react try: @@ -218,18 +258,23 @@ async def prev_group(ctx: commands.Context, pages: list, pass page = PAGE_GROUPS[bisect.bisect_left(PAGE_GROUPS, page) - 1] - return await menu(ctx, pages, controls, message=message, - page=page, timeout=timeout) + return await menu(ctx, pages, controls, message=message, page=page, timeout=timeout) def role_from_alignment(alignment): - return [role_embed(idx, role, ALIGNMENT_COLORS[role.alignment - 1]) - for idx, role in enumerate(ROLE_LIST) if alignment == role.alignment] + return [ + role_embed(idx, role, ALIGNMENT_COLORS[role.alignment - 1]) + for idx, role in enumerate(ROLE_LIST) + if alignment == role.alignment + ] def role_from_category(category): - return [role_embed(idx, role, ALIGNMENT_COLORS[role.alignment - 1]) - for idx, role in enumerate(ROLE_LIST) if category in role.category] + return [ + role_embed(idx, role, ALIGNMENT_COLORS[role.alignment - 1]) + for idx, role in enumerate(ROLE_LIST) + if category in role.category + ] def role_from_id(idx): @@ -242,8 +287,11 @@ def role_from_id(idx): def role_from_name(name: str): - return [role_embed(idx, role, ALIGNMENT_COLORS[role.alignment - 1]) - for idx, role in enumerate(ROLE_LIST) if name in role.__name__] + return [ + role_embed(idx, role, ALIGNMENT_COLORS[role.alignment - 1]) + for idx, role in enumerate(ROLE_LIST) + if name in role.__name__ + ] def say_role_list(code_list, rand_roles): @@ -268,7 +316,6 @@ def say_role_list(code_list, rand_roles): class GameBuilder: - def __init__(self): self.code = [] self.rand_roles = [] @@ -276,13 +323,13 @@ class GameBuilder: async def build_game(self, ctx: commands.Context): new_controls = { - '⏪': prev_group, + "⏪": prev_group, "⬅": prev_page, - '☑': self.select_page, + "☑": self.select_page, "➡": next_page, - '⏩': next_group, - '📇': self.list_roles, - "❌": close_menu + "⏩": next_group, + "📇": self.list_roles, + "❌": close_menu, } await ctx.send("Browse through roles and add the ones you want using the check mark") @@ -292,9 +339,16 @@ class GameBuilder: out = await encode(self.code, self.rand_roles) return out - async def list_roles(self, ctx: commands.Context, pages: list, - controls: dict, message: discord.Message, page: int, - timeout: float, emoji: str): + async def list_roles( + self, + ctx: commands.Context, + pages: list, + controls: dict, + message: discord.Message, + page: int, + timeout: float, + emoji: str, + ): perms = message.channel.permissions_for(ctx.me) if perms.manage_messages: # Can manage messages, so remove react try: @@ -304,12 +358,18 @@ class GameBuilder: await ctx.send(embed=say_role_list(self.code, self.rand_roles)) - return await menu(ctx, pages, controls, message=message, - page=page, timeout=timeout) - - async def select_page(self, ctx: commands.Context, pages: list, - controls: dict, message: discord.Message, page: int, - timeout: float, emoji: str): + return await menu(ctx, pages, controls, message=message, page=page, timeout=timeout) + + async def select_page( + self, + ctx: commands.Context, + pages: list, + controls: dict, + message: discord.Message, + page: int, + timeout: float, + emoji: str, + ): perms = message.channel.permissions_for(ctx.me) if perms.manage_messages: # Can manage messages, so remove react try: @@ -322,5 +382,4 @@ class GameBuilder: else: self.code.append(page) - return await menu(ctx, pages, controls, message=message, - page=page, timeout=timeout) + return await menu(ctx, pages, controls, message=message, page=page, timeout=timeout) diff --git a/werewolf/game.py b/werewolf/game.py index a31518f..95807a1 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -15,6 +15,7 @@ class Game: """ Base class to run a single game of Werewolf """ + vote_groups: Dict[str, VoteGroup] roles: List[Role] players: List[Player] @@ -22,19 +23,25 @@ class Game: default_secret_channel = { "channel": None, "players": [], - "votegroup": None # uninitialized VoteGroup + "votegroup": None, # uninitialized VoteGroup } morning_messages = [ "**The sun rises on day {} in the village..**", - "**Morning has arrived on day {}..**" + "**Morning has arrived on day {}..**", ] day_vote_count = 3 - def __init__(self, guild: discord.Guild, role: discord.Role = None, - category: discord.CategoryChannel = None, village: discord.TextChannel = None, - log_channel: discord.TextChannel = None, game_code=None): + def __init__( + self, + guild: discord.Guild, + role: discord.Role = None, + category: discord.CategoryChannel = None, + village: discord.TextChannel = None, + log_channel: discord.TextChannel = None, + game_code=None, + ): self.guild = guild self.game_code = game_code @@ -97,22 +104,28 @@ class Game: await self.get_roles(ctx) if len(self.players) != len(self.roles): - await ctx.maybe_send_embed("Player count does not match role count, cannot start\n" - "Currently **{} / {}**\n" - "Use `{}ww code` to pick a new game" - "".format(len(self.players), len(self.roles), ctx.prefix)) + await ctx.maybe_send_embed( + "Player count does not match role count, cannot start\n" + "Currently **{} / {}**\n" + "Use `{}ww code` to pick a new game" + "".format(len(self.players), len(self.roles), ctx.prefix) + ) self.roles = [] return False if self.game_role is None: try: - self.game_role = await ctx.guild.create_role(name="WW Players", - hoist=True, - mentionable=True, - reason="(BOT) Werewolf game role") + self.game_role = await ctx.guild.create_role( + name="WW Players", + hoist=True, + mentionable=True, + reason="(BOT) Werewolf game role", + ) self.to_delete.add(self.game_role) except (discord.Forbidden, discord.HTTPException): - await ctx.maybe_send_embed("Game role not configured and unable to generate one, cannot start") + await ctx.maybe_send_embed( + "Game role not configured and unable to generate one, cannot start" + ) self.roles = [] return False try: @@ -120,24 +133,33 @@ class Game: await player.member.add_roles(*[self.game_role]) except discord.Forbidden: await ctx.send( - "Unable to add role **{}**\nBot is missing `manage_roles` permissions".format(self.game_role.name)) + "Unable to add role **{}**\nBot is missing `manage_roles` permissions".format( + self.game_role.name + ) + ) return False await self.assign_roles() # Create category and channel with individual overwrites overwrite = { - self.guild.default_role: discord.PermissionOverwrite(read_messages=True, send_messages=False, - add_reactions=False), - self.guild.me: discord.PermissionOverwrite(read_messages=True, send_messages=True, add_reactions=True, - manage_messages=True, manage_channels=True, - manage_roles=True), - self.game_role: discord.PermissionOverwrite(read_messages=True, send_messages=True) + self.guild.default_role: discord.PermissionOverwrite( + read_messages=True, send_messages=False, add_reactions=False + ), + self.guild.me: discord.PermissionOverwrite( + read_messages=True, + send_messages=True, + add_reactions=True, + manage_messages=True, + manage_channels=True, + manage_roles=True, + ), + self.game_role: discord.PermissionOverwrite(read_messages=True, send_messages=True), } if self.channel_category is None: - self.channel_category = await self.guild.create_category("Werewolf Game", - overwrites=overwrite, - reason="(BOT) New game of werewolf") + self.channel_category = await self.guild.create_category( + "Werewolf Game", overwrites=overwrite, reason="(BOT) New game of werewolf" + ) else: # No need to modify categories pass # await self.channel_category.edit(name="🔴 Werewolf Game (ACTIVE)", reason="(BOT) New game of werewolf") @@ -147,20 +169,26 @@ class Game: # reason="(BOT) New game of werewolf") if self.village_channel is None: try: - self.village_channel = await self.guild.create_text_channel("🔵Werewolf", - overwrites=overwrite, - reason="(BOT) New game of werewolf", - category=self.channel_category) + self.village_channel = await self.guild.create_text_channel( + "🔵Werewolf", + overwrites=overwrite, + reason="(BOT) New game of werewolf", + category=self.channel_category, + ) except discord.Forbidden: - await ctx.send("Unable to create Game Channel and none was provided\n" - "Grant Bot appropriate permissions or assign a game_channel") + await ctx.send( + "Unable to create Game Channel and none was provided\n" + "Grant Bot appropriate permissions or assign a game_channel" + ) return False else: self.save_perms[self.village_channel] = self.village_channel.overwrites try: - await self.village_channel.edit(name="🔵Werewolf", - category=self.channel_category, - reason="(BOT) New game of werewolf") + await self.village_channel.edit( + name="🔵Werewolf", + category=self.channel_category, + reason="(BOT) New game of werewolf", + ) except discord.Forbidden as e: print("Unable to rename Game Channel") print(e) @@ -170,12 +198,14 @@ class Game: for target, ow in overwrite.items(): curr = self.village_channel.overwrites_for(target) curr.update(**{perm: value for perm, value in ow}) - await self.village_channel.set_permissions(target=target, - overwrite=curr, - reason="(BOT) New game of werewolf") + await self.village_channel.set_permissions( + target=target, overwrite=curr, reason="(BOT) New game of werewolf" + ) except discord.Forbidden: - await ctx.send("Unable to edit Game Channel permissions\n" - "Grant Bot appropriate permissions to manage permissions") + await ctx.send( + "Unable to edit Game Channel permissions\n" + "Grant Bot appropriate permissions to manage permissions" + ) return self.started = True # Assuming everything worked so far @@ -186,18 +216,25 @@ class Game: print("Channel id: " + channel_id) overwrite = { self.guild.default_role: discord.PermissionOverwrite(read_messages=False), - self.guild.me: discord.PermissionOverwrite(read_messages=True, send_messages=True, add_reactions=True, - manage_messages=True, manage_channels=True, - manage_roles=True) + self.guild.me: discord.PermissionOverwrite( + read_messages=True, + send_messages=True, + add_reactions=True, + manage_messages=True, + manage_channels=True, + manage_roles=True, + ), } for player in self.p_channels[channel_id]["players"]: overwrite[player.member] = discord.PermissionOverwrite(read_messages=True) - channel = await self.guild.create_text_channel(channel_id, - overwrites=overwrite, - reason="(BOT) WW game secret channel", - category=self.channel_category) + channel = await self.guild.create_text_channel( + channel_id, + overwrites=overwrite, + reason="(BOT) WW game secret channel", + category=self.channel_category, + ) self.p_channels[channel_id]["channel"] = channel @@ -216,8 +253,8 @@ class Game: async def _cycle(self): """ Each event calls the next event - - + + _at_day_start() _at_voted() @@ -225,7 +262,7 @@ class Game: _at_day_end() _at_night_begin() _at_night_end() - + and repeat with _at_day_start() again """ await self._at_day_start() @@ -237,7 +274,8 @@ class Game: return await self.village_channel.send( - embed=discord.Embed(title="Game is starting, please wait for setup to complete")) + embed=discord.Embed(title="Game is starting, please wait for setup to complete") + ) await self._notify(0) @@ -271,7 +309,9 @@ class Game: await asyncio.sleep(24) # 4 minute days FixMe to 120 later if check(): return - await self.village_channel.send(embed=discord.Embed(title="**Two minutes of daylight remain...**")) + await self.village_channel.send( + embed=discord.Embed(title="**Two minutes of daylight remain...**") + ) await asyncio.sleep(24) # 4 minute days FixMe to 120 later # Need a loop here to wait for trial to end (can_vote?) @@ -295,7 +335,10 @@ class Game: await self.speech_perms(self.village_channel, target.member) # Only target can talk await self.village_channel.send( - "**{} will be put to trial and has 30 seconds to defend themselves**".format(target.mention)) + "**{} will be put to trial and has 30 seconds to defend themselves**".format( + target.mention + ) + ) await asyncio.sleep(30) @@ -305,7 +348,8 @@ class Game: "Everyone will now vote whether to lynch {}\n" "👍 to save, 👎 to lynch\n" "*Majority rules, no-lynch on ties, " - "vote both or neither to abstain, 15 seconds to vote*".format(target.mention)) + "vote both or neither to abstain, 15 seconds to vote*".format(target.mention) + ) await message.add_reaction("👍") await message.add_reaction("👎") @@ -317,9 +361,9 @@ class Game: down_votes = sum(p for p in reaction_list if p.emoji == "👎" and not p.me) if down_votes > up_votes: - embed = discord.Embed(title="Vote Results", color=0xff0000) + embed = discord.Embed(title="Vote Results", color=0xFF0000) else: - embed = discord.Embed(title="Vote Results", color=0x80ff80) + embed = discord.Embed(title="Vote Results", color=0x80FF80) embed.add_field(name="👎", value="**{}**".format(up_votes), inline=True) embed.add_field(name="👍", value="**{}**".format(down_votes), inline=True) @@ -339,7 +383,8 @@ class Game: else: await self.village_channel.send( "**{}**/**{}** of today's votes have been used!\n" - "Nominate carefully..".format(self.used_votes, self.day_vote_count)) + "Nominate carefully..".format(self.used_votes, self.day_vote_count) + ) self.ongoing_vote = False @@ -373,7 +418,9 @@ class Game: await self.night_perms(self.village_channel) - await self.village_channel.send(embed=discord.Embed(title="**The sun sets on the village...**")) + await self.village_channel.send( + embed=discord.Embed(title="**The sun sets on the village...**") + ) await self._notify(5) await asyncio.sleep(5) @@ -385,9 +432,13 @@ class Game: await self._notify(6) await asyncio.sleep(12) # 2 minutes FixMe to 120 later - await self.village_channel.send(embed=discord.Embed(title="**Two minutes of night remain...**")) + await self.village_channel.send( + embed=discord.Embed(title="**Two minutes of night remain...**") + ) await asyncio.sleep(9) # 1.5 minutes FixMe to 90 later - await self.village_channel.send(embed=discord.Embed(title="**Thirty seconds until sunrise...**")) + await self.village_channel.send( + embed=discord.Embed(title="**Thirty seconds until sunrise...**") + ) await asyncio.sleep(3) # .5 minutes FixMe to 3 Later await self._at_night_end() @@ -413,10 +464,12 @@ class Game: role_order = [role for role in self.roles if role.action_list[event][1] == i] for role in role_order: tasks.append(asyncio.ensure_future(role.on_event(event, data), loop=self.loop)) - # VoteGroup priorities + # VoteGroup priorities vote_order = [vg for vg in self.vote_groups.values() if vg.action_list[event][1] == i] for vote_group in vote_order: - tasks.append(asyncio.ensure_future(vote_group.on_event(event, data), loop=self.loop)) + tasks.append( + asyncio.ensure_future(vote_group.on_event(event, data), loop=self.loop) + ) if tasks: await asyncio.gather(*tasks) # Run same-priority task simultaneously @@ -432,13 +485,17 @@ class Game: else: status = "*[Dead]*-" if with_roles or not player.alive: - embed.add_field(name="ID# **{}**".format(i), - value="{}{}-{}".format(status, player.member.display_name, str(player.role)), - inline=True) + embed.add_field( + name="ID# **{}**".format(i), + value="{}{}-{}".format(status, player.member.display_name, str(player.role)), + inline=True, + ) else: - embed.add_field(name="ID# **{}**".format(i), - value="{}{}".format(status, player.member.display_name), - inline=True) + embed.add_field( + name="ID# **{}**".format(i), + value="{}{}".format(status, player.member.display_name), + inline=True, + ) return await channel.send(embed=embed) @@ -479,10 +536,15 @@ class Game: await member.add_roles(*[self.game_role]) except discord.Forbidden: await channel.send( - "Unable to add role **{}**\nBot is missing `manage_roles` permissions".format(self.game_role.name)) + "Unable to add role **{}**\nBot is missing `manage_roles` permissions".format( + self.game_role.name + ) + ) - await channel.send("{} has been added to the game, " - "total players is **{}**".format(member.mention, len(self.players))) + await channel.send( + "{} has been added to the game, " + "total players is **{}**".format(member.mention, len(self.players)) + ) async def quit(self, member: discord.Member, channel: discord.TextChannel = None): """ @@ -499,7 +561,11 @@ class Game: else: self.players = [player for player in self.players if player.member != member] await member.remove_roles(*[self.game_role]) - await channel.send("{} chickened out, player count is now **{}**".format(member.mention, len(self.players))) + await channel.send( + "{} chickened out, player count is now **{}**".format( + member.mention, len(self.players) + ) + ) async def choose(self, ctx, data): """ @@ -598,14 +664,20 @@ class Game: required_votes = len([player for player in self.players if player.alive]) // 7 + 2 if self.vote_totals[target_id] < required_votes: - await self.village_channel.send("" - "{} has voted to put {} to trial. " - "{} more votes needed".format(author.mention, - target.member.mention, - required_votes - self.vote_totals[target_id])) + await self.village_channel.send( + "" + "{} has voted to put {} to trial. " + "{} more votes needed".format( + author.mention, + target.member.mention, + required_votes - self.vote_totals[target_id], + ) + ) else: self.vote_totals[target_id] = 0 - self.day_vote = {k: v for k, v in self.day_vote.items() if v != target_id} # Remove votes for this id + self.day_vote = { + k: v for k, v in self.day_vote.items() if v != target_id + } # Remove votes for this id await self._at_voted(target) async def eval_results(self, target, source=None, method=None): @@ -613,9 +685,9 @@ class Game: out = "**{ID}** - " + method return out.format(ID=target.id, target=target.member.display_name) else: - return "**{ID}** - {target} the {role} was found dead".format(ID=target.id, - target=target.member.display_name, - role=await target.role.get_role()) + return "**{ID}** - {target} the {role} was found dead".format( + ID=target.id, target=target.member.display_name, role=await target.role.get_role() + ) async def _quit(self, player): """ @@ -748,7 +820,9 @@ class Game: alive_players = [player for player in self.players if player.alive] if len(alive_players) <= 0: - await self.village_channel.send(embed=discord.Embed(title="**Everyone is dead! Game Over!**")) + await self.village_channel.send( + embed=discord.Embed(title="**Everyone is dead! Game Over!**") + ) self.game_over = True elif len(alive_players) == 1: self.game_over = True @@ -780,17 +854,19 @@ class Game: async def _announce_winners(self, winnerlist): await self.village_channel.send(self.game_role.mention) - embed = discord.Embed(title='Game Over', description='The Following Players have won!') + embed = discord.Embed(title="Game Over", description="The Following Players have won!") for player in winnerlist: embed.add_field(name=player.member.display_name, value=str(player.role), inline=True) - embed.set_thumbnail(url='https://emojipedia-us.s3.amazonaws.com/thumbs/160/twitter/134/trophy_1f3c6.png') + embed.set_thumbnail( + url="https://emojipedia-us.s3.amazonaws.com/thumbs/160/twitter/134/trophy_1f3c6.png" + ) await self.village_channel.send(embed=embed) await self.generate_targets(self.village_channel, True) async def _end_game(self): # Remove game_role access for potential archiving for now - reason = '(BOT) End of WW game' + reason = "(BOT) End of WW game" for obj in self.to_delete: print(obj) await obj.delete(reason=reason) @@ -798,8 +874,12 @@ class Game: try: await self.village_channel.edit(reason=reason, name="Werewolf") for target, overwrites in self.save_perms[self.village_channel]: - await self.village_channel.set_permissions(target, overwrite=overwrites, reason=reason) - await self.village_channel.set_permissions(self.game_role, overwrite=None, reason=reason) + await self.village_channel.set_permissions( + target, overwrite=overwrites, reason=reason + ) + await self.village_channel.set_permissions( + self.game_role, overwrite=None, reason=reason + ) except (discord.HTTPException, discord.NotFound, discord.errors.NotFound): pass diff --git a/werewolf/role.py b/werewolf/role.py index 3e4124d..af2d38b 100644 --- a/werewolf/role.py +++ b/werewolf/role.py @@ -1,31 +1,31 @@ class Role: """ Base Role class for werewolf game - + Category enrollment guide as follows (category property): Town: 1: Random, 2: Investigative, 3: Protective, 4: Government, 5: Killing, 6: Power (Special night action) - + Werewolf: 11: Random, 12: Deception, 15: Killing, 16: Support - + Neutral: 21: Benign, 22: Evil, 23: Killing - - + + Example category: category = [1, 5, 6] Could be Veteran category = [1, 5] Could be Bodyguard category = [11, 16] Could be Werewolf Silencer - - + + Action guide as follows (on_event function): _at_night_start 0. No Action 1. Detain actions (Jailer/Kidnapper) 2. Group discussions and choose targets - + _at_night_end 0. No Action 1. Self actions (Veteran) @@ -68,7 +68,7 @@ class Role: (self._at_day_end, 0), (self._at_night_start, 0), (self._at_night_end, 0), - (self._at_visit, 0) + (self._at_visit, 0), ] def __repr__(self): diff --git a/werewolf/votegroup.py b/werewolf/votegroup.py index bf07c8c..a754247 100644 --- a/werewolf/votegroup.py +++ b/werewolf/votegroup.py @@ -23,7 +23,7 @@ class VoteGroup: (self._at_day_end, 0), (self._at_night_start, 2), (self._at_night_end, 0), - (self._at_visit, 0) + (self._at_visit, 0), ] async def on_event(self, event, data): From 7e1a6e108ea804fc3a35601d8740cc555f3db6ed Mon Sep 17 00:00:00 2001 From: bobloy Date: Fri, 18 Sep 2020 17:02:37 -0400 Subject: [PATCH 05/73] Logs mainly --- werewolf/builder.py | 3 +++ werewolf/game.py | 3 +++ werewolf/night_powers.py | 4 ++++ werewolf/player.py | 4 ++++ werewolf/role.py | 5 +++++ werewolf/votegroup.py | 5 +++++ werewolf/werewolf.py | 4 ++++ 7 files changed, 28 insertions(+) diff --git a/werewolf/builder.py b/werewolf/builder.py index 1648b45..4632a21 100644 --- a/werewolf/builder.py +++ b/werewolf/builder.py @@ -1,4 +1,5 @@ import bisect +import logging from collections import defaultdict from random import choice @@ -13,6 +14,8 @@ from .roles.vanillawerewolf import VanillaWerewolf from .roles.villager import Villager from redbot.core.utils.menus import menu, prev_page, next_page, close_menu +log = logging.getLogger("red.fox_v3.werewolf.builder") + # All roles in this list for iterating ROLE_LIST = sorted([Villager, Seer, VanillaWerewolf], key=lambda x: x.alignment) diff --git a/werewolf/game.py b/werewolf/game.py index 95807a1..7c6a5c8 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -1,4 +1,5 @@ import asyncio +import logging import random from typing import List, Any, Dict, Set, Union @@ -10,6 +11,8 @@ from .player import Player from .role import Role from .votegroup import VoteGroup +log = logging.getLogger("red.fox_v3.werewolf.game") + class Game: """ diff --git a/werewolf/night_powers.py b/werewolf/night_powers.py index b50929b..a39ad26 100644 --- a/werewolf/night_powers.py +++ b/werewolf/night_powers.py @@ -1,5 +1,9 @@ +import logging + from .role import Role +log = logging.getLogger("red.fox_v3.werewolf.night_powers") + def night_immune(role: Role): role.player.alive = True diff --git a/werewolf/player.py b/werewolf/player.py index c84d87f..8ee437e 100644 --- a/werewolf/player.py +++ b/werewolf/player.py @@ -1,5 +1,9 @@ +import logging + import discord +log = logging.getLogger("red.fox_v3.werewolf.player") + class Player: """ diff --git a/werewolf/role.py b/werewolf/role.py index af2d38b..8c4d989 100644 --- a/werewolf/role.py +++ b/werewolf/role.py @@ -1,3 +1,8 @@ +import logging + +log = logging.getLogger("red.fox_v3.werewolf.role") + + class Role: """ Base Role class for werewolf game diff --git a/werewolf/votegroup.py b/werewolf/votegroup.py index a754247..d60f451 100644 --- a/werewolf/votegroup.py +++ b/werewolf/votegroup.py @@ -1,3 +1,8 @@ +import logging + +log = logging.getLogger("red.fox_v3.werewolf.votegroup") + + class VoteGroup: """ Base VoteGroup class for werewolf game diff --git a/werewolf/werewolf.py b/werewolf/werewolf.py index 1f8fc3f..c90982f 100644 --- a/werewolf/werewolf.py +++ b/werewolf/werewolf.py @@ -1,3 +1,5 @@ +import logging + import discord from redbot.core import Config, checks, commands from redbot.core.bot import Red @@ -13,6 +15,8 @@ from .builder import ( ) from .game import Game +log = logging.getLogger("red.fox_v3.werewolf") + class Werewolf(Cog): """ From fe1f11b2ebc1b27c9461d9a632aa69c520744acf Mon Sep 17 00:00:00 2001 From: bobloy Date: Sat, 19 Sep 2020 20:21:28 -0400 Subject: [PATCH 06/73] Maybe dispatch? WIP --- werewolf/game.py | 46 +++++++++++++++++++++------------ werewolf/role.py | 2 +- werewolf/roles/seer.py | 14 +++++++--- werewolf/roles/shifter.py | 2 +- werewolf/votegroups/wolfvote.py | 17 ++++++++---- werewolf/werewolf.py | 18 ++++++------- 6 files changed, 63 insertions(+), 36 deletions(-) diff --git a/werewolf/game.py b/werewolf/game.py index 7c6a5c8..ba63211 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -5,6 +5,7 @@ from typing import List, Any, Dict, Set, Union import discord from redbot.core import commands +from redbot.core.bot import Red from .builder import parse_code from .player import Player @@ -34,10 +35,15 @@ class Game: "**Morning has arrived on day {}..**", ] + night_messages = [ + "**Dawn falls on day {}..****" + ] + day_vote_count = 3 def __init__( self, + bot: Red, guild: discord.Guild, role: discord.Role = None, category: discord.CategoryChannel = None, @@ -45,6 +51,7 @@ class Game: log_channel: discord.TextChannel = None, game_code=None, ): + self.bot = bot self.guild = guild self.game_code = game_code @@ -340,6 +347,9 @@ class Game: await self.village_channel.send( "**{} will be put to trial and has 30 seconds to defend themselves**".format( target.mention + ), + allowed_mentions=discord.AllowedMentions( + everyone=False, users=[target] ) ) @@ -347,11 +357,14 @@ class Game: await self.speech_perms(self.village_channel, target.member, undo=True) # No one can talk - message = await self.village_channel.send( + message: discord.Message = await self.village_channel.send( "Everyone will now vote whether to lynch {}\n" "👍 to save, 👎 to lynch\n" "*Majority rules, no-lynch on ties, " - "vote both or neither to abstain, 15 seconds to vote*".format(target.mention) + "vote both or neither to abstain, 15 seconds to vote*".format(target.mention), + allowed_mentions=discord.AllowedMentions( + everyone=False, users=[target] + ) ) await message.add_reaction("👍") @@ -442,7 +455,7 @@ class Game: await self.village_channel.send( embed=discord.Embed(title="**Thirty seconds until sunrise...**") ) - await asyncio.sleep(3) # .5 minutes FixMe to 3 Later + await asyncio.sleep(3) # .5 minutes FixMe to 30 Later await self._at_night_end() @@ -462,19 +475,20 @@ class Game: async def _notify(self, event, data=None): for i in range(1, 7): # action guide 1-6 (0 is no action) - tasks = [] - # Role priorities - role_order = [role for role in self.roles if role.action_list[event][1] == i] - for role in role_order: - tasks.append(asyncio.ensure_future(role.on_event(event, data), loop=self.loop)) - # VoteGroup priorities - vote_order = [vg for vg in self.vote_groups.values() if vg.action_list[event][1] == i] - for vote_group in vote_order: - tasks.append( - asyncio.ensure_future(vote_group.on_event(event, data), loop=self.loop) - ) - if tasks: - await asyncio.gather(*tasks) + self.bot.dispatch(f"red.fox.werewolf.{event}", data=data, priority=i) + # tasks = [] + # # Role priorities + # role_order = [role for role in self.roles if role.action_list[event][1] == i] + # for role in role_order: + # tasks.append(asyncio.ensure_future(role.on_event(event, data), loop=self.loop)) + # # VoteGroup priorities + # vote_order = [vg for vg in self.vote_groups.values() if vg.action_list[event][1] == i] + # for vote_group in vote_order: + # tasks.append( + # asyncio.ensure_future(vote_group.on_event(event, data), loop=self.loop) + # ) + # if tasks: + # await asyncio.gather(*tasks) # Run same-priority task simultaneously ############END Notify structure############ diff --git a/werewolf/role.py b/werewolf/role.py index 8c4d989..2a08f71 100644 --- a/werewolf/role.py +++ b/werewolf/role.py @@ -38,7 +38,7 @@ class Role: 3. Protection / Preempt actions (bodyguard/framer) 4. Non-disruptive actions (seer/silencer) 5. Disruptive actions (Killing) - 6. Role altering actions (Cult / Mason) + 6. Role altering actions (Cult / Mason / Shifter) """ rand_choice = False # Determines if it can be picked as a random role (False for unusually disruptive roles) diff --git a/werewolf/roles/seer.py b/werewolf/roles/seer.py index 35c8271..8d0fc1f 100644 --- a/werewolf/roles/seer.py +++ b/werewolf/roles/seer.py @@ -14,8 +14,10 @@ class Seer(Role): "Lynch players during the day with `[p]ww vote `\n" "Check for werewolves at night with `[p]ww choose `" ) - description = "A mystic in search of answers in a chaotic town.\n" \ - "Calls upon the cosmos to discern those of Lycan blood" + description = ( + "A mystic in search of answers in a chaotic town.\n" + "Calls upon the cosmos to discern those of Lycan blood" + ) def __init__(self, game): super().__init__(game) @@ -33,7 +35,7 @@ class Seer(Role): (self._at_day_end, 0), (self._at_night_start, 2), (self._at_night_end, 4), - (self._at_visit, 0) + (self._at_visit, 0), ] async def see_alignment(self, source=None): @@ -87,4 +89,8 @@ class Seer(Role): await super().choose(ctx, data) self.see_target, target = await pick_target(self, ctx, data) - await ctx.send("**You will attempt to see the role of {} tonight...**".format(target.member.display_name)) + await ctx.send( + "**You will attempt to see the role of {} tonight...**".format( + target.member.display_name + ) + ) diff --git a/werewolf/roles/shifter.py b/werewolf/roles/shifter.py index 4c550dc..8d4b4f5 100644 --- a/werewolf/roles/shifter.py +++ b/werewolf/roles/shifter.py @@ -37,7 +37,7 @@ class Shifter(Role): 3. Protection / Preempt actions (bodyguard/framer) 4. Non-disruptive actions (seer/silencer) 5. Disruptive actions (Killing) - 6. Role altering actions (Cult / Mason) + 6. Role altering actions (Cult / Mason / Shifter) """ rand_choice = False # Determines if it can be picked as a random role (False for unusually disruptive roles) diff --git a/werewolf/votegroups/wolfvote.py b/werewolf/votegroups/wolfvote.py index 9c068d5..fb98b20 100644 --- a/werewolf/votegroups/wolfvote.py +++ b/werewolf/votegroups/wolfvote.py @@ -13,7 +13,8 @@ class WolfVote(VoteGroup): kill_messages = [ "**{ID}** - {target} was mauled by wolves", - "**{ID}** - {target} was found torn to shreds"] + "**{ID}** - {target} was found torn to shreds", + ] def __init__(self, game, channel): super().__init__(game, channel) @@ -34,7 +35,7 @@ class WolfVote(VoteGroup): (self._at_day_end, 0), (self._at_night_start, 2), (self._at_night_end, 5), # Kill priority - (self._at_visit, 0) + (self._at_visit, 0), ] # async def on_event(self, event, data): @@ -75,7 +76,9 @@ class WolfVote(VoteGroup): await self.channel.send(mention_list) self.killer = random.choice(self.players) - await self.channel.send("{} has been selected as tonight's killer".format(self.killer.member.display_name)) + await self.channel.send( + "{} has been selected as tonight's killer".format(self.killer.member.display_name) + ) async def _at_night_end(self, data=None): if self.channel is None: @@ -90,7 +93,9 @@ class WolfVote(VoteGroup): print("Target id: {}\nKiller: {}".format(target_id, self.killer.member.display_name)) if target_id is not None and self.killer: await self.game.kill(target_id, self.killer, random.choice(self.kill_messages)) - await self.channel.send("**{} has left to complete the kill...**".format(self.killer.member.display_name)) + await self.channel.send( + "**{} has left to complete the kill...**".format(self.killer.member.display_name) + ) else: await self.channel.send("**No kill will be attempted tonight...**") @@ -117,4 +122,6 @@ class WolfVote(VoteGroup): self.vote_results[author.id] = target_id - await self.channel.send("{} has voted to kill {}".format(author.mention, target.member.display_name)) + await self.channel.send( + "{} has voted to kill {}".format(author.mention, target.member.display_name) + ) diff --git a/werewolf/werewolf.py b/werewolf/werewolf.py index c90982f..0c8374c 100644 --- a/werewolf/werewolf.py +++ b/werewolf/werewolf.py @@ -97,7 +97,7 @@ class Werewolf(Cog): @wwset.command(name="role") async def wwset_role(self, ctx: commands.Context, role: discord.Role = None): """ - Assign the game role + Set the game role This role should not be manually assigned """ if role is None: @@ -222,7 +222,7 @@ class Werewolf(Cog): await ctx.send("No game running, cannot start") if not await game.setup(ctx): - pass # Do something? + pass # ToDo something? @commands.guild_only() @ww.command(name="stop") @@ -230,10 +230,10 @@ class Werewolf(Cog): """ Stops the current game """ - if ctx.guild is None: - # Private message, can't get guild - await ctx.send("Cannot start game from PM!") - return + # if ctx.guild is None: + # # Private message, can't get guild + # await ctx.send("Cannot stop game from PM!") + # return if ctx.guild.id not in self.games or self.games[ctx.guild.id].game_over: await ctx.send("No game to stop") return @@ -354,11 +354,11 @@ class Werewolf(Cog): await ctx.send("Role ID not found") async def _get_game(self, ctx: commands.Context, game_code=None): - guild: discord.Guild = ctx.guild + guild: discord.Guild = getattr(ctx, "guild", None) if guild is None: # Private message, can't get guild - await ctx.send("Cannot start game from PM!") + await ctx.send("Cannot start game from DM!") return None if guild.id not in self.games or self.games[guild.id].game_over: await ctx.send("Starting a new game...") @@ -368,7 +368,7 @@ class Werewolf(Cog): await ctx.send("Cannot start a new game") return None - self.games[guild.id] = Game(guild, role, category, channel, log_channel, game_code) + self.games[guild.id] = Game(self.bot, guild, role, category, channel, log_channel, game_code) return self.games[guild.id] From 28bf2a73e15625564dbe2d28b0a2c19879f54ede Mon Sep 17 00:00:00 2001 From: bobloy Date: Mon, 21 Sep 2020 11:17:59 -0400 Subject: [PATCH 07/73] Still going on events --- chatter/chat.py | 2 +- werewolf/game.py | 52 +++++++++++++++++++++++++++++------------------- 2 files changed, 32 insertions(+), 22 deletions(-) diff --git a/chatter/chat.py b/chatter/chat.py index ad8e37b..ef75bb8 100644 --- a/chatter/chat.py +++ b/chatter/chat.py @@ -434,7 +434,7 @@ class Chatter(Cog): else: await ctx.maybe_send_embed("Error occurred :(") - @commands.Cog.listener() + @Cog.listener() async def on_message_without_command(self, message: discord.Message): """ Credit to https://github.com/Twentysix26/26-Cogs/blob/master/cleverbot/cleverbot.py diff --git a/werewolf/game.py b/werewolf/game.py index ba63211..c3aa813 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -1,6 +1,7 @@ import asyncio import logging import random +from collections import deque from typing import List, Any, Dict, Set, Union import discord @@ -85,6 +86,8 @@ class Game: self.loop = asyncio.get_event_loop() + self.action_queue = deque() + # def __del__(self): # """ # Cleanup channels as necessary @@ -275,8 +278,14 @@ class Game: and repeat with _at_day_start() again """ - await self._at_day_start() - # Once cycle ends, this will trigger end_game + + self.action_queue.append(self._at_day_start()) + + while self.action_queue: + await self.action_queue.popleft() + # + # await self._at_day_start() + # # Once cycle ends, this will trigger end_game await self._end_game() # Handle open channels async def _at_game_start(self): # ID 0 @@ -331,7 +340,7 @@ class Game: if check(): return - await self._at_day_end() + self.action_queue.append(self._at_day_end()) async def _at_voted(self, target): # ID 2 if self.game_over: @@ -405,7 +414,7 @@ class Game: self.ongoing_vote = False if not self.can_vote: - await self._at_day_end() + self.action_queue.append(self._at_day_end()) else: await self.normal_perms(self.village_channel) # No point if about to be night @@ -440,7 +449,7 @@ class Game: await self._notify(5) await asyncio.sleep(5) - await self._at_night_start() + self.action_queue.append(self._at_night_start()) async def _at_night_start(self): # ID 6 if self.game_over: @@ -457,7 +466,7 @@ class Game: ) await asyncio.sleep(3) # .5 minutes FixMe to 30 Later - await self._at_night_end() + self.action_queue.append(self._at_night_end()) async def _at_night_end(self): # ID 7 if self.game_over: @@ -465,7 +474,7 @@ class Game: await self._notify(7) await asyncio.sleep(10) - await self._at_day_start() + self.action_queue.append(self._at_day_start()) async def _at_visit(self, target, source): # ID 8 if self.game_over: @@ -475,20 +484,21 @@ class Game: async def _notify(self, event, data=None): for i in range(1, 7): # action guide 1-6 (0 is no action) - self.bot.dispatch(f"red.fox.werewolf.{event}", data=data, priority=i) - # tasks = [] - # # Role priorities - # role_order = [role for role in self.roles if role.action_list[event][1] == i] - # for role in role_order: - # tasks.append(asyncio.ensure_future(role.on_event(event, data), loop=self.loop)) - # # VoteGroup priorities - # vote_order = [vg for vg in self.vote_groups.values() if vg.action_list[event][1] == i] - # for vote_group in vote_order: - # tasks.append( - # asyncio.ensure_future(vote_group.on_event(event, data), loop=self.loop) - # ) - # if tasks: - # await asyncio.gather(*tasks) + # self.bot.dispatch(f"red.fox.werewolf.{event}", data=data, priority=i) + # self.bot.extra_events + tasks = [] + # Role priorities + role_order = [role for role in self.roles if role.action_list[event][1] == i] + for role in role_order: + tasks.append(asyncio.ensure_future(role.on_event(event, data), loop=self.loop)) + # VoteGroup priorities + vote_order = [vg for vg in self.vote_groups.values() if vg.action_list[event][1] == i] + for vote_group in vote_order: + tasks.append( + asyncio.ensure_future(vote_group.on_event(event, data), loop=self.loop) + ) + if tasks: + await asyncio.gather(*tasks) # Run same-priority task simultaneously ############END Notify structure############ From 1723dc381d4a49fc5f2c14a666499967c8bd7f33 Mon Sep 17 00:00:00 2001 From: bobloy Date: Mon, 21 Sep 2020 13:42:43 -0400 Subject: [PATCH 08/73] More listener --- werewolf/game.py | 183 ++++++++++++++++++++++++++++++++++++++--------- werewolf/role.py | 84 +++++++++++----------- 2 files changed, 192 insertions(+), 75 deletions(-) diff --git a/werewolf/game.py b/werewolf/game.py index c3aa813..e9d3763 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -1,4 +1,5 @@ import asyncio +import inspect import logging import random from collections import deque @@ -36,9 +37,7 @@ class Game: "**Morning has arrived on day {}..**", ] - night_messages = [ - "**Dawn falls on day {}..****" - ] + night_messages = ["**Dawn falls on day {}..****"] day_vote_count = 3 @@ -87,6 +86,7 @@ class Game: self.loop = asyncio.get_event_loop() self.action_queue = deque() + self.listeners = {} # def __del__(self): # """ @@ -265,9 +265,7 @@ class Game: ############START Notify structure############ async def _cycle(self): """ - Each event calls the next event - - + Each event enqueues the next event _at_day_start() _at_voted() @@ -296,7 +294,7 @@ class Game: embed=discord.Embed(title="Game is starting, please wait for setup to complete") ) - await self._notify(0) + await self._notify("at_game_start") async def _at_day_start(self): # ID 1 if self.game_over: @@ -318,7 +316,7 @@ class Game: await self.generate_targets(self.village_channel) await self.day_perms(self.village_channel) - await self._notify(1) + await self._notify("at_day_start") await self._check_game_over() if self.game_over: @@ -346,7 +344,7 @@ class Game: if self.game_over: return data = {"player": target} - await self._notify(2, data) + await self._notify("at_voted", player=target) self.ongoing_vote = True @@ -357,9 +355,7 @@ class Game: "**{} will be put to trial and has 30 seconds to defend themselves**".format( target.mention ), - allowed_mentions=discord.AllowedMentions( - everyone=False, users=[target] - ) + allowed_mentions=discord.AllowedMentions(everyone=False, users=[target]), ) await asyncio.sleep(30) @@ -371,9 +367,7 @@ class Game: "👍 to save, 👎 to lynch\n" "*Majority rules, no-lynch on ties, " "vote both or neither to abstain, 15 seconds to vote*".format(target.mention), - allowed_mentions=discord.AllowedMentions( - everyone=False, users=[target] - ) + allowed_mentions=discord.AllowedMentions(everyone=False, users=[target]), ) await message.add_reaction("👍") @@ -422,13 +416,13 @@ class Game: if self.game_over: return data = {"player": target} - await self._notify(3, data) + await self._notify("at_kill", player=target) async def _at_hang(self, target): # ID 4 if self.game_over: return data = {"player": target} - await self._notify(4, data) + await self._notify("at_hang", player=target) async def _at_day_end(self): # ID 5 await self._check_game_over() @@ -447,14 +441,14 @@ class Game: embed=discord.Embed(title="**The sun sets on the village...**") ) - await self._notify(5) + await self._notify("at_day_end") await asyncio.sleep(5) self.action_queue.append(self._at_night_start()) async def _at_night_start(self): # ID 6 if self.game_over: return - await self._notify(6) + await self._notify("at_night_start") await asyncio.sleep(12) # 2 minutes FixMe to 120 later await self.village_channel.send( @@ -471,7 +465,7 @@ class Game: async def _at_night_end(self): # ID 7 if self.game_over: return - await self._notify(7) + await self._notify("at_night_end") await asyncio.sleep(10) self.action_queue.append(self._at_day_start()) @@ -480,25 +474,30 @@ class Game: if self.game_over: return data = {"target": target, "source": source} - await self._notify(8, data) + await self._notify("at_visit", target=target, source=source) - async def _notify(self, event, data=None): + async def _notify(self, event, **kwargs): for i in range(1, 7): # action guide 1-6 (0 is no action) + tasks = [] + for event in self.listeners.get(event, []): + tasks.append(asyncio.ensure_future(event(**kwargs), loop=self.loop)) + await asyncio.gather(*tasks) + # self.bot.dispatch(f"red.fox.werewolf.{event}", data=data, priority=i) # self.bot.extra_events - tasks = [] - # Role priorities - role_order = [role for role in self.roles if role.action_list[event][1] == i] - for role in role_order: - tasks.append(asyncio.ensure_future(role.on_event(event, data), loop=self.loop)) - # VoteGroup priorities - vote_order = [vg for vg in self.vote_groups.values() if vg.action_list[event][1] == i] - for vote_group in vote_order: - tasks.append( - asyncio.ensure_future(vote_group.on_event(event, data), loop=self.loop) - ) - if tasks: - await asyncio.gather(*tasks) + # tasks = [] + # # Role priorities + # role_order = [role for role in self.roles if role.action_list[event][1] == i] + # for role in role_order: + # tasks.append(asyncio.ensure_future(role.on_event(event, data), loop=self.loop)) + # # VoteGroup priorities + # vote_order = [vg for vg in self.vote_groups.values() if vg.action_list[event][1] == i] + # for vote_group in vote_order: + # tasks.append( + # asyncio.ensure_future(vote_group.on_event(event, data), loop=self.loop) + # ) + # if tasks: + # await asyncio.gather(*tasks) # Run same-priority task simultaneously ############END Notify structure############ @@ -911,3 +910,117 @@ class Game: pass # Optional dynamic channels/categories + + @classmethod + def wolflistener(cls, name=None): + """A decorator that marks a function as a listener. + + This is the cog equivalent of :meth:`.Bot.listen`. + + Parameters + ------------ + name: :class:`str` + The name of the event being listened to. If not provided, it + defaults to the function's name. + + Raises + -------- + TypeError + The function is not a coroutine function or a string was not passed as + the name. + """ + + if name is not None and not isinstance(name, str): + raise TypeError( + "Cog.listener expected str but received {0.__class__.__name__!r} instead.".format( + name + ) + ) + + def decorator(func): + actual = func + if isinstance(actual, staticmethod): + actual = actual.__func__ + if not inspect.iscoroutinefunction(actual): + raise TypeError("Listener function must be a coroutine function.") + actual.__werewolf_listener__ = True + to_assign = name or actual.__name__ + try: + actual.__cog_listener_names__.append(to_assign) + except AttributeError: + actual.__cog_listener_names__ = [to_assign] + # we have to return `func` instead of `actual` because + # we need the type to be `staticmethod` for the metaclass + # to pick it up but the metaclass unfurls the function and + # thus the assignments need to be on the actual function + return func + + return decorator + + def wolflisten(self, name=None): + """A decorator that registers another function as an external + event listener. Basically this allows you to listen to multiple + events from different places e.g. such as :func:`.on_ready` + + The functions being listened to must be a :ref:`coroutine `. + + Example + -------- + + .. code-block:: python3 + + @bot.listen() + async def on_message(message): + print('one') + + # in some other file... + + @bot.listen('on_message') + async def my_message(message): + print('two') + + Would print one and two in an unspecified order. + + Raises + ------- + TypeError + The function being listened to is not a coroutine. + """ + + def decorator(func): + self.add_wolflistener(func, name) + return func + + return decorator + + def add_wolflistener(self, func, name=None): + """The non decorator alternative to :meth:`.listen`. + + Parameters + ----------- + func: :ref:`coroutine ` + The function to call. + name: Optional[:class:`str`] + The name of the event to listen for. Defaults to ``func.__name__``. + + Example + -------- + + .. code-block:: python3 + + async def on_ready(): pass + async def my_message(message): pass + + bot.add_listener(on_ready) + bot.add_listener(my_message, 'on_message') + + """ + name = func.__name__ if name is None else name + + if not asyncio.iscoroutinefunction(func): + raise TypeError('Listeners must be coroutines') + + if name in self.listeners: + self.listeners[name].append(func) + else: + self.listeners[name] = [func] diff --git a/werewolf/role.py b/werewolf/role.py index 2a08f71..bdcec71 100644 --- a/werewolf/role.py +++ b/werewolf/role.py @@ -1,5 +1,8 @@ +import inspect import logging +from werewolf import Werewolf + log = logging.getLogger("red.fox_v3.werewolf.role") @@ -64,27 +67,27 @@ class Role: self.blocked = False self.properties = {} # Extra data for other roles (i.e. arsonist) - self.action_list = [ - (self._at_game_start, 1), # (Action, Priority) - (self._at_day_start, 0), - (self._at_voted, 0), - (self._at_kill, 0), - (self._at_hang, 0), - (self._at_day_end, 0), - (self._at_night_start, 0), - (self._at_night_end, 0), - (self._at_visit, 0), - ] + # self.action_list = [ + # (self._at_game_start, 1), # (Action, Priority) + # (self._at_day_start, 0), + # (self._at_voted, 0), + # (self._at_kill, 0), + # (self._at_hang, 0), + # (self._at_day_end, 0), + # (self._at_night_start, 0), + # (self._at_night_end, 0), + # (self._at_visit, 0), + # ] def __repr__(self): return self.__class__.__name__ - async def on_event(self, event, data): - """ - See Game class for event guide - """ - - await self.action_list[event][0](data) + # async def on_event(self, event, data): + # """ + # See Game class for event guide + # """ + # + # await self.action_list[event][0](data) async def assign_player(self, player): """ @@ -124,35 +127,36 @@ class Role: """ return "Default" + @wolflistener("at_game_start") async def _at_game_start(self, data=None): if self.channel_id: await self.game.register_channel(self.channel_id, self) await self.player.send_dm(self.game_start_message) # Maybe embeds eventually - async def _at_day_start(self, data=None): - pass - - async def _at_voted(self, data=None): - pass - - async def _at_kill(self, data=None): - pass - - async def _at_hang(self, data=None): - pass - - async def _at_day_end(self, data=None): - pass - - async def _at_night_start(self, data=None): - pass - - async def _at_night_end(self, data=None): - pass - - async def _at_visit(self, data=None): - pass + # async def _at_day_start(self, data=None): + # pass + # + # async def _at_voted(self, data=None): + # pass + # + # async def _at_kill(self, data=None): + # pass + # + # async def _at_hang(self, data=None): + # pass + # + # async def _at_day_end(self, data=None): + # pass + # + # async def _at_night_start(self, data=None): + # pass + # + # async def _at_night_end(self, data=None): + # pass + # + # async def _at_visit(self, data=None): + # pass async def kill(self, source): """ From 8a3f45bdc1231b606f7dfecd096dbd2c670eda13 Mon Sep 17 00:00:00 2001 From: bobloy Date: Mon, 21 Sep 2020 17:11:55 -0400 Subject: [PATCH 09/73] Listener structure major change Still need priority --- werewolf/game.py | 113 +++++++----------------------- werewolf/listener.py | 91 ++++++++++++++++++++++++ werewolf/role.py | 7 +- werewolf/roles/seer.py | 3 + werewolf/roles/shifter.py | 55 ++++++++------- werewolf/roles/vanillawerewolf.py | 24 ++++--- werewolf/votegroup.py | 67 +++++++++--------- 7 files changed, 200 insertions(+), 160 deletions(-) create mode 100644 werewolf/listener.py diff --git a/werewolf/game.py b/werewolf/game.py index e9d3763..df5d263 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -16,6 +16,7 @@ from .votegroup import VoteGroup log = logging.getLogger("red.fox_v3.werewolf.game") +HALF_DAY_LENGTH = 24 # FixMe: to 120 later for 4 minute days class Game: """ @@ -262,7 +263,7 @@ class Game: await asyncio.sleep(1) await asyncio.ensure_future(self._cycle()) # Start the loop - ############START Notify structure############ + # ###########START Notify structure############ async def _cycle(self): """ Each event enqueues the next event @@ -323,13 +324,13 @@ class Game: return self.can_vote = True - await asyncio.sleep(24) # 4 minute days FixMe to 120 later + await asyncio.sleep(HALF_DAY_LENGTH) # 4 minute days FixMe to 120 later if check(): return await self.village_channel.send( embed=discord.Embed(title="**Two minutes of daylight remain...**") ) - await asyncio.sleep(24) # 4 minute days FixMe to 120 later + await asyncio.sleep(HALF_DAY_LENGTH) # 4 minute days FixMe to 120 later # Need a loop here to wait for trial to end (can_vote?) while self.ongoing_vote: @@ -500,7 +501,7 @@ class Game: # await asyncio.gather(*tasks) # Run same-priority task simultaneously - ############END Notify structure############ + # ###########END Notify structure############ async def generate_targets(self, channel, with_roles=False): embed = discord.Embed(title="Remaining Players") @@ -911,89 +912,7 @@ class Game: # Optional dynamic channels/categories - @classmethod - def wolflistener(cls, name=None): - """A decorator that marks a function as a listener. - - This is the cog equivalent of :meth:`.Bot.listen`. - - Parameters - ------------ - name: :class:`str` - The name of the event being listened to. If not provided, it - defaults to the function's name. - - Raises - -------- - TypeError - The function is not a coroutine function or a string was not passed as - the name. - """ - - if name is not None and not isinstance(name, str): - raise TypeError( - "Cog.listener expected str but received {0.__class__.__name__!r} instead.".format( - name - ) - ) - - def decorator(func): - actual = func - if isinstance(actual, staticmethod): - actual = actual.__func__ - if not inspect.iscoroutinefunction(actual): - raise TypeError("Listener function must be a coroutine function.") - actual.__werewolf_listener__ = True - to_assign = name or actual.__name__ - try: - actual.__cog_listener_names__.append(to_assign) - except AttributeError: - actual.__cog_listener_names__ = [to_assign] - # we have to return `func` instead of `actual` because - # we need the type to be `staticmethod` for the metaclass - # to pick it up but the metaclass unfurls the function and - # thus the assignments need to be on the actual function - return func - - return decorator - - def wolflisten(self, name=None): - """A decorator that registers another function as an external - event listener. Basically this allows you to listen to multiple - events from different places e.g. such as :func:`.on_ready` - - The functions being listened to must be a :ref:`coroutine `. - - Example - -------- - - .. code-block:: python3 - - @bot.listen() - async def on_message(message): - print('one') - - # in some other file... - - @bot.listen('on_message') - async def my_message(message): - print('two') - - Would print one and two in an unspecified order. - - Raises - ------- - TypeError - The function being listened to is not a coroutine. - """ - - def decorator(func): - self.add_wolflistener(func, name) - return func - - return decorator - - def add_wolflistener(self, func, name=None): + def add_listener(self, func, name=None): """The non decorator alternative to :meth:`.listen`. Parameters @@ -1024,3 +943,23 @@ class Game: self.listeners[name].append(func) else: self.listeners[name] = [func] + + def remove_listener(self, func, name=None): + """Removes a listener from the pool of listeners. + + Parameters + ----------- + func + The function that was used as a listener to remove. + name: :class:`str` + The name of the event we want to remove. Defaults to + ``func.__name__``. + """ + + name = func.__name__ if name is None else name + + if name in self.listeners: + try: + self.listeners[name].remove(func) + except ValueError: + pass diff --git a/werewolf/listener.py b/werewolf/listener.py new file mode 100644 index 0000000..9c36400 --- /dev/null +++ b/werewolf/listener.py @@ -0,0 +1,91 @@ +import inspect + + +def wolflistener(name=None): + """A decorator that marks a function as a listener. + + This is the werewolf.Game equivalent of :meth:`.Cog.listener`. + + Parameters + ------------ + name: :class:`str` + The name of the event being listened to. If not provided, it + defaults to the function's name. + + Raises + -------- + TypeError + The function is not a coroutine function or a string was not passed as + the name. + """ + + if name is not None and not isinstance(name, str): + raise TypeError( + "Game.listener expected str but received {0.__class__.__name__!r} instead.".format( + name + ) + ) + + def decorator(func): + actual = func + if isinstance(actual, staticmethod): + actual = actual.__func__ + if not inspect.iscoroutinefunction(actual): + raise TypeError("Listener function must be a coroutine function.") + actual.__wolf_listener__ = True + to_assign = name or actual.__name__ + try: + actual.__wolf_listener_names__.append(to_assign) + except AttributeError: + actual.__wolf_listener_names__ = [to_assign] + # we have to return `func` instead of `actual` because + # we need the type to be `staticmethod` for the metaclass + # to pick it up but the metaclass unfurls the function and + # thus the assignments need to be on the actual function + return func + + return decorator + + +class WolfListenerMeta(type): + def __new__(mcs, cls, *args, **kwargs): + name, bases = args + + commands = {} + listeners = {} + need_at_msg = "Listeners must start with at_ (in method {0.__name__}.{1})" + + new_cls = super().__new__(cls, name, bases, **kwargs) + for base in reversed(new_cls.__mro__): + for elem, value in base.__dict__.items(): + if elem in listeners: + del listeners[elem] + + is_static_method = isinstance(value, staticmethod) + if is_static_method: + value = value.__func__ + if inspect.iscoroutinefunction(value): + try: + is_listener = getattr(value, "__wolf_listener__") + except AttributeError: + continue + else: + if not elem.startswith("at_"): + raise TypeError(need_at_msg.format(mcs, elem)) + listeners[elem] = value + + listeners_as_list = [] + for listener in listeners.values(): + for listener_name in listener.__wolf_listener_names__: + # I use __name__ instead of just storing the value so I can inject + # the self attribute when the time comes to add them to the bot + listeners_as_list.append((listener_name, listener.__name__)) + + new_cls.__wolf_listeners__ = listeners_as_list + return new_cls + + +class WolfListener(metaclass=WolfListenerMeta): + def __init__(self, game): + for name, method_name in self.__wolf_listeners__: + game.add_listener(getattr(self, method_name), name) diff --git a/werewolf/role.py b/werewolf/role.py index bdcec71..4ae10ad 100644 --- a/werewolf/role.py +++ b/werewolf/role.py @@ -1,12 +1,12 @@ import inspect import logging -from werewolf import Werewolf +from werewolf.listener import WolfListener, wolflistener log = logging.getLogger("red.fox_v3.werewolf.role") -class Role: +class Role(WolfListener): """ Base Role class for werewolf game @@ -28,7 +28,7 @@ class Role: category = [11, 16] Could be Werewolf Silencer - Action guide as follows (on_event function): + Action priority guide as follows (on_event function): _at_night_start 0. No Action 1. Detain actions (Jailer/Kidnapper) @@ -62,6 +62,7 @@ class Role: icon_url = None # Adding a URL here will enable a thumbnail of the role def __init__(self, game): + super().__init__(game) self.game = game self.player = None self.blocked = False diff --git a/werewolf/roles/seer.py b/werewolf/roles/seer.py index 8d0fc1f..603d197 100644 --- a/werewolf/roles/seer.py +++ b/werewolf/roles/seer.py @@ -1,3 +1,4 @@ +from ..listener import wolflistener from ..night_powers import pick_target from ..role import Role @@ -59,6 +60,7 @@ class Seer(Role): """ return "Villager" + @wolflistener("at_night_start") async def _at_night_start(self, data=None): if not self.player.alive: return @@ -66,6 +68,7 @@ class Seer(Role): await self.game.generate_targets(self.player.member) await self.player.send_dm("**Pick a target to see tonight**") + @wolflistener("at_night_end") async def _at_night_end(self, data=None): if self.see_target is None: if self.player.alive: diff --git a/werewolf/roles/shifter.py b/werewolf/roles/shifter.py index 8d4b4f5..85c3ec5 100644 --- a/werewolf/roles/shifter.py +++ b/werewolf/roles/shifter.py @@ -1,3 +1,4 @@ +from ..listener import wolflistener from ..night_powers import pick_target from ..role import Role @@ -5,31 +6,31 @@ from ..role import Role class Shifter(Role): """ Base Role class for werewolf game - + Category enrollment guide as follows (category property): Town: 1: Random, 2: Investigative, 3: Protective, 4: Government, 5: Killing, 6: Power (Special night action) - + Werewolf: 11: Random, 12: Deception, 15: Killing, 16: Support - + Neutral: 21: Benign, 22: Evil, 23: Killing - - + + Example category: category = [1, 5, 6] Could be Veteran category = [1, 5] Could be Bodyguard category = [11, 16] Could be Werewolf Silencer - - + + Action guide as follows (on_event function): _at_night_start 0. No Action 1. Detain actions (Jailer/Kidnapper) 2. Group discussions and choose targets - + _at_night_end 0. No Action 1. Self actions (Veteran) @@ -61,17 +62,17 @@ class Shifter(Role): super().__init__(game) self.shift_target = None - self.action_list = [ - (self._at_game_start, 1), # (Action, Priority) - (self._at_day_start, 0), - (self._at_voted, 0), - (self._at_kill, 0), - (self._at_hang, 0), - (self._at_day_end, 0), - (self._at_night_start, 2), # Chooses targets - (self._at_night_end, 6), # Role Swap - (self._at_visit, 0) - ] + # self.action_list = [ + # (self._at_game_start, 1), # (Action, Priority) + # (self._at_day_start, 0), + # (self._at_voted, 0), + # (self._at_kill, 0), + # (self._at_hang, 0), + # (self._at_day_end, 0), + # (self._at_night_start, 2), # Chooses targets + # (self._at_night_end, 6), # Role Swap + # (self._at_visit, 0), + # ] async def see_alignment(self, source=None): """ @@ -94,14 +95,14 @@ class Shifter(Role): """ return "Shifter" + @wolflistener("at_night_start") async def _at_night_start(self, data=None): - await super()._at_night_start(data) self.shift_target = None await self.game.generate_targets(self.player.member) await self.player.send_dm("**Pick a target to shift into**") + @wolflistener("at_night_end") async def _at_night_end(self, data=None): - await super()._at_night_end(data) if self.shift_target is None: if self.player.alive: await self.player.send_dm("You will not use your powers tonight...") @@ -114,16 +115,22 @@ class Shifter(Role): # Roles have now been swapped - await self.player.send_dm("Your role has been stolen...\n" - "You are now a **Shifter**.") + await self.player.send_dm( + "Your role has been stolen...\n" "You are now a **Shifter**." + ) await self.player.send_dm(self.game_start_message) await target.send_dm(target.role.game_start_message) else: await self.player.send_dm("**Your shift failed...**") + async def choose(self, ctx, data): """Handle night actions""" await super().choose(ctx, data) self.shift_target, target = await pick_target(self, ctx, data) - await ctx.send("**You will attempt to see the role of {} tonight...**".format(target.member.display_name)) + await ctx.send( + "**You will attempt to see the role of {} tonight...**".format( + target.member.display_name + ) + ) diff --git a/werewolf/roles/vanillawerewolf.py b/werewolf/roles/vanillawerewolf.py index c8050da..5f7407b 100644 --- a/werewolf/roles/vanillawerewolf.py +++ b/werewolf/roles/vanillawerewolf.py @@ -1,3 +1,4 @@ +from ..listener import wolflistener from ..role import Role from ..votegroups.wolfvote import WolfVote @@ -19,17 +20,17 @@ class VanillaWerewolf(Role): def __init__(self, game): super().__init__(game) - self.action_list = [ - (self._at_game_start, 1), # (Action, Priority) - (self._at_day_start, 0), - (self._at_voted, 0), - (self._at_kill, 0), - (self._at_hang, 0), - (self._at_day_end, 0), - (self._at_night_start, 0), - (self._at_night_end, 0), - (self._at_visit, 0) - ] + # self.action_list = [ + # (self._at_game_start, 1), # (Action, Priority) + # (self._at_day_start, 0), + # (self._at_voted, 0), + # (self._at_kill, 0), + # (self._at_hang, 0), + # (self._at_day_end, 0), + # (self._at_night_start, 0), + # (self._at_night_end, 0), + # (self._at_visit, 0) + # ] async def see_alignment(self, source=None): """ @@ -52,6 +53,7 @@ class VanillaWerewolf(Role): """ return "Werewolf" + @wolflistener("at_game_start") async def _at_game_start(self, data=None): if self.channel_id: print("Wolf has channel_id: " + self.channel_id) diff --git a/werewolf/votegroup.py b/werewolf/votegroup.py index d60f451..19ebd9e 100644 --- a/werewolf/votegroup.py +++ b/werewolf/votegroup.py @@ -1,9 +1,11 @@ import logging +from werewolf.listener import WolfListener, wolflistener + log = logging.getLogger("red.fox_v3.werewolf.votegroup") -class VoteGroup: +class VoteGroup(WolfListener): """ Base VoteGroup class for werewolf game Handles secret channels and group decisions @@ -13,57 +15,55 @@ class VoteGroup: channel_id = "" def __init__(self, game, channel): + super().__init__(game) self.game = game self.channel = channel self.players = [] self.vote_results = {} self.properties = {} # Extra data for other options - self.action_list = [ - (self._at_game_start, 1), # (Action, Priority) - (self._at_day_start, 0), - (self._at_voted, 0), - (self._at_kill, 1), - (self._at_hang, 1), - (self._at_day_end, 0), - (self._at_night_start, 2), - (self._at_night_end, 0), - (self._at_visit, 0), - ] - - async def on_event(self, event, data): - """ - See Game class for event guide - """ - - await self.action_list[event][0](data) - + # self.action_list = [ + # (self._at_game_start, 1), # (Action, Priority) + # (self._at_day_start, 0), + # (self._at_voted, 0), + # (self._at_kill, 1), + # (self._at_hang, 1), + # (self._at_day_end, 0), + # (self._at_night_start, 2), + # (self._at_night_end, 0), + # (self._at_visit, 0), + # ] + + # async def on_event(self, event, data): + # """ + # See Game class for event guide + # """ + # + # await self.action_list[event][0](data) + + @wolflistener("at_game_start") async def _at_game_start(self, data=None): await self.channel.send(" ".join(player.mention for player in self.players)) - async def _at_day_start(self, data=None): - pass - - async def _at_voted(self, data=None): - pass - + @wolflistener("at_kill") async def _at_kill(self, data=None): if data["player"] in self.players: self.players.remove(data["player"]) - async def _at_hang(self, data=None): - if data["player"] in self.players: - self.players.remove(data["player"]) - - async def _at_day_end(self, data=None): - pass + # Removed, only if they actually die + # @wolflistener("at_hang") + # async def _at_hang(self, data=None): + # if data["player"] in self.players: + # self.players.remove(data["player"]) + @wolflistener("at_night_start") async def _at_night_start(self, data=None): if self.channel is None: return await self.game.generate_targets(self.channel) + @wolflistener("at_night_end") async def _at_night_end(self, data=None): if self.channel is None: return @@ -78,9 +78,6 @@ class VoteGroup: # Do what you voted on pass - async def _at_visit(self, data=None): - pass - async def register_players(self, *players): """ Extend players by passed list From 06af229a62f1a81c57f2c233bd96f3dfea57b3e9 Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 22 Sep 2020 16:23:26 -0400 Subject: [PATCH 10/73] non-relative imports --- werewolf/roles/shifter.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/werewolf/roles/shifter.py b/werewolf/roles/shifter.py index 85c3ec5..6e2a850 100644 --- a/werewolf/roles/shifter.py +++ b/werewolf/roles/shifter.py @@ -1,6 +1,6 @@ -from ..listener import wolflistener -from ..night_powers import pick_target -from ..role import Role +from werewolf.listener import wolflistener +from werewolf.night_powers import pick_target +from werewolf.role import Role class Shifter(Role): From a2eaf555159682478398228a465e5de3b34c86fb Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 22 Sep 2020 16:46:38 -0400 Subject: [PATCH 11/73] Priority update for listeners --- werewolf/game.py | 73 +++++++++++++++---------------- werewolf/listener.py | 33 ++++++++++---- werewolf/night_powers.py | 2 +- werewolf/roles/seer.py | 32 +++++++------- werewolf/roles/vanillawerewolf.py | 7 ++- werewolf/roles/villager.py | 2 +- werewolf/votegroups/__init__.py | 1 + werewolf/votegroups/wolfvote.py | 2 +- werewolf/werewolf.py | 16 ++++--- 9 files changed, 92 insertions(+), 76 deletions(-) create mode 100644 werewolf/votegroups/__init__.py diff --git a/werewolf/game.py b/werewolf/game.py index df5d263..98475ba 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -9,10 +9,10 @@ import discord from redbot.core import commands from redbot.core.bot import Red -from .builder import parse_code -from .player import Player -from .role import Role -from .votegroup import VoteGroup +from werewolf.builder import parse_code +from werewolf.player import Player +from werewolf.role import Role +from werewolf.votegroup import VoteGroup log = logging.getLogger("red.fox_v3.werewolf.game") @@ -480,7 +480,7 @@ class Game: async def _notify(self, event, **kwargs): for i in range(1, 7): # action guide 1-6 (0 is no action) tasks = [] - for event in self.listeners.get(event, []): + for event in self.listeners.get(event, {}).get(i, []): tasks.append(asyncio.ensure_future(event(**kwargs), loop=self.loop)) await asyncio.gather(*tasks) @@ -912,26 +912,19 @@ class Game: # Optional dynamic channels/categories - def add_listener(self, func, name=None): - """The non decorator alternative to :meth:`.listen`. + def add_ww_listener(self, func, priority=0, name=None): + """Adds a listener from the pool of listeners. Parameters ----------- func: :ref:`coroutine ` The function to call. + priority: Optional[:class:`int`] + Priority of the listener. Defaults to 0 (no-action) name: Optional[:class:`str`] The name of the event to listen for. Defaults to ``func.__name__``. - - Example - -------- - - .. code-block:: python3 - - async def on_ready(): pass - async def my_message(message): pass - - bot.add_listener(on_ready) - bot.add_listener(my_message, 'on_message') + do_sort: Optional[:class:`bool`] + Whether or not to sort listeners after. Skip sorting during mass appending """ name = func.__name__ if name is None else name @@ -940,26 +933,32 @@ class Game: raise TypeError('Listeners must be coroutines') if name in self.listeners: - self.listeners[name].append(func) + if priority in self.listeners[name]: + self.listeners[name][priority].append(func) + else: + self.listeners[name][priority] = [func] else: - self.listeners[name] = [func] - - def remove_listener(self, func, name=None): - """Removes a listener from the pool of listeners. + self.listeners[name] = {priority: [func]} - Parameters - ----------- - func - The function that was used as a listener to remove. - name: :class:`str` - The name of the event we want to remove. Defaults to - ``func.__name__``. - """ + # self.listeners[name].sort(reverse=True) - name = func.__name__ if name is None else name - if name in self.listeners: - try: - self.listeners[name].remove(func) - except ValueError: - pass + # def remove_wolf_listener(self, func, name=None): + # """Removes a listener from the pool of listeners. + # + # Parameters + # ----------- + # func + # The function that was used as a listener to remove. + # name: :class:`str` + # The name of the event we want to remove. Defaults to + # ``func.__name__``. + # """ + # + # name = func.__name__ if name is None else name + # + # if name in self.listeners: + # try: + # self.listeners[name].remove(func) + # except ValueError: + # pass diff --git a/werewolf/listener.py b/werewolf/listener.py index 9c36400..e14994a 100644 --- a/werewolf/listener.py +++ b/werewolf/listener.py @@ -1,7 +1,7 @@ import inspect -def wolflistener(name=None): +def wolflistener(name=None, priority=0): """A decorator that marks a function as a listener. This is the werewolf.Game equivalent of :meth:`.Cog.listener`. @@ -11,6 +11,22 @@ def wolflistener(name=None): name: :class:`str` The name of the event being listened to. If not provided, it defaults to the function's name. + priority: :class:`int` + The priority of the listener. + Priority guide as follows: + _at_night_start + 0. No Action + 1. Detain actions (Jailer/Kidnapper) + 2. Group discussions and choose targets + + _at_night_end + 0. No Action + 1. Self actions (Veteran) + 2. Target switching and role blocks (bus driver, witch, escort) + 3. Protection / Preempt actions (bodyguard/framer) + 4. Non-disruptive actions (seer/silencer) + 5. Disruptive actions (Killing) + 6. Role altering actions (Cult / Mason / Shifter) Raises -------- @@ -32,12 +48,12 @@ def wolflistener(name=None): actual = actual.__func__ if not inspect.iscoroutinefunction(actual): raise TypeError("Listener function must be a coroutine function.") - actual.__wolf_listener__ = True + actual.__wolf_listener__ = priority to_assign = name or actual.__name__ try: - actual.__wolf_listener_names__.append(to_assign) + actual.__wolf_listener_names__.append((priority, to_assign)) except AttributeError: - actual.__wolf_listener_names__ = [to_assign] + actual.__wolf_listener_names__ = [(priority, to_assign)] # we have to return `func` instead of `actual` because # we need the type to be `staticmethod` for the metaclass # to pick it up but the metaclass unfurls the function and @@ -51,7 +67,6 @@ class WolfListenerMeta(type): def __new__(mcs, cls, *args, **kwargs): name, bases = args - commands = {} listeners = {} need_at_msg = "Listeners must start with at_ (in method {0.__name__}.{1})" @@ -76,10 +91,10 @@ class WolfListenerMeta(type): listeners_as_list = [] for listener in listeners.values(): - for listener_name in listener.__wolf_listener_names__: + for priority, listener_name in listener.__wolf_listener_names__: # I use __name__ instead of just storing the value so I can inject # the self attribute when the time comes to add them to the bot - listeners_as_list.append((listener_name, listener.__name__)) + listeners_as_list.append((priority, listener_name, listener.__name__)) new_cls.__wolf_listeners__ = listeners_as_list return new_cls @@ -87,5 +102,5 @@ class WolfListenerMeta(type): class WolfListener(metaclass=WolfListenerMeta): def __init__(self, game): - for name, method_name in self.__wolf_listeners__: - game.add_listener(getattr(self, method_name), name) + for priority, name, method_name in self.__wolf_listeners__: + game.add_ww_listener(getattr(self, method_name), priority, name) diff --git a/werewolf/night_powers.py b/werewolf/night_powers.py index a39ad26..ab82e87 100644 --- a/werewolf/night_powers.py +++ b/werewolf/night_powers.py @@ -1,6 +1,6 @@ import logging -from .role import Role +from werewolf.role import Role log = logging.getLogger("red.fox_v3.werewolf.night_powers") diff --git a/werewolf/roles/seer.py b/werewolf/roles/seer.py index 603d197..a7a9aa8 100644 --- a/werewolf/roles/seer.py +++ b/werewolf/roles/seer.py @@ -1,6 +1,6 @@ -from ..listener import wolflistener -from ..night_powers import pick_target -from ..role import Role +from werewolf.listener import wolflistener +from werewolf.night_powers import pick_target +from werewolf.role import Role class Seer(Role): @@ -27,17 +27,17 @@ class Seer(Role): # self.blocked = False # self.properties = {} # Extra data for other roles (i.e. arsonist) self.see_target = None - self.action_list = [ - (self._at_game_start, 1), # (Action, Priority) - (self._at_day_start, 0), - (self._at_voted, 0), - (self._at_kill, 0), - (self._at_hang, 0), - (self._at_day_end, 0), - (self._at_night_start, 2), - (self._at_night_end, 4), - (self._at_visit, 0), - ] + # self.action_list = [ + # (self._at_game_start, 1), # (Action, Priority) + # (self._at_day_start, 0), + # (self._at_voted, 0), + # (self._at_kill, 0), + # (self._at_hang, 0), + # (self._at_day_end, 0), + # (self._at_night_start, 2), + # (self._at_night_end, 4), + # (self._at_visit, 0), + # ] async def see_alignment(self, source=None): """ @@ -60,7 +60,7 @@ class Seer(Role): """ return "Villager" - @wolflistener("at_night_start") + @wolflistener("at_night_start", priority=2) async def _at_night_start(self, data=None): if not self.player.alive: return @@ -68,7 +68,7 @@ class Seer(Role): await self.game.generate_targets(self.player.member) await self.player.send_dm("**Pick a target to see tonight**") - @wolflistener("at_night_end") + @wolflistener("at_night_end", priority=4) async def _at_night_end(self, data=None): if self.see_target is None: if self.player.alive: diff --git a/werewolf/roles/vanillawerewolf.py b/werewolf/roles/vanillawerewolf.py index 5f7407b..e6938eb 100644 --- a/werewolf/roles/vanillawerewolf.py +++ b/werewolf/roles/vanillawerewolf.py @@ -1,7 +1,6 @@ -from ..listener import wolflistener -from ..role import Role - -from ..votegroups.wolfvote import WolfVote +from werewolf.listener import wolflistener +from werewolf.role import Role +from werewolf.votegroups.wolfvote import WolfVote class VanillaWerewolf(Role): diff --git a/werewolf/roles/villager.py b/werewolf/roles/villager.py index bda51d2..040e34d 100644 --- a/werewolf/roles/villager.py +++ b/werewolf/roles/villager.py @@ -1,4 +1,4 @@ -from ..role import Role +from werewolf.role import Role class Villager(Role): diff --git a/werewolf/votegroups/__init__.py b/werewolf/votegroups/__init__.py new file mode 100644 index 0000000..03abc1b --- /dev/null +++ b/werewolf/votegroups/__init__.py @@ -0,0 +1 @@ +from .wolfvote import WolfVote \ No newline at end of file diff --git a/werewolf/votegroups/wolfvote.py b/werewolf/votegroups/wolfvote.py index fb98b20..f990eef 100644 --- a/werewolf/votegroups/wolfvote.py +++ b/werewolf/votegroups/wolfvote.py @@ -1,6 +1,6 @@ import random -from ..votegroup import VoteGroup +from werewolf.votegroup import VoteGroup class WolfVote(VoteGroup): diff --git a/werewolf/werewolf.py b/werewolf/werewolf.py index 0c8374c..abed258 100644 --- a/werewolf/werewolf.py +++ b/werewolf/werewolf.py @@ -6,14 +6,14 @@ from redbot.core.bot import Red from redbot.core.commands import Cog from redbot.core.utils.menus import DEFAULT_CONTROLS, menu -from .builder import ( +from werewolf.builder import ( GameBuilder, role_from_alignment, role_from_category, role_from_id, role_from_name, ) -from .game import Game +from werewolf.game import Game log = logging.getLogger("red.fox_v3.werewolf") @@ -81,8 +81,8 @@ class Werewolf(Cog): """ Lists current guild settings """ - success, role, category, channel, log_channel = await self._get_settings(ctx) - if not success: + valid, role, category, channel, log_channel = await self._get_settings(ctx) + if not valid: await ctx.send("Failed to get settings") return None @@ -362,13 +362,15 @@ class Werewolf(Cog): return None if guild.id not in self.games or self.games[guild.id].game_over: await ctx.send("Starting a new game...") - success, role, category, channel, log_channel = await self._get_settings(ctx) + valid, role, category, channel, log_channel = await self._get_settings(ctx) - if not success: + if not valid: await ctx.send("Cannot start a new game") return None - self.games[guild.id] = Game(self.bot, guild, role, category, channel, log_channel, game_code) + self.games[guild.id] = Game( + self.bot, guild, role, category, channel, log_channel, game_code + ) return self.games[guild.id] From 7109471c35194f284ea8f58a107b6ed89c05a37c Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 22 Sep 2020 17:30:58 -0400 Subject: [PATCH 12/73] WIP listeners, switch to f strings, and overall rewrite --- werewolf/builder.py | 16 +++---- werewolf/game.py | 73 ++++++++++++++++--------------- werewolf/listener.py | 10 ++--- werewolf/player.py | 2 +- werewolf/role.py | 2 +- werewolf/roles/seer.py | 8 ++-- werewolf/roles/shifter.py | 8 ++-- werewolf/roles/vanillawerewolf.py | 8 +++- werewolf/roles/villager.py | 4 ++ werewolf/votegroups/wolfvote.py | 7 ++- werewolf/werewolf.py | 2 +- 11 files changed, 80 insertions(+), 60 deletions(-) diff --git a/werewolf/builder.py b/werewolf/builder.py index 4632a21..ca90eca 100644 --- a/werewolf/builder.py +++ b/werewolf/builder.py @@ -49,7 +49,7 @@ CATEGORY_COUNT = [] def role_embed(idx, role, color): embed = discord.Embed( - title="**{}** - {}".format(idx, str(role.__name__)), + title=f"**{idx}** - {role.__name__}", description=role.game_start_message, color=color, ) @@ -82,7 +82,7 @@ def setup(): if 0 < k <= 6: ROLE_PAGES.append( discord.Embed( - title="RANDOM:Town Role", description="Town {}".format(v), color=0x008000 + title="RANDOM:Town Role", description=f"Town {v}", color=0x008000 ) ) CATEGORY_COUNT.append(k) @@ -95,7 +95,7 @@ def setup(): ROLE_PAGES.append( discord.Embed( title="RANDOM:Werewolf Role", - description="Werewolf {}".format(v), + description=f"Werewolf {v}", color=0xFF0000, ) ) @@ -107,7 +107,7 @@ def setup(): if 20 < k <= 26: ROLE_PAGES.append( discord.Embed( - title="RANDOM:Neutral Role", description="Neutral {}".format(v), color=0xC0C0C0 + title=f"RANDOM:Neutral Role", description="Neutral {v}", color=0xC0C0C0 ) ) CATEGORY_COUNT.append(k) @@ -306,14 +306,14 @@ def say_role_list(code_list, rand_roles): for role in rand_roles: if 0 < role <= 6: - role_dict["Town {}".format(ROLE_CATEGORIES[role])] += 1 + role_dict[f"Town {ROLE_CATEGORIES[role]}"] += 1 if 10 < role <= 16: - role_dict["Werewolf {}".format(ROLE_CATEGORIES[role])] += 1 + role_dict[f"Werewolf {ROLE_CATEGORIES[role]}"] += 1 if 20 < role <= 26: - role_dict["Neutral {}".format(ROLE_CATEGORIES[role])] += 1 + role_dict[f"Neutral {ROLE_CATEGORIES[role]}"] += 1 for k, v in role_dict.items(): - embed.add_field(name=k, value="Count: {}".format(v), inline=True) + embed.add_field(name=k, value=f"Count: {v}", inline=True) return embed diff --git a/werewolf/game.py b/werewolf/game.py index 98475ba..7372923 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -18,6 +18,7 @@ log = logging.getLogger("red.fox_v3.werewolf.game") HALF_DAY_LENGTH = 24 # FixMe: to 120 later for 4 minute days + class Game: """ Base class to run a single game of Werewolf @@ -119,10 +120,10 @@ class Game: if len(self.players) != len(self.roles): await ctx.maybe_send_embed( - "Player count does not match role count, cannot start\n" - "Currently **{} / {}**\n" - "Use `{}ww code` to pick a new game" - "".format(len(self.players), len(self.roles), ctx.prefix) + f"Player count does not match role count, cannot start\n" + f"Currently **{len(self.players)} / {len(self.roles)}**\n" + f"Use `{ctx.prefix}ww code` to pick a game setup\n" + f"Use `{ctx.prefix}buildgame` to generate a new game" ) self.roles = [] return False @@ -147,9 +148,7 @@ class Game: await player.member.add_roles(*[self.game_role]) except discord.Forbidden: await ctx.send( - "Unable to add role **{}**\nBot is missing `manage_roles` permissions".format( - self.game_role.name - ) + f"Unable to add role **{self.game_role.name}**\nBot is missing `manage_roles` permissions" ) return False @@ -353,9 +352,7 @@ class Game: await self.speech_perms(self.village_channel, target.member) # Only target can talk await self.village_channel.send( - "**{} will be put to trial and has 30 seconds to defend themselves**".format( - target.mention - ), + f"**{target.mention} will be put to trial and has 30 seconds to defend themselves**", allowed_mentions=discord.AllowedMentions(everyone=False, users=[target]), ) @@ -364,10 +361,10 @@ class Game: await self.speech_perms(self.village_channel, target.member, undo=True) # No one can talk message: discord.Message = await self.village_channel.send( - "Everyone will now vote whether to lynch {}\n" + f"Everyone will now vote whether to lynch {target.mention}\n" "👍 to save, 👎 to lynch\n" "*Majority rules, no-lynch on ties, " - "vote both or neither to abstain, 15 seconds to vote*".format(target.mention), + "vote both or neither to abstain, 15 seconds to vote*", allowed_mentions=discord.AllowedMentions(everyone=False, users=[target]), ) @@ -385,25 +382,31 @@ class Game: else: embed = discord.Embed(title="Vote Results", color=0x80FF80) - embed.add_field(name="👎", value="**{}**".format(up_votes), inline=True) - embed.add_field(name="👍", value="**{}**".format(down_votes), inline=True) + embed.add_field(name="👎", value=f"**{up_votes}**", inline=True) + embed.add_field(name="👍", value=f"**{down_votes}**", inline=True) await self.village_channel.send(embed=embed) if down_votes > up_votes: - await self.village_channel.send("**Voted to lynch {}!**".format(target.mention)) + await self.village_channel.send( + f"**Voted to lynch {target.mention}!**", + allowed_mentions=discord.AllowedMentions(everyone=False, users=[target]), + ) await self.lynch(target) self.can_vote = False else: - await self.village_channel.send("**{} has been spared!**".format(target.mention)) + await self.village_channel.send( + f"**{target.mention} has been spared!**", + allowed_mentions=discord.AllowedMentions(everyone=False, users=[target]), + ) if self.used_votes >= self.day_vote_count: await self.village_channel.send("**All votes have been used! Day is now over!**") self.can_vote = False else: await self.village_channel.send( - "**{}**/**{}** of today's votes have been used!\n" - "Nominate carefully..".format(self.used_votes, self.day_vote_count) + f"**{self.used_votes}**/**{self.day_vote_count}** of today's votes have been used!\n" + "Nominate carefully.." ) self.ongoing_vote = False @@ -513,14 +516,14 @@ class Game: status = "*[Dead]*-" if with_roles or not player.alive: embed.add_field( - name="ID# **{}**".format(i), - value="{}{}-{}".format(status, player.member.display_name, str(player.role)), + name=f"ID# **{i}**", + value=f"{status}{player.member.display_name}-{player.role}", inline=True, ) else: embed.add_field( - name="ID# **{}**".format(i), - value="{}{}".format(status, player.member.display_name), + name=f"ID# **{i}**", + value=f"{status}{player.member.display_name}", inline=True, ) @@ -553,7 +556,7 @@ class Game: return if await self.get_player_by_member(member) is not None: - await channel.send("{} is already in the game!".format(member.mention)) + await channel.send(f"{member.display_name} is already in the game!") return self.players.append(Player(member)) @@ -563,14 +566,12 @@ class Game: await member.add_roles(*[self.game_role]) except discord.Forbidden: await channel.send( - "Unable to add role **{}**\nBot is missing `manage_roles` permissions".format( - self.game_role.name - ) + f"Unable to add role **{self.game_role.name}**\nBot is missing `manage_roles` permissions" ) await channel.send( - "{} has been added to the game, " - "total players is **{}**".format(member.mention, len(self.players)) + f"{member.display_name} has been added to the game, " + f"total players is **{len(self.players)}**" ) async def quit(self, member: discord.Member, channel: discord.TextChannel = None): @@ -584,14 +585,16 @@ class Game: if self.started: await self._quit(player) - await channel.send("{} has left the game".format(member.mention)) + await channel.send( + f"{member.mention} has left the game", + allowed_mentions=discord.AllowedMentions(everyone=False, users=[member]), + ) else: self.players = [player for player in self.players if player.member != member] await member.remove_roles(*[self.game_role]) await channel.send( - "{} chickened out, player count is now **{}**".format( - member.mention, len(self.players) - ) + f"{member.mention} chickened out, player count is now **{len(self.players)}**", + allowed_mentions=discord.AllowedMentions(everyone=False, users=[member]), ) async def choose(self, ctx, data): @@ -698,7 +701,8 @@ class Game: author.mention, target.member.mention, required_votes - self.vote_totals[target_id], - ) + ), + allowed_mentions=discord.AllowedMentions(everyone=False, users=[author, target]), ) else: self.vote_totals[target_id] = 0 @@ -930,7 +934,7 @@ class Game: name = func.__name__ if name is None else name if not asyncio.iscoroutinefunction(func): - raise TypeError('Listeners must be coroutines') + raise TypeError("Listeners must be coroutines") if name in self.listeners: if priority in self.listeners[name]: @@ -942,7 +946,6 @@ class Game: # self.listeners[name].sort(reverse=True) - # def remove_wolf_listener(self, func, name=None): # """Removes a listener from the pool of listeners. # diff --git a/werewolf/listener.py b/werewolf/listener.py index e14994a..29ef7dd 100644 --- a/werewolf/listener.py +++ b/werewolf/listener.py @@ -64,13 +64,13 @@ def wolflistener(name=None, priority=0): class WolfListenerMeta(type): - def __new__(mcs, cls, *args, **kwargs): - name, bases = args + def __new__(mcs, *args, **kwargs): + name, bases, attrs = args listeners = {} need_at_msg = "Listeners must start with at_ (in method {0.__name__}.{1})" - new_cls = super().__new__(cls, name, bases, **kwargs) + new_cls = super().__new__(mcs, name, bases, attrs, **kwargs) for base in reversed(new_cls.__mro__): for elem, value in base.__dict__.items(): if elem in listeners: @@ -85,8 +85,8 @@ class WolfListenerMeta(type): except AttributeError: continue else: - if not elem.startswith("at_"): - raise TypeError(need_at_msg.format(mcs, elem)) + # if not elem.startswith("at_"): + # raise TypeError(need_at_msg.format(base, elem)) listeners[elem] = value listeners_as_list = [] diff --git a/werewolf/player.py b/werewolf/player.py index 8ee437e..201b781 100644 --- a/werewolf/player.py +++ b/werewolf/player.py @@ -34,4 +34,4 @@ class Player: try: await self.member.send(message) # Lets do embeds later except discord.Forbidden: - await self.role.game.village_channel.send("Couldn't DM {}, uh oh".format(self.mention)) + await self.role.game.village_channel.send(f"Couldn't DM {self.mention}, uh oh", allowed_mentions=discord.AllowedMentions(users=[self.member])) diff --git a/werewolf/role.py b/werewolf/role.py index 4ae10ad..0693f46 100644 --- a/werewolf/role.py +++ b/werewolf/role.py @@ -128,7 +128,7 @@ class Role(WolfListener): """ return "Default" - @wolflistener("at_game_start") + @wolflistener("at_game_start", priority=1) async def _at_game_start(self, data=None): if self.channel_id: await self.game.register_channel(self.channel_id, self) diff --git a/werewolf/roles/seer.py b/werewolf/roles/seer.py index a7a9aa8..f01c5c2 100644 --- a/werewolf/roles/seer.py +++ b/werewolf/roles/seer.py @@ -1,7 +1,11 @@ +import logging + from werewolf.listener import wolflistener from werewolf.night_powers import pick_target from werewolf.role import Role +log = logging.getLogger("red.fox_v3.werewolf.role.seer") + class Seer(Role): rand_choice = True # Determines if it can be picked as a random role (False for unusually disruptive roles) @@ -93,7 +97,5 @@ class Seer(Role): self.see_target, target = await pick_target(self, ctx, data) await ctx.send( - "**You will attempt to see the role of {} tonight...**".format( - target.member.display_name - ) + f"**You will attempt to see the role of {target.member.display_name} tonight...**" ) diff --git a/werewolf/roles/shifter.py b/werewolf/roles/shifter.py index 6e2a850..8f93d76 100644 --- a/werewolf/roles/shifter.py +++ b/werewolf/roles/shifter.py @@ -1,7 +1,11 @@ +import logging + from werewolf.listener import wolflistener from werewolf.night_powers import pick_target from werewolf.role import Role +log = logging.getLogger("red.fox_v3.werewolf.role.shifter") + class Shifter(Role): """ @@ -130,7 +134,5 @@ class Shifter(Role): self.shift_target, target = await pick_target(self, ctx, data) await ctx.send( - "**You will attempt to see the role of {} tonight...**".format( - target.member.display_name - ) + f"**You will attempt to see the role of {target.member.display_name} tonight...**" ) diff --git a/werewolf/roles/vanillawerewolf.py b/werewolf/roles/vanillawerewolf.py index e6938eb..db70eb5 100644 --- a/werewolf/roles/vanillawerewolf.py +++ b/werewolf/roles/vanillawerewolf.py @@ -1,7 +1,11 @@ +import logging + from werewolf.listener import wolflistener from werewolf.role import Role from werewolf.votegroups.wolfvote import WolfVote +log = logging.getLogger("red.fox_v3.werewolf.role.vanillawerewolf") + class VanillaWerewolf(Role): rand_choice = True @@ -56,7 +60,9 @@ class VanillaWerewolf(Role): async def _at_game_start(self, data=None): if self.channel_id: print("Wolf has channel_id: " + self.channel_id) - await self.game.register_channel(self.channel_id, self, WolfVote) # Add VoteGroup WolfVote + await self.game.register_channel( + self.channel_id, self, WolfVote + ) # Add VoteGroup WolfVote await self.player.send_dm(self.game_start_message) diff --git a/werewolf/roles/villager.py b/werewolf/roles/villager.py index 040e34d..f225e0d 100644 --- a/werewolf/roles/villager.py +++ b/werewolf/roles/villager.py @@ -1,5 +1,9 @@ +import logging + from werewolf.role import Role +log = logging.getLogger("red.fox_v3.werewolf.role.villager") + class Villager(Role): rand_choice = True # Determines if it can be picked as a random role (False for unusually disruptive roles) diff --git a/werewolf/votegroups/wolfvote.py b/werewolf/votegroups/wolfvote.py index f990eef..0823d77 100644 --- a/werewolf/votegroups/wolfvote.py +++ b/werewolf/votegroups/wolfvote.py @@ -1,7 +1,10 @@ +import logging import random from werewolf.votegroup import VoteGroup +log = logging.getLogger("red.fox_v3.werewolf.votegroup.wolfvote") + class WolfVote(VoteGroup): """ @@ -77,7 +80,7 @@ class WolfVote(VoteGroup): self.killer = random.choice(self.players) await self.channel.send( - "{} has been selected as tonight's killer".format(self.killer.member.display_name) + f"{self.killer.member.display_name} has been selected as tonight's killer" ) async def _at_night_end(self, data=None): @@ -90,7 +93,7 @@ class WolfVote(VoteGroup): if vote_list: target_id = max(set(vote_list), key=vote_list.count) - print("Target id: {}\nKiller: {}".format(target_id, self.killer.member.display_name)) + log.debug("Target id: {target_id}\nKiller: {self.killer.member.display_name}") if target_id is not None and self.killer: await self.game.kill(target_id, self.killer, random.choice(self.kill_messages)) await self.channel.send( diff --git a/werewolf/werewolf.py b/werewolf/werewolf.py index abed258..70574d2 100644 --- a/werewolf/werewolf.py +++ b/werewolf/werewolf.py @@ -62,7 +62,7 @@ class Werewolf(Cog): code = await gb.build_game(ctx) if code != "": - await ctx.send("Your game code is **{}**".format(code)) + await ctx.send(f"Your game code is **{code}**") else: await ctx.send("No code generated") From 29aa4930333d96f97049e88fb3b6eb1820273448 Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 22 Sep 2020 17:50:05 -0400 Subject: [PATCH 13/73] Better checking of valid settings --- werewolf/werewolf.py | 46 +++++++++++++++++++++++++++----------------- 1 file changed, 28 insertions(+), 18 deletions(-) diff --git a/werewolf/werewolf.py b/werewolf/werewolf.py index 70574d2..677e5c0 100644 --- a/werewolf/werewolf.py +++ b/werewolf/werewolf.py @@ -82,15 +82,18 @@ class Werewolf(Cog): Lists current guild settings """ valid, role, category, channel, log_channel = await self._get_settings(ctx) - if not valid: - await ctx.send("Failed to get settings") - return None + # if not valid: + # await ctx.send("Failed to get settings") + # return None - embed = discord.Embed(title="Current Guild Settings") + embed = discord.Embed( + title="Current Guild Settings", description=f"Valid: {valid}", color=0xFF0000 + ) embed.add_field(name="Role", value=str(role)) embed.add_field(name="Category", value=str(category)) embed.add_field(name="Channel", value=str(channel)) embed.add_field(name="Log Channel", value=str(log_channel)) + await ctx.send(embed=embed) @commands.guild_only() @@ -391,23 +394,30 @@ class Werewolf(Cog): if role_id is not None: role = discord.utils.get(guild.roles, id=role_id) - if role is None: - await ctx.send("Game Role is invalid") - return False, None, None, None, None + # if role is None: + # # await ctx.send("Game Role is invalid") + # return False, None, None, None, None if category_id is not None: category = discord.utils.get(guild.categories, id=category_id) - if category is None: - await ctx.send("Game Category is invalid") - return False, None, None, None, None + # if category is None: + # # await ctx.send("Game Category is invalid") + # return False, role, None, None, None if channel_id is not None: channel = discord.utils.get(guild.text_channels, id=channel_id) - if channel is None: - await ctx.send("Village Channel is invalid") - return False, None, None, None, None + # if channel is None: + # # await ctx.send("Village Channel is invalid") + # return False, role, category, None, None + if log_channel_id is not None: log_channel = discord.utils.get(guild.text_channels, id=log_channel_id) - if log_channel is None: - await ctx.send("Log Channel is invalid") - return False, None, None, None, None - - return True, role, category, channel, log_channel + # if log_channel is None: + # # await ctx.send("Log Channel is invalid") + # return False, None, None, None, None + + return ( + role is not None and category is not None and channel is not None, + role, + category, + channel, + log_channel, + ) From 98ae481d14e5851201ec740da530c69f185fe92a Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 22 Sep 2020 17:53:54 -0400 Subject: [PATCH 14/73] Show valid settings correctly --- werewolf/werewolf.py | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/werewolf/werewolf.py b/werewolf/werewolf.py index 677e5c0..a011ccb 100644 --- a/werewolf/werewolf.py +++ b/werewolf/werewolf.py @@ -87,7 +87,9 @@ class Werewolf(Cog): # return None embed = discord.Embed( - title="Current Guild Settings", description=f"Valid: {valid}", color=0xFF0000 + title="Current Guild Settings", + description=f"Valid: {valid}", + color=0x008000 if valid else 0xFF0000, ) embed.add_field(name="Role", value=str(role)) embed.add_field(name="Category", value=str(category)) From 84ed2728e7ce29f3e97282b43f8af0b8776a742e Mon Sep 17 00:00:00 2001 From: bobloy Date: Thu, 24 Sep 2020 13:36:42 -0400 Subject: [PATCH 15/73] switch to log --- werewolf/game.py | 16 ++++++++-------- werewolf/role.py | 2 +- werewolf/roles/vanillawerewolf.py | 4 ++-- werewolf/werewolf.py | 4 ++-- 4 files changed, 13 insertions(+), 13 deletions(-) diff --git a/werewolf/game.py b/werewolf/game.py index 7372923..45f0cbd 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -203,8 +203,7 @@ class Game: reason="(BOT) New game of werewolf", ) except discord.Forbidden as e: - print("Unable to rename Game Channel") - print(e) + log.exception("Unable to rename Game Channel") await ctx.send("Unable to rename Game Channel, ignoring") try: @@ -222,11 +221,11 @@ class Game: return self.started = True # Assuming everything worked so far - print("Pre at_game_start") + log.debug("Pre at_game_start") await self._at_game_start() # This will queue channels and votegroups to be made - print("Post at_game_start") + log.debug("Post at_game_start") for channel_id in self.p_channels: - print("Channel id: " + channel_id) + log.debug("Setup Channel id: " + channel_id) overwrite = { self.guild.default_role: discord.PermissionOverwrite(read_messages=False), self.guild.me: discord.PermissionOverwrite( @@ -258,7 +257,7 @@ class Game: self.vote_groups[channel_id] = vote_group - print("Pre-cycle") + log.debug("Pre-cycle") await asyncio.sleep(1) await asyncio.ensure_future(self._cycle()) # Start the loop @@ -566,7 +565,8 @@ class Game: await member.add_roles(*[self.game_role]) except discord.Forbidden: await channel.send( - f"Unable to add role **{self.game_role.name}**\nBot is missing `manage_roles` permissions" + f"Unable to add role **{self.game_role.name}**\n" + f"Bot is missing `manage_roles` permissions" ) await channel.send( @@ -899,7 +899,7 @@ class Game: # Remove game_role access for potential archiving for now reason = "(BOT) End of WW game" for obj in self.to_delete: - print(obj) + log.debug(f"End_game: Deleting object {obj}") await obj.delete(reason=reason) try: diff --git a/werewolf/role.py b/werewolf/role.py index 0693f46..ccc20ae 100644 --- a/werewolf/role.py +++ b/werewolf/role.py @@ -129,7 +129,7 @@ class Role(WolfListener): return "Default" @wolflistener("at_game_start", priority=1) - async def _at_game_start(self, data=None): + async def _at_game_start(self): if self.channel_id: await self.game.register_channel(self.channel_id, self) diff --git a/werewolf/roles/vanillawerewolf.py b/werewolf/roles/vanillawerewolf.py index db70eb5..f6eea81 100644 --- a/werewolf/roles/vanillawerewolf.py +++ b/werewolf/roles/vanillawerewolf.py @@ -57,9 +57,9 @@ class VanillaWerewolf(Role): return "Werewolf" @wolflistener("at_game_start") - async def _at_game_start(self, data=None): + async def _at_game_start(self): if self.channel_id: - print("Wolf has channel_id: " + self.channel_id) + log.debug("Wolf has channel_id: " + self.channel_id) await self.game.register_channel( self.channel_id, self, WolfVote ) # Add VoteGroup WolfVote diff --git a/werewolf/werewolf.py b/werewolf/werewolf.py index a011ccb..742a890 100644 --- a/werewolf/werewolf.py +++ b/werewolf/werewolf.py @@ -47,7 +47,7 @@ class Werewolf(Cog): return def __unload(self): - print("Unload called") + log.debug("Unload called") for game in self.games.values(): del game @@ -182,7 +182,7 @@ class Werewolf(Cog): Joins a game of Werewolf """ - game = await self._get_game(ctx) + game: Game = await self._get_game(ctx) if not game: await ctx.send("No game to join!\nCreate a new one with `[p]ww new`") From 8ffc8cc707b7c3fd666949b4b94584a36b95cda5 Mon Sep 17 00:00:00 2001 From: bobloy Date: Thu, 24 Sep 2020 17:03:39 -0400 Subject: [PATCH 16/73] Missed the listener update --- werewolf/votegroups/wolfvote.py | 79 +++++---------------------------- 1 file changed, 11 insertions(+), 68 deletions(-) diff --git a/werewolf/votegroups/wolfvote.py b/werewolf/votegroups/wolfvote.py index 0823d77..7f6bbde 100644 --- a/werewolf/votegroups/wolfvote.py +++ b/werewolf/votegroups/wolfvote.py @@ -1,6 +1,9 @@ import logging import random +import discord + +from werewolf.listener import wolflistener from werewolf.votegroup import VoteGroup log = logging.getLogger("red.fox_v3.werewolf.votegroup.wolfvote") @@ -21,59 +24,13 @@ class WolfVote(VoteGroup): def __init__(self, game, channel): super().__init__(game, channel) - # self.game = game - # self.channel = channel - # self.players = [] - # self.vote_results = {} - # self.properties = {} # Extra data for other options self.killer = None # Added killer - self.action_list = [ - (self._at_game_start, 1), # (Action, Priority) - (self._at_day_start, 0), - (self._at_voted, 0), - (self._at_kill, 1), - (self._at_hang, 1), - (self._at_day_end, 0), - (self._at_night_start, 2), - (self._at_night_end, 5), # Kill priority - (self._at_visit, 0), - ] - - # async def on_event(self, event, data): - - # """ - # See Game class for event guide - # """ - # - # await action_list[event][0](data) - # - # async def _at_game_start(self, data=None): - # await self.channel.send(" ".join(player.mention for player in self.players)) - # - # async def _at_day_start(self, data=None): - # pass - # - # async def _at_voted(self, data=None): - # pass - # - # async def _at_kill(self, data=None): - # if data["player"] in self.players: - # self.players.pop(data["player"]) - # - # async def _at_hang(self, data=None): - # if data["player"] in self.players: - # self.players.pop(data["player"]) - # - # async def _at_day_end(self, data=None): - # pass - - async def _at_night_start(self, data=None): - if self.channel is None: - return + @wolflistener("at_night_start", priority=2) + async def _at_night_start(self): + await super()._at_night_start() - await self.game.generate_targets(self.channel) mention_list = " ".join(player.mention for player in self.players) if mention_list != "": await self.channel.send(mention_list) @@ -83,7 +40,8 @@ class WolfVote(VoteGroup): f"{self.killer.member.display_name} has been selected as tonight's killer" ) - async def _at_night_end(self, data=None): + @wolflistener("at_night_end", priority=5) + async def _at_night_end(self): if self.channel is None: return @@ -102,29 +60,14 @@ class WolfVote(VoteGroup): else: await self.channel.send("**No kill will be attempted tonight...**") - # async def _at_visit(self, data=None): - # pass - # - # async def register_players(self, *players): - # """ - # Extend players by passed list - # """ - # self.players.extend(players) - # - # async def remove_player(self, player): - # """ - # Remove a player from player list - # """ - # if player.id in self.players: - # self.players.remove(player) - async def vote(self, target, author, target_id): """ Receive vote from game """ - self.vote_results[author.id] = target_id + await super().vote(target, author, target_id) await self.channel.send( - "{} has voted to kill {}".format(author.mention, target.member.display_name) + "{} has voted to kill {}".format(author.mention, target.member.display_name), + allowed_mentions=discord.AllowedMentions(everyone=False, users=[author]) ) From eaa3e0a2f792d1f4dba535f60c4aa9c7475ea5f7 Mon Sep 17 00:00:00 2001 From: bobloy Date: Thu, 24 Sep 2020 17:04:11 -0400 Subject: [PATCH 17/73] Fix listener parameters and priority --- werewolf/votegroup.py | 57 +++++++++++++++---------------------------- 1 file changed, 20 insertions(+), 37 deletions(-) diff --git a/werewolf/votegroup.py b/werewolf/votegroup.py index 19ebd9e..2f0b3a0 100644 --- a/werewolf/votegroup.py +++ b/werewolf/votegroup.py @@ -22,49 +22,31 @@ class VoteGroup(WolfListener): self.vote_results = {} self.properties = {} # Extra data for other options - # self.action_list = [ - # (self._at_game_start, 1), # (Action, Priority) - # (self._at_day_start, 0), - # (self._at_voted, 0), - # (self._at_kill, 1), - # (self._at_hang, 1), - # (self._at_day_end, 0), - # (self._at_night_start, 2), - # (self._at_night_end, 0), - # (self._at_visit, 0), - # ] - - # async def on_event(self, event, data): - # """ - # See Game class for event guide - # """ - # - # await self.action_list[event][0](data) - - @wolflistener("at_game_start") - async def _at_game_start(self, data=None): + @wolflistener("at_game_start", priority=1) + async def _at_game_start(self): await self.channel.send(" ".join(player.mention for player in self.players)) - @wolflistener("at_kill") - async def _at_kill(self, data=None): - if data["player"] in self.players: - self.players.remove(data["player"]) + @wolflistener("at_kill", priority=1) + async def _at_kill(self, player): + if player in self.players: + self.players.remove(player) - # Removed, only if they actually die - # @wolflistener("at_hang") - # async def _at_hang(self, data=None): - # if data["player"] in self.players: - # self.players.remove(data["player"]) + @wolflistener("at_hang", priority=1) + async def _at_hang(self, player): + if player in self.players: + self.players.remove(player) - @wolflistener("at_night_start") - async def _at_night_start(self, data=None): + @wolflistener("at_night_start", priority=2) + async def _at_night_start(self): if self.channel is None: return + self.vote_results = {} + await self.game.generate_targets(self.channel) - @wolflistener("at_night_end") - async def _at_night_end(self, data=None): + @wolflistener("at_night_end", priority=5) + async def _at_night_end(self): if self.channel is None: return @@ -75,8 +57,8 @@ class VoteGroup(WolfListener): target = max(set(vote_list), key=vote_list.count) if target: - # Do what you voted on - pass + # Do what the votegroup votes on + raise NotImplementedError async def register_players(self, *players): """ @@ -92,7 +74,8 @@ class VoteGroup(WolfListener): self.players.remove(player) if not self.players: - # ToDo: Trigger deletion of votegroup + # TODO: Confirm deletion + self.game.to_delete.add(self) pass async def vote(self, target, author, target_id): From 39801aada912b3ac4bf61226f7207f8518dbdac2 Mon Sep 17 00:00:00 2001 From: bobloy Date: Thu, 24 Sep 2020 17:04:25 -0400 Subject: [PATCH 18/73] Missed priority --- werewolf/roles/vanillawerewolf.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/werewolf/roles/vanillawerewolf.py b/werewolf/roles/vanillawerewolf.py index f6eea81..58b474e 100644 --- a/werewolf/roles/vanillawerewolf.py +++ b/werewolf/roles/vanillawerewolf.py @@ -56,7 +56,7 @@ class VanillaWerewolf(Role): """ return "Werewolf" - @wolflistener("at_game_start") + @wolflistener("at_game_start", priority=1) async def _at_game_start(self): if self.channel_id: log.debug("Wolf has channel_id: " + self.channel_id) From cb0a7f1041a76202a1622d61c7af77a592b667d7 Mon Sep 17 00:00:00 2001 From: bobloy Date: Thu, 24 Sep 2020 17:04:49 -0400 Subject: [PATCH 19/73] Add priority and parameters --- werewolf/roles/seer.py | 4 ++-- werewolf/roles/shifter.py | 8 ++++---- 2 files changed, 6 insertions(+), 6 deletions(-) diff --git a/werewolf/roles/seer.py b/werewolf/roles/seer.py index f01c5c2..56624c9 100644 --- a/werewolf/roles/seer.py +++ b/werewolf/roles/seer.py @@ -65,7 +65,7 @@ class Seer(Role): return "Villager" @wolflistener("at_night_start", priority=2) - async def _at_night_start(self, data=None): + async def _at_night_start(self): if not self.player.alive: return self.see_target = None @@ -73,7 +73,7 @@ class Seer(Role): await self.player.send_dm("**Pick a target to see tonight**") @wolflistener("at_night_end", priority=4) - async def _at_night_end(self, data=None): + async def _at_night_end(self): if self.see_target is None: if self.player.alive: await self.player.send_dm("You will not use your powers tonight...") diff --git a/werewolf/roles/shifter.py b/werewolf/roles/shifter.py index 8f93d76..a7ea058 100644 --- a/werewolf/roles/shifter.py +++ b/werewolf/roles/shifter.py @@ -99,14 +99,14 @@ class Shifter(Role): """ return "Shifter" - @wolflistener("at_night_start") - async def _at_night_start(self, data=None): + @wolflistener("at_night_start", priority=2) + async def _at_night_start(self): self.shift_target = None await self.game.generate_targets(self.player.member) await self.player.send_dm("**Pick a target to shift into**") - @wolflistener("at_night_end") - async def _at_night_end(self, data=None): + @wolflistener("at_night_end", priority=6) + async def _at_night_end(self): if self.shift_target is None: if self.player.alive: await self.player.send_dm("You will not use your powers tonight...") From 61d131341154401c659e5f717b5b4af8a201e3fd Mon Sep 17 00:00:00 2001 From: bobloy Date: Thu, 24 Sep 2020 17:05:52 -0400 Subject: [PATCH 20/73] Switch game to handle daytime smoother allowing cancellation --- werewolf/game.py | 121 +++++++++++++++++++++++++---------------------- 1 file changed, 65 insertions(+), 56 deletions(-) diff --git a/werewolf/game.py b/werewolf/game.py index 45f0cbd..dd95edb 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -65,7 +65,7 @@ class Game: self.started = False self.game_over = False - self.can_vote = False + self.any_votes_remaining = False self.used_votes = 0 self.day_time = False @@ -88,6 +88,7 @@ class Game: self.loop = asyncio.get_event_loop() self.action_queue = deque() + self.current_action = None self.listeners = {} # def __del__(self): @@ -278,8 +279,12 @@ class Game: self.action_queue.append(self._at_day_start()) - while self.action_queue: - await self.action_queue.popleft() + while self.action_queue and not self.game_over: + current_action = asyncio.create_task(self.action_queue.popleft()) + try: + await current_action + except asyncio.CancelledError: + log.debug("Cancelled task") # # await self._at_day_start() # # Once cycle ends, this will trigger end_game @@ -299,52 +304,57 @@ class Game: if self.game_over: return + self.action_queue.append(self._at_day_end()) # Get this ready in case day is cancelled + def check(): - return not self.can_vote or not self.day_time or self.game_over + return not self.any_votes_remaining or not self.day_time or self.game_over self.day_count += 1 + + # Print the results of who died during the night embed = discord.Embed(title=random.choice(self.morning_messages).format(self.day_count)) for result in self.night_results: embed.add_field(name=result, value="________", inline=False) - self.day_time = True + self.day_time = True # True while day self.night_results = [] # Clear for next day await self.village_channel.send(embed=embed) - await self.generate_targets(self.village_channel) + await self.generate_targets(self.village_channel) # Print remaining players for voting await self.day_perms(self.village_channel) - await self._notify("at_day_start") + await self._notify("at_day_start") # Wait for day_start actions await self._check_game_over() - if self.game_over: + if self.game_over: # If game ended because of _notify return - self.can_vote = True + self.any_votes_remaining = True + + # Now we sleep and let the day happen. Print the remaining daylight half way through await asyncio.sleep(HALF_DAY_LENGTH) # 4 minute days FixMe to 120 later if check(): return await self.village_channel.send( - embed=discord.Embed(title="**Two minutes of daylight remain...**") + embed=discord.Embed(title=f"**{HALF_DAY_LENGTH/60} minutes of daylight remain...**") ) await asyncio.sleep(HALF_DAY_LENGTH) # 4 minute days FixMe to 120 later - # Need a loop here to wait for trial to end (can_vote?) + # Need a loop here to wait for trial to end while self.ongoing_vote: await asyncio.sleep(5) - if check(): - return - - self.action_queue.append(self._at_day_end()) + # Abruptly ends, assuming _day_end is next in queue async def _at_voted(self, target): # ID 2 if self.game_over: return - data = {"player": target} + + # Notify that a target has been chosen await self._notify("at_voted", player=target) + # TODO: Support pre-vote target modifying roles self.ongoing_vote = True self.used_votes += 1 @@ -359,7 +369,7 @@ class Game: await self.speech_perms(self.village_channel, target.member, undo=True) # No one can talk - message: discord.Message = await self.village_channel.send( + vote_message: discord.Message = await self.village_channel.send( f"Everyone will now vote whether to lynch {target.mention}\n" "👍 to save, 👎 to lynch\n" "*Majority rules, no-lynch on ties, " @@ -367,41 +377,47 @@ class Game: allowed_mentions=discord.AllowedMentions(everyone=False, users=[target]), ) - await message.add_reaction("👍") - await message.add_reaction("👎") + await vote_message.add_reaction("👍") + await vote_message.add_reaction("👎") await asyncio.sleep(15) - reaction_list = message.reactions + reaction_list = vote_message.reactions + + if True: # TODO: Allow customizable vote history deletion. + await vote_message.delete() - up_votes = sum(p for p in reaction_list if p.emoji == "👍" and not p.me) - down_votes = sum(p for p in reaction_list if p.emoji == "👎" and not p.me) + raw_up_votes = sum(p for p in reaction_list if p.emoji == "👍" and not p.me) + raw_down_votes = sum(p for p in reaction_list if p.emoji == "👎" and not p.me) - if down_votes > up_votes: - embed = discord.Embed(title="Vote Results", color=0xFF0000) + # TODO: Support vote count modifying roles. (Need notify and count function) + voted_to_lynch = raw_down_votes > raw_up_votes + + if voted_to_lynch: + embed = discord.Embed( + title="Vote Results", + description=f"**Voted to lynch {target.mention}!**", + color=0xFF0000, + ) else: - embed = discord.Embed(title="Vote Results", color=0x80FF80) + embed = discord.Embed( + title="Vote Results", + description=f"**{target.mention} has been spared!**", + color=0x80FF80, + ) - embed.add_field(name="👎", value=f"**{up_votes}**", inline=True) - embed.add_field(name="👍", value=f"**{down_votes}**", inline=True) + embed.add_field(name="👎", value=f"**{raw_up_votes}**", inline=True) + embed.add_field(name="👍", value=f"**{raw_down_votes}**", inline=True) await self.village_channel.send(embed=embed) - if down_votes > up_votes: - await self.village_channel.send( - f"**Voted to lynch {target.mention}!**", - allowed_mentions=discord.AllowedMentions(everyone=False, users=[target]), - ) + if voted_to_lynch: await self.lynch(target) - self.can_vote = False + self.any_votes_remaining = False else: - await self.village_channel.send( - f"**{target.mention} has been spared!**", - allowed_mentions=discord.AllowedMentions(everyone=False, users=[target]), - ) if self.used_votes >= self.day_vote_count: await self.village_channel.send("**All votes have been used! Day is now over!**") - self.can_vote = False + self.any_votes_remaining = False else: await self.village_channel.send( f"**{self.used_votes}**/**{self.day_vote_count}** of today's votes have been used!\n" @@ -410,21 +426,19 @@ class Game: self.ongoing_vote = False - if not self.can_vote: - self.action_queue.append(self._at_day_end()) + if not self.any_votes_remaining and self.day_time: + self.current_action.cancel() else: await self.normal_perms(self.village_channel) # No point if about to be night async def _at_kill(self, target): # ID 3 if self.game_over: return - data = {"player": target} await self._notify("at_kill", player=target) async def _at_hang(self, target): # ID 4 if self.game_over: return - data = {"player": target} await self._notify("at_hang", player=target) async def _at_day_end(self): # ID 5 @@ -433,7 +447,7 @@ class Game: if self.game_over: return - self.can_vote = False + self.any_votes_remaining = False self.day_vote = {} self.vote_totals = {} self.day_time = False @@ -476,7 +490,6 @@ class Game: async def _at_visit(self, target, source): # ID 8 if self.game_over: return - data = {"target": target, "source": source} await self._notify("at_visit", target=target, source=source) async def _notify(self, event, **kwargs): @@ -484,6 +497,8 @@ class Game: tasks = [] for event in self.listeners.get(event, {}).get(i, []): tasks.append(asyncio.ensure_future(event(**kwargs), loop=self.loop)) + + # Run same-priority task simultaneously await asyncio.gather(*tasks) # self.bot.dispatch(f"red.fox.werewolf.{event}", data=data, priority=i) @@ -507,8 +522,7 @@ class Game: async def generate_targets(self, channel, with_roles=False): embed = discord.Embed(title="Remaining Players") - for i in range(len(self.players)): - player = self.players[i] + for i, player in enumerate(self.players): if player.alive: status = "" else: @@ -653,7 +667,7 @@ class Game: return if channel == self.village_channel: - if not self.can_vote: + if not self.any_votes_remaining: await channel.send("Voting is not allowed right now") return elif channel.name in self.p_channels: @@ -695,13 +709,8 @@ class Game: if self.vote_totals[target_id] < required_votes: await self.village_channel.send( - "" - "{} has voted to put {} to trial. " - "{} more votes needed".format( - author.mention, - target.member.mention, - required_votes - self.vote_totals[target_id], - ), + f"{author.mention} has voted to put {target.member.mention} to trial. " + f"{required_votes - self.vote_totals[target_id]} more votes needed", allowed_mentions=discord.AllowedMentions(everyone=False, users=[author, target]), ) else: @@ -771,8 +780,8 @@ class Game: Attempt to lynch a target Important to finish execution before triggering notify """ - target = await self.get_day_target(target_id) - target.alive = False + target = await self.get_day_target(target_id) # Allows target modification + target.alive = False # Kill them, await self._at_hang(target) if not target.alive: # Still dead after notifying await self.dead_perms(self.village_channel, target.member) From f263f97cc27bef8af33cd5998fd632a97db38845 Mon Sep 17 00:00:00 2001 From: bobloy Date: Thu, 24 Sep 2020 17:06:03 -0400 Subject: [PATCH 21/73] Update builder to accept any number of roles --- werewolf/builder.py | 30 ++++++++++++++++++++---------- 1 file changed, 20 insertions(+), 10 deletions(-) diff --git a/werewolf/builder.py b/werewolf/builder.py index ca90eca..62d290f 100644 --- a/werewolf/builder.py +++ b/werewolf/builder.py @@ -9,21 +9,33 @@ import discord # Import all roles here from redbot.core import commands -from .roles.seer import Seer -from .roles.vanillawerewolf import VanillaWerewolf -from .roles.villager import Villager +# from .roles.seer import Seer +# from .roles.vanillawerewolf import VanillaWerewolf +# from .roles.villager import Villager + +from werewolf import roles from redbot.core.utils.menus import menu, prev_page, next_page, close_menu +from werewolf.role import Role + log = logging.getLogger("red.fox_v3.werewolf.builder") # All roles in this list for iterating -ROLE_LIST = sorted([Villager, Seer, VanillaWerewolf], key=lambda x: x.alignment) +ROLE_DICT = {name: cls for name, cls in roles.__dict__.items() if isinstance(cls, type)} +ROLE_LIST = sorted( + [cls for cls in ROLE_DICT.values()], + key=lambda x: x.alignment, +) + +log.debug(f"{ROLE_DICT=}") ALIGNMENT_COLORS = [0x008000, 0xFF0000, 0xC0C0C0] -TOWN_ROLES = [(idx, role) for idx, role in enumerate(ROLE_LIST) if role.alignment == 1] -WW_ROLES = [(idx, role) for idx, role in enumerate(ROLE_LIST) if role.alignment == 2] -OTHER_ROLES = [(idx, role) for idx, role in enumerate(ROLE_LIST) if role.alignment not in [0, 1]] +# TOWN_ROLES = [(idx, role) for idx, r_tuple in enumerate(ROLE_LIST) if role.alignment == 1] +# WW_ROLES = [(idx, role) for idx, r_tuple in enumerate(ROLE_LIST) if role.alignment == 2] +# OTHER_ROLES = [ +# (idx, role) for idx, r_tuple in enumerate(ROLE_LIST) if role.alignment not in [0, 1] +# ] ROLE_PAGES = [] PAGE_GROUPS = [0] @@ -81,9 +93,7 @@ def setup(): for k, v in ROLE_CATEGORIES.items(): if 0 < k <= 6: ROLE_PAGES.append( - discord.Embed( - title="RANDOM:Town Role", description=f"Town {v}", color=0x008000 - ) + discord.Embed(title="RANDOM:Town Role", description=f"Town {v}", color=0x008000) ) CATEGORY_COUNT.append(k) From 224ff93531179c1ccc73dbbd060172ca4e26b7dd Mon Sep 17 00:00:00 2001 From: bobloy Date: Thu, 24 Sep 2020 17:06:17 -0400 Subject: [PATCH 22/73] black and __all__ --- werewolf/roles/__init__.py | 1 + werewolf/votegroups/__init__.py | 2 +- 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/werewolf/roles/__init__.py b/werewolf/roles/__init__.py index ba929e5..201799a 100644 --- a/werewolf/roles/__init__.py +++ b/werewolf/roles/__init__.py @@ -3,3 +3,4 @@ from .shifter import Shifter from .vanillawerewolf import VanillaWerewolf from .villager import Villager +__all__ = ["Seer", "Shifter", "VanillaWerewolf", "Villager"] diff --git a/werewolf/votegroups/__init__.py b/werewolf/votegroups/__init__.py index 03abc1b..6b99b1e 100644 --- a/werewolf/votegroups/__init__.py +++ b/werewolf/votegroups/__init__.py @@ -1 +1 @@ -from .wolfvote import WolfVote \ No newline at end of file +from .wolfvote import WolfVote From a0c645bd28f9aa07029776e7b1c99baae4f6919d Mon Sep 17 00:00:00 2001 From: bobloy Date: Fri, 25 Sep 2020 15:56:33 -0400 Subject: [PATCH 23/73] Precarious import order --- werewolf/roles/__init__.py | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/werewolf/roles/__init__.py b/werewolf/roles/__init__.py index 201799a..3f58a76 100644 --- a/werewolf/roles/__init__.py +++ b/werewolf/roles/__init__.py @@ -1,6 +1,11 @@ +from .villager import Villager from .seer import Seer -from .shifter import Shifter + from .vanillawerewolf import VanillaWerewolf -from .villager import Villager + +from .shifter import Shifter + +# Don't sort these imports. They are unstably in order +# TODO: Replace with unique IDs for roles in the future __all__ = ["Seer", "Shifter", "VanillaWerewolf", "Villager"] From eb0c79ef1d57924c650c96efc90998fe326d4d01 Mon Sep 17 00:00:00 2001 From: bobloy Date: Fri, 25 Sep 2020 15:56:49 -0400 Subject: [PATCH 24/73] Introduction of The Blob --- werewolf/roles/blob.py | 101 +++++++++++++++++++++++++++++++++++++++++ 1 file changed, 101 insertions(+) create mode 100644 werewolf/roles/blob.py diff --git a/werewolf/roles/blob.py b/werewolf/roles/blob.py new file mode 100644 index 0000000..bd7b598 --- /dev/null +++ b/werewolf/roles/blob.py @@ -0,0 +1,101 @@ +import logging +import random + +from werewolf.constants import ALIGNMENT_NEUTRAL, CATEGORY_NEUTRAL_EVIL +from werewolf.listener import wolflistener +from werewolf.player import Player +from werewolf.role import Role + +log = logging.getLogger("red.fox_v3.werewolf.role.blob") + + +class TheBlob(Role): + rand_choice = True + category = [CATEGORY_NEUTRAL_EVIL] # List of enrolled categories + alignment = ALIGNMENT_NEUTRAL # 1: Town, 2: Werewolf, 3: Neutral + channel_id = "" # Empty for no private channel + unique = True # Only one of this role per game + game_start_message = ( + "Your role is **The Blob**\n" + "You win by absorbing everyone town\n" + "Lynch players during the day with `[p]ww vote `\n" + "Each night you will absorb an adjacent player" + ) + description = ( + "A mysterious green blob of jelly, slowly growing in size.\n" + "The Blob fears no evil, must be dealt with in town" + ) + + def __init__(self, game): + super().__init__(game) + + self.blob_target = None + + async def see_alignment(self, source=None): + """ + Interaction for investigative roles attempting + to see team (Village, Werewolf, Other) + """ + return ALIGNMENT_NEUTRAL + + async def get_role(self, source=None): + """ + Interaction for powerful access of role + Unlikely to be able to deceive this + """ + return "The Blob" + + async def see_role(self, source=None): + """ + Interaction for investigative roles. + More common to be able to deceive these roles + """ + return "The Blob" + + async def kill(self, source): + """ + Called when someone is trying to kill you! + Can you do anything about it? + self.player.alive is now set to False, set to True to stay alive + """ + + # Blob cannot simply be killed + self.player.alive = True + + @wolflistener("at_night_start", priority=2) + async def _at_night_start(self): + if not self.player.alive: + return + + self.blob_target = None + idx = self.player.id + left_or_right = random.choice((-1, 1)) + while self.blob_target is None: + idx += left_or_right + if idx >= len(self.game.players): + idx = 0 + + player = self.game.players[idx] + + # you went full circle, everyone is a blob or something else is wrong + if player == self.player: + break + + if player.role.properties.get("been_blobbed", False): + self.blob_target = player + + if self.blob_target is not None: + await self.player.send_dm(f"**You will attempt to absorb {self.blob_target} tonight**") + else: + await self.player.send_dm(f"**No player will be absorbed tonight**") + + @wolflistener("at_night_end", priority=4) + async def _at_night_end(self): + if self.blob_target is None or not self.player.alive: + return + + target: "Player" = await self.game.visit(self.blob_target, self.player) + + if target is not None: + target.role.properties["been_blobbed"] = True + self.game.night_messages.append("The Blob grows...") From 61049c23432cd8262e5f1805a30cdae7cc2af569 Mon Sep 17 00:00:00 2001 From: bobloy Date: Fri, 25 Sep 2020 15:59:21 -0400 Subject: [PATCH 25/73] Adding constants --- werewolf/builder.py | 258 +++++++++++++++--------------- werewolf/constants.py | 91 +++++++++++ werewolf/game.py | 10 +- werewolf/role.py | 59 ++----- werewolf/roles/seer.py | 23 +-- werewolf/roles/shifter.py | 8 +- werewolf/roles/vanillawerewolf.py | 20 +-- werewolf/roles/villager.py | 16 +- werewolf/werewolf.py | 2 +- 9 files changed, 273 insertions(+), 214 deletions(-) create mode 100644 werewolf/constants.py diff --git a/werewolf/builder.py b/werewolf/builder.py index 62d290f..4c803cc 100644 --- a/werewolf/builder.py +++ b/werewolf/builder.py @@ -1,6 +1,7 @@ import bisect import logging from collections import defaultdict +from operator import attrgetter from random import choice import discord @@ -16,6 +17,7 @@ from redbot.core import commands from werewolf import roles from redbot.core.utils.menus import menu, prev_page, next_page, close_menu +from werewolf.constants import ROLE_CATEGORY_DESCRIPTIONS from werewolf.role import Role log = logging.getLogger("red.fox_v3.werewolf.builder") @@ -25,104 +27,40 @@ log = logging.getLogger("red.fox_v3.werewolf.builder") ROLE_DICT = {name: cls for name, cls in roles.__dict__.items() if isinstance(cls, type)} ROLE_LIST = sorted( [cls for cls in ROLE_DICT.values()], - key=lambda x: x.alignment, + key=attrgetter('alignment'), ) log.debug(f"{ROLE_DICT=}") +# Town, Werewolf, Neutral ALIGNMENT_COLORS = [0x008000, 0xFF0000, 0xC0C0C0] -# TOWN_ROLES = [(idx, role) for idx, r_tuple in enumerate(ROLE_LIST) if role.alignment == 1] -# WW_ROLES = [(idx, role) for idx, r_tuple in enumerate(ROLE_LIST) if role.alignment == 2] -# OTHER_ROLES = [ -# (idx, role) for idx, r_tuple in enumerate(ROLE_LIST) if role.alignment not in [0, 1] -# ] ROLE_PAGES = [] -PAGE_GROUPS = [0] - -ROLE_CATEGORIES = { - 1: "Random", - 2: "Investigative", - 3: "Protective", - 4: "Government", - 5: "Killing", - 6: "Power (Special night action)", - 11: "Random", - 12: "Deception", - 15: "Killing", - 16: "Support", - 21: "Benign", - 22: "Evil", - 23: "Killing", -} - -CATEGORY_COUNT = [] - - -def role_embed(idx, role, color): + + +def role_embed(idx, role: Role, color): embed = discord.Embed( title=f"**{idx}** - {role.__name__}", description=role.game_start_message, color=color, ) + if role.icon_url is not None: + embed.set_thumbnail(url=role.icon_url) + embed.add_field( - name="Alignment", value=["Town", "Werewolf", "Neutral"][role.alignment - 1], inline=True + name="Alignment", value=["Town", "Werewolf", "Neutral"][role.alignment - 1], inline=False ) - embed.add_field(name="Multiples Allowed", value=str(not role.unique), inline=True) + embed.add_field(name="Multiples Allowed", value=str(not role.unique), inline=False) embed.add_field( - name="Role Type", value=", ".join(ROLE_CATEGORIES[x] for x in role.category), inline=True + name="Role Types", + value=", ".join(ROLE_CATEGORY_DESCRIPTIONS[x] for x in role.category), + inline=False, ) - embed.add_field(name="Random Option", value=str(role.rand_choice), inline=True) + embed.add_field(name="Random Option", value=str(role.rand_choice), inline=False) return embed -def setup(): - # Roles - last_alignment = ROLE_LIST[0].alignment - for idx, role in enumerate(ROLE_LIST): - if role.alignment != last_alignment and len(ROLE_PAGES) - 1 not in PAGE_GROUPS: - PAGE_GROUPS.append(len(ROLE_PAGES) - 1) - last_alignment = role.alignment - - ROLE_PAGES.append(role_embed(idx, role, ALIGNMENT_COLORS[role.alignment - 1])) - - # Random Town Roles - if len(ROLE_PAGES) - 1 not in PAGE_GROUPS: - PAGE_GROUPS.append(len(ROLE_PAGES) - 1) - for k, v in ROLE_CATEGORIES.items(): - if 0 < k <= 6: - ROLE_PAGES.append( - discord.Embed(title="RANDOM:Town Role", description=f"Town {v}", color=0x008000) - ) - CATEGORY_COUNT.append(k) - - # Random WW Roles - if len(ROLE_PAGES) - 1 not in PAGE_GROUPS: - PAGE_GROUPS.append(len(ROLE_PAGES) - 1) - for k, v in ROLE_CATEGORIES.items(): - if 10 < k <= 16: - ROLE_PAGES.append( - discord.Embed( - title="RANDOM:Werewolf Role", - description=f"Werewolf {v}", - color=0xFF0000, - ) - ) - CATEGORY_COUNT.append(k) - # Random Neutral Roles - if len(ROLE_PAGES) - 1 not in PAGE_GROUPS: - PAGE_GROUPS.append(len(ROLE_PAGES) - 1) - for k, v in ROLE_CATEGORIES.items(): - if 20 < k <= 26: - ROLE_PAGES.append( - discord.Embed( - title=f"RANDOM:Neutral Role", description="Neutral {v}", color=0xC0C0C0 - ) - ) - CATEGORY_COUNT.append(k) - - """ Example code: 0 = Villager @@ -189,15 +127,15 @@ async def parse_code(code, game): return decode -async def encode(roles, rand_roles): +async def encode(role_list, rand_roles): """Convert role list to code""" out_code = "" - digit_sort = sorted(role for role in roles if role < 10) + digit_sort = sorted(role for role in role_list if role < 10) for role in digit_sort: out_code += str(role) - digit_sort = sorted(role for role in roles if 10 <= role < 100) + digit_sort = sorted(role for role in role_list if 10 <= role < 100) if digit_sort: out_code += "-" for role in digit_sort: @@ -229,51 +167,6 @@ async def encode(roles, rand_roles): return out_code -async def next_group( - ctx: commands.Context, - pages: list, - controls: dict, - message: discord.Message, - page: int, - timeout: float, - emoji: str, -): - perms = message.channel.permissions_for(ctx.me) - if perms.manage_messages: # Can manage messages, so remove react - try: - await message.remove_reaction(emoji, ctx.author) - except discord.NotFound: - pass - page = bisect.bisect_right(PAGE_GROUPS, page) - - if page == len(PAGE_GROUPS): - page = PAGE_GROUPS[0] - else: - page = PAGE_GROUPS[page] - - return await menu(ctx, pages, controls, message=message, page=page, timeout=timeout) - - -async def prev_group( - ctx: commands.Context, - pages: list, - controls: dict, - message: discord.Message, - page: int, - timeout: float, - emoji: str, -): - perms = message.channel.permissions_for(ctx.me) - if perms.manage_messages: # Can manage messages, so remove react - try: - await message.remove_reaction(emoji, ctx.author) - except discord.NotFound: - pass - page = PAGE_GROUPS[bisect.bisect_left(PAGE_GROUPS, page) - 1] - - return await menu(ctx, pages, controls, message=message, page=page, timeout=timeout) - - def role_from_alignment(alignment): return [ role_embed(idx, role, ALIGNMENT_COLORS[role.alignment - 1]) @@ -316,11 +209,11 @@ def say_role_list(code_list, rand_roles): for role in rand_roles: if 0 < role <= 6: - role_dict[f"Town {ROLE_CATEGORIES[role]}"] += 1 + role_dict[f"Town {ROLE_CATEGORY_DESCRIPTIONS[role]}"] += 1 if 10 < role <= 16: - role_dict[f"Werewolf {ROLE_CATEGORIES[role]}"] += 1 + role_dict[f"Werewolf {ROLE_CATEGORY_DESCRIPTIONS[role]}"] += 1 if 20 < role <= 26: - role_dict[f"Neutral {ROLE_CATEGORIES[role]}"] += 1 + role_dict[f"Neutral {ROLE_CATEGORY_DESCRIPTIONS[role]}"] += 1 for k, v in role_dict.items(): embed.add_field(name=k, value=f"Count: {v}", inline=True) @@ -332,15 +225,69 @@ class GameBuilder: def __init__(self): self.code = [] self.rand_roles = [] - setup() + self.page_groups = [0] + self.category_count = [] + + self.setup() + + def setup(self): + # Roles + last_alignment = ROLE_LIST[0].alignment + for idx, role in enumerate(ROLE_LIST): + if role.alignment != last_alignment and len(ROLE_PAGES) - 1 not in self.page_groups: + self.page_groups.append(len(ROLE_PAGES) - 1) + last_alignment = role.alignment + + ROLE_PAGES.append(role_embed(idx, role, ALIGNMENT_COLORS[role.alignment - 1])) + + # Random Town Roles + if len(ROLE_PAGES) - 1 not in self.page_groups: + self.page_groups.append(len(ROLE_PAGES) - 1) + for k, v in ROLE_CATEGORY_DESCRIPTIONS.items(): + if 0 < k <= 9: + ROLE_PAGES.append( + discord.Embed( + title="RANDOM:Town Role", + description=f"Town {v}", + color=ALIGNMENT_COLORS[0], + ) + ) + self.category_count.append(k) + + # Random WW Roles + if len(ROLE_PAGES) - 1 not in self.page_groups: + self.page_groups.append(len(ROLE_PAGES) - 1) + for k, v in ROLE_CATEGORY_DESCRIPTIONS.items(): + if 10 < k <= 19: + ROLE_PAGES.append( + discord.Embed( + title="RANDOM:Werewolf Role", + description=f"Werewolf {v}", + color=ALIGNMENT_COLORS[1], + ) + ) + self.category_count.append(k) + # Random Neutral Roles + if len(ROLE_PAGES) - 1 not in self.page_groups: + self.page_groups.append(len(ROLE_PAGES) - 1) + for k, v in ROLE_CATEGORY_DESCRIPTIONS.items(): + if 20 < k <= 29: + ROLE_PAGES.append( + discord.Embed( + title=f"RANDOM:Neutral Role", + description=f"Neutral {v}", + color=ALIGNMENT_COLORS[2], + ) + ) + self.category_count.append(k) async def build_game(self, ctx: commands.Context): new_controls = { - "⏪": prev_group, + "⏪": self.prev_group, "⬅": prev_page, "☑": self.select_page, "➡": next_page, - "⏩": next_group, + "⏩": self.next_group, "📇": self.list_roles, "❌": close_menu, } @@ -391,8 +338,53 @@ class GameBuilder: pass if page >= len(ROLE_LIST): - self.rand_roles.append(CATEGORY_COUNT[page - len(ROLE_LIST)]) + self.rand_roles.append(self.category_count[page - len(ROLE_LIST)]) else: self.code.append(page) return await menu(ctx, pages, controls, message=message, page=page, timeout=timeout) + + async def next_group( + self, + ctx: commands.Context, + pages: list, + controls: dict, + message: discord.Message, + page: int, + timeout: float, + emoji: str, + ): + perms = message.channel.permissions_for(ctx.me) + if perms.manage_messages: # Can manage messages, so remove react + try: + await message.remove_reaction(emoji, ctx.author) + except discord.NotFound: + pass + page = bisect.bisect_right(self.page_groups, page) + + if page == len(self.page_groups): + page = self.page_groups[0] + else: + page = self.page_groups[page] + + return await menu(ctx, pages, controls, message=message, page=page, timeout=timeout) + + async def prev_group( + self, + ctx: commands.Context, + pages: list, + controls: dict, + message: discord.Message, + page: int, + timeout: float, + emoji: str, + ): + perms = message.channel.permissions_for(ctx.me) + if perms.manage_messages: # Can manage messages, so remove react + try: + await message.remove_reaction(emoji, ctx.author) + except discord.NotFound: + pass + page = self.page_groups[bisect.bisect_left(self.page_groups, page) - 1] + + return await menu(ctx, pages, controls, message=message, page=page, timeout=timeout) diff --git a/werewolf/constants.py b/werewolf/constants.py new file mode 100644 index 0000000..bb77421 --- /dev/null +++ b/werewolf/constants.py @@ -0,0 +1,91 @@ +""" +Role Constants + + Role Alignment guide as follows: + Town: 1 + Werewolf: 2 + Neutral: 3 + + Additional alignments may be added when warring factions are added + (Rival werewolves, cultists, vampires) + + Role Category enrollment guide as follows (See Role.category): + Town: + 1: Random, 2: Investigative, 3: Protective, 4: Government, + 5: Killing, 6: Power (Special night action) + + Werewolf: + 11: Random, 12: Deception, 15: Killing, 16: Support + + Neutral: + 21: Benign, 22: Evil, 23: Killing + + + Example category: + category = [1, 5, 6] Could be Veteran + category = [1, 5] Could be Bodyguard + category = [11, 16] Could be Werewolf Silencer + category = [22] Could be Blob (non-killing) + category = [22, 23] Could be Serial-Killer +""" + + +ALIGNMENT_TOWN = 1 +ALIGNMENT_WEREWOLF = 2 +ALIGNMENT_NEUTRAL = 3 +ALIGNMENT_MAP = {"Town": 1, "Werewolf": 2, "Neutral": 3} + +# 0-9: Town Role Categories +# 10-19: Werewolf Role Categories +# 20-29: Neutral Role Categories +CATEGORY_TOWN_RANDOM = 1 +CATEGORY_TOWN_INVESTIGATIVE = 2 +CATEGORY_TOWN_PROTECTIVE = 3 +CATEGORY_TOWN_GOVERNMENT = 4 +CATEGORY_TOWN_KILLING = 5 +CATEGORY_TOWN_POWER = 6 + +CATEGORY_WW_RANDOM = 11 +CATEGORY_WW_DECEPTION = 12 +CATEGORY_WW_KILLING = 15 +CATEGORY_WW_SUPPORT = 16 + +CATEGORY_NEUTRAL_BENIGN = 21 +CATEGORY_NEUTRAL_EVIL = 22 +CATEGORY_NEUTRAL_KILLING = 23 + +ROLE_CATEGORY_DESCRIPTIONS = { + CATEGORY_TOWN_RANDOM: "Random", + CATEGORY_TOWN_INVESTIGATIVE: "Investigative", + CATEGORY_TOWN_PROTECTIVE: "Protective", + CATEGORY_TOWN_GOVERNMENT: "Government", + CATEGORY_TOWN_KILLING: "Killing", + CATEGORY_TOWN_POWER: "Power (Special night action)", + CATEGORY_WW_RANDOM: "Random", + CATEGORY_WW_DECEPTION: "Deception", + CATEGORY_WW_KILLING: "Killing", + CATEGORY_WW_SUPPORT: "Support", + CATEGORY_NEUTRAL_BENIGN: "Benign", + CATEGORY_NEUTRAL_EVIL: "Evil", + CATEGORY_NEUTRAL_KILLING: "Killing", +} + + +""" +Listener Actions Priority Guide + + Action priority guide as follows (see listeners.py for wolflistener): + _at_night_start + 0. No Action + 1. Detain actions (Jailer/Kidnapper) + 2. Group discussions and choose targets + + _at_night_end + 0. No Action + 1. Self actions (Veteran) + 2. Target switching and role blocks (bus driver, witch, escort) + 3. Protection / Preempt actions (bodyguard/framer) + 4. Non-disruptive actions (seer/silencer) + 5. Disruptive actions (Killing) + 6. Role altering actions (Cult / Mason / Shifter) +""" diff --git a/werewolf/game.py b/werewolf/game.py index dd95edb..13db415 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -639,14 +639,14 @@ class Game: await target.role.visit(source) await self._at_visit(target, source) - async def visit(self, target_id, source): + async def visit(self, target_id, source) -> Union[Player, None]: """ Night visit target_id Returns a target for role information (i.e. Seer) """ if source.role.blocked: # Blocker handles text - return + return None target = await self.get_night_target(target_id, source) await self._visit(target, source) return target @@ -786,10 +786,10 @@ class Game: if not target.alive: # Still dead after notifying await self.dead_perms(self.village_channel, target.member) - async def get_night_target(self, target_id, source=None): + async def get_night_target(self, target_id, source=None) -> Player: return self.players[target_id] # ToDo check source - async def get_day_target(self, target_id, source=None): + async def get_day_target(self, target_id, source=None) -> Player: return self.players[target_id] # ToDo check source async def set_code(self, ctx: commands.Context, game_code): @@ -829,7 +829,7 @@ class Game: # Sorted players, now assign id's await self.players[idx].assign_id(idx) - async def get_player_by_member(self, member): + async def get_player_by_member(self, member: discord.Member): for player in self.players: if player.member == member: return player diff --git a/werewolf/role.py b/werewolf/role.py index ccc20ae..db7b852 100644 --- a/werewolf/role.py +++ b/werewolf/role.py @@ -26,6 +26,8 @@ class Role(WolfListener): category = [1, 5, 6] Could be Veteran category = [1, 5] Could be Bodyguard category = [11, 16] Could be Werewolf Silencer + category = [22] Could be Blob (non-killing) + category = [22, 23] Could be Serial-Killer Action priority guide as follows (on_event function): @@ -44,10 +46,12 @@ class Role(WolfListener): 6. Role altering actions (Cult / Mason / Shifter) """ - rand_choice = False # Determines if it can be picked as a random role (False for unusually disruptive roles) + # Determines if it can be picked as a random role (False for unusually disruptive roles) + rand_choice = False # TODO: Rework random with categories + town_balance = 0 # Guess at power level and it's balance on the town category = [0] # List of enrolled categories (listed above) alignment = 0 # 1: Town, 2: Werewolf, 3: Neutral - channel_id = "" # Empty for no private channel + channel_name = "" # Empty for no private channel unique = False # Only one of this role per game game_start_message = ( "Your role is **Default**\n" @@ -68,28 +72,9 @@ class Role(WolfListener): self.blocked = False self.properties = {} # Extra data for other roles (i.e. arsonist) - # self.action_list = [ - # (self._at_game_start, 1), # (Action, Priority) - # (self._at_day_start, 0), - # (self._at_voted, 0), - # (self._at_kill, 0), - # (self._at_hang, 0), - # (self._at_day_end, 0), - # (self._at_night_start, 0), - # (self._at_night_end, 0), - # (self._at_visit, 0), - # ] - def __repr__(self): return self.__class__.__name__ - # async def on_event(self, event, data): - # """ - # See Game class for event guide - # """ - # - # await self.action_list[event][0](data) - async def assign_player(self, player): """ Give this role a player @@ -110,7 +95,7 @@ class Role(WolfListener): async def see_alignment(self, source=None): """ Interaction for investigative roles attempting - to see alignment (Village, Werewolf Other) + to see alignment (Village, Werewolf, Other) """ return "Other" @@ -128,37 +113,13 @@ class Role(WolfListener): """ return "Default" - @wolflistener("at_game_start", priority=1) + @wolflistener("at_game_start", priority=2) async def _at_game_start(self): - if self.channel_id: - await self.game.register_channel(self.channel_id, self) + if self.channel_name: + await self.game.register_channel(self.channel_name, self) await self.player.send_dm(self.game_start_message) # Maybe embeds eventually - # async def _at_day_start(self, data=None): - # pass - # - # async def _at_voted(self, data=None): - # pass - # - # async def _at_kill(self, data=None): - # pass - # - # async def _at_hang(self, data=None): - # pass - # - # async def _at_day_end(self, data=None): - # pass - # - # async def _at_night_start(self, data=None): - # pass - # - # async def _at_night_end(self, data=None): - # pass - # - # async def _at_visit(self, data=None): - # pass - async def kill(self, source): """ Called when someone is trying to kill you! diff --git a/werewolf/roles/seer.py b/werewolf/roles/seer.py index 56624c9..32ace18 100644 --- a/werewolf/roles/seer.py +++ b/werewolf/roles/seer.py @@ -1,5 +1,7 @@ import logging +from werewolf.constants import ALIGNMENT_TOWN, ALIGNMENT_WEREWOLF, CATEGORY_TOWN_INVESTIGATIVE, \ + CATEGORY_TOWN_RANDOM from werewolf.listener import wolflistener from werewolf.night_powers import pick_target from werewolf.role import Role @@ -8,9 +10,12 @@ log = logging.getLogger("red.fox_v3.werewolf.role.seer") class Seer(Role): - rand_choice = True # Determines if it can be picked as a random role (False for unusually disruptive roles) - category = [1, 2] # List of enrolled categories (listed above) - alignment = 1 # 1: Town, 2: Werewolf, 3: Neutral + rand_choice = True + category = [ + CATEGORY_TOWN_RANDOM, + CATEGORY_TOWN_INVESTIGATIVE, + ] # List of enrolled categories (listed above) + alignment = ALIGNMENT_TOWN # 1: Town, 2: Werewolf, 3: Neutral channel_id = "" # Empty for no private channel unique = False # Only one of this role per game game_start_message = ( @@ -46,23 +51,23 @@ class Seer(Role): async def see_alignment(self, source=None): """ Interaction for investigative roles attempting - to see team (Village, Werewolf Other) + to see team (Village, Werewolf, Other) """ - return "Village" + return ALIGNMENT_TOWN async def get_role(self, source=None): """ Interaction for powerful access of role Unlikely to be able to deceive this """ - return "Villager" + return "Seer" async def see_role(self, source=None): """ Interaction for investigative roles. More common to be able to deceive these roles """ - return "Villager" + return "Seer" @wolflistener("at_night_start", priority=2) async def _at_night_start(self): @@ -84,9 +89,9 @@ class Seer(Role): if target: alignment = await target.role.see_alignment(self.player) - if alignment == "Werewolf": + if alignment == ALIGNMENT_WEREWOLF: out = "Your insight reveals this player to be a **Werewolf!**" - else: + else: # Don't reveal neutrals out = "You fail to find anything suspicious about this player..." await self.player.send_dm(out) diff --git a/werewolf/roles/shifter.py b/werewolf/roles/shifter.py index a7ea058..9685e20 100644 --- a/werewolf/roles/shifter.py +++ b/werewolf/roles/shifter.py @@ -1,5 +1,6 @@ import logging +from werewolf.constants import ALIGNMENT_NEUTRAL, CATEGORY_NEUTRAL_BENIGN from werewolf.listener import wolflistener from werewolf.night_powers import pick_target from werewolf.role import Role @@ -46,8 +47,9 @@ class Shifter(Role): """ rand_choice = False # Determines if it can be picked as a random role (False for unusually disruptive roles) - category = [22] # List of enrolled categories (listed above) - alignment = 3 # 1: Town, 2: Werewolf, 3: Neutral + town_balance = -3 + category = [CATEGORY_NEUTRAL_BENIGN] # List of enrolled categories (listed above) + alignment = ALIGNMENT_NEUTRAL # 1: Town, 2: Werewolf, 3: Neutral channel_id = "" # Empty for no private channel unique = False # Only one of this role per game game_start_message = ( @@ -81,7 +83,7 @@ class Shifter(Role): async def see_alignment(self, source=None): """ Interaction for investigative roles attempting - to see alignment (Village, Werewolf, Other) + to see alignment (Village, Werewolf,, Other) """ return "Other" diff --git a/werewolf/roles/vanillawerewolf.py b/werewolf/roles/vanillawerewolf.py index 58b474e..74e8d96 100644 --- a/werewolf/roles/vanillawerewolf.py +++ b/werewolf/roles/vanillawerewolf.py @@ -1,5 +1,6 @@ import logging +from werewolf.constants import ALIGNMENT_WEREWOLF, CATEGORY_WW_KILLING, CATEGORY_WW_RANDOM from werewolf.listener import wolflistener from werewolf.role import Role from werewolf.votegroups.wolfvote import WolfVote @@ -9,9 +10,10 @@ log = logging.getLogger("red.fox_v3.werewolf.role.vanillawerewolf") class VanillaWerewolf(Role): rand_choice = True - category = [11, 15] - alignment = 2 # 1: Town, 2: Werewolf, 3: Neutral - channel_id = "werewolves" + town_balance = -6 + category = [CATEGORY_WW_RANDOM, CATEGORY_WW_KILLING] + alignment = ALIGNMENT_WEREWOLF # 1: Town, 2: Werewolf, 3: Neutral + channel_name = "werewolves" unique = False game_start_message = ( "Your role is **Werewolf**\n" @@ -40,14 +42,14 @@ class VanillaWerewolf(Role): Interaction for investigative roles attempting to see team (Village, Werewolf Other) """ - return "Werewolf" + return ALIGNMENT_WEREWOLF async def get_role(self, source=None): """ Interaction for powerful access of role Unlikely to be able to deceive this """ - return "Werewolf" + return "VanillaWerewolf" async def see_role(self, source=None): """ @@ -56,12 +58,12 @@ class VanillaWerewolf(Role): """ return "Werewolf" - @wolflistener("at_game_start", priority=1) + @wolflistener("at_game_start", priority=2) async def _at_game_start(self): - if self.channel_id: - log.debug("Wolf has channel_id: " + self.channel_id) + if self.channel_name: + log.debug("Wolf has channel_name: " + self.channel_name) await self.game.register_channel( - self.channel_id, self, WolfVote + self.channel_name, self, WolfVote ) # Add VoteGroup WolfVote await self.player.send_dm(self.game_start_message) diff --git a/werewolf/roles/villager.py b/werewolf/roles/villager.py index f225e0d..d669ef9 100644 --- a/werewolf/roles/villager.py +++ b/werewolf/roles/villager.py @@ -1,14 +1,20 @@ import logging +from werewolf.constants import ALIGNMENT_TOWN, CATEGORY_TOWN_RANDOM from werewolf.role import Role log = logging.getLogger("red.fox_v3.werewolf.role.villager") class Villager(Role): - rand_choice = True # Determines if it can be picked as a random role (False for unusually disruptive roles) - category = [1] # List of enrolled categories (listed above) - alignment = 1 # 1: Town, 2: Werewolf, 3: Neutral + + # Determines if it can be picked as a random role (False for unusually disruptive roles) + rand_choice = True + + town_balance = 1 + + category = [CATEGORY_TOWN_RANDOM] # List of enrolled categories (listed above) + alignment = ALIGNMENT_TOWN # 1: Town, 2: Werewolf, 3: Neutral channel_id = "" # Empty for no private channel unique = False # Only one of this role per game game_start_message = ( @@ -23,9 +29,9 @@ class Villager(Role): async def see_alignment(self, source=None): """ Interaction for investigative roles attempting - to see team (Village, Werewolf Other) + to see team (Village, Werewolf, Other) """ - return "Village" + return ALIGNMENT_TOWN async def get_role(self, source=None): """ diff --git a/werewolf/werewolf.py b/werewolf/werewolf.py index 742a890..d26dc38 100644 --- a/werewolf/werewolf.py +++ b/werewolf/werewolf.py @@ -33,7 +33,7 @@ class Werewolf(Cog): default_guild = { "role_id": None, "category_id": None, - "channel_id": None, + "channel_name": None, "log_channel_id": None, } From 029b6a51b141c4c21c425bf8d3f8b771042dd5ee Mon Sep 17 00:00:00 2001 From: bobloy Date: Fri, 25 Sep 2020 15:59:29 -0400 Subject: [PATCH 26/73] black --- werewolf/player.py | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/werewolf/player.py b/werewolf/player.py index 201b781..7f10758 100644 --- a/werewolf/player.py +++ b/werewolf/player.py @@ -34,4 +34,7 @@ class Player: try: await self.member.send(message) # Lets do embeds later except discord.Forbidden: - await self.role.game.village_channel.send(f"Couldn't DM {self.mention}, uh oh", allowed_mentions=discord.AllowedMentions(users=[self.member])) + await self.role.game.village_channel.send( + f"Couldn't DM {self.mention}, uh oh", + allowed_mentions=discord.AllowedMentions(users=[self.member]), + ) From db1d64ae3e3865a00443c0d53773dc2c8ba7f4c8 Mon Sep 17 00:00:00 2001 From: bobloy Date: Fri, 25 Sep 2020 16:59:30 -0400 Subject: [PATCH 27/73] More async iters --- timerole/timerole.py | 9 +++------ 1 file changed, 3 insertions(+), 6 deletions(-) diff --git a/timerole/timerole.py b/timerole/timerole.py index 3815dcd..7484267 100644 --- a/timerole/timerole.py +++ b/timerole/timerole.py @@ -157,7 +157,7 @@ class Timerole(Cog): await ctx.maybe_send_embed(out) async def timerole_update(self): - for guild in self.bot.guilds: + async for guild in AsyncIter(self.bot.guilds): addlist = [] removelist = [] @@ -200,7 +200,7 @@ class Timerole(Cog): async def announce_roles(self, title, role_list, channel, guild, to_add: True): results = "" - for member, role_id in role_list: + async for member, role_id in AsyncIter(role_list): role = discord.utils.get(guild.roles, id=role_id) try: if to_add: @@ -219,7 +219,7 @@ class Timerole(Cog): log.info(results) async def check_required_and_date(self, role_list, check_roles, has_roles, member, role_dict): - for role_id in check_roles: + async for role_id in AsyncIter(check_roles): # Check for required role if "required" in role_dict[str(role_id)]: if not set(role_dict[str(role_id)]["required"]) & set(has_roles): @@ -242,6 +242,3 @@ class Timerole(Cog): while self is self.bot.get_cog("Timerole"): await self.timerole_update() await sleep_till_next_hour() - - - From 03f0ef17be2d8b3ea937cfcc14eb22aa14ddf90c Mon Sep 17 00:00:00 2001 From: bobloy Date: Fri, 25 Sep 2020 22:04:35 -0400 Subject: [PATCH 28/73] Fix aggressive refactor --- werewolf/werewolf.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/werewolf/werewolf.py b/werewolf/werewolf.py index d26dc38..742a890 100644 --- a/werewolf/werewolf.py +++ b/werewolf/werewolf.py @@ -33,7 +33,7 @@ class Werewolf(Cog): default_guild = { "role_id": None, "category_id": None, - "channel_name": None, + "channel_id": None, "log_channel_id": None, } From af4cd92488ac3aae415aff445a3a2673ae5d67e4 Mon Sep 17 00:00:00 2001 From: bobloy Date: Fri, 25 Sep 2020 22:05:22 -0400 Subject: [PATCH 29/73] Correctly reset channels --- werewolf/game.py | 12 ++++++++---- 1 file changed, 8 insertions(+), 4 deletions(-) diff --git a/werewolf/game.py b/werewolf/game.py index 13db415..df345af 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -8,6 +8,7 @@ from typing import List, Any, Dict, Set, Union import discord from redbot.core import commands from redbot.core.bot import Red +from redbot.core.utils import AsyncIter from werewolf.builder import parse_code from werewolf.player import Player @@ -913,10 +914,13 @@ class Game: try: await self.village_channel.edit(reason=reason, name="Werewolf") - for target, overwrites in self.save_perms[self.village_channel]: - await self.village_channel.set_permissions( - target, overwrite=overwrites, reason=reason - ) + async for channel, overwrites in AsyncIter(self.save_perms.items()): + async for target, overwrite in AsyncIter(overwrites.items()): + await channel.set_permissions(target, overwrite=overwrite, reason=reason) + # for target, overwrites in self.save_perms[self.village_channel]: + # await self.village_channel.set_permissions( + # target, overwrite=overwrites, reason=reason + # ) await self.village_channel.set_permissions( self.game_role, overwrite=None, reason=reason ) From ab1b069ee98435ff14db8a27e1eb41941d1e85f6 Mon Sep 17 00:00:00 2001 From: bobloy Date: Mon, 28 Sep 2020 08:44:03 -0400 Subject: [PATCH 30/73] Move games to non-functional --- audiotrivia/data/lists/{ => non_funcitonal_lists}/games.yaml | 1 + 1 file changed, 1 insertion(+) rename audiotrivia/data/lists/{ => non_funcitonal_lists}/games.yaml (99%) diff --git a/audiotrivia/data/lists/games.yaml b/audiotrivia/data/lists/non_funcitonal_lists/games.yaml similarity index 99% rename from audiotrivia/data/lists/games.yaml rename to audiotrivia/data/lists/non_funcitonal_lists/games.yaml index 4de795a..da1dcb6 100644 --- a/audiotrivia/data/lists/games.yaml +++ b/audiotrivia/data/lists/non_funcitonal_lists/games.yaml @@ -1,4 +1,5 @@ AUTHOR: Plab +NEEDS: New links for all songs. https://www.youtube.com/watch?v=f9O2Rjn1azc: - Transistor https://www.youtube.com/watch?v=PgUhYFkVdSY: From b27b252e6f28ef5a23218933efd0b885c3fde529 Mon Sep 17 00:00:00 2001 From: bobloy Date: Mon, 28 Sep 2020 08:44:14 -0400 Subject: [PATCH 31/73] Add new videogames trivia list --- audiotrivia/data/lists/videogames.yaml | 1762 ++++++++++++++++++++++++ 1 file changed, 1762 insertions(+) create mode 100644 audiotrivia/data/lists/videogames.yaml diff --git a/audiotrivia/data/lists/videogames.yaml b/audiotrivia/data/lists/videogames.yaml new file mode 100644 index 0000000..eec01b6 --- /dev/null +++ b/audiotrivia/data/lists/videogames.yaml @@ -0,0 +1,1762 @@ +Author: Bobloy +https://www.youtube.com/watch?v=GBPbJyxqHV0: +- Super Mario 64 +https://www.youtube.com/watch?v=0jXTBAGv9ZQ: +- Halo +https://www.youtube.com/watch?v=ZksNhHyEhE0: +- Sonic Generations +https://www.youtube.com/watch?v=4qJ-xEZhGms: +- The Legend of Zelda A Link to the Past +- A Link to the Past +https://www.youtube.com/watch?v=0J5gN1DwNDE: +- The Elder Scrolls 4 Oblivion +- Elder Scrolls 4 +- Oblivion +https://www.youtube.com/watch?v=VTsD2FjmLsw: +- Mass Effect 2 +https://www.youtube.com/watch?v=yYTY9EkFSRU: +- Mortal Kombat Deception +https://www.youtube.com/watch?v=uhscMsBhNhw: +- Super Mario Bros +https://www.youtube.com/watch?v=Jk4P10nsq4c: +- Kingdom Hearts +https://www.youtube.com/watch?v=qDnaIfiH37w: +- Super Smash Bros for Wii U +- Super Smash Bros Wii U +https://www.youtube.com/watch?v=-8wo0KBQ3oI: +- Doom +https://www.youtube.com/watch?v=jLJLyneZGKc: +- Super Street Fighter 2 +https://www.youtube.com/watch?v=T5ASJvTgpJo: +- Call Of Duty Black Ops +https://www.youtube.com/watch?v=Gb33Qnbw520: +- Super Mario Land +https://www.youtube.com/watch?v=omvdFcr0z08: +- Mortal Kombat +https://www.youtube.com/watch?v=W4VTq0sa9yg: +- GTA San Andreas +- Grand Theft Auto San Andreas +https://www.youtube.com/watch?v=7Lj1aw4J8ZA: +- Kirby's Dream Land +- Kirbys Dream Land +https://www.youtube.com/watch?v=O5FsjHRIVrc: +- Overwatch +https://www.youtube.com/watch?v=kSA4U4jNq4E: +- Hitman +https://www.youtube.com/watch?v=kzvZE4BY0hY: +- Fallout 4 +https://www.youtube.com/watch?v=5vYsVk23oxA: +- Super Metroid +https://www.youtube.com/watch?v=TVEyGntyOZQ: +- Sonic 2 +https://www.youtube.com/watch?v=EWbfixMWg4g: +- Super Smash Bros +https://www.youtube.com/watch?v=Kxp1qIYy2jQ: +- Super Smash Bros Melee +https://www.youtube.com/watch?v=zeKE0NHUtUw: +- Super Smash Bros Brawl +https://www.youtube.com/watch?v=eWSU8YOa3jU: +- Super Smash Bros 4 +https://www.youtube.com/watch?v=oC83pBZ0Z2I: +- Super Smash Bros Ultimate +https://www.youtube.com/watch?v=b8JbXCe3NJk: +- Pokémon Red/Blue/Yellow +- Pokémon Red +- Pokemon Red +- Pokémon Blue +- Pokemon Blue +- Pokémon Yellow +- Pokemon Yellow +https://www.youtube.com/watch?v=O9FeFkMyBIY: +- The Legend of Zelda Ocarina of Time +- Zelda Ocarina of Time +https://www.youtube.com/watch?v=xtJcoZ9E9ZQ: +- WOW +- World of Warcraft +https://www.youtube.com/watch?v=oDo3wlDretk: +- Street Fighter 2 +https://www.youtube.com/watch?v=l2qD1h9mGF0: +- Sonic Unleashed +https://www.youtube.com/watch?v=Oa2TCaE4MGk: +- Sonic Adventure 2 +https://www.youtube.com/watch?v=WJRoRt155mA: +- Mega Man 2 +https://www.youtube.com/watch?v=N6noiPFB3io: +- Minecraft +https://www.youtube.com/watch?v=XcHDVwL-1W8: +- Call Of Duty WWII +- Call of Duty WW2 +https://www.youtube.com/watch?v=Cgh4CFzKCps: +- Cuphead +https://www.youtube.com/watch?v=mEgE5z6Sl7Y: +- Assassin's Creed 2 +https://www.youtube.com/watch?v=KmmNYTg-wSA: +- Donkey Kong +https://www.youtube.com/watch?v=dVPF-lboK14: +- God of War 3 +https://www.youtube.com/watch?v=vf2yKCCRE50: +- New Super Mario Bros Wii +https://www.youtube.com/watch?v=BqPgeWf8vNM: +- Minecraft +https://www.youtube.com/watch?v=23hJeaLotEw: +- The Legend of Zelda Skyward Sword +- Skyward Sword +https://www.youtube.com/watch?v=hMa4hZQbrms: +- Undertale +https://www.youtube.com/watch?v=LWVjNNKDYz8: +- Ryse Son of Rome +https://www.youtube.com/watch?v=EAwWPadFsOA: +- Mortal Kombat +https://www.youtube.com/watch?v=YEZhF_98cIc: +- Uncharted 4 +https://www.youtube.com/watch?v=LRKfd5EKBtI: +- Destiny +https://www.youtube.com/watch?v=TgjOBqD_ljk: +- Super Mario Galaxy +https://www.youtube.com/watch?v=cPWBG6_jn4Y: +- The Legend of Zelda Breath of the Wild +- Breath of the Wild +- Zelda Botw +https://www.youtube.com/watch?v=M-U3sVX2G3w: +- Metroid +https://www.youtube.com/watch?v=b62RgDBmly8: +- Fire Emblem +https://www.youtube.com/watch?v=d2tdSiQAF20: +- Sonic the Hedgehog 2006 +- Sonic the Hedgehog 06 +- Sonic 06 +- Sonic 2006 +https://www.youtube.com/watch?v=qI-Takf76RY: +- Mortal Kombat +https://www.youtube.com/watch?v=GFDvcR7Ozbg: +- Kingdom hearts Birth By Sleep +https://www.youtube.com/watch?v=F-QCjTBR0Pg: +- Halo 2 Anniversary +https://www.youtube.com/watch?v=c0SuIMUoShI: +- Super Mario Bros +https://www.youtube.com/watch?v=LnSp9rgfel8: +- Borderlands +https://www.youtube.com/watch?v=cpXXfCzSmJQ: +- Red Dead Redemption +https://www.youtube.com/watch?v=CMfx3MT8ge0: +- The Elder Scrolls 5 Skyrim +- Elder Scrolls 5 +- Skyrim +https://www.youtube.com/watch?v=LtmZJwbLjb0: +- Super Mario 3D World +https://www.youtube.com/watch?v=HGXo7LExG6o: +- Tetris +https://www.youtube.com/watch?v=aUS36INQoUw: +- Fire Emblem Shadow Dragon and the Blade of Light +- Shadow Dragon and the Blade of Light +https://www.youtube.com/watch?v=s7RRgF5Ve_E: +- Undertale +https://www.youtube.com/watch?v=MuVAaU63K7k: +- Sonic Heroes +https://www.youtube.com/watch?v=7lq5rr0RTeU: +- The Legend of Zelda Twilight Princess +- Twilight Princess +https://www.youtube.com/watch?v=S62IDJBdGWA: +- Mortal Kombat X +- Mortal Komba 10 +https://www.youtube.com/watch?v=_50vhftK66I: +- Super Mario World +https://www.youtube.com/watch?v=cZjSrW4p7r8: +- Sonic Adventure 2 +https://www.youtube.com/watch?v=e-nXonpnFHI: +- Monster Hunter Frontier G6 +https://www.youtube.com/watch?v=_dWMv2hfck0: +- Halo 2 +https://www.youtube.com/watch?v=SV0IF4XVBlE: +- Battlefield 1 +https://www.youtube.com/watch?v=rqIRZHPuiqE: +- Middle Earth Shadow of War +- Shadow of War +https://www.youtube.com/watch?v=qQGh4dy1cCA: +- Killzone 3 +https://www.youtube.com/watch?v=e9r5hx47kxM: +- Super Mario Odyssey +https://www.youtube.com/watch?v=SXKrsJZWqK0: +- Batman Arkham City +https://www.youtube.com/watch?v=Wbk8WeQU0rc: +- Far Cry +https://www.youtube.com/watch?v=4i8qAZOu5-g: +- Crash Bandicoot +https://www.youtube.com/watch?v=qEpaZR2Dvlg: +- Super Mario World +https://www.youtube.com/watch?v=-uEc8_dcYUc: +- Battlefield 4 +https://www.youtube.com/watch?v=jqE8M2ZnFL8: +- Grand Theft Auto 4 +https://www.youtube.com/watch?v=_kTBbTSjZpI: +- Shadow the Hedgehog +https://www.youtube.com/watch?v=QG77HTdreh0: +- Kid Icarus Uprising +https://www.youtube.com/watch?v=Z6NaZrPQGfY: +- Sonic Adventure 2 +https://www.youtube.com/watch?v=GkQUDi0pMAE: +- Marvel Ultimate Alliance +https://www.youtube.com/watch?v=o78T9-I4OGA: +- The Legend of Zelda The Wind Waker +- Zelda Wind Waker +- Zelda The Wind Waker +https://www.youtube.com/watch?v=dLDWYSQkUbE: +- Halo Combat Evolved +https://www.youtube.com/watch?v=mp9mzmq5Oas: +- Tokyo Mirage Sessions FE +https://www.youtube.com/watch?v=T5_vKsROSU0: +- Halo 2 +https://www.youtube.com/watch?v=MC0hV3dea9g: +- Sonic R +https://www.youtube.com/watch?v=dHkyqHlQ_is: +- Marvel Ultimate Alliance +https://www.youtube.com/watch?v=VfGeeZwv4Os: +- Diablo III +- Diablo 3 +https://www.youtube.com/watch?v=E3tkgU0pQmQ: +- Super Mario Sunshine +https://www.youtube.com/watch?v=53aDI5K49F4: +- Resident Evil 2 +https://www.youtube.com/watch?v=f43swtBDkuo: +- Marvel Ultimate Alliance +https://www.youtube.com/watch?v=cn144yWiZF4: +- Silent Hill 2 +https://www.youtube.com/watch?v=JSR5rROsvL0: +- Super Mario 3D Land +https://www.youtube.com/watch?v=CXxEUEK2_cs: +- Forza Motorsport 6 +https://www.youtube.com/watch?v=r8sB8N44eBE: +- Sonic Adventure (Theme of Knuckles) +- Sonic Adventure +https://www.youtube.com/watch?v=jrB6DLpIDaA: +- Rocket League +https://www.youtube.com/watch?v=6vY7S6iwwlQ: +- GTA San Andreas +- Grand Theft Auto San Andreas +https://www.youtube.com/watch?v=ndaJphlTPY8: +- Gears of War +https://www.youtube.com/watch?v=j9JWZ0mlX94: +- Luigi's Mansion +- Luigis Mansion +https://www.youtube.com/watch?v=zh4r_sXHyjc: +- Marvel Ultimate Alliance +https://www.youtube.com/watch?v=CvERHiTfx9w: +- Sonic Adventure 2 +https://www.youtube.com/watch?v=1TxEdJaZkY4: +- Dead by Daylight +https://www.youtube.com/watch?v=rxkiOLs06Z8: +- Injustice 2 +https://www.youtube.com/watch?v=bG0xho_yoaU: +- Marvel Ultimate Alliance +https://www.youtube.com/watch?v=xsyPHkeyz6o: +- Super Mario Galaxy 2 +https://www.youtube.com/watch?v=MdQvcgBmexw: +- Battlefront 2 +https://www.youtube.com/watch?v=6lR4nexN7RE: +- DOTA 2 +https://www.youtube.com/watch?v=m0NhFUw8mpI: +- Super Smash Bros Wii U +https://www.youtube.com/watch?v=E9s1ltPGQOo: +- Mii Channel +https://www.youtube.com/watch?v=QSBco8kVuZM: +- Fallout 3 +https://www.youtube.com/watch?v=Tz5Ld2STm2M: +- Ghost Recon +https://www.youtube.com/watch?v=oY9m2sHQwLs: +- Sonic R +https://www.youtube.com/watch?v=NF8kT_6giu8: +- Halo 2 +https://www.youtube.com/watch?v=EK3q3Jb3TCQ: +- The Elder Scrolls 5 Skyrim +- Elder Scrolls 5 +- Skyrim +https://www.youtube.com/watch?v=h4fOIVlEOIY: +- Assassin’s Creed Origins +- Assasins Creed Origins +https://www.youtube.com/watch?v=mziw3FQkZYg: +- Metroid Prime +https://www.youtube.com/watch?v=CXdNLpOrwFQ: +- Pikmin 3 +https://www.youtube.com/watch?v=RZFhVs6OGAc: +- Wii Fit +https://www.youtube.com/watch?v=UW-7Em3BYiA: +- Star Fox 64 +https://www.youtube.com/watch?v=9k-XrIGobsA: +- Super Mario 3D World +https://www.youtube.com/watch?v=kohY_vaMp0c: +- Bayonetta +https://www.youtube.com/watch?v=4roOuDyLkNo: +- Fallout 3 +https://www.youtube.com/watch?v=4wzRjZh3EkM: +- DOTA 2 +https://www.youtube.com/watch?v=OiwLQAA_V8E: +- Fire Emblem Fates +https://www.youtube.com/watch?v=UigzN-4JR14: +- Kingdom Hearts +https://www.youtube.com/watch?v=bIoPaeigMJw: +- Halo 2 +https://www.youtube.com/watch?v=JIE94pn4iJs: +- Killer Instinct +https://www.youtube.com/watch?v=Y9Jt52Q6GD4: +- Deadpool +https://www.youtube.com/watch?v=uNBzfe3TAEs: +- Shadow the Hedgehog +https://www.youtube.com/watch?v=PZVNi6L7g54: +- Mario Party 8 +https://www.youtube.com/watch?v=0uom8gJxe_8: +- Overwatch +https://www.youtube.com/watch?v=OCH6nQYflwY: +- Kirby's Dream Land +- Kirbys Dream Land +https://www.youtube.com/watch?v=PDURJZp3Sv4: +- World of Warcraft +- WoW +https://www.youtube.com/watch?v=3DYqiG1VmMQ: +- Diablo 2 +https://www.youtube.com/watch?v=xWSiztHV0us: +- Mortal Kombat Armageddon +https://www.youtube.com/watch?v=D6fkkVMsrAA: +- Super Smash Bros 3DS +- Mr Game and Watch +https://www.youtube.com/watch?v=tz82xbLvK_k: +- Undertale +https://www.youtube.com/watch?v=EV6E13xODyA: +- Bayonetta +https://www.youtube.com/watch?v=bq_jS6o3OoY: +- Super Mario 64 +https://www.youtube.com/watch?v=M3CSWLFL5u8: +- The Legend of Zelda The Wind Waker +- Zelda Wind Waker +- Zelda The Wind Waker +https://www.youtube.com/watch?v=PnzjqTvHJto: +- Destiny +https://www.youtube.com/watch?v=JntQ1X46qgg: +- Sonic Adventure 2 +https://www.youtube.com/watch?v=M9Ki3br3Qec: +- Deadrising +https://www.youtube.com/watch?v=bwaFADxmoeo: +- Splatoon +https://www.youtube.com/watch?v=Tc0fDGVQtJk: +- Metal Gear Solid +https://www.youtube.com/watch?v=ZvwQNlNIcr8: +- Forza Horizon 3 +https://www.youtube.com/watch?v=B7eRmpqtL40: +- Call of Duty Ghosts +https://www.youtube.com/watch?v=qURei6svd90: +- Doom II +- Doom 2 +https://www.youtube.com/watch?v=U1sYDWzhEog: +- Mario Party 8 +https://www.youtube.com/watch?v=aKKFjNVZkHU: +- Marvel vs Capcom 3 +https://www.youtube.com/watch?v=PDM2qukzKwg: +- Team Fortress 2 +https://www.youtube.com/watch?v=0nlJuwO0GDs: +- League of Legends +https://www.youtube.com/watch?v=UuL8nnyzf14: +- Fire Emblem The Sacred Stones +https://www.youtube.com/watch?v=ARFeJ3z6wes: +- Mario Party DS +https://www.youtube.com/watch?v=pNB4Cy4mftI: +- Sonic and the Black Knight +https://www.youtube.com/watch?v=GWba_XNUxtA: +- Quantum Break +https://www.youtube.com/watch?v=7XjrTV6M84c: +- Dead Rising 3 +https://www.youtube.com/watch?v=LjYWmShOoA4: +- Overwatch +https://www.youtube.com/watch?v=1pga6OLyrAg: +- Borderlands +https://www.youtube.com/watch?v=1OEyASR5C5U: +- Uncharted +https://www.youtube.com/watch?v=K-pJ1xMF7QE: +- League of Legends +https://www.youtube.com/watch?v=9q_MEnn5w0I: +- Minecraft +https://www.youtube.com/watch?v=5WduvI0CZ4c: +- Mega Man +https://www.youtube.com/watch?v=mB7veHQPF6M: +- Halo 2 +https://www.youtube.com/watch?v=mtxlFKP0OgY: +- Call of Duty Modern Warfare 2 +- Modern Warfare 2 +https://www.youtube.com/watch?v=2Jmty_NiaXc: +- Pokémon Red/Blue/Yellow +- Pokémon Red +- Pokemon Red +- Pokémon Blue +- Pokemon Blue +- Pokémon Yellow +- Pokemon Yellow +https://www.youtube.com/watch?v=6YQm2M-v4l0: +- Metal Gear +https://www.youtube.com/watch?v=r9spthA_nAA: +- Earthbound +https://www.youtube.com/watch?v=M_hUepNd52Q: +- Fallout New Vegas +https://www.youtube.com/watch?v=_X2A7qLDuLQ: +- GTA San Andreas +- Grand Theft Auto San Andreas +https://www.youtube.com/watch?v=8SM_39H-ztc: +- Kingdom Hearts +https://www.youtube.com/watch?v=Vj3dgToY_Fg: +- Ratchet and clank +https://www.youtube.com/watch?v=jgn3GSiYtEc: +- Middle earth Shadow of Mordor +- Shadow of Mordor +https://www.youtube.com/watch?v=sZ1f6BEkn_g: +- Paper Mario The Thousand-Year Door +- Paper Mario The Thousand Year Door +https://www.youtube.com/watch?v=T7AkuzZOSWM: +- DOTA 2 +https://www.youtube.com/watch?v=ixR9kOAR3ZY: +- Dragon Quest IX Sentinels of the Starry Skies +- Dragon Quest 9 +https://www.youtube.com/watch?v=WiQ1O5of0u4: +- Street Fighter II +- Street Fighter 2 +https://www.youtube.com/watch?v=0hEYvdMoF2g: +- The Legend of Zelda Ocarina of Time +- Zelda Ocarina of Time +https://www.youtube.com/watch?v=mK4mP7grUWY: +- Final Fantasy IX +- Final Fantasy 9 +https://www.youtube.com/watch?v=YzfEjZV46wM: +- Forza Motorsport +https://www.youtube.com/watch?v=vX7G4Qq6Li0: +- Shadow the Hedgehog +https://www.youtube.com/watch?v=95jDylAOsBQ: +- Assassins Creed +https://www.youtube.com/watch?v=EjazC45Qkww: +- Castlevania II Simon's Quest +- Castlevania 2 +https://www.youtube.com/watch?v=w09rc1VfWRQ: +- Forza Motorsport 4 +https://www.youtube.com/watch?v=Nc9w4QoSdfk: +- Tomb Raider +https://www.youtube.com/watch?v=VD9XmYnml_M: +- GTA V +- GTA 5 +- Grand Theft Auto 5 +https://www.youtube.com/watch?v=D5Z95cH7iDA: +- Ghost Recon Island Thunder +https://www.youtube.com/watch?v=o9Pu92St5O0: +- Metal Gear Solid 4 +https://www.youtube.com/watch?v=Bm2U7fqtSak: +- Ninja Gaiden +https://www.youtube.com/watch?v=NfCwWu8z8zE: +- Resident Evil +https://www.youtube.com/watch?v=zWHMj6ccSwA: +- Nintendo Land +https://www.youtube.com/watch?v=L8Vbg-1kPvg: +- Far Cry 5 +https://www.youtube.com/watch?v=hOju2ThS--M: +- Sonic the Hedgehog 2006 +- Sonic the Hedgehog 06 +- Sonic 06 +- Sonic 2006 +https://www.youtube.com/watch?v=EYW7-hNXZlM: +- Sonic CD +https://www.youtube.com/watch?v=ijEOqab3StE: +- Shadow the Hedgehog +https://www.youtube.com/watch?v=nJD-Ufi1jGk: +- The Elder Scrolls 3 Morrowind +- Elder Scrolls 3 +- Morrowind +https://www.youtube.com/watch?v=Iy4iQvJo24U: +- Crysis 2 +https://www.youtube.com/watch?v=o_ayLF9vdls: +- Dragon Age Origins +https://www.youtube.com/watch?v=LFcH84oNU6s: +- Skies of Arcadia +https://www.youtube.com/watch?v=9wMjq58Fjvo: +- Castle Crashers +https://www.youtube.com/watch?v=WA2WjP6sgrc: +- Donkey Kong Country Tropical Freeze +- Donkey Kong Tropical Freeze +https://www.youtube.com/watch?v=_Uzlm2MaCWw: +- MegaMan Maverick Hunter X +https://www.youtube.com/watch?v=Vm7aNxzWTJk: +- Prince of Persia +https://www.youtube.com/watch?v=viM0-3PXef0: +- The Witcher 3 +- Witcher 3 +https://www.youtube.com/watch?v=TO7UI0WIqVw: +- Super Smash Bros Brawl +https://www.youtube.com/watch?v=Y6ljFaKRTrI: +- Portal +https://www.youtube.com/watch?v=8yj-25MOgOM: +- Star Fox Zero +https://www.youtube.com/watch?v=W7rhEKTX-sE: +- Shovel Knight +https://www.youtube.com/watch?v=dTZ8uhJ5hIE: +- Kirby's Epic Yarn Music +- Kirbys Epic Yarn Music +https://www.youtube.com/watch?v=GQZLEegUK74: +- Goldeneye 007 +https://www.youtube.com/watch?v=zz8m1oEkW5k: +- Tetris Blitz +https://www.youtube.com/watch?v=ya3yxTbkh5s: +- Okami +https://www.youtube.com/watch?v=OYc_Vosp9_Y: +- The Legend of Zelda A Link Between Worlds +- Zelda A Link Between Worlds +https://www.youtube.com/watch?v=6LB7LZZGpkw: +- Silent Hill 2 +https://www.youtube.com/watch?v=TLYimAlnxEk: +- Mario Kart 8 +https://www.youtube.com/watch?v=-_Lwf6uaYOs: +- Kid Icarus Uprising +https://www.youtube.com/watch?v=H1etAFiAPYQ: +- Pikmin 3 +https://www.youtube.com/watch?v=jrbN2y5r-vU: +- Starcraft +https://www.youtube.com/watch?v=3o_RwQysgA8: +- The Last of Us +https://www.youtube.com/watch?v=75OlLWtV8gc: +- Pokémon Diamond/Pearl/Platinum +- Pokémon Diamond +- Pokemon Diamond +- Pokémon Pearl +- Pokemon Pearl +- Pokémon Platinum +- Pokemon Platinum +https://www.youtube.com/watch?v=sWTqNGUFF0g: +- Gunman Clive +https://www.youtube.com/watch?v=D4ragdexomw: +- Apollo Justice Ace Attorney +- Ace Attorney Apollo Justice +https://www.youtube.com/watch?v=poxDJHoYxCc: +- Gravity Rush +- Gravity Daze +https://www.youtube.com/watch?v=kwmOSMttZtM: +- Luigi's Mansion Dark Moon +- Luigis Mansion Dark Moon +https://www.youtube.com/watch?v=Ib19evMJJVI: +- PAYDAY The Heist +https://www.youtube.com/watch?v=7jVPwHqXz8A: +- Metroid Fusion +https://www.youtube.com/watch?v=utOa8ZMKwSA: +- Super Smash Bros Brawl +https://www.youtube.com/watch?v=2OOYZcVIp8w: +- Super Mario Galaxy +https://www.youtube.com/watch?v=u4jlk2x3Om0: +- Banjo-Kazooie Nuts & Bolts +- Banjo-Kazooie Nuts and Bolts +- Banjo Kazooie Nuts and Bolts +https://www.youtube.com/watch?v=mbsm2YXfAzo: +- Mortal Kombat Deception +https://www.youtube.com/watch?v=WB1I3PV7Ml8: +- Crash Bandicoot +https://www.youtube.com/watch?v=gy0T8nmw5GQ: +- Hitman 2 Silent Assassin +- Hitman 2 +https://www.youtube.com/watch?v=WuUVKOwUEA0: +- Super Mario Bros 3 +https://www.youtube.com/watch?v=6hgK3XJWSOk: +- Startropics +https://www.youtube.com/watch?v=_5iRzoSLWMM: +- The Legend of Zelda A Link to the Past +- A Link to the Past +https://www.youtube.com/watch?v=bV80prS5hBk: +- Silent Hill +https://www.youtube.com/watch?v=RDeJMAsrZMU: +- Sonic Adventure 2 +https://www.youtube.com/watch?v=QEmbOL3AAEs: +- Final Fantasy VI +- Final Fantasy 6 +https://www.youtube.com/watch?v=CoOTDI2KZco: +- Mega Man 3 +https://www.youtube.com/watch?v=Xeloqt4Wkcw: +- Streets Of Rage 2 +https://www.youtube.com/watch?v=wOQ5YyAorrw: +- Pokémon X & Y +- Pokemon X +- Pokemon Y +https://www.youtube.com/watch?v=CU1goWmsTRg: +- Duck Hunt +https://www.youtube.com/watch?v=oJqlpwN9mUM: +- Mario Party 2 +https://www.youtube.com/watch?v=ye78aoUtqOY: +- Pokémon GO +- Pokemon GO +https://www.youtube.com/watch?v=hf_D3pK6LVI: +- Phoenix Write Ace Attorney +https://www.youtube.com/watch?v=7qvudKHKozc: +- Shining Force II +- Shining Force 2 +https://www.youtube.com/watch?v=xPJcOybiCKM: +- Super Mario 64 +https://www.youtube.com/watch?v=IHxFTQYAqfc: +- Halo 3 +https://www.youtube.com/watch?v=0t9TlXAnDDM: +- Call of Juarez Gunslinger +https://www.youtube.com/watch?v=LuZy6A9luhU: +- Far Cry 3 +https://www.youtube.com/watch?v=HcvX3Md4uvM: +- Rainbow Six Vegas 2 +https://www.youtube.com/watch?v=RQYh342H5ec: +- Nintendo 3DS Nintendo Video +- Nintendo Video +https://www.youtube.com/watch?v=EMlQiS372Y0: +- Wii sports resort +https://www.youtube.com/watch?v=OtXHRSRQZvU: +- DK Rap +- Super Smash Bros Melee +- Donkey Kong 64 +https://www.youtube.com/watch?v=fmHEwxYdwcA: +- Mario & Luigi Bowsers Inside Story +- Bowsers Inside Story +- Mario and Luigi Bowsers Inside Story +https://www.youtube.com/watch?v=9cXiGjjLZAI: +- Pikmin 3 +https://www.youtube.com/watch?v=Q47iqNfsA80: +- The Legend of Zelda A Link Between Worlds +- A Link Between Worlds +https://www.youtube.com/watch?v=uQjcAvGTD7M: +- Professor Layton vs. Phoenix Wright Ace Attorney +- Professor Layton vs Phoenix Wright +https://www.youtube.com/watch?v=tBTb8H6vJe8: +- Donkey Kong 64 +https://www.youtube.com/watch?v=qQFseWcceDs: +- Pilotwings +https://www.youtube.com/watch?v=OVZ7EaqplsI: +- eShop Wii U +- eShop +https://www.youtube.com/watch?v=2hVsp2ncHoA: +- Marvel Ultimate Alliance +https://www.youtube.com/watch?v=kU36OmpCK_c: +- The Legend of Zelda Skyward Sword +- Skyward Sword +https://www.youtube.com/watch?v=7TCI6rdokD4: +- Street Fighter II +- Street Fighter 2 +https://www.youtube.com/watch?v=-unUysuRmWo: +- Sonic Adventure 2 +https://www.youtube.com/watch?v=QoIE04AqDgI: +- Kirby's Epic Yarn +- Kirbys Epic Yarn +https://www.youtube.com/watch?v=lSaDibas_Sw: +- Call of Duty Black Ops 3 +- Black Ops 3 +https://www.youtube.com/watch?v=HTUq3Ik1GHM: +- Kingdom Hearts II +- Kingdom Hearts 2 +https://www.youtube.com/watch?v=6ylyRyoUyH8: +- One Piece Pirate Warriors +https://www.youtube.com/watch?v=FJd6TMOZuJQ: +- Pac-Man +- Pac Man +https://www.youtube.com/watch?v=mBwmfDh7qtA: +- Dig Dug +https://www.youtube.com/watch?v=XjPF3AwVPM4: +- Finaly Fantasy +https://www.youtube.com/watch?v=KnoXA_EAnjo: +- Sonic Heroes +https://www.youtube.com/watch?v=XHpXEfa1kzk: +- Super Smash Bros Brawl +https://www.youtube.com/watch?v=mld2IxYXgt0: +- Super Robot Wars K +- SRW K +https://www.youtube.com/watch?v=Xi31NAktYDw: +- BioShock 2 +https://www.youtube.com/watch?v=YqyhezUoDj4: +- Call of Duty World at War +- World at War +https://www.youtube.com/watch?v=hOB4q01VCVg: +- Battlefield 1943 +https://www.youtube.com/watch?v=_cZoUYndFHo: +- Cuphead +https://www.youtube.com/watch?v=ojS8lCeUgsg: +- Deadpool +https://www.youtube.com/watch?v=2HTs2n0q-RA: +- Sonic Riders Zero Gravity +https://www.youtube.com/watch?v=yykgQClYlCk: +- TitanFall +https://www.youtube.com/watch?v=I6XMQYkPczI: +- Bravely Default +https://www.youtube.com/watch?v=q_TxCygjNIM: +- Deadrising +https://www.youtube.com/watch?v=2iLl_uusujo: +- STAR WARS The Old Republic +https://www.youtube.com/watch?v=KDyTjQmIWAw: +- Check Mii Out Channel Contest Menu Wii +- Check Mii Out +https://www.youtube.com/watch?v=VhjwXsDJQ2Y: +- Wii Sports +https://www.youtube.com/watch?v=fxMW5__vaDQ: +- Super Mario 64 +https://www.youtube.com/watch?v=4DJzm9cFGkg: +- Puyopuyo Tetris +- Puyo puyo Tetris +https://www.youtube.com/watch?v=G1MfL1oGN0E: +- Call of Duty Modern Warfare 3 +- Modern Warfare 3 +https://www.youtube.com/watch?v=bRLzfoySFWc: +- Assassin's Creed 4 Black Flag +- Assasins Creed 4 +- Assasins Creed Black Flag +https://www.youtube.com/watch?v=30lnA-haJiM: +- XBOX 360 Avatar Editor +- Avatar Editor +https://www.youtube.com/watch?v=czTksCF6X8Y: +- Spider-Man 2 The Game +- Spider Man 2 +- Spider-Man 2 +https://www.youtube.com/watch?v=_FAYOrpr-Fk: +- Wii Sports +https://www.youtube.com/watch?v=UbqoImQZ-EY: +- Touhou Embodiment of Scarlet Devil +- Touhou EoSD +- Embodiment of Scarlet Devil +https://www.youtube.com/watch?v=qIHqpjVxTtY: +- Super Stardust Delta +https://www.youtube.com/watch?v=VU7qTVkjMWU: +- World of Warcraft Wrath of the Lich King +- Wrath of the Lich King +- WoW WotLK +https://www.youtube.com/watch?v=YB2hQicc13Q: +- Super Smash Bros Melee +https://www.youtube.com/watch?v=9NBu_N9paNM: +- Metro Last Light +https://www.youtube.com/watch?v=aJOovInlk-w: +- Assassin's Creed 4 Black Flag +- Assasins Creed 4 +- Assasins Creed Black Flag +https://www.youtube.com/watch?v=6OYeyGi67UI: +- Call of Duty Ghosts +https://www.youtube.com/watch?v=IuLMkfG_Xu0: +- World of Warcraft +- WoW +https://www.youtube.com/watch?v=gR03h48G9rw: +- Destiny +https://www.youtube.com/watch?v=e87BLzzcwqQ: +- Halo Combat Evolved +https://www.youtube.com/watch?v=0_FquzflYYU: +- Luigis Mansion +- Luigi's Mansion +https://www.youtube.com/watch?v=I3XPus2T58o: +- The Legend of Zelda Twilight Princess +- Twilight Princess +https://www.youtube.com/watch?v=pVFYljt05hg: +- Metal Gear Solid 3 +https://www.youtube.com/watch?v=XHXVFBKZ9i8: +- Rhythm Heaven +https://www.youtube.com/watch?v=9hUT8VsLIG4: +- Pac-Man Championship Edition 2 +- Pac Man Championship Edition 2 +- Pac-Man CE 2 +- Pac Man CE 2 +https://www.youtube.com/watch?v=OziIZxI-Am4: +- Super Mario Maker +https://www.youtube.com/watch?v=frWaTmdN1Yk: +- Saga Frontier 2 +https://www.youtube.com/watch?v=r3m8_TDltwg: +- The Simpsons Hit & Run +- The Simpsons Hit and Run +https://www.youtube.com/watch?v=6kZ5TiSl2d8: +- Metal Gear Solid 2 +https://www.youtube.com/watch?v=CvL6fzXOJyY: +- Middle Earth Shadow of War +- Shadow of War +https://www.youtube.com/watch?v=zsxwbxS6LRM: +- Pocky & Rocky +- Pocky and Rocky +https://www.youtube.com/watch?v=YEo8vfvU6t8: +- Mario Golf World Tour +https://www.youtube.com/watch?v=q38jsbJ0afg: +- Marvel Ultimate Alliance 2 +https://www.youtube.com/watch?v=jbppvtGktvA: +- Super Street Fighter IV +- Super Street Fighter 4 +https://www.youtube.com/watch?v=kEdRtzNAh0g: +- Mortal Kombat +https://www.youtube.com/watch?v=SiLykPb_14g: +- Snipperclips +https://www.youtube.com/watch?v=2LhOV_ygYlY: +- Ghostbusters The Videogame +- Ghostbusters +https://www.youtube.com/watch?v=TUA9WwTV10M: +- Back to the Future the Game +- Back to the Future +https://www.youtube.com/watch?v=krISRDj6_2w: +- DSi Shop Theme +- DSi Shop +https://www.youtube.com/watch?v=R5DJENl__ZE: +- The Elder Scrolls 3 Morrowind +- Elder Scrolls 3 +- Morrowind +https://www.youtube.com/watch?v=2MDXwJ7Skik: +- Mortal Kombat 9 +https://www.youtube.com/watch?v=NKkXYxFh9M0: +- Killzone 3 +https://www.youtube.com/watch?v=5rfpGcIWe7E: +- BioShock 2 +https://www.youtube.com/watch?v=euJNO3xzprQ: +- Call of Duty Black Ops 3 +- Black Ops 3 +https://www.youtube.com/watch?v=L8EfpcF9rRc: +- Super Smash Bros Brawl +https://www.youtube.com/watch?v=LaM9YRUsDXs: +- Dark Souls +https://www.youtube.com/watch?v=nghTrcPBp3s: +- Yoshi's Story +- Yoshis Story +https://www.youtube.com/watch?v=l7vpszYDFEo: +- Destiny 2 +https://www.youtube.com/watch?v=ukMWZymYV-E: +- Borderlands +https://www.youtube.com/watch?v=ybhjox7LhmY: +- Mario Kart Wii +https://www.youtube.com/watch?v=9vevdlA1mFE: +- Pocky & Rocky 2 +- Pocky and Rocky 2 +https://www.youtube.com/watch?v=sZZgenIEL9o: +- Call of Duty Black Ops 4 +- Black Ops 4 +https://www.youtube.com/watch?v=OTF5gn5S0zs: +- Super Mario Bros Super Smash Bros Brawl +- Super Mario Bros +- Super Smash Bros Brawl +https://www.youtube.com/watch?v=OiWIpzor_Vk: +- Far Cry 5 +https://www.youtube.com/watch?v=h-xih93YIwk: +- Super Mario Bros 3 +https://www.youtube.com/watch?v=aHW0Vn-wRzs: +- Team Sonic Racing +https://www.youtube.com/watch?v=DehK_Y0TUbE: +- Angry Birds +https://www.youtube.com/watch?v=eVKj3u8JUm0: +- Super Mario 64 +https://www.youtube.com/watch?v=vCF_mG9T6-Y: +- Super Smash Bros Ultimate +- Super Smash Bros Melee +https://www.youtube.com/watch?v=o1rXwfuffwk: +- Battle Stadium D.O.N. +- Battle Stadium DON +https://www.youtube.com/watch?v=y_qHuDjE3CQ: +- Undertale +https://www.youtube.com/watch?v=kNK3gZnd3Qc: +- LEGO Marvel Super Heroes +https://www.youtube.com/watch?v=a6t_uyg_pF8: +- Final Fantasy VI +- Final Fantasy 6 +https://www.youtube.com/watch?v=Vc1wzDWFvf8: +- Super Smash Bros Melee +https://www.youtube.com/watch?v=4QR21UOXUTk: +- J Stars Victory Vs +https://www.youtube.com/watch?v=8avMLHvLwRQ: +- Wii Shop Channel +- Wii Shop +https://www.youtube.com/watch?v=d5c4KOopwLs: +- Wii Sports +https://www.youtube.com/watch?v=K5PNe0a04_I: +- Fire Emblem Echoes Shadows of Valentia +- Fire Emblem Echoes +- Shadows of Valentia +https://www.youtube.com/watch?v=8GxAi9b0ZvM: +- Super Mario Odyssey +https://www.youtube.com/watch?v=uTcOC04eEro: +- Yu-Gi-Oh! Duel Links +- Yu-Gi-Oh Duel Links +- YuGiOh Duel Links +- Yu Gi Oh Duel Links +https://www.youtube.com/watch?v=nTSXf2YVNUs: +- Final Fantasy III +- Final Fantasy 3 +https://www.youtube.com/watch?v=96uAHbo2f0o: +- Super Smash Bros Brawl +https://www.youtube.com/watch?v=AD0SP68Z2D0: +- Tom Clancys The Division +- The Division +https://www.youtube.com/watch?v=XmK8vhe6_yI: +- Destiny +https://www.youtube.com/watch?v=OM_qluyUGtY: +- Halo 2 +https://www.youtube.com/watch?v=KCQpsPFOEyU: +- Tom Clancys Rainbow Six Siege +- Rainbow Six Siege +- R6 Siege +https://www.youtube.com/watch?v=Rvi6c8toWJM: +- Counter-Strike Global Offensive +- CS GO +- CSGO +- Counter Strike GO +- Counter Strike Global Offensive +https://www.youtube.com/watch?v=WGkBan-hrR0: +- Ninja Gaiden 2 +https://www.youtube.com/watch?v=NPKG8HOyH50: +- Call of Duty Black Ops 4 +- Black Ops 4 +https://www.youtube.com/watch?v=Pva3UHPUHns: +- Luigis Mansion +https://www.youtube.com/watch?v=Gv__pDOd9R4: +- Donkey Kong 64 +https://www.youtube.com/watch?v=IJxIR35cWFg: +- Mario Party 8 +https://www.youtube.com/watch?v=6bWqgqcaQVs: +- Batman Arkham City +https://www.youtube.com/watch?v=FL6BRBzv8Ec: +- Anthem +https://www.youtube.com/watch?v=FG9uTzsqLNc: +- Counter-Strike Global Offensive +- CS GO +- CSGO +- Counter Strike GO +- Counter Strike Global Offensive +https://www.youtube.com/watch?v=-s9hGP8EPC4: +- Crackdown +https://www.youtube.com/watch?v=p7ew-8C3G_M: +- Killzone +https://www.youtube.com/watch?v=HyoZspjzaT8: +- Dragonball Z Budokai Tenkaichi 2 +https://www.youtube.com/watch?v=KveGWSYDxKY: +- Minecraft +https://www.youtube.com/watch?v=rXGdYmigdzw: +- Conker's Bad Fur Day +- Conkers Bad Fur Day +https://www.youtube.com/watch?v=bxJaHn4OcOk: +- Nintendo 3DS Camera Music +- 3DS Camera +https://www.youtube.com/watch?v=YZ3XjVVNagU: +- Undertale +https://www.youtube.com/watch?v=Rdeuye6BfmA: +- Marvel Ultimate Alliance +https://www.youtube.com/watch?v=rVzkd6B7LCA: +- Halo 3 +https://www.youtube.com/watch?v=IhQ45MHHZGA: +- Anthem +https://www.youtube.com/watch?v=rYpqRmsA94Y: +- Spider-Man PS4 +- Spider Man +https://www.youtube.com/watch?v=dFkuCBF-bf0: +- Sonic Colors +https://www.youtube.com/watch?v=jPfcnzPixBI: +- One Piece World Seeker +https://www.youtube.com/watch?v=KamjTqi4VjI: +- Kirby Air Ride +https://www.youtube.com/watch?v=7kCXNPZscFU: +- Dance Dance Revolution Mario Mix +- DDR Mario Mix +https://www.youtube.com/watch?v=CaXapWQeeeg: +- Spongebob Battle for Bikini Bottom +https://www.youtube.com/watch?v=cSiyDP_B5vY: +- The Witcher 3 Wild Hunt +- Witcher 3 +- Witcher Wild Hunt +https://www.youtube.com/watch?v=bGTWOWeFFw8: +- Marvel vs Capcom 3 +https://www.youtube.com/watch?v=Du0r5FOKbis: +- Bayonetta +https://www.youtube.com/watch?v=07mbmJHeBhg: +- Monopoly +https://www.youtube.com/watch?v=tPbnW2HmuIY: +- Super Mario Odyssey +https://www.youtube.com/watch?v=X6SVxsouuvE: +- Call of Duty Black Ops +- Black Ops +https://www.youtube.com/watch?v=zxgJTtumaV8: +- Tetris (Game Boy) - 10. Fanfare Victory (Varation) +https://www.youtube.com/watch?v=cOlCCc-_2sg: +- Dark Castle - Kirby's Dream Land 2 +https://www.youtube.com/watch?v=UDJtsfR51To: +- Xenoblade Chronicles OST - Gaur Plain +https://www.youtube.com/watch?v=tg-yC0xcI9s: +- Triggernometry - Red Dead Redemption Soundtrack +https://www.youtube.com/watch?v=rNNQBAj10n4: +- Marvel vs Capcom 3 +https://www.youtube.com/watch?v=bwhHD92bQUk: +- Sonic CD +https://www.youtube.com/watch?v=ZlTnLsjg59s: +- GTA V +- GTA 5 +- Grand Theft Auto 5 +https://www.youtube.com/watch?v=i-x22mnsUJ8: +- Gears 5 +- Gears of War 5 +https://www.youtube.com/watch?v=mOw3pZQctJ0: +- Call of Duty Advanced Warfare +https://www.youtube.com/watch?v=cnosICv2Mjg: +- Hotel Mario +https://www.youtube.com/watch?v=MMwM6hLEyeY: +- Sonic Mania +https://www.youtube.com/watch?v=XsliycURi0Y: +- Crackdown +https://www.youtube.com/watch?v=a51CkkWec1Q: +- Half-Life 2 +https://www.youtube.com/watch?v=iKBVXX8Vh7g: +- Call of Duty WWII +- Call of Duty WW2 +https://www.youtube.com/watch?v=qEvxbJROGuQ: +- Kirby Super Star +https://www.youtube.com/watch?v=_J4R6WsSKVs: +- Cuphead +https://www.youtube.com/watch?v=vcEGFAaSpzI: +- Final Fantasy VII +- Final Fantasy 7 +https://www.youtube.com/watch?v=cOWRNLaCMJg: +- Pokémon Red/Blue/Yellow +- Pokémon Red +- Pokemon Red +- Pokémon Blue +- Pokemon Blue +- Pokémon Yellow +- Pokemon Yellow +https://www.youtube.com/watch?v=XF2fA1epvLE: +- Super Mario World +https://www.youtube.com/watch?v=-tv0ZXF7uA4: +- GTA V +- GTA 5 +- Grand Theft Auto 5 +https://www.youtube.com/watch?v=3JncraxIMXY: +- Battlefield V +- Battlefield 5 +https://www.youtube.com/watch?v=fYnw_xoZCsg: +- Mario Teaches Typing 2 +https://www.youtube.com/watch?v=FQ73YtVm3Js: +- Spider-Man 2 The Game +- Spider Man 2 +- Spider-Man 2 +https://www.youtube.com/watch?v=nC5FK864qTk: +- Mortal Kombat Armageddon +https://www.youtube.com/watch?v=SnqPJv0UqKo: +- The Elder Scrolls 5 Skyrim +- Elder Scrolls 5 +- Skyrim +https://www.youtube.com/watch?v=pTE54izXTqc: +- Arkanoid R 2000 +- Arkanoid 2000 +- Arkanoid Returns +https://www.youtube.com/watch?v=FezNgPThD3M: +- Undertale +https://www.youtube.com/watch?v=DCX6c07rcJw: +- The World Ends With You +https://www.youtube.com/watch?v=hyQ6bn1Dno4: +- Sonic & All-Stars Racing Transformed +- Sonic and All Stars Racing Transformed +- Sonic and All-Stars Racing Transformed +- Sonic and Sega All-Stars Racing Transformed +- Sonic and Sega All Stars Racing Transformed +https://www.youtube.com/watch?v=okksreh2eQ4: +- Disney Universe +https://www.youtube.com/watch?v=3L97-DxwgRU: +- Billy The Wizard +https://www.youtube.com/watch?v=kY1F5QO5VOo: +- LittleBigPlanet +- Little Big Planet +https://www.youtube.com/watch?v=QEEPfzrDpoc: +- Bust a Groove +https://www.youtube.com/watch?v=JkO05j0zzZQ: +- LA Noire +https://www.youtube.com/watch?v=TNosg0EXQlU: +- Tom Clancys Rainbow Six Siege +- Rainbow Six Siege +- R6 Siege +https://www.youtube.com/watch?v=h38zN_vF_dQ: +- Fallout +https://www.youtube.com/watch?v=P9GX6Fyxr90: +- Snipperclips +https://www.youtube.com/watch?v=aKTJNO3HLEE: +- Xenoblade Chronicles +- Super Smash Bros Ultimate +https://www.youtube.com/watch?v=0gs56c9qGKA: +- Marvel Ultimate Alliance +https://www.youtube.com/watch?v=SzDmvY2pRJI: +- Yoshi's New Island +- Yoshis New Island +https://www.youtube.com/watch?v=IVH6Gl7W0hI: +- Super Mario Galaxy +https://www.youtube.com/watch?v=tZHcVMqXGhQ: +- Fire Emblem Echoes Shadows of Valentia +- Fire Emblem Echoes +- Shadows of Valentia +https://www.youtube.com/watch?v=Ltay2kMtCeM: +- Fortune Street +https://www.youtube.com/watch?v=mi3SE5j5XrU: +- Wizard101 +- Wizard 101 +https://www.youtube.com/watch?v=_nfld1C7RuE: +- Sonic and the Secret Rings +https://www.youtube.com/watch?v=nJB0OvU_vkg: +- Animal Crossing New Leaf +https://www.youtube.com/watch?v=xi0M9SIaLb4: +- Xenoblade Chronicles +https://www.youtube.com/watch?v=qw2lEHk5kDI: +- Gears of War 5 +- Gears 5 +https://www.youtube.com/watch?v=zlUWnUzpzSA: +- Halo 5 Guardians +https://www.youtube.com/watch?v=iUZyJHDasqA: +- Dark Souls II +- Dark Souls 2 +https://www.youtube.com/watch?v=jO3BZ9b1YsE: +- Nintendogs + Cats +- Nintendogs and Cats +https://www.youtube.com/watch?v=wDgQdr8ZkTw: +- Undertale +https://www.youtube.com/watch?v=O2MrJvaSjDE: +- Paper Mario Sticker Star +https://www.youtube.com/watch?v=umSJ--aGeYo: +- Portal +https://www.youtube.com/watch?v=JJOJdWubO-o: +- Plants vs Zombies Battle For Neighborville +- PvZ Battle for Neighborville +- Plants vs Zombies +https://www.youtube.com/watch?v=hUiAmrtxits: +- The Legend of Zelda The Wind Waker +- Zelda Wind Waker +- Zelda The Wind Waker +https://www.youtube.com/watch?v=1h9bU-y49-A: +- Hyper Street Fighter II The Anniversary Edition +- Hyper Street Fighter 2 +https://www.youtube.com/watch?v=sPWnlR5PEdQ: +- Call of Duty Advanced Warfare +https://www.youtube.com/watch?v=KF32DRg9opA: +- DuckTales +https://www.youtube.com/watch?v=rWatUe5yOP8: +- Sonic Heroes +https://www.youtube.com/watch?v=OTyghyyX34I: +- Paperboy 2 +https://www.youtube.com/watch?v=bGIP3IUgOg8: +- Super Mario Odyssey +https://www.youtube.com/watch?v=M8gJxIWSxGE: +- Battletoads +https://www.youtube.com/watch?v=btgi3TPL3AE: +- Castlevania +https://www.youtube.com/watch?v=rct233J9lsE: +- Dead Rising 2 Off The Record +- Dead Rising 2 +https://www.youtube.com/watch?v=Bz-QyMQDg_8: +- Life is Strange +https://www.youtube.com/watch?v=l2ETBzOAsV0: +- Sonic Forces +https://www.youtube.com/watch?v=6xzRa1EtOaw: +- Mario Strikers Charged +https://www.youtube.com/watch?v=6QpG48UEcI4: +- Fruit Ninja +https://www.youtube.com/watch?v=4n6WP9qHyRM: +- World of Warcraft +https://www.youtube.com/watch?v=R0cia14N9no: +- League of Legends +https://www.youtube.com/watch?v=iiNyQD5Yq3E: +- Star Wars Republic Commando +https://www.youtube.com/watch?v=nJ3o3kQZWm8: +- Parappa the Rapper +https://www.youtube.com/watch?v=2aV-ej4gwJU: +- Destiny 2 +https://www.youtube.com/watch?v=eCwzdl2q6Ho: +- King of Fighters XIII +- King of Fighters 13 +https://www.youtube.com/watch?v=PLDyWLbuptQ: +- Undertale +https://www.youtube.com/watch?v=3EdeEKDxhPE: +- Halo 3 +https://www.youtube.com/watch?v=em_oGE5Ht2Y: +- Resident Evil +https://www.youtube.com/watch?v=bQYuuUN7pSY: +- Mario Sports Mix +https://www.youtube.com/watch?v=cHJLdF2WNxI: +- World of Warcraft +https://www.youtube.com/watch?v=EXAb0Df3Rpc: +- Clannad +https://www.youtube.com/watch?v=LeeD7lWh2RE: +- Super Smash Bros Ultimate +https://www.youtube.com/watch?v=nJa7hJEfdww: +- Witcher 3 +https://www.youtube.com/watch?v=-cg5QXLS4nc: +- Cuphead +https://www.youtube.com/watch?v=8eDRM-YzpIs: +- Mario Sports Mix +https://www.youtube.com/watch?v=sunu9ShZ4xs: +- Mortal Kombat Armageddon +https://www.youtube.com/watch?v=n25nqibaIDg: +- Super Smash Bros Ultimate +https://www.youtube.com/watch?v=8KT7jcB72fQ: +- Donkey Kong 64 +https://www.youtube.com/watch?v=QCQ3DpI5hHk: +- Plants Vs Zombies +https://www.youtube.com/watch?v=Ze3G5EksIQw: +- The Stanley Parable +- Stanley Parable +https://www.youtube.com/watch?v=9pPQnhhUqtU: +- Total Distortion +https://www.youtube.com/watch?v=WSig9qC9tWE: +- Pokemon Sword and Shield +- Pokemon Sword +- Pokemon Shield +https://www.youtube.com/watch?v=5_E_y1AWAfc: +- Undertale +https://www.youtube.com/watch?v=TsfRhKE5iP8: +- Tomb Raider +https://www.youtube.com/watch?v=n5mo2zPBl3k: +- Middle Earth Shadow of Modor +- Shadow of Modor +https://www.youtube.com/watch?v=Z2g-xpGOusI: +- Tom Clancys The Division +- The Division +https://www.youtube.com/watch?v=S5DFhaprlRM: +- Super Smash Bros +https://www.youtube.com/watch?v=mtfsZv_wr98: +- Tekken +https://www.youtube.com/watch?v=Y8V_MvH8y8U: +- Fallout New Vegas +https://www.youtube.com/watch?v=4eGCH6tAycE: +- Xenoblade 2 +https://www.youtube.com/watch?v=HwJgHF0xDbU: +- Street Fighter V +- Street Fighter 5 +https://www.youtube.com/watch?v=taNV00DWLq8: +- ARMS +https://www.youtube.com/watch?v=9XzAPa4ji7M: +- Punch-Out!! +- Punch Out +- Punch-Out +https://www.youtube.com/watch?v=el_r5y_AcrE: +- Super Mario Sunshine +https://www.youtube.com/watch?v=7dRVuXYmB1M: +- Deltarune +https://www.youtube.com/watch?v=E_KMghGlDY8: +- Nintendo 3DS Music StreetPass Mii Plaza +- Nintendo 3DS +- StreetPass +- Mii Plaza +https://www.youtube.com/watch?v=QJbYjj5cNqQ: +- Wolfenstein The New Order +https://www.youtube.com/watch?v=qHpKdriwcOg: +- Kingdom Hearts +https://www.youtube.com/watch?v=jxMzCya8pQM: +- Payday 2 +https://www.youtube.com/watch?v=LQdUAMrcEnw: +- Farcry 5 +https://www.youtube.com/watch?v=r5Q4FOSPSx0: +- The Witcher 2 Assasins of Kings +- The Witcher 2 +https://www.youtube.com/watch?v=hU3K2a4uXxs: +- Sonic Unleashed +https://www.youtube.com/watch?v=2UKw9aheG3Y: +- Mafia 2 +https://www.youtube.com/watch?v=YTy9v9a7Tmo: +- Undertale +https://www.youtube.com/watch?v=rU_im8hx2TI: +- Earthbound +https://www.youtube.com/watch?v=sIL2fuxlMkc: +- Super Mario Galaxy +https://www.youtube.com/watch?v=-KBLgp5_smc: +- Mortal Kombat 9 +https://www.youtube.com/watch?v=Nsps0I58yUM: +- Dark Souls +https://www.youtube.com/watch?v=RjLWGyx4zoA: +- Persona 5 +https://www.youtube.com/watch?v=XXNVnq1r3yY: +- Castlevania Harmony of Despair +https://www.youtube.com/watch?v=6qIjHtI3kGU: +- Far Cry 5 +https://www.youtube.com/watch?v=3FRJU4DGZSc: +- Red Dead Redemption 2 +https://www.youtube.com/watch?v=zKYmW19Sk8s: +- Sekiro +https://www.youtube.com/watch?v=NYoP3_p2-N8: +- Super Mario 3D Land +https://www.youtube.com/watch?v=XnV_wHQ01dU: +- Cuphead +https://www.youtube.com/watch?v=MXAHCZXaoy8: +- Pokémon Red/Blue/Yellow +- Pokémon Red +- Pokemon Red +- Pokémon Blue +- Pokemon Blue +- Pokémon Yellow +- Pokemon Yellow +https://www.youtube.com/watch?v=56xw3mryadY: +- Mega Man 11 +https://www.youtube.com/watch?v=1K1rV9kFs6I: +- Sonic 3 +https://www.youtube.com/watch?v=7RiHmQcHa84: +- Shin Megami Tensei III Nocturne +- Shin Megami Tensei 3 Nocturne +- Shin Megami Tensei 3 +https://www.youtube.com/watch?v=aPGwniiEfIk: +- Escape from Mars Starring Taz +- Taz in Escape from Mars +- Escape from Mars +https://www.youtube.com/watch?v=kZnS_HxSdvo: +- Nintendo 3DS Find Mii +- Find Mii +https://www.youtube.com/watch?v=JEj_8VkTHVw: +- Mario + Rabbids Kingdom Battle +- Mario and Rabbids Kingdom Battle +https://www.youtube.com/watch?v=n8yKpcyU0fg: +- Hot Wheels Beat That +https://www.youtube.com/watch?v=cIKpAjB5B00: +- Super Hero Squad Online +https://www.youtube.com/watch?v=dYvvKyYIc5s: +- Touhou Embodiment of Scarlet Devil +- Touhou EoSD +- Embodiment of Scarlet Devil +https://www.youtube.com/watch?v=0BkHICB4eJs: +- Half Life +https://www.youtube.com/watch?v=dqwjST3UYcc: +- Mass Effect +https://www.youtube.com/watch?v=WQJxqXo8_cY: +- Tom Clancys Rainbow Six Siege +- Rainbow Six Siege +- R6 Siege +https://www.youtube.com/watch?v=4xbXFcadgy0: +- For Honor +https://www.youtube.com/watch?v=MJrZKCZ2pqo: +- Mario Kart Wii +https://www.youtube.com/watch?v=GgbXr4rN4D4: +- Banjo-Kazooie +- Banjo Kazooi +https://www.youtube.com/watch?v=oL5fbozc3kU: +- Yoshi's Island +- Yoshis Island +https://www.youtube.com/watch?v=DMDYpRaUvIQ: +- Sonic R +https://www.youtube.com/watch?v=oOMy4AZTPLc: +- Metropolis Street Racer +https://www.youtube.com/watch?v=Zf2qOWmKiz0: +- Deltarune +https://www.youtube.com/watch?v=6jFaoLrLzd4: +- Persona 3 +https://www.youtube.com/watch?v=UKYXOobaceQ: +- Sonic 3D Blast +https://www.youtube.com/watch?v=R9awS7E0jDc: +- Donkey Konga +https://www.youtube.com/watch?v=-VhWTfHZBOI: +- Contrast +https://www.youtube.com/watch?v=6QaH6tOXHek: +- Miitomo +https://www.youtube.com/watch?v=-GK41pLUuP0: +- New Super Mario Bros +https://www.youtube.com/watch?v=xjUHMsAZti4: +- Earthbound +https://www.youtube.com/watch?v=y4y13vnGXec: +- Psycho Soldier +https://www.youtube.com/watch?v=sEhg1qAmWMg: +- Codename Kids Next Door Operation V.I.D.E.O.G.A.M.E. +- Codename KND Operation VIDEOGAME +- Codename Kids Next Door Operation VIDEOGAME +- Codename Kids Next Door +https://www.youtube.com/watch?v=TAQVE2e3u8o: +- Power Rangers +https://www.youtube.com/watch?v=NRcsLULM2ns: +- Happy Wheels +https://www.youtube.com/watch?v=pU_zIsgnb3w: +- Hitman Blood Money +https://www.youtube.com/watch?v=ZXX8f0JcSG4: +- Pokémon Black & White +- Pokemon Black and White +- Pokemon Black +- Pokemon White +https://www.youtube.com/watch?v=AWkOMVUJ7gs: +- Fire Emblem Fates Condemnation +- Fire Emblem Fates +https://www.youtube.com/watch?v=QyPR77rg1to: +- Undertale +https://www.youtube.com/watch?v=sxJ-LSBfrQ4: +- Splatoon 2 +https://www.youtube.com/watch?v=RZVyHH-voR8: +- Dark Souls +https://www.youtube.com/watch?v=nWfRO1yxaXs: +- Wario world +- Warioworld +https://www.youtube.com/watch?v=bMF_PRSXbXk: +- Half-Life Opposing Force +https://www.youtube.com/watch?v=XVpC7Tg-hH4: +- Dreams +https://www.youtube.com/watch?v=EkLvDaWNez8: +- New Super Mario Bros +https://www.youtube.com/watch?v=7LvsWVE9-es: +- Pokémon X & Y +- Pokemon X +- Pokemon Y +https://www.youtube.com/watch?v=iAWXT2HJFTI: +- The Legend of Zelda Breath of the Wild +- Breath of the Wild +- Zelda Botw +https://www.youtube.com/watch?v=JZw9wWmcXH4: +- Super Smash Bros Brawl +https://www.youtube.com/watch?v=XJsy5jJ7Dp0: +- Trauma Center Under the Knife 2 +- Trauma Center 2 +- Trauma Center Under the Knife +https://www.youtube.com/watch?v=gd22rCkvJoA: +- Assassin's Creed 4 Black Flag +- Assasins Creed 4 +- Assasins Creed Black Flag +https://www.youtube.com/watch?v=ECE7ZeDsuPk: +- Destiny 2 +https://www.youtube.com/watch?v=v0nlnUDKT1s: +- Fallout 4 +https://www.youtube.com/watch?v=zgLgSQPN0Gw: +- Super Mario Party +https://www.youtube.com/watch?v=Ib3FTf3Frbo: +- Fire Emblem Three Houses +- Super Smash Bros Ultimate +https://www.youtube.com/watch?v=LG0-UMOljBU: +- Pokémon Gold/Silver/Crystal +- Pokemon Gold +- Pokemon Silver +- Pokemon Crystal +https://www.youtube.com/watch?v=0cuhPPuTA3s: +- Saints Row The Third +https://www.youtube.com/watch?v=1scgMvQoLVg: +- Tomodachi Life +https://www.youtube.com/watch?v=D8ShsHeYkeI: +- Dark Souls 2 +https://www.youtube.com/watch?v=BG-0WXsZR-A: +- Marvel vs Capcom 2 +https://www.youtube.com/watch?v=NPjLgDF1hUg: +- The Legend of Zelda Ocarina of Time +- Zelda Ocarina of Time +https://www.youtube.com/watch?v=upPtP_qtQHM: +- Super Paper Mario +https://www.youtube.com/watch?v=5xk8-dDqu50: +- Luigi's Mansion +- Luigis Mansion +https://www.youtube.com/watch?v=zop1cFVz1So: +- Killzone +https://www.youtube.com/watch?v=WLLE_PvJ5mY: +- Metroid Prime +https://www.youtube.com/watch?v=oMed0AQt6g4: +- Mass Effect +https://www.youtube.com/watch?v=S7NuuyVIYO8: +- Hitman +https://www.youtube.com/watch?v=ENWDVA0Xz88: +- Crackdown +https://www.youtube.com/watch?v=dQNAVqW1shA: +- Banjo-Kazooie +- Banjo Kazooie +https://www.youtube.com/watch?v=tnzziAe9tYA: +- Super Mario Galaxy 2 +https://www.youtube.com/watch?v=Ma7mdIHy-kA: +- Sonic Forces +https://www.youtube.com/watch?v=avyasO9uqfo: +- Donkey Kong Country +https://www.youtube.com/watch?v=i6v6w0mbPvU: +- A Hat in Time Music +https://www.youtube.com/watch?v=Df1nqnxRe7g: +- Left 4 Dead 2 +https://www.youtube.com/watch?v=QunjkirEY_g: +- Fire Emblem Fates +https://www.youtube.com/watch?v=-9bcB4_adYo: +- Crysis +https://www.youtube.com/watch?v=QMEqfXYjyvM: +- Deadspace +https://www.youtube.com/watch?v=a75SxyVBvKU: +- Half-Life 2 +- Half Life 2 +https://www.youtube.com/watch?v=5h1t94LxGd0: +- The Witcher 2 Assasins of Kings +- The Witcher 2 +https://www.youtube.com/watch?v=Eoo5oifOsUs: +- DayZ +https://www.youtube.com/watch?v=24ijaaCSrYM: +- For Honor +https://www.youtube.com/watch?v=sGjdjdyt5s4: +- Kirby Planet Robobot +https://www.youtube.com/watch?v=oA6kuCUmBpo: +- Captain Toad Treasure Tracker +https://www.youtube.com/watch?v=x9dJrEvHk3o: +- Mario & Sonic at the London 2012 Olympic Games +- Mario and Sonic at the London 2012 Olympic Games +https://www.youtube.com/watch?v=DheEtF1G_Rs: +- Splatoon 2 +https://www.youtube.com/watch?v=TcufdwbnF0k: +- Rhythm Heaven Fever +https://www.youtube.com/watch?v=4GChwuX81o4: +- Grabbed by the Ghoulies +https://www.youtube.com/watch?v=PhWge4ro354: +- Call of Duty Black Ops 2 +- Black Ops 2 +https://www.youtube.com/watch?v=bzbWrGr_of0: +- The Legend of Zelda Majora's Mask +- Zelda Majora's Mask +- Zelda Majoras Mask +https://www.youtube.com/watch?v=2RKnfOAI1A0: +- 3D Dot Game Heroes +https://www.youtube.com/watch?v=L2f5v1w_v8g: +- Conker Live & Reloaded +- Conker Live and Reloaded +https://www.youtube.com/watch?v=nVRQJ2i59XE: +- Wii Sports +https://www.youtube.com/watch?v=JDqJa1RC3q8: +- Kid Icarus Uprising +https://www.youtube.com/watch?v=Jq6oRi6UTXY: +- Sonic the Hedgehog 3 +https://www.youtube.com/watch?v=JzzJwrIN6Mc: +- Mega Man 3 +https://www.youtube.com/watch?v=s1AkaZuRaM4: +- Sonic Dash +https://www.youtube.com/watch?v=7S7WEkopK8k: +- Dance Dance Revolution Mario Mix +- Dance Dance Revolution Mario +https://www.youtube.com/watch?v=KVwdKcwWe2k: +- Yooka-Laylee Music +- Yooka Laylee +https://www.youtube.com/watch?v=ZdBO_V-guJ4: +- Call of Duty Modern Warfare +- Modern Warfare +https://www.youtube.com/watch?v=NQIeAbFT4Kc: +- League of Legends +https://www.youtube.com/watch?v=cnAoE7tcrBI: +- Max Payne +https://www.youtube.com/watch?v=pYf0C4Yh8pE: +- Apex Legends +https://www.youtube.com/watch?v=0W-trrXXcS8: +- Far Cry New Dawn +https://www.youtube.com/watch?v=jv249KSsjcM: +- Red Dead Redemption 2 +https://www.youtube.com/watch?v=z5JhMEP0GSU: +- Mario Party The Top 100 +- Mario Party Top 100 +https://www.youtube.com/watch?v=3nxFYoGNJKc: +- Banjo-Kazooie +- Banjo Kazooie +https://www.youtube.com/watch?v=6vICKfFo6bU: +- Wii Fit +https://www.youtube.com/watch?v=EJCq_hnFZyk: +- Fire Emblem Three Houses +https://www.youtube.com/watch?v=bXM7CvKCSx0: +- GTA San Andreas +- Grand Theft Auto San Andreas +https://www.youtube.com/watch?v=5YjGmTI_fPU: +- Bayonetta +https://www.youtube.com/watch?v=5wApmRF5gyM: +- Super Mario Bros 3 +https://www.youtube.com/watch?v=kRMTqfxkCsc: +- Mortal Kombat Deadly Alliance +https://www.youtube.com/watch?v=Ljqe4Nj7nBA: +- The Legend of Zelda Ocarina of Time +- Zelda Ocarina of Time +https://www.youtube.com/watch?v=RB4nFoA63rs: +- Gravity Rush +https://www.youtube.com/watch?v=UsaoAl5EIH8: +- Punch-Out!! +- Punch Out +- Punch-Out +https://www.youtube.com/watch?v=rMudHClToL0: +- Donkey Kong Country Tropical Freeze +- Donkey Kong Tropical Freeze +https://www.youtube.com/watch?v=iT8T-dBwztg: +- FlingSmash +https://www.youtube.com/watch?v=dcp5gPf7mhI: +- Jet Set Radio Future +https://www.youtube.com/watch?v=sC0cvwnG0Ik: +- Crazy Bus +https://www.youtube.com/watch?v=KJfMS5XzmIc: +- Animal Crossing New Horizons +https://www.youtube.com/watch?v=HAYPkVeWjXM: +- Halo 5 Guardians +https://www.youtube.com/watch?v=ozLqx3EboU4: +- Shantae Half-Genie Hero +- Shantae Half Genie Hero +- 'Shantae 1/2 Genie Hero' +https://www.youtube.com/watch?v=e2Gyaqf7EoU: +- Persona 3 +https://www.youtube.com/watch?v=KC5kHf58GMI: +- The Legend of Zelda Skyward Sword +- Skyward Sword +https://www.youtube.com/watch?v=XHzlfCpmqsw: +- Super Mario Kart +https://www.youtube.com/watch?v=6l1HUDuzyrk: +- Sonic Lost World +https://www.youtube.com/watch?v=512pcIZOjm0: +- Xenoblade Chronicles +https://www.youtube.com/watch?v=7ePFxuu9lik: +- The Walking Dead Game +- The Walking Dead +https://www.youtube.com/watch?v=rgq0wF_ByO8: +- Contrast +https://www.youtube.com/watch?v=O7eGxoFLOI4: +- Mario Kart Double Dash!! +- Mario Kart Double Dash +https://www.youtube.com/watch?v=6ppBSY92rzg: +- EarthBound +https://www.youtube.com/watch?v=mmiWjG0N21w: +- Kingdom Hearts +https://www.youtube.com/watch?v=kNlVo5udnPs: +- Team Fortress 2 +https://www.youtube.com/watch?v=-t_C-CgJ-3A: +- Pokemon Sun & Moon +- Pokemon Sun +- Pokemon Moon +https://www.youtube.com/watch?v=7G_aaak-tDE: +- The Legend of Zelda Majora's Mask +- Zelda Majora's Mask +- Zelda Majoras Mask +https://www.youtube.com/watch?v=rGFic2e-uTE: +- Kirby's Epic Yarn +- Kirbys Epic Yarn +https://www.youtube.com/watch?v=1Cfin7GJUGI: +- Mario vs Donkey Kong 2 March of the Minis +- Mario vs Donkey Kong 2 +https://www.youtube.com/watch?v=gtKxgf_lETg: +- Rhythm Heaven Fever +https://www.youtube.com/watch?v=xnmxhE9usbI: +- Animal Crossing +https://www.youtube.com/watch?v=kwQ6_rwvMp8: +- ARMS +https://www.youtube.com/watch?v=2odP8HQuPgs: +- Nintendogs +https://www.youtube.com/watch?v=9ICz9BArKiM: +- Splatoon 2 +https://www.youtube.com/watch?v=IeosKeE1psE: +- Super Street Fighter IV +- Super Street Fighter 4 +https://www.youtube.com/watch?v=P9lLD_hgbU4: +- Bayonetta +https://www.youtube.com/watch?v=QTRsbrFWomM: +- Fire Emblem Fates +https://www.youtube.com/watch?v=zwOnn-ObtNo: +- Chibi-Robo! +- Chibi-Robo +- Chibi Robo +https://www.youtube.com/watch?v=5FYneJvqtMc: +- Lego Dimensions +https://www.youtube.com/watch?v=jkk6f817nNU: +- Xenoblade Chronicles +https://www.youtube.com/watch?v=N8OHSXvneOE: +- Celeste +https://www.youtube.com/watch?v=ryAv7ESrJtQ: +- Half Life 2 +https://www.youtube.com/watch?v=vkailb3xcTI: +- Super Mario RPG +https://www.youtube.com/watch?v=MOkWNuGfCD0: +- Pokémon X & Y +- Pokemon X +- Pokemon Y +https://www.youtube.com/watch?v=JUYAX3z2JYc: +- Call of Duty Black Ops 2 +- Black Ops 2 +https://www.youtube.com/watch?v=kfo7hmy-qso: +- Bomberman Hero +https://www.youtube.com/watch?v=myxM9Q72hxE: +- NES Remix Pack +https://www.youtube.com/watch?v=PbYoMpwHq5c: +- Wii Play +https://www.youtube.com/watch?v=-NuIPxOmpmc: +- Star Fox +https://www.youtube.com/watch?v=Ga9wY3CzEJQ: +- Gex Enter the Gecko +- Gex +https://www.youtube.com/watch?v=mpqct1zMqTE: +- Killer Instinct S1 +- Killer Instinct +https://www.youtube.com/watch?v=g3jCAyPai2Y: +- Yakuza +https://www.youtube.com/watch?v=4GfXKM-2nak: +- Paper Mario The Origami King +https://www.youtube.com/watch?v=6rW52ajkrZ0: +- Street Fighter V +- Street Fighter 5 +https://www.youtube.com/watch?v=zLs96hgloOA: +- Sekiro +https://www.youtube.com/watch?v=qG_lzlLH3UU: +- Titanfall +https://www.youtube.com/watch?v=U8ilLSEjlAQ: +- Destiny 2 +https://www.youtube.com/watch?v=QFaZC_R3WoE: +- Resident Evil 3 +https://www.youtube.com/watch?v=1fmhHRNzZLo: +- Gears Of War +https://www.youtube.com/watch?v=Oc44myRhNRM: +- Pokemon Omega Ruby/Alpha Sapphire +- Pokemon Omega Ruby +- Pokemon Alpha Sapphire +https://www.youtube.com/watch?v=vNVz6N-bNZs: +- Sonic Adventure +https://www.youtube.com/watch?v=WcabIuZOWYY: +- DragonBall Xenoverse 2 +https://www.youtube.com/watch?v=GmpB0btoC6M: +- Yoshi's Cookie +- Yoshis Cookie +https://www.youtube.com/watch?v=KC-B3Q0cRrY: +- Streets Of Rage 2 +https://www.youtube.com/watch?v=DekKId4f-Yk: +- Super Mario 3D World +https://www.youtube.com/watch?v=xorWgC8vf3I: +- Fallout 3 +https://www.youtube.com/watch?v=X9Q4Zg1tYno: +- Super Smash Bros Wii U +https://www.youtube.com/watch?v=llOvd2yGEio: +- Team Fortress 2 +https://www.youtube.com/watch?v=RR_ifniLcW4: +- Donkey Kong Country 2 Diddy's Kong Quest +- Donkey Kong Country 2 +- Diddys Kong Quest +https://www.youtube.com/watch?v=NoBRdekWKxI: +- The Legend of Zelda Twilight Princess +- Twilight Princess +https://www.youtube.com/watch?v=rpocm85Hauc: +- Crash Tag Team Racing +https://www.youtube.com/watch?v=-4FvqXfQYbk: +- Dragon Ball FighterZ +https://www.youtube.com/watch?v=xgcyH9I1NBE: +- Sonic Adventure +https://www.youtube.com/watch?v=gESYyh_tsGA: +- Halo Infinite +https://www.youtube.com/watch?v=fXWCotTeNgk: +- Donkey Kong 64 +https://www.youtube.com/watch?v=tWBKtLiYe5w: +- Persona 4 Dancing All Night +- Persona 4 +https://www.youtube.com/watch?v=pQ-bjZD1EnI: +- Sonic Heroes +https://www.youtube.com/watch?v=4S9qb1jZGlA: +- Ghost of Tsushima +https://www.youtube.com/watch?v=v1M8VVDFxgo: +- Celeste +https://www.youtube.com/watch?v=ROBc_MVW-Lg: +- Fall Guys Ultimate Knockout +- Fall Guys +https://www.youtube.com/watch?v=r5EjvtyLJ1g: +- Sonic the Hedgehog CD +- Sonic CD +https://www.youtube.com/watch?v=JRPXRHS4XNQ: +- Battletoads +https://www.youtube.com/watch?v=hHKmORyO8WA: +- Mario Super Sluggers +https://www.youtube.com/watch?v=oeEiOtndLB0: +- Crusader Kings 2 +https://www.youtube.com/watch?v=TKvjEQXKeec: +- Earthbound +https://www.youtube.com/watch?v=m65ns26m9IE: +- A Hat in Time +https://www.youtube.com/watch?v=feM7-QL9x38: +- Bomberman Max 2 +https://www.youtube.com/watch?v=EGeoG0KQa68: +- Mappy Arcade +- Mappy +https://www.youtube.com/watch?v=ySYwqTKRtgQ: +- Mega Man 2 +https://www.youtube.com/watch?v=VCCZe8S5miY: +- Pac-Man World +- Pac Man World +https://www.youtube.com/watch?v=g-aNgman8Wo: +- Street Fighter 5 +https://www.youtube.com/watch?v=U1_PStq_tTM: +- Watch Dogs +https://www.youtube.com/watch?v=vLljFmx7yF8: +- Counter-Strike Global Offensive +- CS GO +- CSGO +- Counter Strike GO +- Counter Strike Global Offensive +https://www.youtube.com/watch?v=gsZJg5mmsHA: +- The Angry Video Game Nerd Adventures +- Angry Video Game Nerd Adventures +https://www.youtube.com/watch?v=n3c2xbYWKY0: +- Double Dragon +https://www.youtube.com/watch?v=JNEKqu6Pe48: +- Pac Man Party +- Pac-Man Party +https://www.youtube.com/watch?v=sEFIZh_Zscc: +- Bioshock +https://www.youtube.com/watch?v=fmU9MXi9Uz0: +- Under Night In-Birth +- Under Night In Birth +https://www.youtube.com/watch?v=EzDLLILqvLU: +- Asura's Wrath +- Asuras Wrath From b9d8be397c04053d3542c2acc8e5b5f30fd9c4ca Mon Sep 17 00:00:00 2001 From: bobloy Date: Mon, 28 Sep 2020 09:50:43 -0400 Subject: [PATCH 32/73] Add workflow labeler --- .github/labeler.yml | 62 +++++++++++++++++++++++++++++++++++ .github/workflows/labeler.yml | 0 2 files changed, 62 insertions(+) create mode 100644 .github/labeler.yml create mode 100644 .github/workflows/labeler.yml diff --git a/.github/labeler.yml b/.github/labeler.yml new file mode 100644 index 0000000..dd944c8 --- /dev/null +++ b/.github/labeler.yml @@ -0,0 +1,62 @@ +'cog: announcedaily': + - announcedaily/* +'cog: audiotrivia': + - audiotrivia/* +'cog: ccrole': + - ccrole/* +'cog: chatter': + - chatter/* +'cog: conquest': + - conquest/* +'cog: dad': + - dad/* +'cog: exclusiverole': + - exclusiverole/* +'cog: fifo': + - fifo/* +'cog: firstmessage': + - firstmessage/* +'cog: flag': + - flag/* +'cog: forcemention': + - forcemention/* +'cog: hangman': + - hangman +'cog: infochannel': + - infochannel/* +'cog: isitdown': + - isitdown/* +'cog: launchlib': + - launchlib/* +'cog: leaver': + - leaver/* +'cog: lovecalculator': + - lovecalculator/* +'cog: lseen': + - lseen/* +'cog: nudity': + - nudity/* +'cog: planttycoon': + - planttycoon/* +'cog: qrinvite': + - qrinvite/* +'cog: reactrestrict': + - reactrestrict/* +'cog: recyclingplant': + - recyclingplant/* +'cog: rpsls': + - rpsls/* +'cog: sayurl': + - sayurl/* +'cog: scp': + - scp/* +'cog: stealemoji': + - stealemoji/* +'cog: timerole': + - timerole/* +'cog: tts': + - tts/* +'cog: unicode': + - unicode/* +'cog: werewolf': + - werewolf \ No newline at end of file diff --git a/.github/workflows/labeler.yml b/.github/workflows/labeler.yml new file mode 100644 index 0000000..e69de29 From ad66d171d4035dc17794b346563a87d5f806d61e Mon Sep 17 00:00:00 2001 From: bobloy Date: Mon, 28 Sep 2020 09:52:17 -0400 Subject: [PATCH 33/73] Forgot the contents --- .github/workflows/labeler.yml | 19 +++++++++++++++++++ 1 file changed, 19 insertions(+) diff --git a/.github/workflows/labeler.yml b/.github/workflows/labeler.yml index e69de29..7c724a6 100644 --- a/.github/workflows/labeler.yml +++ b/.github/workflows/labeler.yml @@ -0,0 +1,19 @@ +# This workflow will triage pull requests and apply a label based on the +# paths that are modified in the pull request. +# +# To use this workflow, you will need to set up a .github/labeler.yml +# file with configuration. For more information, see: +# https://github.com/actions/labeler + +name: Labeler +on: [pull_request] + +jobs: + label: + + runs-on: ubuntu-latest + + steps: + - uses: actions/labeler@v2 + with: + repo-token: "${{ secrets.GITHUB_TOKEN }}" From af41d079d336426626f7dbfb5c948579fd17781f Mon Sep 17 00:00:00 2001 From: bobloy Date: Mon, 28 Sep 2020 10:52:45 -0400 Subject: [PATCH 34/73] Add black check --- .github/workflows/black_check.yml | 20 ++++++++++++++++++++ 1 file changed, 20 insertions(+) create mode 100644 .github/workflows/black_check.yml diff --git a/.github/workflows/black_check.yml b/.github/workflows/black_check.yml new file mode 100644 index 0000000..ab2a435 --- /dev/null +++ b/.github/workflows/black_check.yml @@ -0,0 +1,20 @@ +# GitHub Action that uses Black to reformat the Python code in an incoming pull request. +# If all Python code in the pull request is compliant with Black then this Action does nothing. +# Othewrwise, Black is run and its changes are committed back to the incoming pull request. +# https://github.com/cclauss/autoblack + +name: autoblack +on: [push, pull_request] +jobs: + build: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v1 + - name: Set up Python 3.8 + uses: actions/setup-python@v2 + with: + python-version: '3.8' + - name: Install Black + run: pip install --upgrade --no-cache-dir black + - name: Run black --check . + run: black --check --diff -l 99 . \ No newline at end of file From a36a800b45d105b6f6dced7c250f0e5efeb178f9 Mon Sep 17 00:00:00 2001 From: bobloy Date: Mon, 28 Sep 2020 10:53:00 -0400 Subject: [PATCH 35/73] named black --- .github/workflows/black_check.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/black_check.yml b/.github/workflows/black_check.yml index ab2a435..a36b5f2 100644 --- a/.github/workflows/black_check.yml +++ b/.github/workflows/black_check.yml @@ -3,7 +3,7 @@ # Othewrwise, Black is run and its changes are committed back to the incoming pull request. # https://github.com/cclauss/autoblack -name: autoblack +name: black on: [push, pull_request] jobs: build: From 8a42b87bd63e0c1660aadef4cb5e76b4ab12159d Mon Sep 17 00:00:00 2001 From: bobloy Date: Mon, 28 Sep 2020 10:54:47 -0400 Subject: [PATCH 36/73] black --- werewolf/builder.py | 2 +- werewolf/game.py | 5 ++--- werewolf/roles/seer.py | 10 +++++++--- werewolf/votegroups/wolfvote.py | 2 +- 4 files changed, 11 insertions(+), 8 deletions(-) diff --git a/werewolf/builder.py b/werewolf/builder.py index 4c803cc..f57a669 100644 --- a/werewolf/builder.py +++ b/werewolf/builder.py @@ -27,7 +27,7 @@ log = logging.getLogger("red.fox_v3.werewolf.builder") ROLE_DICT = {name: cls for name, cls in roles.__dict__.items() if isinstance(cls, type)} ROLE_LIST = sorted( [cls for cls in ROLE_DICT.values()], - key=attrgetter('alignment'), + key=attrgetter("alignment"), ) log.debug(f"{ROLE_DICT=}") diff --git a/werewolf/game.py b/werewolf/game.py index df345af..79d8455 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -1,9 +1,8 @@ import asyncio -import inspect import logging import random from collections import deque -from typing import List, Any, Dict, Set, Union +from typing import Dict, List, Union import discord from redbot.core import commands @@ -338,7 +337,7 @@ class Game: if check(): return await self.village_channel.send( - embed=discord.Embed(title=f"**{HALF_DAY_LENGTH/60} minutes of daylight remain...**") + embed=discord.Embed(title=f"**{HALF_DAY_LENGTH / 60} minutes of daylight remain...**") ) await asyncio.sleep(HALF_DAY_LENGTH) # 4 minute days FixMe to 120 later diff --git a/werewolf/roles/seer.py b/werewolf/roles/seer.py index 32ace18..f6bd857 100644 --- a/werewolf/roles/seer.py +++ b/werewolf/roles/seer.py @@ -1,7 +1,11 @@ import logging -from werewolf.constants import ALIGNMENT_TOWN, ALIGNMENT_WEREWOLF, CATEGORY_TOWN_INVESTIGATIVE, \ - CATEGORY_TOWN_RANDOM +from werewolf.constants import ( + ALIGNMENT_TOWN, + ALIGNMENT_WEREWOLF, + CATEGORY_TOWN_INVESTIGATIVE, + CATEGORY_TOWN_RANDOM, +) from werewolf.listener import wolflistener from werewolf.night_powers import pick_target from werewolf.role import Role @@ -91,7 +95,7 @@ class Seer(Role): if alignment == ALIGNMENT_WEREWOLF: out = "Your insight reveals this player to be a **Werewolf!**" - else: # Don't reveal neutrals + else: # Don't reveal neutrals out = "You fail to find anything suspicious about this player..." await self.player.send_dm(out) diff --git a/werewolf/votegroups/wolfvote.py b/werewolf/votegroups/wolfvote.py index 7f6bbde..75fb01c 100644 --- a/werewolf/votegroups/wolfvote.py +++ b/werewolf/votegroups/wolfvote.py @@ -69,5 +69,5 @@ class WolfVote(VoteGroup): await self.channel.send( "{} has voted to kill {}".format(author.mention, target.member.display_name), - allowed_mentions=discord.AllowedMentions(everyone=False, users=[author]) + allowed_mentions=discord.AllowedMentions(everyone=False, users=[author]), ) From 693964183cce5e8056492a4a4374d560b75fecf3 Mon Sep 17 00:00:00 2001 From: bobloy Date: Mon, 28 Sep 2020 12:45:03 -0400 Subject: [PATCH 37/73] Update issue templates --- .github/ISSUE_TEMPLATE/bug_report.md | 26 +++++++++++++++++++ .github/ISSUE_TEMPLATE/feature_request.md | 14 ++++++++++ .../ISSUE_TEMPLATE/new-audiotrivia-list.md | 26 +++++++++++++++++++ 3 files changed, 66 insertions(+) create mode 100644 .github/ISSUE_TEMPLATE/bug_report.md create mode 100644 .github/ISSUE_TEMPLATE/feature_request.md create mode 100644 .github/ISSUE_TEMPLATE/new-audiotrivia-list.md diff --git a/.github/ISSUE_TEMPLATE/bug_report.md b/.github/ISSUE_TEMPLATE/bug_report.md new file mode 100644 index 0000000..7a6e260 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/bug_report.md @@ -0,0 +1,26 @@ +--- +name: Bug report +about: Create an issue to report a bug +title: '' +labels: bug +assignees: bobloy + +--- + +**Describe the bug** + + +**To Reproduce** + +1. Load cog '...' +2. Run command '....' +3. See error + +**Expected behavior** + + +**Screenshots or Error Messages** + + +**Additional context** + diff --git a/.github/ISSUE_TEMPLATE/feature_request.md b/.github/ISSUE_TEMPLATE/feature_request.md new file mode 100644 index 0000000..42f5500 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/feature_request.md @@ -0,0 +1,14 @@ +--- +name: Feature request +about: Suggest an idea for this project +title: "[Feature Request]" +labels: enhancement +assignees: '' + +--- + +**Is your feature request related to a problem? Please describe.** + + +**Describe the solution you'd like** + diff --git a/.github/ISSUE_TEMPLATE/new-audiotrivia-list.md b/.github/ISSUE_TEMPLATE/new-audiotrivia-list.md new file mode 100644 index 0000000..b6be14e --- /dev/null +++ b/.github/ISSUE_TEMPLATE/new-audiotrivia-list.md @@ -0,0 +1,26 @@ +--- +name: New AudioTrivia List +about: Submit a new AudioTrivia list to be added +title: "[AudioTrivia Submission]" +labels: 'cog: audiotrivia' +assignees: bobloy + +--- + +**What is this trivia list?** + + +**Number of Questions** + + +**Original Content?** + + +-[ ] Yes +-[ ] No + + +**Did I test the list?** + +-[ ] Yes +-[ ] No From 94aceb32e8e37681a7d3f9b4b8fd528b639f15d1 Mon Sep 17 00:00:00 2001 From: bobloy Date: Mon, 28 Sep 2020 12:46:10 -0400 Subject: [PATCH 38/73] Update issue templates --- .github/ISSUE_TEMPLATE/new-audiotrivia-list.md | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/.github/ISSUE_TEMPLATE/new-audiotrivia-list.md b/.github/ISSUE_TEMPLATE/new-audiotrivia-list.md index b6be14e..25bcc81 100644 --- a/.github/ISSUE_TEMPLATE/new-audiotrivia-list.md +++ b/.github/ISSUE_TEMPLATE/new-audiotrivia-list.md @@ -16,11 +16,11 @@ assignees: bobloy **Original Content?** --[ ] Yes --[ ] No +- [ ] Yes +- [ ] No **Did I test the list?** --[ ] Yes --[ ] No +- [ ] Yes +- [ ] No From 3a6d3df374e49bd0005e2cec2c7f456f3900f9ad Mon Sep 17 00:00:00 2001 From: bobloy Date: Mon, 28 Sep 2020 17:24:35 -0400 Subject: [PATCH 39/73] More WIP progress --- werewolf/game.py | 84 ++++++++++++++++++++++--------- werewolf/player.py | 3 ++ werewolf/role.py | 4 +- werewolf/roles/seer.py | 1 + werewolf/roles/vanillawerewolf.py | 15 ------ werewolf/roles/villager.py | 6 --- werewolf/votegroup.py | 1 - werewolf/werewolf.py | 22 +++++++- 8 files changed, 88 insertions(+), 48 deletions(-) diff --git a/werewolf/game.py b/werewolf/game.py index 79d8455..ee84b09 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -19,6 +19,14 @@ log = logging.getLogger("red.fox_v3.werewolf.game") HALF_DAY_LENGTH = 24 # FixMe: to 120 later for 4 minute days +async def anyone_has_role( + member_list: List[discord.Member], role: discord.Role +) -> Union[None, discord.Member]: + return await AsyncIter(member_list).find( + lambda m: AsyncIter(m.roles).find(lambda r: r.id == role.id) + ) + + class Game: """ Base class to run a single game of Werewolf @@ -129,6 +137,7 @@ class Game: self.roles = [] return False + # If there's no game role, make the role and delete it later in `self.to_delete` if self.game_role is None: try: self.game_role = await ctx.guild.create_role( @@ -144,14 +153,25 @@ class Game: ) self.roles = [] return False - try: - for player in self.players: - await player.member.add_roles(*[self.game_role]) - except discord.Forbidden: - await ctx.send( - f"Unable to add role **{self.game_role.name}**\nBot is missing `manage_roles` permissions" - ) - return False + + anyone_with_role = await anyone_has_role(self.guild.members, self.game_role) + if anyone_with_role is not None: + await ctx.maybe_send_embed( + f"{anyone_with_role.display_name} has the game role, " + f"can't continue until no one has the role" + ) + return False + + try: + for player in self.players: + await player.member.add_roles(*[self.game_role]) + except discord.Forbidden: + log.exception(f"Unable to add role **{self.game_role.name}**") + await ctx.send( + f"Unable to add role **{self.game_role.name}**\n" + f"Bot is missing `manage_roles` permissions" + ) + return False await self.assign_roles() @@ -223,9 +243,10 @@ class Game: self.started = True # Assuming everything worked so far log.debug("Pre at_game_start") - await self._at_game_start() # This will queue channels and votegroups to be made + await self._at_game_start() # This will add votegroups to self.p_channels log.debug("Post at_game_start") - for channel_id in self.p_channels: + log.debug(f"Private channels: {self.p_channels}") + for channel_id in self.p_channels.keys(): log.debug("Setup Channel id: " + channel_id) overwrite = { self.guild.default_role: discord.PermissionOverwrite(read_messages=False), @@ -251,6 +272,8 @@ class Game: self.p_channels[channel_id]["channel"] = channel + self.to_delete.add(channel) + if self.p_channels[channel_id]["votegroup"] is not None: vote_group = self.p_channels[channel_id]["votegroup"](self, channel) @@ -259,8 +282,10 @@ class Game: self.vote_groups[channel_id] = vote_group log.debug("Pre-cycle") - await asyncio.sleep(1) - await asyncio.ensure_future(self._cycle()) # Start the loop + await asyncio.sleep(0) + + asyncio.create_task(self._cycle()) # Start the loop + return True # ###########START Notify structure############ async def _cycle(self): @@ -553,13 +578,14 @@ class Game: try: await asyncio.sleep(1) # This will have multiple calls self.p_channels[channel_id]["players"].append(role.player) - if votegroup is not None: - self.p_channels[channel_id]["votegroup"] = votegroup except AttributeError: continue else: break + if votegroup is not None: + self.p_channels[channel_id]["votegroup"] = votegroup + async def join(self, member: discord.Member, channel: discord.TextChannel): """ Have a member join a game @@ -574,14 +600,15 @@ class Game: self.players.append(Player(member)) - if self.game_role is not None: - try: - await member.add_roles(*[self.game_role]) - except discord.Forbidden: - await channel.send( - f"Unable to add role **{self.game_role.name}**\n" - f"Bot is missing `manage_roles` permissions" - ) + # Add the role during setup, not before + # if self.game_role is not None: + # try: + # await member.add_roles(*[self.game_role]) + # except discord.Forbidden: + # await channel.send( + # f"Unable to add role **{self.game_role.name}**\n" + # f"Bot is missing `manage_roles` permissions" + # ) await channel.send( f"{member.display_name} has been added to the game, " @@ -908,7 +935,7 @@ class Game: # Remove game_role access for potential archiving for now reason = "(BOT) End of WW game" for obj in self.to_delete: - log.debug(f"End_game: Deleting object {obj}") + log.debug(f"End_game: Deleting object {obj.__repr__()}") await obj.delete(reason=reason) try: @@ -926,6 +953,17 @@ class Game: except (discord.HTTPException, discord.NotFound, discord.errors.NotFound): pass + for player in self.players: + try: + await player.member.remove_roles(*[self.game_role]) + except discord.Forbidden: + log.exception(f"Unable to add remove **{self.game_role.name}**") + # await ctx.send( + # f"Unable to add role **{self.game_role.name}**\n" + # f"Bot is missing `manage_roles` permissions" + # ) + pass + # Optional dynamic channels/categories def add_ww_listener(self, func, priority=0, name=None): diff --git a/werewolf/player.py b/werewolf/player.py index 7f10758..48885a8 100644 --- a/werewolf/player.py +++ b/werewolf/player.py @@ -20,6 +20,9 @@ class Player: self.muted = False self.protected = False + def __repr__(self): + return f"{self.__class__.__name__}({self.member})" + async def assign_role(self, role): """ Give this player a role diff --git a/werewolf/role.py b/werewolf/role.py index db7b852..0997b56 100644 --- a/werewolf/role.py +++ b/werewolf/role.py @@ -73,7 +73,7 @@ class Role(WolfListener): self.properties = {} # Extra data for other roles (i.e. arsonist) def __repr__(self): - return self.__class__.__name__ + return f"{self.__class__.__name__}({self.player.__repr__()})" async def assign_player(self, player): """ @@ -84,6 +84,8 @@ class Role(WolfListener): player.role = self self.player = player + log.debug(f"Assigned {self} to {player}") + async def get_alignment(self, source=None): """ Interaction for powerful access of alignment diff --git a/werewolf/roles/seer.py b/werewolf/roles/seer.py index f6bd857..983fd14 100644 --- a/werewolf/roles/seer.py +++ b/werewolf/roles/seer.py @@ -15,6 +15,7 @@ log = logging.getLogger("red.fox_v3.werewolf.role.seer") class Seer(Role): rand_choice = True + town_balance = 4 category = [ CATEGORY_TOWN_RANDOM, CATEGORY_TOWN_INVESTIGATIVE, diff --git a/werewolf/roles/vanillawerewolf.py b/werewolf/roles/vanillawerewolf.py index 74e8d96..8abdea2 100644 --- a/werewolf/roles/vanillawerewolf.py +++ b/werewolf/roles/vanillawerewolf.py @@ -22,21 +22,6 @@ class VanillaWerewolf(Role): "Vote to kill players at night with `[p]ww vote `" ) - def __init__(self, game): - super().__init__(game) - - # self.action_list = [ - # (self._at_game_start, 1), # (Action, Priority) - # (self._at_day_start, 0), - # (self._at_voted, 0), - # (self._at_kill, 0), - # (self._at_hang, 0), - # (self._at_day_end, 0), - # (self._at_night_start, 0), - # (self._at_night_end, 0), - # (self._at_visit, 0) - # ] - async def see_alignment(self, source=None): """ Interaction for investigative roles attempting diff --git a/werewolf/roles/villager.py b/werewolf/roles/villager.py index d669ef9..eb0b2c9 100644 --- a/werewolf/roles/villager.py +++ b/werewolf/roles/villager.py @@ -7,12 +7,9 @@ log = logging.getLogger("red.fox_v3.werewolf.role.villager") class Villager(Role): - # Determines if it can be picked as a random role (False for unusually disruptive roles) rand_choice = True - town_balance = 1 - category = [CATEGORY_TOWN_RANDOM] # List of enrolled categories (listed above) alignment = ALIGNMENT_TOWN # 1: Town, 2: Werewolf, 3: Neutral channel_id = "" # Empty for no private channel @@ -23,9 +20,6 @@ class Villager(Role): "Lynch players during the day with `[p]ww vote `" ) - def __init__(self, game): - super().__init__(game) - async def see_alignment(self, source=None): """ Interaction for investigative roles attempting diff --git a/werewolf/votegroup.py b/werewolf/votegroup.py index 2f0b3a0..d8411fb 100644 --- a/werewolf/votegroup.py +++ b/werewolf/votegroup.py @@ -75,7 +75,6 @@ class VoteGroup(WolfListener): if not self.players: # TODO: Confirm deletion - self.game.to_delete.add(self) pass async def vote(self, target, author, target_id): diff --git a/werewolf/werewolf.py b/werewolf/werewolf.py index 742a890..599796c 100644 --- a/werewolf/werewolf.py +++ b/werewolf/werewolf.py @@ -1,9 +1,11 @@ import logging +from typing import List, Union import discord from redbot.core import Config, checks, commands from redbot.core.bot import Red from redbot.core.commands import Cog +from redbot.core.utils import AsyncIter from redbot.core.utils.menus import DEFAULT_CONTROLS, menu from werewolf.builder import ( @@ -18,6 +20,14 @@ from werewolf.game import Game log = logging.getLogger("red.fox_v3.werewolf") +async def anyone_has_role( + member_list: List[discord.Member], role: discord.Role +) -> Union[None, discord.Member]: + return await AsyncIter(member_list).find( + lambda m: AsyncIter(m.roles).find(lambda r: r.id == role.id) + ) + + class Werewolf(Cog): """ Base to host werewolf on a guild @@ -189,12 +199,15 @@ class Werewolf(Cog): return await game.join(ctx.author, ctx.channel) + await ctx.tick() @commands.guild_only() @ww.command(name="code") async def ww_code(self, ctx: commands.Context, code): """ - Adjust game code + Adjusts the game code. + + See `[p]buildgame` to generate a new code """ game = await self._get_game(ctx) @@ -204,6 +217,7 @@ class Werewolf(Cog): return await game.set_code(ctx, code) + await ctx.tick() @commands.guild_only() @ww.command(name="quit") @@ -215,6 +229,7 @@ class Werewolf(Cog): game = await self._get_game(ctx) await game.quit(ctx.author, ctx.channel) + await ctx.tick() @commands.guild_only() @ww.command(name="start") @@ -229,6 +244,8 @@ class Werewolf(Cog): if not await game.setup(ctx): pass # ToDo something? + await ctx.tick() + @commands.guild_only() @ww.command(name="stop") async def ww_stop(self, ctx: commands.Context): @@ -245,6 +262,7 @@ class Werewolf(Cog): game = await self._get_game(ctx) game.game_over = True + await game.current_action.cancel() await ctx.send("Game has been stopped") @commands.guild_only() @@ -358,7 +376,7 @@ class Werewolf(Cog): else: await ctx.send("Role ID not found") - async def _get_game(self, ctx: commands.Context, game_code=None): + async def _get_game(self, ctx: commands.Context, game_code=None) -> Union[Game, None]: guild: discord.Guild = getattr(ctx, "guild", None) if guild is None: From cd89bd87e9a852269aaa548ac8a0d37adb0aa572 Mon Sep 17 00:00:00 2001 From: bobloy Date: Mon, 28 Sep 2020 17:30:02 -0400 Subject: [PATCH 40/73] Only pull requests --- .github/workflows/black_check.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/black_check.yml b/.github/workflows/black_check.yml index a36b5f2..076e238 100644 --- a/.github/workflows/black_check.yml +++ b/.github/workflows/black_check.yml @@ -4,7 +4,7 @@ # https://github.com/cclauss/autoblack name: black -on: [push, pull_request] +on: [pull_request] jobs: build: runs-on: ubuntu-latest From 8c0a1db06fb13087fb03158a9d5f683ff6bd7b64 Mon Sep 17 00:00:00 2001 From: bobloy Date: Mon, 28 Sep 2020 17:30:23 -0400 Subject: [PATCH 41/73] v2 checkout --- .github/workflows/black_check.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/black_check.yml b/.github/workflows/black_check.yml index 076e238..5350f98 100644 --- a/.github/workflows/black_check.yml +++ b/.github/workflows/black_check.yml @@ -9,7 +9,7 @@ jobs: build: runs-on: ubuntu-latest steps: - - uses: actions/checkout@v1 + - uses: actions/checkout@v2 - name: Set up Python 3.8 uses: actions/setup-python@v2 with: From 2ab87866ddf2d4056c6d5266a4db733befe98d1c Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 29 Sep 2020 12:41:15 -0400 Subject: [PATCH 42/73] Adjust to italics, fix generate targets to be more obvious and readable, fix `night_messages` confusion with `night_results` --- werewolf/game.py | 33 ++++++++++++++++----------------- 1 file changed, 16 insertions(+), 17 deletions(-) diff --git a/werewolf/game.py b/werewolf/game.py index ee84b09..c5aaad6 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -42,12 +42,12 @@ class Game: "votegroup": None, # uninitialized VoteGroup } - morning_messages = [ - "**The sun rises on day {} in the village..**", - "**Morning has arrived on day {}..**", + day_start_messages = [ + "*The sun rises on day {} in the village..*", + "*Morning has arrived on day {}..*", ] - night_messages = ["**Dawn falls on day {}..****"] + day_end_messages = ["*Dawn falls..*", "*The sun sets on the village*"] day_vote_count = 3 @@ -305,9 +305,9 @@ class Game: self.action_queue.append(self._at_day_start()) while self.action_queue and not self.game_over: - current_action = asyncio.create_task(self.action_queue.popleft()) + self.current_action = asyncio.create_task(self.action_queue.popleft()) try: - await current_action + await self.current_action except asyncio.CancelledError: log.debug("Cancelled task") # @@ -337,7 +337,7 @@ class Game: self.day_count += 1 # Print the results of who died during the night - embed = discord.Embed(title=random.choice(self.morning_messages).format(self.day_count)) + embed = discord.Embed(title=random.choice(self.day_start_messages).format(self.day_count)) for result in self.night_results: embed.add_field(name=result, value="________", inline=False) @@ -362,7 +362,7 @@ class Game: if check(): return await self.village_channel.send( - embed=discord.Embed(title=f"**{HALF_DAY_LENGTH / 60} minutes of daylight remain...**") + embed=discord.Embed(title=f"*{HALF_DAY_LENGTH / 60} minutes of daylight remain...*") ) await asyncio.sleep(HALF_DAY_LENGTH) # 4 minute days FixMe to 120 later @@ -386,7 +386,7 @@ class Game: await self.speech_perms(self.village_channel, target.member) # Only target can talk await self.village_channel.send( - f"**{target.mention} will be put to trial and has 30 seconds to defend themselves**", + f"*{target.mention} will be put to trial and has 30 seconds to defend themselves**", allowed_mentions=discord.AllowedMentions(everyone=False, users=[target]), ) @@ -480,7 +480,7 @@ class Game: await self.night_perms(self.village_channel) await self.village_channel.send( - embed=discord.Embed(title="**The sun sets on the village...**") + embed=discord.Embed(title=random.choice(self.day_end_messages)) ) await self._notify("at_day_end") @@ -546,7 +546,7 @@ class Game: # ###########END Notify structure############ async def generate_targets(self, channel, with_roles=False): - embed = discord.Embed(title="Remaining Players") + embed = discord.Embed(title="Remaining Players", description="[ID] - [Name]") for i, player in enumerate(self.players): if player.alive: status = "" @@ -554,15 +554,14 @@ class Game: status = "*[Dead]*-" if with_roles or not player.alive: embed.add_field( - name=f"ID# **{i}**", - value=f"{status}{player.member.display_name}-{player.role}", - inline=True, + name=f"{i} - {status}{player.member.display_name}", + value=f"{player.role}", + inline=False, ) else: embed.add_field( - name=f"ID# **{i}**", - value=f"{status}{player.member.display_name}", - inline=True, + name=f"{i} - {status}{player.member.display_name}", + inline=False, ) return await channel.send(embed=embed) From c7d320ccaa351c383968330faf21dda696ac4958 Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 29 Sep 2020 12:41:26 -0400 Subject: [PATCH 43/73] WIP Player converter --- werewolf/converters.py | 28 ++++++++++++++++++++++++++++ 1 file changed, 28 insertions(+) create mode 100644 werewolf/converters.py diff --git a/werewolf/converters.py b/werewolf/converters.py new file mode 100644 index 0000000..376749d --- /dev/null +++ b/werewolf/converters.py @@ -0,0 +1,28 @@ +from typing import TYPE_CHECKING, Union + +import discord +from discord.ext.commands import BadArgument, Converter +from redbot.core import commands + +from werewolf.player import Player + +if TYPE_CHECKING: + PlayerConverter = Union[int, discord.Member] + CronConverter = str +else: + + class PlayerConverter(Converter): + async def convert(self, ctx, argument) -> Player: + + try: + target = await commands.MemberConverter().convert(ctx, argument) + except BadArgument: + try: + target = int(argument) + assert target >= 0 + except (ValueError, AssertionError): + raise BadArgument + + # TODO: Get the game for context without making a new one + # TODO: Get player from game based on either ID or member object + return target \ No newline at end of file From 443c84ccabd38fed7c7a06a4939b24708d527c86 Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 29 Sep 2020 12:41:43 -0400 Subject: [PATCH 44/73] Fix `night_messages` to `night_results` --- werewolf/roles/blob.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/werewolf/roles/blob.py b/werewolf/roles/blob.py index bd7b598..af18983 100644 --- a/werewolf/roles/blob.py +++ b/werewolf/roles/blob.py @@ -98,4 +98,4 @@ class TheBlob(Role): if target is not None: target.role.properties["been_blobbed"] = True - self.game.night_messages.append("The Blob grows...") + self.game.night_results.append("The Blob grows...") From 211df56e1b6070196c226728b206b0f935ae529a Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 29 Sep 2020 12:41:57 -0400 Subject: [PATCH 45/73] Add repr --- werewolf/votegroup.py | 3 +++ 1 file changed, 3 insertions(+) diff --git a/werewolf/votegroup.py b/werewolf/votegroup.py index d8411fb..e651eda 100644 --- a/werewolf/votegroup.py +++ b/werewolf/votegroup.py @@ -22,6 +22,9 @@ class VoteGroup(WolfListener): self.vote_results = {} self.properties = {} # Extra data for other options + def __repr__(self): + return f"{self.__class__.__name__}({self.channel},{self.players})" + @wolflistener("at_game_start", priority=1) async def _at_game_start(self): await self.channel.send(" ".join(player.mention for player in self.players)) From e27cfba763b4d596fbed7568aeb64ed5c8389b2d Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 29 Sep 2020 12:42:05 -0400 Subject: [PATCH 46/73] Move to italics --- werewolf/votegroups/wolfvote.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/werewolf/votegroups/wolfvote.py b/werewolf/votegroups/wolfvote.py index 75fb01c..d637c87 100644 --- a/werewolf/votegroups/wolfvote.py +++ b/werewolf/votegroups/wolfvote.py @@ -55,10 +55,10 @@ class WolfVote(VoteGroup): if target_id is not None and self.killer: await self.game.kill(target_id, self.killer, random.choice(self.kill_messages)) await self.channel.send( - "**{} has left to complete the kill...**".format(self.killer.member.display_name) + "*{} has left to complete the kill...*".format(self.killer.member.display_name) ) else: - await self.channel.send("**No kill will be attempted tonight...**") + await self.channel.send("*No kill will be attempted tonight...*") async def vote(self, target, author, target_id): """ From f3965b73d831bd86c36d38d270b5f38259057fac Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 29 Sep 2020 14:25:19 -0400 Subject: [PATCH 47/73] Mostly messaging adjustments, fix for failing to talley votes --- werewolf/game.py | 57 ++++++++++++++++------------- werewolf/player.py | 6 +++- werewolf/werewolf.py | 85 ++++++++++++++++++++++++++++---------------- 3 files changed, 91 insertions(+), 57 deletions(-) diff --git a/werewolf/game.py b/werewolf/game.py index c5aaad6..c0a9db4 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -10,13 +10,15 @@ from redbot.core.bot import Red from redbot.core.utils import AsyncIter from werewolf.builder import parse_code +from werewolf.constants import ALIGNMENT_NEUTRAL from werewolf.player import Player from werewolf.role import Role from werewolf.votegroup import VoteGroup log = logging.getLogger("red.fox_v3.werewolf.game") -HALF_DAY_LENGTH = 24 # FixMe: to 120 later for 4 minute days +HALF_DAY_LENGTH = 60 # FixMe: Make configurable +HALF_NIGHT_LENGTH = 60 async def anyone_has_role( @@ -167,7 +169,7 @@ class Game: await player.member.add_roles(*[self.game_role]) except discord.Forbidden: log.exception(f"Unable to add role **{self.game_role.name}**") - await ctx.send( + await ctx.maybe_send_embed( f"Unable to add role **{self.game_role.name}**\n" f"Bot is missing `manage_roles` permissions" ) @@ -210,7 +212,7 @@ class Game: category=self.channel_category, ) except discord.Forbidden: - await ctx.send( + await ctx.maybe_send_embed( "Unable to create Game Channel and none was provided\n" "Grant Bot appropriate permissions or assign a game_channel" ) @@ -225,7 +227,7 @@ class Game: ) except discord.Forbidden as e: log.exception("Unable to rename Game Channel") - await ctx.send("Unable to rename Game Channel, ignoring") + await ctx.maybe_send_embed("Unable to rename Game Channel, ignoring") try: for target, ow in overwrite.items(): @@ -235,7 +237,7 @@ class Game: target=target, overwrite=curr, reason="(BOT) New game of werewolf" ) except discord.Forbidden: - await ctx.send( + await ctx.maybe_send_embed( "Unable to edit Game Channel permissions\n" "Grant Bot appropriate permissions to manage permissions" ) @@ -406,14 +408,17 @@ class Game: await vote_message.add_reaction("👎") await asyncio.sleep(15) - reaction_list = vote_message.reactions - if True: # TODO: Allow customizable vote history deletion. - await vote_message.delete() + # Refetch for reactions + vote_message = await self.village_channel.fetch_message(id=vote_message.id) + reaction_list = vote_message.reactions raw_up_votes = sum(p for p in reaction_list if p.emoji == "👍" and not p.me) raw_down_votes = sum(p for p in reaction_list if p.emoji == "👎" and not p.me) + if True: # TODO: Allow customizable vote history deletion. + await vote_message.delete() + # TODO: Support vote count modifying roles. (Need notify and count function) voted_to_lynch = raw_down_votes > raw_up_votes @@ -492,13 +497,13 @@ class Game: return await self._notify("at_night_start") - await asyncio.sleep(12) # 2 minutes FixMe to 120 later + await asyncio.sleep(HALF_NIGHT_LENGTH) # 2 minutes FixMe to 120 later await self.village_channel.send( - embed=discord.Embed(title="**Two minutes of night remain...**") + embed=discord.Embed(title=f"**{HALF_NIGHT_LENGTH / 60} minutes of night remain...**") ) - await asyncio.sleep(9) # 1.5 minutes FixMe to 90 later + await asyncio.sleep(HALF_NIGHT_LENGTH) # 1.5 minutes FixMe to 90 later await self.village_channel.send( - embed=discord.Embed(title="**Thirty seconds until sunrise...**") + embed=discord.Embed(title=f"**{HALF_NIGHT_LENGTH / 60} minutes until sunrise...**") ) await asyncio.sleep(3) # .5 minutes FixMe to 30 Later @@ -560,8 +565,7 @@ class Game: ) else: embed.add_field( - name=f"{i} - {status}{player.member.display_name}", - inline=False, + name=f"{i} - {status}{player.member.display_name}", inline=False, value="" ) return await channel.send(embed=embed) @@ -585,16 +589,16 @@ class Game: if votegroup is not None: self.p_channels[channel_id]["votegroup"] = votegroup - async def join(self, member: discord.Member, channel: discord.TextChannel): + async def join(self, ctx, member: discord.Member): """ Have a member join a game """ if self.started: - await channel.send("**Game has already started!**") + await ctx.maybe_send_embed("**Game has already started!**") return if await self.get_player_by_member(member) is not None: - await channel.send(f"{member.display_name} is already in the game!") + await ctx.maybe_send_embed(f"{member.display_name} is already in the game!") return self.players.append(Player(member)) @@ -609,7 +613,7 @@ class Game: # f"Bot is missing `manage_roles` permissions" # ) - await channel.send( + await ctx.maybe_send_embed( f"{member.display_name} has been added to the game, " f"total players is **{len(self.players)}**" ) @@ -645,15 +649,15 @@ class Game: player = await self.get_player_by_member(ctx.author) if player is None: - await ctx.send("You're not in this game!") + await ctx.maybe_send_embed("You're not in this game!") return if not player.alive: - await ctx.send("**Corpses** can't participate...") + await ctx.maybe_send_embed("**Corpses** can't participate...") return if player.role.blocked: - await ctx.send("Something is preventing you from doing this...") + await ctx.maybe_send_embed("Something is preventing you from doing this...") return # Let role do target validation, might be alternate targets @@ -821,7 +825,7 @@ class Game: async def set_code(self, ctx: commands.Context, game_code): if game_code is not None: self.game_code = game_code - await ctx.send("Code has been set") + await ctx.maybe_send_embed("Code has been set") async def get_roles(self, ctx, game_code=None): if game_code is not None: @@ -833,10 +837,12 @@ class Game: try: self.roles = await parse_code(self.game_code, self) except ValueError as e: - await ctx.send("Invalid Code: Code contains unknown character\n{}".format(e)) + await ctx.maybe_send_embed( + "Invalid Code: Code contains unknown character\n{}".format(e) + ) return False except IndexError as e: - await ctx.send("Invalid Code: Code references unknown role\n{}".format(e)) + await ctx.maybe_send_embed("Invalid Code: Code references unknown role\n{}".format(e)) if not self.roles: return False @@ -898,7 +904,8 @@ class Game: self.game_over = True alignment1 = alive_players[0].role.alignment alignment2 = alive_players[1].role.alignment - if alignment1 == alignment2: # Same team + # Same team and not neutral + if alignment1 == alignment2 and alignment1 != ALIGNMENT_NEUTRAL: winners = alive_players else: winners = [max(alive_players, key=lambda p: p.role.alignment)] diff --git a/werewolf/player.py b/werewolf/player.py index 48885a8..7aec179 100644 --- a/werewolf/player.py +++ b/werewolf/player.py @@ -35,9 +35,13 @@ class Player: async def send_dm(self, message): try: - await self.member.send(message) # Lets do embeds later + await self.member.send(message) # Lets ToDo embeds later except discord.Forbidden: + log.info(f"Unable to mention {self.member.__repr__()}") await self.role.game.village_channel.send( f"Couldn't DM {self.mention}, uh oh", allowed_mentions=discord.AllowedMentions(users=[self.member]), ) + except AttributeError: + log.exception("Someone messed up and added a bot to the game (I think)") + await self.role.game.village_channel.send("Someone messed up and added a bot to the game :eyes:") diff --git a/werewolf/werewolf.py b/werewolf/werewolf.py index 599796c..dc27338 100644 --- a/werewolf/werewolf.py +++ b/werewolf/werewolf.py @@ -72,9 +72,9 @@ class Werewolf(Cog): code = await gb.build_game(ctx) if code != "": - await ctx.send(f"Your game code is **{code}**") + await ctx.maybe_send_embed(f"Your game code is **{code}**") else: - await ctx.send("No code generated") + await ctx.maybe_send_embed("No code generated") @checks.guildowner() @commands.group() @@ -117,10 +117,10 @@ class Werewolf(Cog): """ if role is None: await self.config.guild(ctx.guild).role_id.set(None) - await ctx.send("Cleared Game Role") + await ctx.maybe_send_embed("Cleared Game Role") else: await self.config.guild(ctx.guild).role_id.set(role.id) - await ctx.send("Game Role has been set to **{}**".format(role.name)) + await ctx.maybe_send_embed("Game Role has been set to **{}**".format(role.name)) @commands.guild_only() @wwset.command(name="category") @@ -130,14 +130,14 @@ class Werewolf(Cog): """ if category_id is None: await self.config.guild(ctx.guild).category_id.set(None) - await ctx.send("Cleared Game Channel Category") + await ctx.maybe_send_embed("Cleared Game Channel Category") else: category = discord.utils.get(ctx.guild.categories, id=int(category_id)) if category is None: - await ctx.send("Category not found") + await ctx.maybe_send_embed("Category not found") return await self.config.guild(ctx.guild).category_id.set(category.id) - await ctx.send("Game Channel Category has been set to **{}**".format(category.name)) + await ctx.maybe_send_embed("Game Channel Category has been set to **{}**".format(category.name)) @commands.guild_only() @wwset.command(name="channel") @@ -147,10 +147,10 @@ class Werewolf(Cog): """ if channel is None: await self.config.guild(ctx.guild).channel_id.set(None) - await ctx.send("Cleared Game Channel") + await ctx.maybe_send_embed("Cleared Game Channel") else: await self.config.guild(ctx.guild).channel_id.set(channel.id) - await ctx.send("Game Channel has been set to **{}**".format(channel.mention)) + await ctx.maybe_send_embed("Game Channel has been set to **{}**".format(channel.mention)) @commands.guild_only() @wwset.command(name="logchannel") @@ -160,10 +160,10 @@ class Werewolf(Cog): """ if channel is None: await self.config.guild(ctx.guild).log_channel_id.set(None) - await ctx.send("Cleared Game Log Channel") + await ctx.maybe_send_embed("Cleared Game Log Channel") else: await self.config.guild(ctx.guild).log_channel_id.set(channel.id) - await ctx.send("Game Log Channel has been set to **{}**".format(channel.mention)) + await ctx.maybe_send_embed("Game Log Channel has been set to **{}**".format(channel.mention)) @commands.group() async def ww(self, ctx: commands.Context): @@ -181,9 +181,9 @@ class Werewolf(Cog): """ game = await self._get_game(ctx, game_code) if not game: - await ctx.send("Failed to start a new game") + await ctx.maybe_send_embed("Failed to start a new game") else: - await ctx.send("Game is ready to join! Use `[p]ww join`") + await ctx.maybe_send_embed("Game is ready to join! Use `[p]ww join`") @commands.guild_only() @ww.command(name="join") @@ -195,10 +195,27 @@ class Werewolf(Cog): game: Game = await self._get_game(ctx) if not game: - await ctx.send("No game to join!\nCreate a new one with `[p]ww new`") + await ctx.maybe_send_embed("Failed to join a game!") return - await game.join(ctx.author, ctx.channel) + await game.join(ctx, ctx.author) + await ctx.tick() + + @commands.guild_only() + @commands.admin() + @ww.command(name="forcejoin") + async def ww_forcejoin(self, ctx: commands.Context, target: discord.Member): + """ + Force someone to join a game of Werewolf + """ + + game: Game = await self._get_game(ctx) + + if not game: + await ctx.maybe_send_embed("Failed to join a game!") + return + + await game.join(ctx, target) await ctx.tick() @commands.guild_only() @@ -213,7 +230,7 @@ class Werewolf(Cog): game = await self._get_game(ctx) if not game: - await ctx.send("No game to join!\nCreate a new one with `[p]ww new`") + await ctx.maybe_send_embed("No game to join!\nCreate a new one with `[p]ww new`") return await game.set_code(ctx, code) @@ -239,7 +256,7 @@ class Werewolf(Cog): """ game = await self._get_game(ctx) if not game: - await ctx.send("No game running, cannot start") + await ctx.maybe_send_embed("No game running, cannot start") if not await game.setup(ctx): pass # ToDo something? @@ -257,13 +274,13 @@ class Werewolf(Cog): # await ctx.send("Cannot stop game from PM!") # return if ctx.guild.id not in self.games or self.games[ctx.guild.id].game_over: - await ctx.send("No game to stop") + await ctx.maybe_send_embed("No game to stop") return game = await self._get_game(ctx) game.game_over = True await game.current_action.cancel() - await ctx.send("Game has been stopped") + await ctx.maybe_send_embed("Game has been stopped") @commands.guild_only() @ww.command(name="vote") @@ -277,7 +294,7 @@ class Werewolf(Cog): target_id = None if target_id is None: - await ctx.send("`id` must be an integer") + await ctx.maybe_send_embed("`id` must be an integer") return # if ctx.guild is None: @@ -294,7 +311,7 @@ class Werewolf(Cog): game = await self._get_game(ctx) if game is None: - await ctx.send("No game running, cannot vote") + await ctx.maybe_send_embed("No game running, cannot vote") return # Game handles response now @@ -304,7 +321,7 @@ class Werewolf(Cog): elif channel in (c["channel"] for c in game.p_channels.values()): await game.vote(ctx.author, target_id, channel) else: - await ctx.send("Nothing to vote for in this channel") + await ctx.maybe_send_embed("Nothing to vote for in this channel") @ww.command(name="choose") async def ww_choose(self, ctx: commands.Context, data): @@ -315,7 +332,7 @@ class Werewolf(Cog): """ if ctx.guild is not None: - await ctx.send("This action is only available in DM's") + await ctx.maybe_send_embed("This action is only available in DM's") return # DM nonsense, find their game # If multiple games, panic @@ -323,7 +340,7 @@ class Werewolf(Cog): if await game.get_player_by_member(ctx.author): break # game = game else: - await ctx.send("You're not part of any werewolf game") + await ctx.maybe_send_embed("You're not part of any werewolf game") return await game.choose(ctx, data) @@ -344,7 +361,7 @@ class Werewolf(Cog): if from_name: await menu(ctx, from_name, DEFAULT_CONTROLS) else: - await ctx.send("No roles containing that name were found") + await ctx.maybe_send_embed("No roles containing that name were found") @ww_search.command(name="alignment") async def ww_search_alignment(self, ctx: commands.Context, alignment: int): @@ -354,7 +371,7 @@ class Werewolf(Cog): if from_alignment: await menu(ctx, from_alignment, DEFAULT_CONTROLS) else: - await ctx.send("No roles with that alignment were found") + await ctx.maybe_send_embed("No roles with that alignment were found") @ww_search.command(name="category") async def ww_search_category(self, ctx: commands.Context, category: int): @@ -364,7 +381,7 @@ class Werewolf(Cog): if pages: await menu(ctx, pages, DEFAULT_CONTROLS) else: - await ctx.send("No roles in that category were found") + await ctx.maybe_send_embed("No roles in that category were found") @ww_search.command(name="index") async def ww_search_index(self, ctx: commands.Context, idx: int): @@ -374,23 +391,29 @@ class Werewolf(Cog): if idx_embed is not None: await ctx.send(embed=idx_embed) else: - await ctx.send("Role ID not found") + await ctx.maybe_send_embed("Role ID not found") async def _get_game(self, ctx: commands.Context, game_code=None) -> Union[Game, None]: guild: discord.Guild = getattr(ctx, "guild", None) if guild is None: # Private message, can't get guild - await ctx.send("Cannot start game from DM!") + await ctx.maybe_send_embed("Cannot start game from DM!") return None if guild.id not in self.games or self.games[guild.id].game_over: - await ctx.send("Starting a new game...") + await ctx.maybe_send_embed("Starting a new game...") valid, role, category, channel, log_channel = await self._get_settings(ctx) if not valid: - await ctx.send("Cannot start a new game") + await ctx.maybe_send_embed("Cannot start a new game") return None + who_has_the_role = await anyone_has_role(guild.members, role) + if who_has_the_role: + await ctx.maybe_send_embed( + f"Cannot continue, {who_has_the_role.display_name} already has the game role." + ) + return None self.games[guild.id] = Game( self.bot, guild, role, category, channel, log_channel, game_code ) From 9ca5d37f7e9bf318c33415e53476f89e99e1950c Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 29 Sep 2020 17:17:11 -0400 Subject: [PATCH 48/73] Fixed a variable reuse, channel naming, bot's can't play, less bold, object deletion error catching --- werewolf/game.py | 36 +++++++++++++++++++++++------------- 1 file changed, 23 insertions(+), 13 deletions(-) diff --git a/werewolf/game.py b/werewolf/game.py index c0a9db4..bb77f02 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -221,8 +221,7 @@ class Game: self.save_perms[self.village_channel] = self.village_channel.overwrites try: await self.village_channel.edit( - name="🔵Werewolf", - category=self.channel_category, + name="🔵werewolf", reason="(BOT) New game of werewolf", ) except discord.Forbidden as e: @@ -298,7 +297,7 @@ class Game: _at_voted() _at_kill() _at_day_end() - _at_night_begin() + _at_night_start() _at_night_end() and repeat with _at_day_start() again @@ -331,6 +330,7 @@ class Game: if self.game_over: return + # await self.village_channel.edit(reason="WW Night Start", name="werewolf-🌞") self.action_queue.append(self._at_day_end()) # Get this ready in case day is cancelled def check(): @@ -413,6 +413,7 @@ class Game: vote_message = await self.village_channel.fetch_message(id=vote_message.id) reaction_list = vote_message.reactions + log.debug(f"Vote results: {[p.emoji.__repr__() for p in reaction_list]}") raw_up_votes = sum(p for p in reaction_list if p.emoji == "👍" and not p.me) raw_down_votes = sum(p for p in reaction_list if p.emoji == "👎" and not p.me) @@ -495,6 +496,9 @@ class Game: async def _at_night_start(self): # ID 6 if self.game_over: return + + # await self.village_channel.edit(reason="WW Night Start", name="werewolf-🌑") + await self._notify("at_night_start") await asyncio.sleep(HALF_NIGHT_LENGTH) # 2 minutes FixMe to 120 later @@ -502,9 +506,7 @@ class Game: embed=discord.Embed(title=f"**{HALF_NIGHT_LENGTH / 60} minutes of night remain...**") ) await asyncio.sleep(HALF_NIGHT_LENGTH) # 1.5 minutes FixMe to 90 later - await self.village_channel.send( - embed=discord.Embed(title=f"**{HALF_NIGHT_LENGTH / 60} minutes until sunrise...**") - ) + await asyncio.sleep(3) # .5 minutes FixMe to 30 Later self.action_queue.append(self._at_night_end()) @@ -522,11 +524,11 @@ class Game: return await self._notify("at_visit", target=target, source=source) - async def _notify(self, event, **kwargs): + async def _notify(self, event_name, **kwargs): for i in range(1, 7): # action guide 1-6 (0 is no action) tasks = [] - for event in self.listeners.get(event, {}).get(i, []): - tasks.append(asyncio.ensure_future(event(**kwargs), loop=self.loop)) + for event in self.listeners.get(event_name, {}).get(i, []): + tasks.append(asyncio.create_task(event(**kwargs))) # Run same-priority task simultaneously await asyncio.gather(*tasks) @@ -565,7 +567,7 @@ class Game: ) else: embed.add_field( - name=f"{i} - {status}{player.member.display_name}", inline=False, value="" + name=f"{i} - {status}{player.member.display_name}", inline=False, value="____" ) return await channel.send(embed=embed) @@ -594,7 +596,11 @@ class Game: Have a member join a game """ if self.started: - await ctx.maybe_send_embed("**Game has already started!**") + await ctx.maybe_send_embed("Game has already started!") + return + + if member.bot: + await ctx.maybe_send_embed("Bots can't play games") return if await self.get_player_by_member(member) is not None: @@ -942,10 +948,14 @@ class Game: reason = "(BOT) End of WW game" for obj in self.to_delete: log.debug(f"End_game: Deleting object {obj.__repr__()}") - await obj.delete(reason=reason) + try: + await obj.delete(reason=reason) + except discord.NotFound: + # Already deleted + pass try: - await self.village_channel.edit(reason=reason, name="Werewolf") + asyncio.create_task(self.village_channel.edit(reason=reason, name="werewolf")) async for channel, overwrites in AsyncIter(self.save_perms.items()): async for target, overwrite in AsyncIter(overwrites.items()): await channel.set_permissions(target, overwrite=overwrite, reason=reason) From 8a4893c5f54996c29c7f1a534675b5c8ffa94f21 Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 29 Sep 2020 17:17:20 -0400 Subject: [PATCH 49/73] Forgot the f in fstring --- werewolf/votegroups/wolfvote.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/werewolf/votegroups/wolfvote.py b/werewolf/votegroups/wolfvote.py index d637c87..dfb4f32 100644 --- a/werewolf/votegroups/wolfvote.py +++ b/werewolf/votegroups/wolfvote.py @@ -51,7 +51,7 @@ class WolfVote(VoteGroup): if vote_list: target_id = max(set(vote_list), key=vote_list.count) - log.debug("Target id: {target_id}\nKiller: {self.killer.member.display_name}") + log.debug(f"Target id: {target_id}\nKiller: {self.killer.member.display_name}") if target_id is not None and self.killer: await self.game.kill(target_id, self.killer, random.choice(self.kill_messages)) await self.channel.send( From 19104241d71391ba4eb103e90343d49c0bbddea8 Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 29 Sep 2020 17:17:29 -0400 Subject: [PATCH 50/73] Don't await task cancels --- werewolf/werewolf.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/werewolf/werewolf.py b/werewolf/werewolf.py index dc27338..a9870ab 100644 --- a/werewolf/werewolf.py +++ b/werewolf/werewolf.py @@ -279,7 +279,7 @@ class Werewolf(Cog): game = await self._get_game(ctx) game.game_over = True - await game.current_action.cancel() + game.current_action.cancel() await ctx.maybe_send_embed("Game has been stopped") @commands.guild_only() From 62a70c52c63055ae88cb76ca71f6810073bf19fd Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 29 Sep 2020 17:17:49 -0400 Subject: [PATCH 51/73] Some weird error with dm-ing keeps happening, add better log to catch it --- werewolf/role.py | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/werewolf/role.py b/werewolf/role.py index 0997b56..e267283 100644 --- a/werewolf/role.py +++ b/werewolf/role.py @@ -120,7 +120,11 @@ class Role(WolfListener): if self.channel_name: await self.game.register_channel(self.channel_name, self) - await self.player.send_dm(self.game_start_message) # Maybe embeds eventually + try: + await self.player.send_dm(self.game_start_message) # Maybe embeds eventually + except AttributeError as e: + log.exception(self.__repr__()) + raise e async def kill(self, source): """ From c529d792e6d9416ff8f093151755e0947099e916 Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 29 Sep 2020 17:31:55 -0400 Subject: [PATCH 52/73] Fix double game_end --- werewolf/game.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/werewolf/game.py b/werewolf/game.py index bb77f02..63b922a 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -929,7 +929,7 @@ class Game: await self._announce_winners(alive_players) # If no return, cleanup and end game - await self._end_game() + # await self._end_game() async def _announce_winners(self, winnerlist): await self.village_channel.send(self.game_role.mention) From 7f8d0f13f7b0ca9866a482fd40a893d72cc33a37 Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 29 Sep 2020 17:32:35 -0400 Subject: [PATCH 53/73] Black format --- werewolf/converters.py | 2 +- werewolf/player.py | 4 +++- werewolf/werewolf.py | 12 +++++++++--- 3 files changed, 13 insertions(+), 5 deletions(-) diff --git a/werewolf/converters.py b/werewolf/converters.py index 376749d..f108666 100644 --- a/werewolf/converters.py +++ b/werewolf/converters.py @@ -25,4 +25,4 @@ else: # TODO: Get the game for context without making a new one # TODO: Get player from game based on either ID or member object - return target \ No newline at end of file + return target diff --git a/werewolf/player.py b/werewolf/player.py index 7aec179..c574109 100644 --- a/werewolf/player.py +++ b/werewolf/player.py @@ -44,4 +44,6 @@ class Player: ) except AttributeError: log.exception("Someone messed up and added a bot to the game (I think)") - await self.role.game.village_channel.send("Someone messed up and added a bot to the game :eyes:") + await self.role.game.village_channel.send( + "Someone messed up and added a bot to the game :eyes:" + ) diff --git a/werewolf/werewolf.py b/werewolf/werewolf.py index a9870ab..bd68a6f 100644 --- a/werewolf/werewolf.py +++ b/werewolf/werewolf.py @@ -137,7 +137,9 @@ class Werewolf(Cog): await ctx.maybe_send_embed("Category not found") return await self.config.guild(ctx.guild).category_id.set(category.id) - await ctx.maybe_send_embed("Game Channel Category has been set to **{}**".format(category.name)) + await ctx.maybe_send_embed( + "Game Channel Category has been set to **{}**".format(category.name) + ) @commands.guild_only() @wwset.command(name="channel") @@ -150,7 +152,9 @@ class Werewolf(Cog): await ctx.maybe_send_embed("Cleared Game Channel") else: await self.config.guild(ctx.guild).channel_id.set(channel.id) - await ctx.maybe_send_embed("Game Channel has been set to **{}**".format(channel.mention)) + await ctx.maybe_send_embed( + "Game Channel has been set to **{}**".format(channel.mention) + ) @commands.guild_only() @wwset.command(name="logchannel") @@ -163,7 +167,9 @@ class Werewolf(Cog): await ctx.maybe_send_embed("Cleared Game Log Channel") else: await self.config.guild(ctx.guild).log_channel_id.set(channel.id) - await ctx.maybe_send_embed("Game Log Channel has been set to **{}**".format(channel.mention)) + await ctx.maybe_send_embed( + "Game Log Channel has been set to **{}**".format(channel.mention) + ) @commands.group() async def ww(self, ctx: commands.Context): From d0445f41c70333f403d464fff44fb427fc7c0a12 Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 29 Sep 2020 17:36:08 -0400 Subject: [PATCH 54/73] Black format update --- audiotrivia/audiotrivia.py | 8 +- ccrole/ccrole.py | 5 +- conquest/conquest.py | 7 +- fifo/timezones.py | 388 +++++++++++++++++++---------------- flag/flag.py | 2 +- forcemention/forcemention.py | 4 +- nudity/nudity.py | 4 +- rpsls/rpsls.py | 4 +- tts/tts.py | 4 +- 9 files changed, 236 insertions(+), 190 deletions(-) diff --git a/audiotrivia/audiotrivia.py b/audiotrivia/audiotrivia.py index 0bab980..c3ac1e9 100644 --- a/audiotrivia/audiotrivia.py +++ b/audiotrivia/audiotrivia.py @@ -90,7 +90,9 @@ class AudioTrivia(Trivia): categories = [c.lower() for c in categories] session = self._get_trivia_session(ctx.channel) if session is not None: - await ctx.maybe_send_embed("There is already an ongoing trivia session in this channel.") + await ctx.maybe_send_embed( + "There is already an ongoing trivia session in this channel." + ) return status = await self.audio.config.status() notify = await self.audio.config.guild(ctx.guild).notify() @@ -110,7 +112,9 @@ class AudioTrivia(Trivia): if not ctx.author.voice.channel.permissions_for( ctx.me ).connect or self.audio.is_vc_full(ctx.author.voice.channel): - return await ctx.maybe_send_embed("I don't have permission to connect to your channel.") + return await ctx.maybe_send_embed( + "I don't have permission to connect to your channel." + ) await lavalink.connect(ctx.author.voice.channel) lavaplayer = lavalink.get_player(ctx.guild.id) lavaplayer.store("connect", datetime.datetime.utcnow()) diff --git a/ccrole/ccrole.py b/ccrole/ccrole.py index eb654b1..59efc55 100644 --- a/ccrole/ccrole.py +++ b/ccrole/ccrole.py @@ -143,8 +143,9 @@ class CCRole(commands.Cog): return # Selfrole - await ctx.send("Is this a targeted command?(yes/no)\n" - "No will make this a selfrole command") + await ctx.send( + "Is this a targeted command?(yes/no)\n" "No will make this a selfrole command" + ) try: answer = await self.bot.wait_for("message", timeout=120, check=check) diff --git a/conquest/conquest.py b/conquest/conquest.py index fb8b280..fdf5e96 100644 --- a/conquest/conquest.py +++ b/conquest/conquest.py @@ -159,7 +159,12 @@ class Conquest(commands.Cog): self.data_path / self.current_map / f"current.{self.ext}", x, y, zoom ) - await ctx.send(file=discord.File(fp=zoomed_path, filename=f"current_zoomed.{self.ext}",)) + await ctx.send( + file=discord.File( + fp=zoomed_path, + filename=f"current_zoomed.{self.ext}", + ) + ) async def _create_zoomed_map(self, map_path, x, y, zoom, **kwargs): current_map = Image.open(map_path) diff --git a/fifo/timezones.py b/fifo/timezones.py index 5fdbdba..54d7c3e 100644 --- a/fifo/timezones.py +++ b/fifo/timezones.py @@ -15,182 +15,216 @@ def assemble_timezones(): """ timezones = {} - timezones['ACDT'] = timezone('Australia/Darwin') # Australian Central Daylight Savings Time (UTC+10:30) - timezones['ACST'] = timezone('Australia/Darwin') # Australian Central Standard Time (UTC+09:30) - timezones['ACT'] = timezone('Brazil/Acre') # Acre Time (UTC−05) - timezones['ADT'] = timezone('America/Halifax') # Atlantic Daylight Time (UTC−03) - timezones['AEDT'] = timezone('Australia/Sydney') # Australian Eastern Daylight Savings Time (UTC+11) - timezones['AEST'] = timezone('Australia/Sydney') # Australian Eastern Standard Time (UTC+10) - timezones['AFT'] = timezone('Asia/Kabul') # Afghanistan Time (UTC+04:30) - timezones['AKDT'] = timezone('America/Juneau') # Alaska Daylight Time (UTC−08) - timezones['AKST'] = timezone('America/Juneau') # Alaska Standard Time (UTC−09) - timezones['AMST'] = timezone('America/Manaus') # Amazon Summer Time (Brazil)[1] (UTC−03) - timezones['AMT'] = timezone('America/Manaus') # Amazon Time (Brazil)[2] (UTC−04) - timezones['ART'] = timezone('America/Cordoba') # Argentina Time (UTC−03) - timezones['AST'] = timezone('Asia/Riyadh') # Arabia Standard Time (UTC+03) - timezones['AWST'] = timezone('Australia/Perth') # Australian Western Standard Time (UTC+08) - timezones['AZOST'] = timezone('Atlantic/Azores') # Azores Summer Time (UTC±00) - timezones['AZOT'] = timezone('Atlantic/Azores') # Azores Standard Time (UTC−01) - timezones['AZT'] = timezone('Asia/Baku') # Azerbaijan Time (UTC+04) - timezones['BDT'] = timezone('Asia/Brunei') # Brunei Time (UTC+08) - timezones['BIOT'] = timezone('Etc/GMT+6') # British Indian Ocean Time (UTC+06) - timezones['BIT'] = timezone('Pacific/Funafuti') # Baker Island Time (UTC−12) - timezones['BOT'] = timezone('America/La_Paz') # Bolivia Time (UTC−04) - timezones['BRST'] = timezone('America/Sao_Paulo') # Brasilia Summer Time (UTC−02) - timezones['BRT'] = timezone('America/Sao_Paulo') # Brasilia Time (UTC−03) - timezones['BST'] = timezone('Asia/Dhaka') # Bangladesh Standard Time (UTC+06) - timezones['BTT'] = timezone('Asia/Thimphu') # Bhutan Time (UTC+06) - timezones['CAT'] = timezone('Africa/Harare') # Central Africa Time (UTC+02) - timezones['CCT'] = timezone('Indian/Cocos') # Cocos Islands Time (UTC+06:30) - timezones['CDT'] = timezone('America/Chicago') # Central Daylight Time (North America) (UTC−05) - timezones['CEST'] = timezone('Europe/Berlin') # Central European Summer Time (Cf. HAEC) (UTC+02) - timezones['CET'] = timezone('Europe/Berlin') # Central European Time (UTC+01) - timezones['CHADT'] = timezone('Pacific/Chatham') # Chatham Daylight Time (UTC+13:45) - timezones['CHAST'] = timezone('Pacific/Chatham') # Chatham Standard Time (UTC+12:45) - timezones['CHOST'] = timezone('Asia/Choibalsan') # Choibalsan Summer Time (UTC+09) - timezones['CHOT'] = timezone('Asia/Choibalsan') # Choibalsan Standard Time (UTC+08) - timezones['CHST'] = timezone('Pacific/Guam') # Chamorro Standard Time (UTC+10) - timezones['CHUT'] = timezone('Pacific/Chuuk') # Chuuk Time (UTC+10) - timezones['CIST'] = timezone('Etc/GMT-8') # Clipperton Island Standard Time (UTC−08) - timezones['CIT'] = timezone('Asia/Makassar') # Central Indonesia Time (UTC+08) - timezones['CKT'] = timezone('Pacific/Rarotonga') # Cook Island Time (UTC−10) - timezones['CLST'] = timezone('America/Santiago') # Chile Summer Time (UTC−03) - timezones['CLT'] = timezone('America/Santiago') # Chile Standard Time (UTC−04) - timezones['COST'] = timezone('America/Bogota') # Colombia Summer Time (UTC−04) - timezones['COT'] = timezone('America/Bogota') # Colombia Time (UTC−05) - timezones['CST'] = timezone('America/Chicago') # Central Standard Time (North America) (UTC−06) - timezones['CT'] = timezone('Asia/Chongqing') # China time (UTC+08) - timezones['CVT'] = timezone('Atlantic/Cape_Verde') # Cape Verde Time (UTC−01) - timezones['CXT'] = timezone('Indian/Christmas') # Christmas Island Time (UTC+07) - timezones['DAVT'] = timezone('Antarctica/Davis') # Davis Time (UTC+07) - timezones['DDUT'] = timezone('Antarctica/DumontDUrville') # Dumont d'Urville Time (UTC+10) - timezones['DFT'] = timezone('Europe/Berlin') # AIX equivalent of Central European Time (UTC+01) - timezones['EASST'] = timezone('Chile/EasterIsland') # Easter Island Summer Time (UTC−05) - timezones['EAST'] = timezone('Chile/EasterIsland') # Easter Island Standard Time (UTC−06) - timezones['EAT'] = timezone('Africa/Mogadishu') # East Africa Time (UTC+03) - timezones['ECT'] = timezone('America/Guayaquil') # Ecuador Time (UTC−05) - timezones['EDT'] = timezone('America/New_York') # Eastern Daylight Time (North America) (UTC−04) - timezones['EEST'] = timezone('Europe/Bucharest') # Eastern European Summer Time (UTC+03) - timezones['EET'] = timezone('Europe/Bucharest') # Eastern European Time (UTC+02) - timezones['EGST'] = timezone('America/Scoresbysund') # Eastern Greenland Summer Time (UTC±00) - timezones['EGT'] = timezone('America/Scoresbysund') # Eastern Greenland Time (UTC−01) - timezones['EIT'] = timezone('Asia/Jayapura') # Eastern Indonesian Time (UTC+09) - timezones['EST'] = timezone('America/New_York') # Eastern Standard Time (North America) (UTC−05) - timezones['FET'] = timezone('Europe/Minsk') # Further-eastern European Time (UTC+03) - timezones['FJT'] = timezone('Pacific/Fiji') # Fiji Time (UTC+12) - timezones['FKST'] = timezone('Atlantic/Stanley') # Falkland Islands Summer Time (UTC−03) - timezones['FKT'] = timezone('Atlantic/Stanley') # Falkland Islands Time (UTC−04) - timezones['FNT'] = timezone('Brazil/DeNoronha') # Fernando de Noronha Time (UTC−02) - timezones['GALT'] = timezone('Pacific/Galapagos') # Galapagos Time (UTC−06) - timezones['GAMT'] = timezone('Pacific/Gambier') # Gambier Islands (UTC−09) - timezones['GET'] = timezone('Asia/Tbilisi') # Georgia Standard Time (UTC+04) - timezones['GFT'] = timezone('America/Cayenne') # French Guiana Time (UTC−03) - timezones['GILT'] = timezone('Pacific/Tarawa') # Gilbert Island Time (UTC+12) - timezones['GIT'] = timezone('Pacific/Gambier') # Gambier Island Time (UTC−09) - timezones['GMT'] = timezone('GMT') # Greenwich Mean Time (UTC±00) - timezones['GST'] = timezone('Asia/Muscat') # Gulf Standard Time (UTC+04) - timezones['GYT'] = timezone('America/Guyana') # Guyana Time (UTC−04) - timezones['HADT'] = timezone('Pacific/Honolulu') # Hawaii-Aleutian Daylight Time (UTC−09) - timezones['HAEC'] = timezone('Europe/Paris') # Heure Avancée d'Europe Centrale (CEST) (UTC+02) - timezones['HAST'] = timezone('Pacific/Honolulu') # Hawaii-Aleutian Standard Time (UTC−10) - timezones['HKT'] = timezone('Asia/Hong_Kong') # Hong Kong Time (UTC+08) - timezones['HMT'] = timezone('Indian/Kerguelen') # Heard and McDonald Islands Time (UTC+05) - timezones['HOVST'] = timezone('Asia/Hovd') # Khovd Summer Time (UTC+08) - timezones['HOVT'] = timezone('Asia/Hovd') # Khovd Standard Time (UTC+07) - timezones['ICT'] = timezone('Asia/Ho_Chi_Minh') # Indochina Time (UTC+07) - timezones['IDT'] = timezone('Asia/Jerusalem') # Israel Daylight Time (UTC+03) - timezones['IOT'] = timezone('Etc/GMT+3') # Indian Ocean Time (UTC+03) - timezones['IRDT'] = timezone('Asia/Tehran') # Iran Daylight Time (UTC+04:30) - timezones['IRKT'] = timezone('Asia/Irkutsk') # Irkutsk Time (UTC+08) - timezones['IRST'] = timezone('Asia/Tehran') # Iran Standard Time (UTC+03:30) - timezones['IST'] = timezone('Asia/Kolkata') # Indian Standard Time (UTC+05:30) - timezones['JST'] = timezone('Asia/Tokyo') # Japan Standard Time (UTC+09) - timezones['KGT'] = timezone('Asia/Bishkek') # Kyrgyzstan time (UTC+06) - timezones['KOST'] = timezone('Pacific/Kosrae') # Kosrae Time (UTC+11) - timezones['KRAT'] = timezone('Asia/Krasnoyarsk') # Krasnoyarsk Time (UTC+07) - timezones['KST'] = timezone('Asia/Seoul') # Korea Standard Time (UTC+09) - timezones['LHST'] = timezone('Australia/Lord_Howe') # Lord Howe Standard Time (UTC+10:30) - timezones['LINT'] = timezone('Pacific/Kiritimati') # Line Islands Time (UTC+14) - timezones['MAGT'] = timezone('Asia/Magadan') # Magadan Time (UTC+12) - timezones['MART'] = timezone('Pacific/Marquesas') # Marquesas Islands Time (UTC−09:30) - timezones['MAWT'] = timezone('Antarctica/Mawson') # Mawson Station Time (UTC+05) - timezones['MDT'] = timezone('America/Denver') # Mountain Daylight Time (North America) (UTC−06) - timezones['MEST'] = timezone('Europe/Paris') # Middle European Summer Time Same zone as CEST (UTC+02) - timezones['MET'] = timezone('Europe/Berlin') # Middle European Time Same zone as CET (UTC+01) - timezones['MHT'] = timezone('Pacific/Kwajalein') # Marshall Islands (UTC+12) - timezones['MIST'] = timezone('Antarctica/Macquarie') # Macquarie Island Station Time (UTC+11) - timezones['MIT'] = timezone('Pacific/Marquesas') # Marquesas Islands Time (UTC−09:30) - timezones['MMT'] = timezone('Asia/Rangoon') # Myanmar Standard Time (UTC+06:30) - timezones['MSK'] = timezone('Europe/Moscow') # Moscow Time (UTC+03) - timezones['MST'] = timezone('America/Denver') # Mountain Standard Time (North America) (UTC−07) - timezones['MUT'] = timezone('Indian/Mauritius') # Mauritius Time (UTC+04) - timezones['MVT'] = timezone('Indian/Maldives') # Maldives Time (UTC+05) - timezones['MYT'] = timezone('Asia/Kuching') # Malaysia Time (UTC+08) - timezones['NCT'] = timezone('Pacific/Noumea') # New Caledonia Time (UTC+11) - timezones['NDT'] = timezone('Canada/Newfoundland') # Newfoundland Daylight Time (UTC−02:30) - timezones['NFT'] = timezone('Pacific/Norfolk') # Norfolk Time (UTC+11) - timezones['NPT'] = timezone('Asia/Kathmandu') # Nepal Time (UTC+05:45) - timezones['NST'] = timezone('Canada/Newfoundland') # Newfoundland Standard Time (UTC−03:30) - timezones['NT'] = timezone('Canada/Newfoundland') # Newfoundland Time (UTC−03:30) - timezones['NUT'] = timezone('Pacific/Niue') # Niue Time (UTC−11) - timezones['NZDT'] = timezone('Pacific/Auckland') # New Zealand Daylight Time (UTC+13) - timezones['NZST'] = timezone('Pacific/Auckland') # New Zealand Standard Time (UTC+12) - timezones['OMST'] = timezone('Asia/Omsk') # Omsk Time (UTC+06) - timezones['ORAT'] = timezone('Asia/Oral') # Oral Time (UTC+05) - timezones['PDT'] = timezone('America/Los_Angeles') # Pacific Daylight Time (North America) (UTC−07) - timezones['PET'] = timezone('America/Lima') # Peru Time (UTC−05) - timezones['PETT'] = timezone('Asia/Kamchatka') # Kamchatka Time (UTC+12) - timezones['PGT'] = timezone('Pacific/Port_Moresby') # Papua New Guinea Time (UTC+10) - timezones['PHOT'] = timezone('Pacific/Enderbury') # Phoenix Island Time (UTC+13) - timezones['PKT'] = timezone('Asia/Karachi') # Pakistan Standard Time (UTC+05) - timezones['PMDT'] = timezone('America/Miquelon') # Saint Pierre and Miquelon Daylight time (UTC−02) - timezones['PMST'] = timezone('America/Miquelon') # Saint Pierre and Miquelon Standard Time (UTC−03) - timezones['PONT'] = timezone('Pacific/Pohnpei') # Pohnpei Standard Time (UTC+11) - timezones['PST'] = timezone('America/Los_Angeles') # Pacific Standard Time (North America) (UTC−08) - timezones['PYST'] = timezone('America/Asuncion') # Paraguay Summer Time (South America)[7] (UTC−03) - timezones['PYT'] = timezone('America/Asuncion') # Paraguay Time (South America)[8] (UTC−04) - timezones['RET'] = timezone('Indian/Reunion') # Réunion Time (UTC+04) - timezones['ROTT'] = timezone('Antarctica/Rothera') # Rothera Research Station Time (UTC−03) - timezones['SAKT'] = timezone('Asia/Vladivostok') # Sakhalin Island time (UTC+11) - timezones['SAMT'] = timezone('Europe/Samara') # Samara Time (UTC+04) - timezones['SAST'] = timezone('Africa/Johannesburg') # South African Standard Time (UTC+02) - timezones['SBT'] = timezone('Pacific/Guadalcanal') # Solomon Islands Time (UTC+11) - timezones['SCT'] = timezone('Indian/Mahe') # Seychelles Time (UTC+04) - timezones['SGT'] = timezone('Asia/Singapore') # Singapore Time (UTC+08) - timezones['SLST'] = timezone('Asia/Colombo') # Sri Lanka Standard Time (UTC+05:30) - timezones['SRET'] = timezone('Asia/Srednekolymsk') # Srednekolymsk Time (UTC+11) - timezones['SRT'] = timezone('America/Paramaribo') # Suriname Time (UTC−03) - timezones['SST'] = timezone('Asia/Singapore') # Singapore Standard Time (UTC+08) - timezones['SYOT'] = timezone('Antarctica/Syowa') # Showa Station Time (UTC+03) - timezones['TAHT'] = timezone('Pacific/Tahiti') # Tahiti Time (UTC−10) - timezones['TFT'] = timezone('Indian/Kerguelen') # Indian/Kerguelen (UTC+05) - timezones['THA'] = timezone('Asia/Bangkok') # Thailand Standard Time (UTC+07) - timezones['TJT'] = timezone('Asia/Dushanbe') # Tajikistan Time (UTC+05) - timezones['TKT'] = timezone('Pacific/Fakaofo') # Tokelau Time (UTC+13) - timezones['TLT'] = timezone('Asia/Dili') # Timor Leste Time (UTC+09) - timezones['TMT'] = timezone('Asia/Ashgabat') # Turkmenistan Time (UTC+05) - timezones['TOT'] = timezone('Pacific/Tongatapu') # Tonga Time (UTC+13) - timezones['TVT'] = timezone('Pacific/Funafuti') # Tuvalu Time (UTC+12) - timezones['ULAST'] = timezone('Asia/Ulan_Bator') # Ulaanbaatar Summer Time (UTC+09) - timezones['ULAT'] = timezone('Asia/Ulan_Bator') # Ulaanbaatar Standard Time (UTC+08) - timezones['USZ1'] = timezone('Europe/Kaliningrad') # Kaliningrad Time (UTC+02) - timezones['UTC'] = timezone('UTC') # Coordinated Universal Time (UTC±00) - timezones['UYST'] = timezone('America/Montevideo') # Uruguay Summer Time (UTC−02) - timezones['UYT'] = timezone('America/Montevideo') # Uruguay Standard Time (UTC−03) - timezones['UZT'] = timezone('Asia/Tashkent') # Uzbekistan Time (UTC+05) - timezones['VET'] = timezone('America/Caracas') # Venezuelan Standard Time (UTC−04) - timezones['VLAT'] = timezone('Asia/Vladivostok') # Vladivostok Time (UTC+10) - timezones['VOLT'] = timezone('Europe/Volgograd') # Volgograd Time (UTC+04) - timezones['VOST'] = timezone('Antarctica/Vostok') # Vostok Station Time (UTC+06) - timezones['VUT'] = timezone('Pacific/Efate') # Vanuatu Time (UTC+11) - timezones['WAKT'] = timezone('Pacific/Wake') # Wake Island Time (UTC+12) - timezones['WAST'] = timezone('Africa/Lagos') # West Africa Summer Time (UTC+02) - timezones['WAT'] = timezone('Africa/Lagos') # West Africa Time (UTC+01) - timezones['WEST'] = timezone('Europe/London') # Western European Summer Time (UTC+01) - timezones['WET'] = timezone('Europe/London') # Western European Time (UTC±00) - timezones['WIT'] = timezone('Asia/Jakarta') # Western Indonesian Time (UTC+07) - timezones['WST'] = timezone('Australia/Perth') # Western Standard Time (UTC+08) - timezones['YAKT'] = timezone('Asia/Yakutsk') # Yakutsk Time (UTC+09) - timezones['YEKT'] = timezone('Asia/Yekaterinburg') # Yekaterinburg Time (UTC+05) + timezones["ACDT"] = timezone( + "Australia/Darwin" + ) # Australian Central Daylight Savings Time (UTC+10:30) + timezones["ACST"] = timezone( + "Australia/Darwin" + ) # Australian Central Standard Time (UTC+09:30) + timezones["ACT"] = timezone("Brazil/Acre") # Acre Time (UTC−05) + timezones["ADT"] = timezone("America/Halifax") # Atlantic Daylight Time (UTC−03) + timezones["AEDT"] = timezone( + "Australia/Sydney" + ) # Australian Eastern Daylight Savings Time (UTC+11) + timezones["AEST"] = timezone("Australia/Sydney") # Australian Eastern Standard Time (UTC+10) + timezones["AFT"] = timezone("Asia/Kabul") # Afghanistan Time (UTC+04:30) + timezones["AKDT"] = timezone("America/Juneau") # Alaska Daylight Time (UTC−08) + timezones["AKST"] = timezone("America/Juneau") # Alaska Standard Time (UTC−09) + timezones["AMST"] = timezone("America/Manaus") # Amazon Summer Time (Brazil)[1] (UTC−03) + timezones["AMT"] = timezone("America/Manaus") # Amazon Time (Brazil)[2] (UTC−04) + timezones["ART"] = timezone("America/Cordoba") # Argentina Time (UTC−03) + timezones["AST"] = timezone("Asia/Riyadh") # Arabia Standard Time (UTC+03) + timezones["AWST"] = timezone("Australia/Perth") # Australian Western Standard Time (UTC+08) + timezones["AZOST"] = timezone("Atlantic/Azores") # Azores Summer Time (UTC±00) + timezones["AZOT"] = timezone("Atlantic/Azores") # Azores Standard Time (UTC−01) + timezones["AZT"] = timezone("Asia/Baku") # Azerbaijan Time (UTC+04) + timezones["BDT"] = timezone("Asia/Brunei") # Brunei Time (UTC+08) + timezones["BIOT"] = timezone("Etc/GMT+6") # British Indian Ocean Time (UTC+06) + timezones["BIT"] = timezone("Pacific/Funafuti") # Baker Island Time (UTC−12) + timezones["BOT"] = timezone("America/La_Paz") # Bolivia Time (UTC−04) + timezones["BRST"] = timezone("America/Sao_Paulo") # Brasilia Summer Time (UTC−02) + timezones["BRT"] = timezone("America/Sao_Paulo") # Brasilia Time (UTC−03) + timezones["BST"] = timezone("Asia/Dhaka") # Bangladesh Standard Time (UTC+06) + timezones["BTT"] = timezone("Asia/Thimphu") # Bhutan Time (UTC+06) + timezones["CAT"] = timezone("Africa/Harare") # Central Africa Time (UTC+02) + timezones["CCT"] = timezone("Indian/Cocos") # Cocos Islands Time (UTC+06:30) + timezones["CDT"] = timezone( + "America/Chicago" + ) # Central Daylight Time (North America) (UTC−05) + timezones["CEST"] = timezone( + "Europe/Berlin" + ) # Central European Summer Time (Cf. HAEC) (UTC+02) + timezones["CET"] = timezone("Europe/Berlin") # Central European Time (UTC+01) + timezones["CHADT"] = timezone("Pacific/Chatham") # Chatham Daylight Time (UTC+13:45) + timezones["CHAST"] = timezone("Pacific/Chatham") # Chatham Standard Time (UTC+12:45) + timezones["CHOST"] = timezone("Asia/Choibalsan") # Choibalsan Summer Time (UTC+09) + timezones["CHOT"] = timezone("Asia/Choibalsan") # Choibalsan Standard Time (UTC+08) + timezones["CHST"] = timezone("Pacific/Guam") # Chamorro Standard Time (UTC+10) + timezones["CHUT"] = timezone("Pacific/Chuuk") # Chuuk Time (UTC+10) + timezones["CIST"] = timezone("Etc/GMT-8") # Clipperton Island Standard Time (UTC−08) + timezones["CIT"] = timezone("Asia/Makassar") # Central Indonesia Time (UTC+08) + timezones["CKT"] = timezone("Pacific/Rarotonga") # Cook Island Time (UTC−10) + timezones["CLST"] = timezone("America/Santiago") # Chile Summer Time (UTC−03) + timezones["CLT"] = timezone("America/Santiago") # Chile Standard Time (UTC−04) + timezones["COST"] = timezone("America/Bogota") # Colombia Summer Time (UTC−04) + timezones["COT"] = timezone("America/Bogota") # Colombia Time (UTC−05) + timezones["CST"] = timezone( + "America/Chicago" + ) # Central Standard Time (North America) (UTC−06) + timezones["CT"] = timezone("Asia/Chongqing") # China time (UTC+08) + timezones["CVT"] = timezone("Atlantic/Cape_Verde") # Cape Verde Time (UTC−01) + timezones["CXT"] = timezone("Indian/Christmas") # Christmas Island Time (UTC+07) + timezones["DAVT"] = timezone("Antarctica/Davis") # Davis Time (UTC+07) + timezones["DDUT"] = timezone("Antarctica/DumontDUrville") # Dumont d'Urville Time (UTC+10) + timezones["DFT"] = timezone( + "Europe/Berlin" + ) # AIX equivalent of Central European Time (UTC+01) + timezones["EASST"] = timezone("Chile/EasterIsland") # Easter Island Summer Time (UTC−05) + timezones["EAST"] = timezone("Chile/EasterIsland") # Easter Island Standard Time (UTC−06) + timezones["EAT"] = timezone("Africa/Mogadishu") # East Africa Time (UTC+03) + timezones["ECT"] = timezone("America/Guayaquil") # Ecuador Time (UTC−05) + timezones["EDT"] = timezone( + "America/New_York" + ) # Eastern Daylight Time (North America) (UTC−04) + timezones["EEST"] = timezone("Europe/Bucharest") # Eastern European Summer Time (UTC+03) + timezones["EET"] = timezone("Europe/Bucharest") # Eastern European Time (UTC+02) + timezones["EGST"] = timezone("America/Scoresbysund") # Eastern Greenland Summer Time (UTC±00) + timezones["EGT"] = timezone("America/Scoresbysund") # Eastern Greenland Time (UTC−01) + timezones["EIT"] = timezone("Asia/Jayapura") # Eastern Indonesian Time (UTC+09) + timezones["EST"] = timezone( + "America/New_York" + ) # Eastern Standard Time (North America) (UTC−05) + timezones["FET"] = timezone("Europe/Minsk") # Further-eastern European Time (UTC+03) + timezones["FJT"] = timezone("Pacific/Fiji") # Fiji Time (UTC+12) + timezones["FKST"] = timezone("Atlantic/Stanley") # Falkland Islands Summer Time (UTC−03) + timezones["FKT"] = timezone("Atlantic/Stanley") # Falkland Islands Time (UTC−04) + timezones["FNT"] = timezone("Brazil/DeNoronha") # Fernando de Noronha Time (UTC−02) + timezones["GALT"] = timezone("Pacific/Galapagos") # Galapagos Time (UTC−06) + timezones["GAMT"] = timezone("Pacific/Gambier") # Gambier Islands (UTC−09) + timezones["GET"] = timezone("Asia/Tbilisi") # Georgia Standard Time (UTC+04) + timezones["GFT"] = timezone("America/Cayenne") # French Guiana Time (UTC−03) + timezones["GILT"] = timezone("Pacific/Tarawa") # Gilbert Island Time (UTC+12) + timezones["GIT"] = timezone("Pacific/Gambier") # Gambier Island Time (UTC−09) + timezones["GMT"] = timezone("GMT") # Greenwich Mean Time (UTC±00) + timezones["GST"] = timezone("Asia/Muscat") # Gulf Standard Time (UTC+04) + timezones["GYT"] = timezone("America/Guyana") # Guyana Time (UTC−04) + timezones["HADT"] = timezone("Pacific/Honolulu") # Hawaii-Aleutian Daylight Time (UTC−09) + timezones["HAEC"] = timezone("Europe/Paris") # Heure Avancée d'Europe Centrale (CEST) (UTC+02) + timezones["HAST"] = timezone("Pacific/Honolulu") # Hawaii-Aleutian Standard Time (UTC−10) + timezones["HKT"] = timezone("Asia/Hong_Kong") # Hong Kong Time (UTC+08) + timezones["HMT"] = timezone("Indian/Kerguelen") # Heard and McDonald Islands Time (UTC+05) + timezones["HOVST"] = timezone("Asia/Hovd") # Khovd Summer Time (UTC+08) + timezones["HOVT"] = timezone("Asia/Hovd") # Khovd Standard Time (UTC+07) + timezones["ICT"] = timezone("Asia/Ho_Chi_Minh") # Indochina Time (UTC+07) + timezones["IDT"] = timezone("Asia/Jerusalem") # Israel Daylight Time (UTC+03) + timezones["IOT"] = timezone("Etc/GMT+3") # Indian Ocean Time (UTC+03) + timezones["IRDT"] = timezone("Asia/Tehran") # Iran Daylight Time (UTC+04:30) + timezones["IRKT"] = timezone("Asia/Irkutsk") # Irkutsk Time (UTC+08) + timezones["IRST"] = timezone("Asia/Tehran") # Iran Standard Time (UTC+03:30) + timezones["IST"] = timezone("Asia/Kolkata") # Indian Standard Time (UTC+05:30) + timezones["JST"] = timezone("Asia/Tokyo") # Japan Standard Time (UTC+09) + timezones["KGT"] = timezone("Asia/Bishkek") # Kyrgyzstan time (UTC+06) + timezones["KOST"] = timezone("Pacific/Kosrae") # Kosrae Time (UTC+11) + timezones["KRAT"] = timezone("Asia/Krasnoyarsk") # Krasnoyarsk Time (UTC+07) + timezones["KST"] = timezone("Asia/Seoul") # Korea Standard Time (UTC+09) + timezones["LHST"] = timezone("Australia/Lord_Howe") # Lord Howe Standard Time (UTC+10:30) + timezones["LINT"] = timezone("Pacific/Kiritimati") # Line Islands Time (UTC+14) + timezones["MAGT"] = timezone("Asia/Magadan") # Magadan Time (UTC+12) + timezones["MART"] = timezone("Pacific/Marquesas") # Marquesas Islands Time (UTC−09:30) + timezones["MAWT"] = timezone("Antarctica/Mawson") # Mawson Station Time (UTC+05) + timezones["MDT"] = timezone( + "America/Denver" + ) # Mountain Daylight Time (North America) (UTC−06) + timezones["MEST"] = timezone( + "Europe/Paris" + ) # Middle European Summer Time Same zone as CEST (UTC+02) + timezones["MET"] = timezone("Europe/Berlin") # Middle European Time Same zone as CET (UTC+01) + timezones["MHT"] = timezone("Pacific/Kwajalein") # Marshall Islands (UTC+12) + timezones["MIST"] = timezone("Antarctica/Macquarie") # Macquarie Island Station Time (UTC+11) + timezones["MIT"] = timezone("Pacific/Marquesas") # Marquesas Islands Time (UTC−09:30) + timezones["MMT"] = timezone("Asia/Rangoon") # Myanmar Standard Time (UTC+06:30) + timezones["MSK"] = timezone("Europe/Moscow") # Moscow Time (UTC+03) + timezones["MST"] = timezone( + "America/Denver" + ) # Mountain Standard Time (North America) (UTC−07) + timezones["MUT"] = timezone("Indian/Mauritius") # Mauritius Time (UTC+04) + timezones["MVT"] = timezone("Indian/Maldives") # Maldives Time (UTC+05) + timezones["MYT"] = timezone("Asia/Kuching") # Malaysia Time (UTC+08) + timezones["NCT"] = timezone("Pacific/Noumea") # New Caledonia Time (UTC+11) + timezones["NDT"] = timezone("Canada/Newfoundland") # Newfoundland Daylight Time (UTC−02:30) + timezones["NFT"] = timezone("Pacific/Norfolk") # Norfolk Time (UTC+11) + timezones["NPT"] = timezone("Asia/Kathmandu") # Nepal Time (UTC+05:45) + timezones["NST"] = timezone("Canada/Newfoundland") # Newfoundland Standard Time (UTC−03:30) + timezones["NT"] = timezone("Canada/Newfoundland") # Newfoundland Time (UTC−03:30) + timezones["NUT"] = timezone("Pacific/Niue") # Niue Time (UTC−11) + timezones["NZDT"] = timezone("Pacific/Auckland") # New Zealand Daylight Time (UTC+13) + timezones["NZST"] = timezone("Pacific/Auckland") # New Zealand Standard Time (UTC+12) + timezones["OMST"] = timezone("Asia/Omsk") # Omsk Time (UTC+06) + timezones["ORAT"] = timezone("Asia/Oral") # Oral Time (UTC+05) + timezones["PDT"] = timezone( + "America/Los_Angeles" + ) # Pacific Daylight Time (North America) (UTC−07) + timezones["PET"] = timezone("America/Lima") # Peru Time (UTC−05) + timezones["PETT"] = timezone("Asia/Kamchatka") # Kamchatka Time (UTC+12) + timezones["PGT"] = timezone("Pacific/Port_Moresby") # Papua New Guinea Time (UTC+10) + timezones["PHOT"] = timezone("Pacific/Enderbury") # Phoenix Island Time (UTC+13) + timezones["PKT"] = timezone("Asia/Karachi") # Pakistan Standard Time (UTC+05) + timezones["PMDT"] = timezone( + "America/Miquelon" + ) # Saint Pierre and Miquelon Daylight time (UTC−02) + timezones["PMST"] = timezone( + "America/Miquelon" + ) # Saint Pierre and Miquelon Standard Time (UTC−03) + timezones["PONT"] = timezone("Pacific/Pohnpei") # Pohnpei Standard Time (UTC+11) + timezones["PST"] = timezone( + "America/Los_Angeles" + ) # Pacific Standard Time (North America) (UTC−08) + timezones["PYST"] = timezone( + "America/Asuncion" + ) # Paraguay Summer Time (South America)[7] (UTC−03) + timezones["PYT"] = timezone("America/Asuncion") # Paraguay Time (South America)[8] (UTC−04) + timezones["RET"] = timezone("Indian/Reunion") # Réunion Time (UTC+04) + timezones["ROTT"] = timezone("Antarctica/Rothera") # Rothera Research Station Time (UTC−03) + timezones["SAKT"] = timezone("Asia/Vladivostok") # Sakhalin Island time (UTC+11) + timezones["SAMT"] = timezone("Europe/Samara") # Samara Time (UTC+04) + timezones["SAST"] = timezone("Africa/Johannesburg") # South African Standard Time (UTC+02) + timezones["SBT"] = timezone("Pacific/Guadalcanal") # Solomon Islands Time (UTC+11) + timezones["SCT"] = timezone("Indian/Mahe") # Seychelles Time (UTC+04) + timezones["SGT"] = timezone("Asia/Singapore") # Singapore Time (UTC+08) + timezones["SLST"] = timezone("Asia/Colombo") # Sri Lanka Standard Time (UTC+05:30) + timezones["SRET"] = timezone("Asia/Srednekolymsk") # Srednekolymsk Time (UTC+11) + timezones["SRT"] = timezone("America/Paramaribo") # Suriname Time (UTC−03) + timezones["SST"] = timezone("Asia/Singapore") # Singapore Standard Time (UTC+08) + timezones["SYOT"] = timezone("Antarctica/Syowa") # Showa Station Time (UTC+03) + timezones["TAHT"] = timezone("Pacific/Tahiti") # Tahiti Time (UTC−10) + timezones["TFT"] = timezone("Indian/Kerguelen") # Indian/Kerguelen (UTC+05) + timezones["THA"] = timezone("Asia/Bangkok") # Thailand Standard Time (UTC+07) + timezones["TJT"] = timezone("Asia/Dushanbe") # Tajikistan Time (UTC+05) + timezones["TKT"] = timezone("Pacific/Fakaofo") # Tokelau Time (UTC+13) + timezones["TLT"] = timezone("Asia/Dili") # Timor Leste Time (UTC+09) + timezones["TMT"] = timezone("Asia/Ashgabat") # Turkmenistan Time (UTC+05) + timezones["TOT"] = timezone("Pacific/Tongatapu") # Tonga Time (UTC+13) + timezones["TVT"] = timezone("Pacific/Funafuti") # Tuvalu Time (UTC+12) + timezones["ULAST"] = timezone("Asia/Ulan_Bator") # Ulaanbaatar Summer Time (UTC+09) + timezones["ULAT"] = timezone("Asia/Ulan_Bator") # Ulaanbaatar Standard Time (UTC+08) + timezones["USZ1"] = timezone("Europe/Kaliningrad") # Kaliningrad Time (UTC+02) + timezones["UTC"] = timezone("UTC") # Coordinated Universal Time (UTC±00) + timezones["UYST"] = timezone("America/Montevideo") # Uruguay Summer Time (UTC−02) + timezones["UYT"] = timezone("America/Montevideo") # Uruguay Standard Time (UTC−03) + timezones["UZT"] = timezone("Asia/Tashkent") # Uzbekistan Time (UTC+05) + timezones["VET"] = timezone("America/Caracas") # Venezuelan Standard Time (UTC−04) + timezones["VLAT"] = timezone("Asia/Vladivostok") # Vladivostok Time (UTC+10) + timezones["VOLT"] = timezone("Europe/Volgograd") # Volgograd Time (UTC+04) + timezones["VOST"] = timezone("Antarctica/Vostok") # Vostok Station Time (UTC+06) + timezones["VUT"] = timezone("Pacific/Efate") # Vanuatu Time (UTC+11) + timezones["WAKT"] = timezone("Pacific/Wake") # Wake Island Time (UTC+12) + timezones["WAST"] = timezone("Africa/Lagos") # West Africa Summer Time (UTC+02) + timezones["WAT"] = timezone("Africa/Lagos") # West Africa Time (UTC+01) + timezones["WEST"] = timezone("Europe/London") # Western European Summer Time (UTC+01) + timezones["WET"] = timezone("Europe/London") # Western European Time (UTC±00) + timezones["WIT"] = timezone("Asia/Jakarta") # Western Indonesian Time (UTC+07) + timezones["WST"] = timezone("Australia/Perth") # Western Standard Time (UTC+08) + timezones["YAKT"] = timezone("Asia/Yakutsk") # Yakutsk Time (UTC+09) + timezones["YEKT"] = timezone("Asia/Yekaterinburg") # Yekaterinburg Time (UTC+05) return timezones diff --git a/flag/flag.py b/flag/flag.py index 035ae1d..6216f65 100644 --- a/flag/flag.py +++ b/flag/flag.py @@ -54,7 +54,7 @@ class Flag(Cog): async def flagset(self, ctx: commands.Context): """ My custom cog - + Extra information goes here """ if ctx.invoked_subcommand is None: diff --git a/forcemention/forcemention.py b/forcemention/forcemention.py index dd8948c..3df68e5 100644 --- a/forcemention/forcemention.py +++ b/forcemention/forcemention.py @@ -30,8 +30,8 @@ class ForceMention(Cog): @commands.command() async def forcemention(self, ctx: commands.Context, role: str, *, message=""): """ - Mentions that role, regardless if it's unmentionable - """ + Mentions that role, regardless if it's unmentionable + """ role_obj = get(ctx.guild.roles, name=role) if role_obj is None: await ctx.maybe_send_embed("Couldn't find role named {}".format(role)) diff --git a/nudity/nudity.py b/nudity/nudity.py index 0d46ca9..4233460 100644 --- a/nudity/nudity.py +++ b/nudity/nudity.py @@ -85,7 +85,9 @@ class Nudity(commands.Cog): if r["unsafe"] > 0.7: await nsfw_channel.send( "NSFW Image from {}".format(message.channel.mention), - file=discord.File(image,), + file=discord.File( + image, + ), ) @commands.Cog.listener() diff --git a/rpsls/rpsls.py b/rpsls/rpsls.py index b831d1c..ed2e1dc 100644 --- a/rpsls/rpsls.py +++ b/rpsls/rpsls.py @@ -28,8 +28,8 @@ class RPSLS(Cog): @commands.command() async def rpsls(self, ctx: commands.Context, choice: str): """ - Play Rock Paper Scissors Lizard Spock by Sam Kass in Discord! - + Play Rock Paper Scissors Lizard Spock by Sam Kass in Discord! + Rules: Scissors cuts Paper Paper covers Rock diff --git a/tts/tts.py b/tts/tts.py index 1291777..235d585 100644 --- a/tts/tts.py +++ b/tts/tts.py @@ -30,8 +30,8 @@ class TTS(Cog): @commands.command(aliases=["t2s", "text2"]) async def tts(self, ctx: commands.Context, *, text: str): """ - Send Text to speech messages as an mp3 - """ + Send Text to speech messages as an mp3 + """ mp3_fp = io.BytesIO() tts = gTTS(text, lang="en") tts.write_to_fp(mp3_fp) From 266b0a485d03d02f40640871d86255fa522e0760 Mon Sep 17 00:00:00 2001 From: bobloy Date: Tue, 29 Sep 2020 17:39:07 -0400 Subject: [PATCH 55/73] Alpha ready changes --- werewolf/game.py | 2 +- werewolf/info.json | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/werewolf/game.py b/werewolf/game.py index 63b922a..668bf16 100644 --- a/werewolf/game.py +++ b/werewolf/game.py @@ -17,7 +17,7 @@ from werewolf.votegroup import VoteGroup log = logging.getLogger("red.fox_v3.werewolf.game") -HALF_DAY_LENGTH = 60 # FixMe: Make configurable +HALF_DAY_LENGTH = 90 # FixMe: Make configurable HALF_NIGHT_LENGTH = 60 diff --git a/werewolf/info.json b/werewolf/info.json index 99bc768..c8ef454 100644 --- a/werewolf/info.json +++ b/werewolf/info.json @@ -4,10 +4,10 @@ ], "min_bot_version": "3.3.0", "description": "Customizable Werewolf Game", - "hidden": true, + "hidden": false, "install_msg": "Thank you for installing Werewolf! Get started with `[p]load werewolf`\n Use `[p]wwset` to run inital setup", "requirements": [], - "short": "Werewolf Game", + "short": "[ALPHA] Play Werewolf (Mafia) Game in discord", "end_user_data_statement": "This stores user IDs in memory while they're actively using the cog, and stores no persistent End User Data.", "tags": [ "mafia", From da754e3cb2f6df5ca5b3ba0f85f3fd71ae03827d Mon Sep 17 00:00:00 2001 From: bobloy Date: Wed, 30 Sep 2020 10:31:53 -0400 Subject: [PATCH 56/73] Update to latest version of labeler --- .github/workflows/labeler.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/labeler.yml b/.github/workflows/labeler.yml index 7c724a6..65e6640 100644 --- a/.github/workflows/labeler.yml +++ b/.github/workflows/labeler.yml @@ -14,6 +14,6 @@ jobs: runs-on: ubuntu-latest steps: - - uses: actions/labeler@v2 + - uses: actions/labeler@2.2.0 with: repo-token: "${{ secrets.GITHUB_TOKEN }}" From b210f4a9ff4f2a637914f9cf5a4fc42efb81ab70 Mon Sep 17 00:00:00 2001 From: bobloy Date: Wed, 30 Sep 2020 12:13:40 -0400 Subject: [PATCH 57/73] Uppercase key --- audiotrivia/data/lists/videogames.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/audiotrivia/data/lists/videogames.yaml b/audiotrivia/data/lists/videogames.yaml index eec01b6..5798f0c 100644 --- a/audiotrivia/data/lists/videogames.yaml +++ b/audiotrivia/data/lists/videogames.yaml @@ -1,4 +1,4 @@ -Author: Bobloy +AUTHOR: Bobloy https://www.youtube.com/watch?v=GBPbJyxqHV0: - Super Mario 64 https://www.youtube.com/watch?v=0jXTBAGv9ZQ: From 3fceea634bb6a9e6e590a013cda8819d653f1532 Mon Sep 17 00:00:00 2001 From: bobloy Date: Thu, 1 Oct 2020 09:10:02 -0400 Subject: [PATCH 58/73] Audiotrivia updates from lessons learned attempting core --- audiotrivia/audiosession.py | 117 ++++++++----- audiotrivia/audiotrivia.py | 159 +++++++++--------- .../lists/{anime.yaml => audioanime.yaml} | 1 + ...lgoalhorns.yaml => audionhlgoalhorns.yaml} | 1 + .../{videogames.yaml => audiovideogames.yaml} | 1 + 5 files changed, 154 insertions(+), 125 deletions(-) rename audiotrivia/data/lists/{anime.yaml => audioanime.yaml} (99%) rename audiotrivia/data/lists/{nhlgoalhorns.yaml => audionhlgoalhorns.yaml} (97%) rename audiotrivia/data/lists/{videogames.yaml => audiovideogames.yaml} (99%) diff --git a/audiotrivia/audiosession.py b/audiotrivia/audiosession.py index 1bdff02..17fc998 100644 --- a/audiotrivia/audiosession.py +++ b/audiotrivia/audiosession.py @@ -2,9 +2,8 @@ import asyncio import logging -import lavalink -from lavalink.enums import LoadType from redbot.cogs.trivia import TriviaSession +from redbot.cogs.trivia.session import _parse_answers from redbot.core.utils.chat_formatting import bold log = logging.getLogger("red.fox_v3.audiotrivia.audiosession") @@ -13,14 +12,14 @@ log = logging.getLogger("red.fox_v3.audiotrivia.audiosession") class AudioSession(TriviaSession): """Class to run a session of audio trivia""" - def __init__(self, ctx, question_list: dict, settings: dict, player: lavalink.Player): + def __init__(self, ctx, question_list: dict, settings: dict, audio = None): super().__init__(ctx, question_list, settings) - self.player = player + self.audio = audio @classmethod - def start(cls, ctx, question_list, settings, player: lavalink.Player = None): - session = cls(ctx, question_list, settings, player) + def start(cls, ctx, question_list, settings, audio = None): + session = cls(ctx, question_list, settings, audio) loop = ctx.bot.loop session._task = loop.create_task(session.run()) return session @@ -34,57 +33,89 @@ class AudioSession(TriviaSession): await self._send_startup_msg() max_score = self.settings["max_score"] delay = self.settings["delay"] + audio_delay = self.settings["audio_delay"] timeout = self.settings["timeout"] - for question, answers in self._iter_questions(): + if self.audio is not None: + import lavalink + + player = lavalink.get_player(self.ctx.guild.id) + player.store("channel", self.ctx.channel.id) # What's this for? I dunno + await self.audio.set_player_settings(self.ctx) + else: + lavalink = None + player = False + + for question, answers, audio_url in self._iter_questions(): async with self.ctx.typing(): await asyncio.sleep(3) self.count += 1 - await self.player.stop() - - msg = bold(f"Question number {self.count}!") + "\n\nName this audio!" + msg = bold(f"Question number {self.count}!") + f"\n\n{question}" + if player: + await player.stop() + if audio_url: + if not player: + log.debug("Got an audio question in a non-audio trivia session") + continue + + load_result = await player.load_tracks(audio_url) + if ( + load_result.has_error + or load_result.load_type != lavalink.enums.LoadType.TRACK_LOADED + ): + await self.ctx.maybe_send_embed( + "Audio Track has an error, skipping. See logs for details" + ) + log.info(f"Track has error: {load_result.exception_message}") + continue + tracks = load_result.tracks + track = tracks[0] + seconds = track.length / 1000 + track.uri = "" # Hide the info from `now` + if self.settings["repeat"] and seconds < audio_delay: + # Append it until it's longer than the delay + tot_length = seconds + 0 + while tot_length < audio_delay: + player.add(self.ctx.author, track) + tot_length += seconds + else: + player.add(self.ctx.author, track) + + if not player.current: + await player.play() await self.ctx.maybe_send_embed(msg) log.debug(f"Audio question: {question}") - # print("Audio question: {}".format(question)) - - # await self.ctx.invoke(self.audio.play(ctx=self.ctx, query=question)) - # ctx_copy = copy(self.ctx) - - # await self.ctx.invoke(self.player.play, query=question) - query = question.strip("<>") - load_result = await self.player.load_tracks(query) - log.debug(f"{load_result.load_type=}") - if load_result.has_error or load_result.load_type != LoadType.TRACK_LOADED: - await self.ctx.maybe_send_embed(f"Track has error, skipping. See logs for details") - log.info(f"Track has error: {load_result.exception_message}") - continue # Skip tracks with error - tracks = load_result.tracks - - track = tracks[0] - seconds = track.length / 1000 - - if self.settings["repeat"] and seconds < delay: - # Append it until it's longer than the delay - tot_length = seconds + 0 - while tot_length < delay: - self.player.add(self.ctx.author, track) - tot_length += seconds - else: - self.player.add(self.ctx.author, track) - - if not self.player.current: - log.debug("Pressing play") - await self.player.play() - continue_ = await self.wait_for_answer(answers, delay, timeout) + continue_ = await self.wait_for_answer( + answers, audio_delay if audio_url else delay, timeout + ) if continue_ is False: break if any(score >= max_score for score in self.scores.values()): await self.end_game() break else: - await self.ctx.send("There are no more questions!") + await self.ctx.maybe_send_embed("There are no more questions!") await self.end_game() async def end_game(self): await super().end_game() - await self.player.disconnect() + if self.audio is not None: + await self.ctx.invoke(self.audio.command_disconnect) + + def _iter_questions(self): + """Iterate over questions and answers for this session. + + Yields + ------ + `tuple` + A tuple containing the question (`str`) and the answers (`tuple` of + `str`). + + """ + for question, q_data in self.question_list: + answers = _parse_answers(q_data["answers"]) + _audio = q_data["audio"] + if _audio: + yield _audio, answers, question.strip("<>") + else: + yield question, answers, _audio \ No newline at end of file diff --git a/audiotrivia/audiotrivia.py b/audiotrivia/audiotrivia.py index c3ac1e9..9617f32 100644 --- a/audiotrivia/audiotrivia.py +++ b/audiotrivia/audiotrivia.py @@ -1,17 +1,17 @@ import datetime import logging import pathlib -from typing import List +from typing import List, Optional +import discord import lavalink import yaml from redbot.cogs.audio import Audio -from redbot.cogs.trivia import LOG -from redbot.cogs.trivia.trivia import InvalidListError, Trivia +from redbot.cogs.trivia.trivia import InvalidListError, Trivia, get_core_lists from redbot.core import Config, checks, commands from redbot.core.bot import Red from redbot.core.data_manager import cog_data_path -from redbot.core.utils.chat_formatting import box +from redbot.core.utils.chat_formatting import bold, box from .audiosession import AudioSession @@ -28,12 +28,11 @@ class AudioTrivia(Trivia): def __init__(self, bot: Red): super().__init__() self.bot = bot - self.audio = None self.audioconf = Config.get_conf( self, identifier=651171001051118411410511810597, force_registration=True ) - self.audioconf.register_guild(delay=30.0, repeat=True) + self.audioconf.register_guild(audio_delay=30.0, repeat=True) @commands.group() @commands.guild_only() @@ -44,49 +43,42 @@ class AudioTrivia(Trivia): settings_dict = await audioset.all() msg = box( "**Audio settings**\n" - "Answer time limit: {delay} seconds\n" + "Answer time limit: {audio_delay} seconds\n" "Repeat Short Audio: {repeat}" "".format(**settings_dict), lang="py", ) await ctx.send(msg) - @atriviaset.command(name="delay") - async def atriviaset_delay(self, ctx: commands.Context, seconds: float): + @atriviaset.command(name="timelimit") + async def atriviaset_timelimit(self, ctx: commands.Context, seconds: float): """Set the maximum seconds permitted to answer a question.""" if seconds < 4.0: await ctx.send("Must be at least 4 seconds.") return settings = self.audioconf.guild(ctx.guild) - await settings.delay.set(seconds) - await ctx.send("Done. Maximum seconds to answer set to {}.".format(seconds)) + await settings.audo_delay.set(seconds) + await ctx.maybe_send_embed(f"Done. Maximum seconds to answer set to {seconds}.") @atriviaset.command(name="repeat") async def atriviaset_repeat(self, ctx: commands.Context, true_or_false: bool): """Set whether or not short audio will be repeated""" settings = self.audioconf.guild(ctx.guild) await settings.repeat.set(true_or_false) - await ctx.send("Done. Repeating short audio is now set to {}.".format(true_or_false)) + await ctx.maybe_send_embed(f"Done. Repeating short audio is now set to {true_or_false}.") @commands.group(invoke_without_command=True) @commands.guild_only() async def audiotrivia(self, ctx: commands.Context, *categories: str): - """Start trivia session on the specified category. + """Start trivia session on the specified category or categories. + Includes Audio categories. You may list multiple categories, in which case the trivia will involve questions from all of them. """ if not categories and ctx.invoked_subcommand is None: await ctx.send_help() return - - if self.audio is None: - self.audio: Audio = self.bot.get_cog("Audio") - - if self.audio is None: - await ctx.maybe_send_embed("Audio is not loaded. Load it and try again") - return - categories = [c.lower() for c in categories] session = self._get_trivia_session(ctx.channel) if session is not None: @@ -94,45 +86,9 @@ class AudioTrivia(Trivia): "There is already an ongoing trivia session in this channel." ) return - status = await self.audio.config.status() - notify = await self.audio.config.guild(ctx.guild).notify() - - if status: - await ctx.maybe_send_embed( - f"It is recommended to disable audio status with `{ctx.prefix}audioset status`" - ) - - if notify: - await ctx.maybe_send_embed( - f"It is recommended to disable audio notify with `{ctx.prefix}audioset notify`" - ) - - if not self.audio._player_check(ctx): - try: - if not ctx.author.voice.channel.permissions_for( - ctx.me - ).connect or self.audio.is_vc_full(ctx.author.voice.channel): - return await ctx.maybe_send_embed( - "I don't have permission to connect to your channel." - ) - await lavalink.connect(ctx.author.voice.channel) - lavaplayer = lavalink.get_player(ctx.guild.id) - lavaplayer.store("connect", datetime.datetime.utcnow()) - except AttributeError: - return await ctx.maybe_send_embed("Connect to a voice channel first.") - - lavaplayer = lavalink.get_player(ctx.guild.id) - lavaplayer.store("channel", ctx.channel.id) # What's this for? I dunno - - await self.audio.set_player_settings(ctx) - - if not ctx.author.voice or ctx.author.voice.channel != lavaplayer.channel: - return await ctx.maybe_send_embed( - "You must be in the voice channel to use the audiotrivia command." - ) - trivia_dict = {} authors = [] + any_audio = False for category in reversed(categories): # We reverse the categories so that the first list's config takes # priority over the others. @@ -140,19 +96,22 @@ class AudioTrivia(Trivia): dict_ = self.get_audio_list(category) except FileNotFoundError: await ctx.maybe_send_embed( - "Invalid category `{0}`. See `{1}audiotrivia list`" + f"Invalid category `{category}`. See `{ctx.prefix}audiotrivia list`" " for a list of trivia categories." - "".format(category, ctx.prefix) ) except InvalidListError: await ctx.maybe_send_embed( "There was an error parsing the trivia list for" - " the `{}` category. It may be formatted" - " incorrectly.".format(category) + f" the `{category}` category. It may be formatted" + " incorrectly." ) else: - trivia_dict.update(dict_) - authors.append(trivia_dict.pop("AUTHOR", None)) + is_audio = dict_.pop("AUDIO", False) + authors.append(dict_.pop("AUTHOR", None)) + trivia_dict.update( + {_q: {"audio": is_audio, "answers": _a} for _q, _a in dict_.items()} + ) + any_audio = any_audio or is_audio continue return if not trivia_dict: @@ -161,9 +120,35 @@ class AudioTrivia(Trivia): ) return + if not any_audio: + audio = None + else: + audio: Optional["Audio"] = self.bot.get_cog("Audio") + if audio is None: + await ctx.send("Audio lists were parsed but Audio is not loaded!") + return + status = await audio.config.status() + notify = await audio.config.guild(ctx.guild).notify() + + if status: + await ctx.maybe_send_embed( + f"It is recommended to disable audio status with `{ctx.prefix}audioset status`" + ) + + if notify: + await ctx.maybe_send_embed( + f"It is recommended to disable audio notify with `{ctx.prefix}audioset notify`" + ) + + failed = await ctx.invoke(audio.command_summon) + if failed: + return + lavaplayer = lavalink.get_player(ctx.guild.id) + lavaplayer.store("channel", ctx.channel.id) # What's this for? I dunno + settings = await self.config.guild(ctx.guild).all() audiosettings = await self.audioconf.guild(ctx.guild).all() - config = trivia_dict.pop("CONFIG", None) + config = trivia_dict.pop("CONFIG", {"answer": None})["answer"] if config and settings["allow_override"]: settings.update(config) settings["lists"] = dict(zip(categories, reversed(authors))) @@ -171,25 +156,33 @@ class AudioTrivia(Trivia): # Delay in audiosettings overwrites delay in settings combined_settings = {**settings, **audiosettings} session = AudioSession.start( - ctx=ctx, - question_list=trivia_dict, - settings=combined_settings, - player=lavaplayer, + ctx, + trivia_dict, + combined_settings, + audio, ) self.trivia_sessions.append(session) - LOG.debug("New audio trivia session; #%s in %d", ctx.channel, ctx.guild.id) + log.debug("New audio trivia session; #%s in %d", ctx.channel, ctx.guild.id) @audiotrivia.command(name="list") @commands.guild_only() async def audiotrivia_list(self, ctx: commands.Context): - """List available trivia categories.""" - lists = set(p.stem for p in self._audio_lists()) - - msg = box("**Available trivia lists**\n\n{}".format(", ".join(sorted(lists)))) - if len(msg) > 1000: - await ctx.author.send(msg) - return - await ctx.send(msg) + """List available trivia including audio categories.""" + lists = set(p.stem for p in self._all_audio_lists()) + if await ctx.embed_requested(): + await ctx.send( + embed=discord.Embed( + title="Available trivia lists", + colour=await ctx.embed_colour(), + description=", ".join(sorted(lists)), + ) + ) + else: + msg = box(bold("Available trivia lists") + "\n\n" + ", ".join(sorted(lists))) + if len(msg) > 1000: + await ctx.author.send(msg) + else: + await ctx.send(msg) def get_audio_list(self, category: str) -> dict: """Get the audiotrivia list corresponding to the given category. @@ -206,7 +199,7 @@ class AudioTrivia(Trivia): """ try: - path = next(p for p in self._audio_lists() if p.stem == category) + path = next(p for p in self._all_audio_lists() if p.stem == category) except StopIteration: raise FileNotFoundError("Could not find the `{}` category.".format(category)) @@ -218,13 +211,15 @@ class AudioTrivia(Trivia): else: return dict_ - def _audio_lists(self) -> List[pathlib.Path]: + def _all_audio_lists(self) -> List[pathlib.Path]: + # Custom trivia lists uploaded with audiotrivia. Not necessarily audio lists personal_lists = [p.resolve() for p in cog_data_path(self).glob("*.yaml")] - return personal_lists + get_core_lists() + # Add to that custom lists uploaded with trivia and core lists + return personal_lists + get_core_audio_lists() + self._all_lists() -def get_core_lists() -> List[pathlib.Path]: +def get_core_audio_lists() -> List[pathlib.Path]: """Return a list of paths for all trivia lists packaged with the bot.""" core_lists_path = pathlib.Path(__file__).parent.resolve() / "data/lists" return list(core_lists_path.glob("*.yaml")) diff --git a/audiotrivia/data/lists/anime.yaml b/audiotrivia/data/lists/audioanime.yaml similarity index 99% rename from audiotrivia/data/lists/anime.yaml rename to audiotrivia/data/lists/audioanime.yaml index 7a27a0e..8aa518d 100644 --- a/audiotrivia/data/lists/anime.yaml +++ b/audiotrivia/data/lists/audioanime.yaml @@ -1,4 +1,5 @@ AUTHOR: Plab +AUDIO: "[Audio] Identify this Anime!" https://www.youtube.com/watch?v=2uq34TeWEdQ: - 'Hagane no Renkinjutsushi (2009)' - '(2009) الخيميائي المعدني الكامل' diff --git a/audiotrivia/data/lists/nhlgoalhorns.yaml b/audiotrivia/data/lists/audionhlgoalhorns.yaml similarity index 97% rename from audiotrivia/data/lists/nhlgoalhorns.yaml rename to audiotrivia/data/lists/audionhlgoalhorns.yaml index 689f478..9e86313 100644 --- a/audiotrivia/data/lists/nhlgoalhorns.yaml +++ b/audiotrivia/data/lists/audionhlgoalhorns.yaml @@ -1,4 +1,5 @@ AUTHOR: Lazar +AUDIO: "[Audio] Identify this NHL Team by their goal horn" https://youtu.be/6OejNXrGkK0: - Anaheim Ducks - Anaheim diff --git a/audiotrivia/data/lists/videogames.yaml b/audiotrivia/data/lists/audiovideogames.yaml similarity index 99% rename from audiotrivia/data/lists/videogames.yaml rename to audiotrivia/data/lists/audiovideogames.yaml index 5798f0c..d7f594c 100644 --- a/audiotrivia/data/lists/videogames.yaml +++ b/audiotrivia/data/lists/audiovideogames.yaml @@ -1,4 +1,5 @@ AUTHOR: Bobloy +AUDIO: "[Audio] Identify this video game" https://www.youtube.com/watch?v=GBPbJyxqHV0: - Super Mario 64 https://www.youtube.com/watch?v=0jXTBAGv9ZQ: From 7c95bd4c0fa9d73fa958e53bbff3428dbf453cb7 Mon Sep 17 00:00:00 2001 From: bobloy Date: Thu, 1 Oct 2020 09:10:26 -0400 Subject: [PATCH 59/73] Black formatting --- audiotrivia/audiosession.py | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/audiotrivia/audiosession.py b/audiotrivia/audiosession.py index 17fc998..1f3297b 100644 --- a/audiotrivia/audiosession.py +++ b/audiotrivia/audiosession.py @@ -12,13 +12,13 @@ log = logging.getLogger("red.fox_v3.audiotrivia.audiosession") class AudioSession(TriviaSession): """Class to run a session of audio trivia""" - def __init__(self, ctx, question_list: dict, settings: dict, audio = None): + def __init__(self, ctx, question_list: dict, settings: dict, audio=None): super().__init__(ctx, question_list, settings) self.audio = audio @classmethod - def start(cls, ctx, question_list, settings, audio = None): + def start(cls, ctx, question_list, settings, audio=None): session = cls(ctx, question_list, settings, audio) loop = ctx.bot.loop session._task = loop.create_task(session.run()) @@ -59,8 +59,8 @@ class AudioSession(TriviaSession): load_result = await player.load_tracks(audio_url) if ( - load_result.has_error - or load_result.load_type != lavalink.enums.LoadType.TRACK_LOADED + load_result.has_error + or load_result.load_type != lavalink.enums.LoadType.TRACK_LOADED ): await self.ctx.maybe_send_embed( "Audio Track has an error, skipping. See logs for details" @@ -118,4 +118,4 @@ class AudioSession(TriviaSession): if _audio: yield _audio, answers, question.strip("<>") else: - yield question, answers, _audio \ No newline at end of file + yield question, answers, _audio From 9440f34669f3be869bd5b9d896e6c16a26984658 Mon Sep 17 00:00:00 2001 From: bobloy Date: Thu, 1 Oct 2020 09:14:46 -0400 Subject: [PATCH 60/73] lovecalculator hotfix ssl error --- lovecalculator/lovecalculator.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lovecalculator/lovecalculator.py b/lovecalculator/lovecalculator.py index 95e9f97..94b6d49 100644 --- a/lovecalculator/lovecalculator.py +++ b/lovecalculator/lovecalculator.py @@ -33,7 +33,7 @@ class LoveCalculator(Cog): x.replace(" ", "+"), y.replace(" ", "+") ) async with aiohttp.ClientSession(headers={"Connection": "keep-alive"}) as session: - async with session.get(url) as response: + async with session.get(url, ssl=False) as response: assert response.status == 200 resp = await response.text() From 479b23f0f33ff960f985e5c38f840472d782df90 Mon Sep 17 00:00:00 2001 From: bobloy Date: Thu, 1 Oct 2020 09:24:26 -0400 Subject: [PATCH 61/73] Get love image right (when cert is fixed) --- lovecalculator/lovecalculator.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/lovecalculator/lovecalculator.py b/lovecalculator/lovecalculator.py index 94b6d49..fba6500 100644 --- a/lovecalculator/lovecalculator.py +++ b/lovecalculator/lovecalculator.py @@ -60,14 +60,14 @@ class LoveCalculator(Cog): else: emoji = "💔" title = f"Dr. Love says that the love percentage for {x} and {y} is: {emoji} {description} {emoji}" - except: + except (TypeError, ValueError): title = "Dr. Love has left a note for you." em = discord.Embed( title=title, description=result_text, color=discord.Color.red(), - url=f"https://www.lovecalculator.com/{result_image}", + url=url ) - + em.set_image(url=f"https://www.lovecalculator.com/{result_image}") await ctx.send(embed=em) From 5752ba605637da4005763e0f55aeab4f229d29af Mon Sep 17 00:00:00 2001 From: bobloy Date: Thu, 1 Oct 2020 09:25:16 -0400 Subject: [PATCH 62/73] Black formatting --- lovecalculator/lovecalculator.py | 5 +---- 1 file changed, 1 insertion(+), 4 deletions(-) diff --git a/lovecalculator/lovecalculator.py b/lovecalculator/lovecalculator.py index fba6500..20504bd 100644 --- a/lovecalculator/lovecalculator.py +++ b/lovecalculator/lovecalculator.py @@ -64,10 +64,7 @@ class LoveCalculator(Cog): title = "Dr. Love has left a note for you." em = discord.Embed( - title=title, - description=result_text, - color=discord.Color.red(), - url=url + title=title, description=result_text, color=discord.Color.red(), url=url ) em.set_image(url=f"https://www.lovecalculator.com/{result_image}") await ctx.send(embed=em) From 815cfcb0315bb88a099c566d7c5faa6aecc11c71 Mon Sep 17 00:00:00 2001 From: bobloy Date: Fri, 2 Oct 2020 12:01:17 -0400 Subject: [PATCH 63/73] Add member role WIP --- timerole/timerole.py | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/timerole/timerole.py b/timerole/timerole.py index 7484267..45d147a 100644 --- a/timerole/timerole.py +++ b/timerole/timerole.py @@ -27,10 +27,15 @@ class Timerole(Cog): self.bot = bot self.config = Config.get_conf(self, identifier=9811198108111121, force_registration=True) default_global = {} - default_guild = {"announce": None, "roles": {}} + default_guild = {"announce": None} + default_memberrole = {"had_role": False, "check_again_time": None} self.config.register_global(**default_global) self.config.register_guild(**default_guild) + + self.config.init_custom("MemberRole", 2) + self.config.register_custom("MemberRole", **default_memberrole) + self.updating = asyncio.create_task(self.check_hour()) async def red_delete_data_for_user(self, **kwargs): From 44035b78f7b5e4cb6f31518173af50e33c85e969 Mon Sep 17 00:00:00 2001 From: bobloy Date: Fri, 2 Oct 2020 15:13:08 -0400 Subject: [PATCH 64/73] Timerole rewrite --- timerole/timerole.py | 236 ++++++++++++++++++++++++++++--------------- 1 file changed, 156 insertions(+), 80 deletions(-) diff --git a/timerole/timerole.py b/timerole/timerole.py index 45d147a..ac23bd7 100644 --- a/timerole/timerole.py +++ b/timerole/timerole.py @@ -19,6 +19,15 @@ async def sleep_till_next_hour(): await asyncio.sleep((next_hour - datetime.utcnow()).seconds) +async def announce_to_channel(channel, remove_results, title): + if channel is not None and remove_results: + await channel.send(title) + for page in pagify(remove_results, shorten_by=50): + await channel.send(page) + elif remove_results: # Channel is None, log the results + log.info(remove_results) + + class Timerole(Cog): """Add roles to users based on time on server""" @@ -27,7 +36,7 @@ class Timerole(Cog): self.bot = bot self.config = Config.get_conf(self, identifier=9811198108111121, force_registration=True) default_global = {} - default_guild = {"announce": None} + default_guild = {"announce": None, "reapply": True, "roles": {}} default_memberrole = {"had_role": False, "check_again_time": None} self.config.register_global(**default_global) @@ -89,9 +98,7 @@ class Timerole(Cog): await self.config.guild(guild).roles.set_raw(role.id, value=to_set) await ctx.maybe_send_embed( - "Time Role for {0} set to {1} days and {2} hours until added".format( - role.name, days, hours - ) + f"Time Role for {role.name} set to {days} days and {hours} hours until added" ) @timerole.command() @@ -119,9 +126,7 @@ class Timerole(Cog): await self.config.guild(guild).roles.set_raw(role.id, value=to_set) await ctx.maybe_send_embed( - "Time Role for {0} set to {1} days and {2} hours until removed".format( - role.name, days, hours - ) + f"Time Role for {role.name} set to {days} days and {hours} hours until removed" ) @timerole.command() @@ -130,7 +135,15 @@ class Timerole(Cog): guild = ctx.guild await self.config.guild(guild).announce.set(channel.id) - await ctx.send("Announce channel set to {0}".format(channel.mention)) + await ctx.send(f"Announce channel set to {channel.mention}") + + @timerole.command() + async def reapply(self, ctx: commands.Context): + """Toggle reapplying roles if the member loses it somehow. Defaults to True""" + guild = ctx.guild + current_setting = await self.config.guild(guild).reapply() + await self.config.guild(guild).reapply.set(not current_setting) + await ctx.send(f"Reapplying roles is now set to: {not current_setting}") @timerole.command() async def delrole(self, ctx: commands.Context, role: discord.Role): @@ -138,7 +151,7 @@ class Timerole(Cog): guild = ctx.guild await self.config.guild(guild).roles.set_raw(role.id, value=None) - await ctx.send("{0} will no longer be applied".format(role.name)) + await ctx.send(f"{role.name} will no longer be applied") @timerole.command() async def list(self, ctx: commands.Context): @@ -158,89 +171,152 @@ class Timerole(Cog): str(discord.utils.get(guild.roles, id=int(new_id))) for new_id in r_data["required"] ] - out += "{} | {} days | requires: {}\n".format(str(role), r_data["days"], r_roles) + out += f"{role} | {r_data['days']} days | requires: {r_roles}\n" await ctx.maybe_send_embed(out) async def timerole_update(self): - async for guild in AsyncIter(self.bot.guilds): - addlist = [] - removelist = [] + utcnow = datetime.utcnow() + all_guilds = await self.config.all_guilds() - role_dict = await self.config.guild(guild).roles() - if not any(role_data for role_data in role_dict.values()): # No roles + all_mrs = await self.config.custom("MemberRole").all() + + for guild in self.bot.guilds: + guild_id = str(guild.id) + if guild_id not in all_guilds: continue - async for member in AsyncIter(guild.members): - has_roles = [r.id for r in member.roles] - - add_roles = [ - int(rID) - for rID, r_data in role_dict.items() - if r_data is not None and not r_data["remove"] - ] - remove_roles = [ - int(rID) - for rID, r_data in role_dict.items() - if r_data is not None and r_data["remove"] - ] - - check_add_roles = set(add_roles) - set(has_roles) - check_remove_roles = set(remove_roles) & set(has_roles) - - await self.check_required_and_date( - addlist, check_add_roles, has_roles, member, role_dict - ) - await self.check_required_and_date( - removelist, check_remove_roles, has_roles, member, role_dict - ) + add_results = "" + remove_results = "" + reapply = all_guilds[guild_id]["reapply"] + role_dict = all_guilds[guild_id]["roles"] + if not any(role_data for role_data in role_dict.values()): # No roles + continue + + async for member in AsyncIter(guild.members, steps=100): + addlist = [] + removelist = [] + + for role_id, role_data in role_dict.items(): + mr_dict = all_mrs[str(member.id)][role_id] + + # Stop if they've had the role and reapplying is disabled + if not reapply and mr_dict["had_role"]: + continue + + # Stop if the check_again_time hasn't passed yet + if ( + mr_dict["check_again_time"] is not None + and datetime.fromisoformat(mr_dict["check_again_time"]) >= utcnow + ): + continue + member: discord.Member + has_roles = set(r.id for r in member.roles) + + # Stop if they currently have the role (and mark they had it) + if role_id in has_roles: + if not mr_dict["had_role"]: + await self.config.custom( + "MemberRole", member.id, role_id + ).had_role.set(True) + continue + + # Stop if they don't have all the required roles + if "required" in role_data and not set(role_data["required"]) & has_roles: + # Doesn't have required role + continue + + check_time = member.joined_at + timedelta( + days=role_data["days"], + hours=role_data.get("hours", 0), + ) + + # Check if enough time has passed to get the role and save the check_again_time + if check_time <= utcnow: + await self.config.custom( + "MemberRole", member.id, role_id + ).check_again_time.set(check_time.isoformat()) + continue + + if role_data["remove"]: + removelist.append(role_id) + else: + addlist.append(role_id) + + # Done iterating through roles, now add or remove the roles + addlist = [discord.utils.get(guild.roles, id=role_id) for role_id in addlist] + removelist = [discord.utils.get(guild.roles, id=role_id) for role_id in removelist] + + if addlist: + try: + await member.add_roles(*addlist, reason="Timerole", atomic=False) + except (discord.Forbidden, discord.NotFound) as e: + log.exception("Failed Adding Roles") + add_results += f"{member.display_name} : **(Failed Adding Roles)**\n" + else: + add_results += "\n".join( + f"{member.display_name} : {role.name}" for role in addlist + ) + + if removelist: + try: + await member.remove_roles(*removelist, reason="Timerole", atomic=False) + except (discord.Forbidden, discord.NotFound) as e: + log.exception("Failed Removing Roles") + remove_results += f"{member.display_name} : **(Failed Removing Roles)**\n" + else: + remove_results += "\n".join( + f"{member.display_name} : {role.name}" for role in removelist + ) + + # Done iterating through members, now maybe announce to the guild channel = await self.config.guild(guild).announce() if channel is not None: channel = guild.get_channel(channel) - title = "**These members have received the following roles**\n" - await self.announce_roles(title, addlist, channel, guild, to_add=True) + await announce_to_channel(channel, remove_results, title) title = "**These members have lost the following roles**\n" - await self.announce_roles(title, removelist, channel, guild, to_add=False) - - async def announce_roles(self, title, role_list, channel, guild, to_add: True): - results = "" - async for member, role_id in AsyncIter(role_list): - role = discord.utils.get(guild.roles, id=role_id) - try: - if to_add: - await member.add_roles(role, reason="Timerole") - else: - await member.remove_roles(role, reason="Timerole") - except (discord.Forbidden, discord.NotFound) as e: - results += "{} : {} **(Failed)**\n".format(member.display_name, role.name) - else: - results += "{} : {}\n".format(member.display_name, role.name) - if channel is not None and results: - await channel.send(title) - for page in pagify(results, shorten_by=50): - await channel.send(page) - elif results: # Channel is None, log the results - log.info(results) - - async def check_required_and_date(self, role_list, check_roles, has_roles, member, role_dict): - async for role_id in AsyncIter(check_roles): - # Check for required role - if "required" in role_dict[str(role_id)]: - if not set(role_dict[str(role_id)]["required"]) & set(has_roles): - # Doesn't have required role - continue - - if ( - member.joined_at - + timedelta( - days=role_dict[str(role_id)]["days"], - hours=role_dict[str(role_id)].get("hours", 0), - ) - <= datetime.today() - ): - # Qualifies - role_list.append((member, role_id)) + await announce_to_channel(channel, remove_results, title) + # End + + # async def announce_roles(self, title, role_list, channel, guild, to_add: True): + # results = "" + # async for member, role_id in AsyncIter(role_list): + # role = discord.utils.get(guild.roles, id=role_id) + # try: + # if to_add: + # await member.add_roles(role, reason="Timerole") + # else: + # await member.remove_roles(role, reason="Timerole") + # except (discord.Forbidden, discord.NotFound) as e: + # results += f"{member.display_name} : {role.name} **(Failed)**\n" + # else: + # results += f"{member.display_name} : {role.name}\n" + # if channel is not None and results: + # await channel.send(title) + # for page in pagify(results, shorten_by=50): + # await channel.send(page) + # elif results: # Channel is None, log the results + # log.info(results) + + # async def check_required_and_date(self, role_list, check_roles, has_roles, member, role_dict): + # async for role_id in AsyncIter(check_roles): + # # Check for required role + # if "required" in role_dict[str(role_id)]: + # if not set(role_dict[str(role_id)]["required"]) & set(has_roles): + # # Doesn't have required role + # continue + # + # if ( + # member.joined_at + # + timedelta( + # days=role_dict[str(role_id)]["days"], + # hours=role_dict[str(role_id)].get("hours", 0), + # ) + # <= datetime.utcnow() + # ): + # # Qualifies + # role_list.append((member, role_id)) async def check_hour(self): await sleep_till_next_hour() From b141accbd96ac861eec9eda4fcefa3e69a909b7c Mon Sep 17 00:00:00 2001 From: bobloy Date: Fri, 2 Oct 2020 15:32:55 -0400 Subject: [PATCH 65/73] Timerole rewrite WIP --- timerole/timerole.py | 14 ++++++++++---- 1 file changed, 10 insertions(+), 4 deletions(-) diff --git a/timerole/timerole.py b/timerole/timerole.py index ac23bd7..af6f4b4 100644 --- a/timerole/timerole.py +++ b/timerole/timerole.py @@ -178,11 +178,14 @@ class Timerole(Cog): utcnow = datetime.utcnow() all_guilds = await self.config.all_guilds() - all_mrs = await self.config.custom("MemberRole").all() + # all_mrs = await self.config.custom("MemberRole").all() + + log.debug(f"Begin timerole update") for guild in self.bot.guilds: - guild_id = str(guild.id) + guild_id = guild.id if guild_id not in all_guilds: + log.debug(f"Guild has no configured settings: {guild}") continue add_results = "" @@ -191,6 +194,7 @@ class Timerole(Cog): role_dict = all_guilds[guild_id]["roles"] if not any(role_data for role_data in role_dict.values()): # No roles + log.debug(f"No roles are configured for guild: {guild}") continue async for member in AsyncIter(guild.members, steps=100): @@ -198,7 +202,7 @@ class Timerole(Cog): removelist = [] for role_id, role_data in role_dict.items(): - mr_dict = all_mrs[str(member.id)][role_id] + mr_dict = await self.config.custom("MemberRole", member.id, role_id).all() # Stop if they've had the role and reapplying is disabled if not reapply and mr_dict["had_role"]: @@ -222,7 +226,9 @@ class Timerole(Cog): continue # Stop if they don't have all the required roles - if "required" in role_data and not set(role_data["required"]) & has_roles: + if role_data is None or ( + "required" in role_data and not set(role_data["required"]) & has_roles + ): # Doesn't have required role continue From c63a4923e7c947a795aac332838b31001ee2422c Mon Sep 17 00:00:00 2001 From: bobloy Date: Mon, 5 Oct 2020 11:21:57 -0400 Subject: [PATCH 66/73] Actually do the logic right --- timerole/timerole.py | 61 ++++++++++++++++++++++++++++++++------------ 1 file changed, 45 insertions(+), 16 deletions(-) diff --git a/timerole/timerole.py b/timerole/timerole.py index af6f4b4..c079596 100644 --- a/timerole/timerole.py +++ b/timerole/timerole.py @@ -19,13 +19,13 @@ async def sleep_till_next_hour(): await asyncio.sleep((next_hour - datetime.utcnow()).seconds) -async def announce_to_channel(channel, remove_results, title): - if channel is not None and remove_results: +async def announce_to_channel(channel, results, title): + if channel is not None and results: await channel.send(title) - for page in pagify(remove_results, shorten_by=50): + for page in pagify(results, shorten_by=50): await channel.send(page) - elif remove_results: # Channel is None, log the results - log.info(remove_results) + elif results: # Channel is None, log the results + log.info(results) class Timerole(Cog): @@ -197,6 +197,9 @@ class Timerole(Cog): log.debug(f"No roles are configured for guild: {guild}") continue + # all_mr = await self.config.all_custom("MemberRole") + # log.debug(f"{all_mr=}") + async for member in AsyncIter(guild.members, steps=100): addlist = [] removelist = [] @@ -206,6 +209,7 @@ class Timerole(Cog): # Stop if they've had the role and reapplying is disabled if not reapply and mr_dict["had_role"]: + log.debug(f"{member.display_name} - Not reapplying") continue # Stop if the check_again_time hasn't passed yet @@ -213,6 +217,7 @@ class Timerole(Cog): mr_dict["check_again_time"] is not None and datetime.fromisoformat(mr_dict["check_again_time"]) >= utcnow ): + log.debug(f"{member.display_name} - Not time to check again yet") continue member: discord.Member has_roles = set(r.id for r in member.roles) @@ -223,6 +228,9 @@ class Timerole(Cog): await self.config.custom( "MemberRole", member.id, role_id ).had_role.set(True) + log.debug( + f"{member.display_name} - Already has the role, maybe applying `had_role`" + ) continue # Stop if they don't have all the required roles @@ -230,6 +238,7 @@ class Timerole(Cog): "required" in role_data and not set(role_data["required"]) & has_roles ): # Doesn't have required role + # log.debug(f"{member.display_name} - Missing required roles") continue check_time = member.joined_at + timedelta( @@ -238,10 +247,14 @@ class Timerole(Cog): ) # Check if enough time has passed to get the role and save the check_again_time - if check_time <= utcnow: + if check_time >= utcnow: await self.config.custom( "MemberRole", member.id, role_id ).check_again_time.set(check_time.isoformat()) + log.debug( + f"{member.display_name} - Not enough time has passed to qualify for the role\n" + f"Waiting until {check_time}" + ) continue if role_data["remove"]: @@ -250,39 +263,55 @@ class Timerole(Cog): addlist.append(role_id) # Done iterating through roles, now add or remove the roles - addlist = [discord.utils.get(guild.roles, id=role_id) for role_id in addlist] - removelist = [discord.utils.get(guild.roles, id=role_id) for role_id in removelist] + if not addlist and not removelist: + continue + + log.debug(f"{addlist=}\n{removelist=}") + add_roles = [ + discord.utils.get(guild.roles, id=int(role_id)) for role_id in addlist + ] + remove_roles = [ + discord.utils.get(guild.roles, id=int(role_id)) for role_id in removelist + ] + + if None in add_roles or None in remove_roles: + log.info( + f"Timerole ran into an error with the roles in: {add_roles + remove_roles}" + ) if addlist: try: - await member.add_roles(*addlist, reason="Timerole", atomic=False) + await member.add_roles(*add_roles, reason="Timerole", atomic=False) except (discord.Forbidden, discord.NotFound) as e: log.exception("Failed Adding Roles") add_results += f"{member.display_name} : **(Failed Adding Roles)**\n" else: add_results += "\n".join( - f"{member.display_name} : {role.name}" for role in addlist + f"{member.display_name} : {role.name}" for role in add_roles ) if removelist: try: - await member.remove_roles(*removelist, reason="Timerole", atomic=False) + await member.remove_roles(*remove_roles, reason="Timerole", atomic=False) except (discord.Forbidden, discord.NotFound) as e: log.exception("Failed Removing Roles") remove_results += f"{member.display_name} : **(Failed Removing Roles)**\n" else: remove_results += "\n".join( - f"{member.display_name} : {role.name}" for role in removelist + f"{member.display_name} : {role.name}" for role in remove_roles ) # Done iterating through members, now maybe announce to the guild channel = await self.config.guild(guild).announce() if channel is not None: channel = guild.get_channel(channel) - title = "**These members have received the following roles**\n" - await announce_to_channel(channel, remove_results, title) - title = "**These members have lost the following roles**\n" - await announce_to_channel(channel, remove_results, title) + + if add_results: + title = "**These members have received the following roles**\n" + await announce_to_channel(channel, add_results, title) + if remove_results: + title = "**These members have lost the following roles**\n" + await announce_to_channel(channel, remove_results, title) # End # async def announce_roles(self, title, role_list, channel, guild, to_add: True): From 10767da507f02c0b273e60d9ce10606a3001837a Mon Sep 17 00:00:00 2001 From: bobloy Date: Mon, 5 Oct 2020 13:00:58 -0400 Subject: [PATCH 67/73] Clear the reapply logic if the role is deleted --- timerole/timerole.py | 69 +++++++++++++++++++++++++++----------------- 1 file changed, 43 insertions(+), 26 deletions(-) diff --git a/timerole/timerole.py b/timerole/timerole.py index c079596..fdfe584 100644 --- a/timerole/timerole.py +++ b/timerole/timerole.py @@ -1,6 +1,7 @@ import asyncio import logging from datetime import datetime, timedelta +from typing import Optional import discord from redbot.core import Config, checks, commands @@ -37,13 +38,13 @@ class Timerole(Cog): self.config = Config.get_conf(self, identifier=9811198108111121, force_registration=True) default_global = {} default_guild = {"announce": None, "reapply": True, "roles": {}} - default_memberrole = {"had_role": False, "check_again_time": None} + default_rolemember = {"had_role": False, "check_again_time": None} self.config.register_global(**default_global) self.config.register_guild(**default_guild) - self.config.init_custom("MemberRole", 2) - self.config.register_custom("MemberRole", **default_memberrole) + self.config.init_custom("RoleMember", 2) + self.config.register_custom("RoleMember", **default_rolemember) self.updating = asyncio.create_task(self.check_hour()) @@ -63,10 +64,12 @@ class Timerole(Cog): Useful for troubleshooting the initial setup """ - + pre_run = datetime.utcnow() async with ctx.typing(): await self.timerole_update() await ctx.tick() + after_run = datetime.utcnow() + await ctx.maybe_send_embed(f"Took {after_run-pre_run} seconds") @commands.group() @checks.mod_or_permissions(administrator=True) @@ -130,12 +133,15 @@ class Timerole(Cog): ) @timerole.command() - async def channel(self, ctx: commands.Context, channel: discord.TextChannel): + async def channel(self, ctx: commands.Context, channel: Optional[discord.TextChannel] = None): """Sets the announce channel for role adds""" guild = ctx.guild - - await self.config.guild(guild).announce.set(channel.id) - await ctx.send(f"Announce channel set to {channel.mention}") + if channel is None: + await self.config.guild(guild).announce.clear() + await ctx.maybe_send_embed(f"Announce channel has been cleared") + else: + await self.config.guild(guild).announce.set(channel.id) + await ctx.send(f"Announce channel set to {channel.mention}") @timerole.command() async def reapply(self, ctx: commands.Context): @@ -143,7 +149,7 @@ class Timerole(Cog): guild = ctx.guild current_setting = await self.config.guild(guild).reapply() await self.config.guild(guild).reapply.set(not current_setting) - await ctx.send(f"Reapplying roles is now set to: {not current_setting}") + await ctx.maybe_send_embed(f"Reapplying roles is now set to: {not current_setting}") @timerole.command() async def delrole(self, ctx: commands.Context, role: discord.Role): @@ -151,7 +157,8 @@ class Timerole(Cog): guild = ctx.guild await self.config.guild(guild).roles.set_raw(role.id, value=None) - await ctx.send(f"{role.name} will no longer be applied") + await self.config.custom("RoleMember", role.id).clear() + await ctx.maybe_send_embed(f"{role.name} will no longer be applied") @timerole.command() async def list(self, ctx: commands.Context): @@ -178,7 +185,7 @@ class Timerole(Cog): utcnow = datetime.utcnow() all_guilds = await self.config.all_guilds() - # all_mrs = await self.config.custom("MemberRole").all() + # all_mrs = await self.config.custom("RoleMember").all() log.debug(f"Begin timerole update") @@ -197,15 +204,19 @@ class Timerole(Cog): log.debug(f"No roles are configured for guild: {guild}") continue - # all_mr = await self.config.all_custom("MemberRole") + # all_mr = await self.config.all_custom("RoleMember") # log.debug(f"{all_mr=}") - async for member in AsyncIter(guild.members, steps=100): + async for member in AsyncIter(guild.members, steps=10): addlist = [] removelist = [] for role_id, role_data in role_dict.items(): - mr_dict = await self.config.custom("MemberRole", member.id, role_id).all() + # Skip non-configured roles + if not role_data: + continue + + mr_dict = await self.config.custom("RoleMember", role_id, member.id).all() # Stop if they've had the role and reapplying is disabled if not reapply and mr_dict["had_role"]: @@ -222,23 +233,21 @@ class Timerole(Cog): member: discord.Member has_roles = set(r.id for r in member.roles) - # Stop if they currently have the role (and mark they had it) - if role_id in has_roles: + # Stop if they currently have or don't have the role, and mark had_role + if (role_id in has_roles and not role_data["remove"]) or ( + role_id not in has_roles and role_data["remove"] + ): if not mr_dict["had_role"]: await self.config.custom( - "MemberRole", member.id, role_id + "RoleMember", role_id, member.id ).had_role.set(True) - log.debug( - f"{member.display_name} - Already has the role, maybe applying `had_role`" - ) + log.debug(f"{member.display_name} - applying had_role") continue # Stop if they don't have all the required roles if role_data is None or ( "required" in role_data and not set(role_data["required"]) & has_roles ): - # Doesn't have required role - # log.debug(f"{member.display_name} - Missing required roles") continue check_time = member.joined_at + timedelta( @@ -249,7 +258,7 @@ class Timerole(Cog): # Check if enough time has passed to get the role and save the check_again_time if check_time >= utcnow: await self.config.custom( - "MemberRole", member.id, role_id + "RoleMember", role_id, member.id ).check_again_time.set(check_time.isoformat()) log.debug( f"{member.display_name} - Not enough time has passed to qualify for the role\n" @@ -266,7 +275,7 @@ class Timerole(Cog): if not addlist and not removelist: continue - log.debug(f"{addlist=}\n{removelist=}") + # log.debug(f"{addlist=}\n{removelist=}") add_roles = [ discord.utils.get(guild.roles, id=int(role_id)) for role_id in addlist ] @@ -286,9 +295,13 @@ class Timerole(Cog): log.exception("Failed Adding Roles") add_results += f"{member.display_name} : **(Failed Adding Roles)**\n" else: - add_results += "\n".join( + add_results += " \n".join( f"{member.display_name} : {role.name}" for role in add_roles ) + for role_id in addlist: + await self.config.custom( + "RoleMember", role_id, member.id + ).had_role.set(True) if removelist: try: @@ -297,9 +310,13 @@ class Timerole(Cog): log.exception("Failed Removing Roles") remove_results += f"{member.display_name} : **(Failed Removing Roles)**\n" else: - remove_results += "\n".join( + remove_results += " \n".join( f"{member.display_name} : {role.name}" for role in remove_roles ) + for role_id in removelist: + await self.config.custom( + "RoleMember", role_id, member.id + ).had_role.set(True) # Done iterating through members, now maybe announce to the guild channel = await self.config.guild(guild).announce() From 4c1cd869303e1407e8d651f11aeffade3b680c52 Mon Sep 17 00:00:00 2001 From: bobloy Date: Mon, 5 Oct 2020 13:06:53 -0400 Subject: [PATCH 68/73] Better time keepking doesn't include tick --- timerole/timerole.py | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/timerole/timerole.py b/timerole/timerole.py index fdfe584..1b56b69 100644 --- a/timerole/timerole.py +++ b/timerole/timerole.py @@ -64,11 +64,12 @@ class Timerole(Cog): Useful for troubleshooting the initial setup """ - pre_run = datetime.utcnow() async with ctx.typing(): + pre_run = datetime.utcnow() await self.timerole_update() + after_run = datetime.utcnow() await ctx.tick() - after_run = datetime.utcnow() + await ctx.maybe_send_embed(f"Took {after_run-pre_run} seconds") @commands.group() @@ -187,7 +188,7 @@ class Timerole(Cog): # all_mrs = await self.config.custom("RoleMember").all() - log.debug(f"Begin timerole update") + # log.debug(f"Begin timerole update") for guild in self.bot.guilds: guild_id = guild.id From a92c373b497402cff00265e98195eacbc3d3f83f Mon Sep 17 00:00:00 2001 From: bobloy Date: Wed, 7 Oct 2020 15:13:53 -0400 Subject: [PATCH 69/73] New gitignore --- .gitignore | 1 + 1 file changed, 1 insertion(+) diff --git a/.gitignore b/.gitignore index 7a224ea..3732d09 100644 --- a/.gitignore +++ b/.gitignore @@ -4,3 +4,4 @@ venv/ v-data/ database.sqlite3 /venv3.4/ +/.venv/ From 1e8d1efb57ffcf764cbfb63b583d760e41b22ec3 Mon Sep 17 00:00:00 2001 From: bobloy Date: Fri, 9 Oct 2020 13:54:05 -0400 Subject: [PATCH 70/73] Respect embed color --- lseen/lseen.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lseen/lseen.py b/lseen/lseen.py index 3348b65..0aff69c 100644 --- a/lseen/lseen.py +++ b/lseen/lseen.py @@ -83,7 +83,7 @@ class LastSeen(Cog): # description="{} was last seen at this date and time".format(member.display_name), # timestamp=last_seen) - embed = discord.Embed(timestamp=last_seen) + embed = discord.Embed(timestamp=last_seen, color=self.bot.get_embed_color(ctx)) await ctx.send(embed=embed) @commands.Cog.listener() From 5611f7abe7e58f6998011cb52455affae543f2aa Mon Sep 17 00:00:00 2001 From: bobloy Date: Fri, 9 Oct 2020 13:55:14 -0400 Subject: [PATCH 71/73] Don't commit before testing --- lseen/lseen.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lseen/lseen.py b/lseen/lseen.py index 0aff69c..69bcf87 100644 --- a/lseen/lseen.py +++ b/lseen/lseen.py @@ -83,7 +83,7 @@ class LastSeen(Cog): # description="{} was last seen at this date and time".format(member.display_name), # timestamp=last_seen) - embed = discord.Embed(timestamp=last_seen, color=self.bot.get_embed_color(ctx)) + embed = discord.Embed(timestamp=last_seen, color=await self.bot.get_embed_color(ctx)) await ctx.send(embed=embed) @commands.Cog.listener() From d71e3afb86573fecd7463787a439d3d77a816ba6 Mon Sep 17 00:00:00 2001 From: bobloy Date: Sun, 11 Oct 2020 14:01:12 -0400 Subject: [PATCH 72/73] Fix re-adding roles bug --- timerole/timerole.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/timerole/timerole.py b/timerole/timerole.py index 1b56b69..714bcc8 100644 --- a/timerole/timerole.py +++ b/timerole/timerole.py @@ -235,8 +235,8 @@ class Timerole(Cog): has_roles = set(r.id for r in member.roles) # Stop if they currently have or don't have the role, and mark had_role - if (role_id in has_roles and not role_data["remove"]) or ( - role_id not in has_roles and role_data["remove"] + if (int(role_id) in has_roles and not role_data["remove"]) or ( + int(role_id) not in has_roles and role_data["remove"] ): if not mr_dict["had_role"]: await self.config.custom( From 5ddafff59f62abb3211aaeec8dbd4b6da844b4de Mon Sep 17 00:00:00 2001 From: bobloy Date: Sat, 17 Oct 2020 12:24:43 -0400 Subject: [PATCH 73/73] Add ability to delete autobanked guildbanks --- stealemoji/stealemoji.py | 49 +++++++++++++++++++++++++++++++++++++++- 1 file changed, 48 insertions(+), 1 deletion(-) diff --git a/stealemoji/stealemoji.py b/stealemoji/stealemoji.py index 492ef70..a492527 100644 --- a/stealemoji/stealemoji.py +++ b/stealemoji/stealemoji.py @@ -50,6 +50,7 @@ class StealEmoji(Cog): default_global = { "stolemoji": {}, "guildbanks": [], + "autobanked_guilds": [], "on": False, "notify": 0, "autobank": False, @@ -145,11 +146,54 @@ class StealEmoji(Cog): await ctx.maybe_send_embed("AutoBanking is now " + str(not curr_setting)) + @checks.is_owner() + @commands.guild_only() + @stealemoji.command(name="deleteserver", aliases=["deleteguild"]) + async def se_deleteserver(self, ctx: commands.Context, guild_id=None): + """Delete servers the bot is the owner of. + + Useful for auto-generated guildbanks.""" + if guild_id is None: + guild = ctx.guild + else: + guild = await self.bot.get_guild(guild_id) + + if guild is None: + await ctx.maybe_send_embed("Failed to get guild, cancelling") + return + guild: discord.Guild + await ctx.maybe_send_embed( + f"Will attempt to delete {guild.name} ({guild.id})\n" f"Okay to continue? (yes/no)" + ) + + def check(m): + return m.author == ctx.author and m.channel == ctx.channel + + try: + answer = await self.bot.wait_for("message", timeout=120, check=check) + except asyncio.TimeoutError: + await ctx.send("Timed out, canceling") + return + + if answer.content.upper() not in ["Y", "YES"]: + await ctx.maybe_send_embed("Cancelling") + return + try: + await guild.delete() + except discord.Forbidden: + log.exception("No permission to delete. I'm probably not the guild owner") + await ctx.maybe_send_embed("No permission to delete. I'm probably not the guild owner") + except discord.HTTPException: + log.exception("Unexpected error when deleting guild") + await ctx.maybe_send_embed("Unexpected error when deleting guild") + else: + await self.bot.send_to_owners(f"Guild {guild.name} deleted") + @checks.is_owner() @commands.guild_only() @stealemoji.command(name="bank") async def se_bank(self, ctx): - """Add current server as emoji bank""" + """Add or remove current server as emoji bank""" def check(m): return ( @@ -235,6 +279,9 @@ class StealEmoji(Cog): return async with self.config.guildbanks() as guildbanks: guildbanks.append(guildbank.id) + # Track generated guilds for easier deletion + async with self.config.autobanked_guilds() as autobanked_guilds: + autobanked_guilds.append(guildbank.id) await asyncio.sleep(2)