diff options
40 files changed, 3342 insertions, 521 deletions
| diff --git a/.github/workflows/lint.yaml b/.github/workflows/lint.yaml index 81706e1e..756b3c16 100644 --- a/.github/workflows/lint.yaml +++ b/.github/workflows/lint.yaml @@ -74,12 +74,14 @@ jobs:            pip install poetry            poetry install --no-interaction --no-ansi -      # Check all the dependencies are compatible with the MIT license. +      # Check all of our dev dependencies are compatible with the MIT license.        # If you added a new dependencies that is being rejected,        # please make sure it is compatible with the license for this project,        # and add it to the ALLOWED_LICENSE variable        - name: Check Dependencies License -        run: pip-licenses --allow-only="$ALLOWED_LICENSES" +        run: | +          pip-licenses --allow-only="$ALLOWED_LICENSE" \ +            --package $(poetry export -f requirements.txt --without-hashes | sed "s/==.*//g" | tr "\n" " ")        # This step caches our pre-commit environment. To make sure we        # do create a new environment when our pre-commit setup changes, diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 7244cb4e..a2e9a398 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -12,6 +12,11 @@ repos:      rev: v1.5.1      hooks:        - id: python-check-blanket-noqa +  - repo: https://github.com/pycqa/isort +    rev: 5.8.0 +    hooks: +      - id: isort +        name: isort (python)    - repo: local      hooks:        - id: flake8 @@ -25,6 +25,3 @@ COPY . .  # Start the bot  CMD ["python", "-m", "bot"] - -# Define docker persistent volumes -VOLUME /bot/bot/log /bot/data diff --git a/bot/__init__.py b/bot/__init__.py index db576cb2..ae53a5a5 100644 --- a/bot/__init__.py +++ b/bot/__init__.py @@ -7,66 +7,36 @@ except ModuleNotFoundError:  import asyncio  import logging -import logging.handlers  import os  from functools import partial, partialmethod -from pathlib import Path  import arrow +import sentry_sdk  from discord.ext import commands +from sentry_sdk.integrations.logging import LoggingIntegration +from sentry_sdk.integrations.redis import RedisIntegration -from bot import monkey_patches -from bot.constants import Client +from bot import log, monkey_patches +sentry_logging = LoggingIntegration( +    level=logging.DEBUG, +    event_level=logging.WARNING +) -# Configure the "TRACE" logging level (e.g. "log.trace(message)") -logging.TRACE = 5 -logging.addLevelName(logging.TRACE, "TRACE") +sentry_sdk.init( +    dsn=os.environ.get("BOT_SENTRY_DSN"), +    integrations=[ +        sentry_logging, +        RedisIntegration() +    ], +    release=f"sir-lancebot@{os.environ.get('GIT_SHA', 'foobar')}" +) -logging.Logger.trace = monkey_patches.trace_log +log.setup()  # Set timestamp of when execution started (approximately)  start_time = arrow.utcnow() -# Set up file logging -log_dir = Path("bot/log") -log_file = log_dir / "hackbot.log" -os.makedirs(log_dir, exist_ok=True) - -# File handler rotates logs every 5 MB -file_handler = logging.handlers.RotatingFileHandler( -    log_file, maxBytes=5 * (2**20), backupCount=10, encoding="utf-8", -) -file_handler.setLevel(logging.TRACE if Client.debug else logging.DEBUG) - -# Console handler prints to terminal -console_handler = logging.StreamHandler() -level = logging.TRACE if Client.debug else logging.INFO -console_handler.setLevel(level) - -# Remove old loggers, if any -root = logging.getLogger() -if root.handlers: -    for handler in root.handlers: -        root.removeHandler(handler) - -# Silence irrelevant loggers -logging.getLogger("discord").setLevel(logging.ERROR) -logging.getLogger("websockets").setLevel(logging.ERROR) -logging.getLogger("PIL").setLevel(logging.ERROR) -logging.getLogger("matplotlib").setLevel(logging.ERROR) -logging.getLogger("async_rediscache").setLevel(logging.WARNING) - -# Setup new logging configuration -logging.basicConfig( -    format="%(asctime)s - %(name)s %(levelname)s: %(message)s", -    datefmt="%D %H:%M:%S", -    level=logging.TRACE if Client.debug else logging.DEBUG, -    handlers=[console_handler, file_handler], -) -logging.getLogger().info("Logging initialization complete") - -  # On Windows, the selector event loop is required for aiodns.  if os.name == "nt":      asyncio.set_event_loop_policy(asyncio.WindowsSelectorEventLoopPolicy()) diff --git a/bot/__main__.py b/bot/__main__.py index c6e5fa57..6889fe2b 100644 --- a/bot/__main__.py +++ b/bot/__main__.py @@ -1,28 +1,10 @@  import logging -import sentry_sdk -from sentry_sdk.integrations.logging import LoggingIntegration -from sentry_sdk.integrations.redis import RedisIntegration -  from bot.bot import bot -from bot.constants import Client, GIT_SHA, STAFF_ROLES, WHITELISTED_CHANNELS +from bot.constants import Client, STAFF_ROLES, WHITELISTED_CHANNELS  from bot.utils.decorators import whitelist_check  from bot.utils.extensions import walk_extensions -sentry_logging = LoggingIntegration( -    level=logging.DEBUG, -    event_level=logging.WARNING -) - -sentry_sdk.init( -    dsn=Client.sentry_dsn, -    integrations=[ -        sentry_logging, -        RedisIntegration() -    ], -    release=f"sir-lancebot@{GIT_SHA}" -) -  log = logging.getLogger(__name__)  bot.add_check(whitelist_check(channels=WHITELISTED_CHANNELS, roles=STAFF_ROLES)) diff --git a/bot/constants.py b/bot/constants.py index bef7f3d1..f4b1cab0 100644 --- a/bot/constants.py +++ b/bot/constants.py @@ -88,6 +88,7 @@ class AdventOfCode:      ignored_days = environ.get("AOC_IGNORED_DAYS", "").split(",")      leaderboard_displayed_members = 10      leaderboard_cache_expiry_seconds = 1800 +    max_day_and_star_results = 15      year = int(environ.get("AOC_YEAR", datetime.utcnow().year))      role_id = int(environ.get("AOC_ROLE_ID", 518565788744024082)) @@ -101,8 +102,8 @@ class Cats:  class Channels(NamedTuple): -    advent_of_code = int(environ.get("AOC_CHANNEL_ID", 782715290437943306)) -    advent_of_code_commands = int(environ.get("AOC_COMMANDS_CHANNEL_ID", 607247579608121354)) +    advent_of_code = int(environ.get("AOC_CHANNEL_ID", 897932085766004786)) +    advent_of_code_commands = int(environ.get("AOC_COMMANDS_CHANNEL_ID", 897932607545823342))      bot = 267659945086812160      organisation = 551789653284356126      devlog = int(environ.get("CHANNEL_DEVLOG", 622895325144940554)) @@ -133,11 +134,12 @@ class Client(NamedTuple):      guild = int(environ.get("BOT_GUILD", 267624335836053506))      prefix = environ.get("PREFIX", ".")      token = environ.get("BOT_TOKEN") -    sentry_dsn = environ.get("BOT_SENTRY_DSN")      debug = environ.get("BOT_DEBUG", "true").lower() == "true" +    file_logs = environ.get("FILE_LOGS", "false").lower() == "true"      github_bot_repo = "https://github.com/python-discord/sir-lancebot"      # Override seasonal locks: 1 (January) to 12 (December)      month_override = int(environ["MONTH_OVERRIDE"]) if "MONTH_OVERRIDE" in environ else None +    trace_loggers = environ.get("BOT_TRACE_LOGGERS")  class Colours: @@ -346,8 +348,6 @@ WHITELISTED_CHANNELS = (      Channels.voice_chat_1,  ) -GIT_SHA = environ.get("GIT_SHA", "foobar") -  # Bot replies  ERROR_REPLIES = [      "Please don't do that.", diff --git a/bot/exts/avatar_modification/avatar_modify.py b/bot/exts/avatar_modification/avatar_modify.py index 87eb05e6..3ee70cfd 100644 --- a/bot/exts/avatar_modification/avatar_modify.py +++ b/bot/exts/avatar_modification/avatar_modify.py @@ -239,7 +239,7 @@ class AvatarModify(commands.Cog):                  description=f"Here is your lovely avatar, surrounded by\n a beautiful {option} flag. Enjoy :D"              )              embed.set_image(url=f"attachment://{file_name}") -            embed.set_footer(text=f"Made by {ctx.author.display_name}.", icon_url=ctx.author.avatar.url) +            embed.set_footer(text=f"Made by {ctx.author.display_name}.", icon_url=ctx.author.display_avatar.url)              await ctx.send(file=file, embed=embed)      @avatar_modify.group( @@ -286,7 +286,7 @@ class AvatarModify(commands.Cog):      @avatar_modify.command(          aliases=("savatar", "spookify"),          root_aliases=("spookyavatar", "spookify", "savatar"), -        brief="Spookify an user's avatar." +        brief="Spookify a user's avatar."      )      async def spookyavatar(self, ctx: commands.Context) -> None:          """This "spookifies" the user's avatar, with a random *spooky* effect.""" diff --git a/bot/exts/core/extensions.py b/bot/exts/core/extensions.py index dbb9e069..d809d2b9 100644 --- a/bot/exts/core/extensions.py +++ b/bot/exts/core/extensions.py @@ -152,7 +152,7 @@ class Extensions(commands.Cog):          Grey indicates that the extension is unloaded.          Green indicates that the extension is currently loaded.          """ -        embed = Embed(colour=Colour.blurple()) +        embed = Embed(colour=Colour.og_blurple())          embed.set_author(              name="Extensions List",              url=Client.github_bot_repo, diff --git a/bot/exts/core/help.py b/bot/exts/core/help.py index 4b766b50..db3c2aa6 100644 --- a/bot/exts/core/help.py +++ b/bot/exts/core/help.py @@ -13,10 +13,7 @@ from rapidfuzz import process  from bot import constants  from bot.bot import Bot  from bot.constants import Emojis -from bot.utils.pagination import ( -    FIRST_EMOJI, LAST_EMOJI, -    LEFT_EMOJI, LinePaginator, RIGHT_EMOJI, -) +from bot.utils.pagination import FIRST_EMOJI, LAST_EMOJI, LEFT_EMOJI, LinePaginator, RIGHT_EMOJI  DELETE_EMOJI = Emojis.trashcan diff --git a/bot/exts/core/internal_eval/_internal_eval.py b/bot/exts/core/internal_eval/_internal_eval.py index 47e564a5..5b5461f0 100644 --- a/bot/exts/core/internal_eval/_internal_eval.py +++ b/bot/exts/core/internal_eval/_internal_eval.py @@ -10,6 +10,7 @@ from bot.bot import Bot  from bot.constants import Client, Roles  from bot.utils.decorators import with_role  from bot.utils.extensions import invoke_help_command +  from ._helpers import EvalContext  __all__ = ["InternalEval"] diff --git a/bot/exts/events/advent_of_code/_cog.py b/bot/exts/events/advent_of_code/_cog.py index 3bd4873c..c3073fd5 100644 --- a/bot/exts/events/advent_of_code/_cog.py +++ b/bot/exts/events/advent_of_code/_cog.py @@ -2,16 +2,16 @@ import json  import logging  from datetime import datetime, timedelta  from pathlib import Path +from typing import Optional  import arrow  import discord  from discord.ext import commands  from bot.bot import Bot -from bot.constants import ( -    AdventOfCode as AocConfig, Channels, Colours, Emojis, Month, Roles, STAFF_ROLES, WHITELISTED_CHANNELS, -) +from bot.constants import AdventOfCode as AocConfig, Channels, Colours, Emojis, Month, Roles, WHITELISTED_CHANNELS  from bot.exts.events.advent_of_code import _helpers +from bot.exts.events.advent_of_code.views.dayandstarview import AoCDropdownView  from bot.utils.decorators import InChannelCheckFailure, in_month, whitelist_override, with_role  from bot.utils.extensions import invoke_help_command @@ -153,7 +153,7 @@ class AdventOfCode(commands.Cog):          else:              try:                  join_code = await _helpers.get_public_join_code(author) -            except _helpers.FetchingLeaderboardFailed: +            except _helpers.FetchingLeaderboardFailedError:                  await ctx.send(":x: Failed to get join code! Notified maintainers.")                  return @@ -183,29 +183,79 @@ class AdventOfCode(commands.Cog):      @in_month(Month.DECEMBER)      @adventofcode_group.command( +        name="dayandstar", +        aliases=("daynstar", "daystar"), +        brief="Get a view that lets you filter the leaderboard by day and star", +    ) +    @whitelist_override(channels=AOC_WHITELIST_RESTRICTED) +    async def aoc_day_and_star_leaderboard( +            self, +            ctx: commands.Context, +            maximum_scorers_day_and_star: Optional[int] = 10 +    ) -> None: +        """Have the bot send a View that will let you filter the leaderboard by day and star.""" +        if maximum_scorers_day_and_star > AocConfig.max_day_and_star_results or maximum_scorers_day_and_star <= 0: +            raise commands.BadArgument( +                f"The maximum number of results you can query is {AocConfig.max_day_and_star_results}" +            ) +        async with ctx.typing(): +            try: +                leaderboard = await _helpers.fetch_leaderboard() +            except _helpers.FetchingLeaderboardFailedError: +                await ctx.send(":x: Unable to fetch leaderboard!") +                return +        # This is a dictionary that contains solvers in respect of day, and star. +        # e.g. 1-1 means the solvers of the first star of the first day and their completion time +        per_day_and_star = json.loads(leaderboard['leaderboard_per_day_and_star']) +        view = AoCDropdownView( +            day_and_star_data=per_day_and_star, +            maximum_scorers=maximum_scorers_day_and_star, +            original_author=ctx.author +        ) +        message = await ctx.send( +            content="Please select a day and a star to filter by!", +            view=view +        ) +        await view.wait() +        await message.edit(view=None) + +    @in_month(Month.DECEMBER) +    @adventofcode_group.command(          name="leaderboard",          aliases=("board", "lb"),          brief="Get a snapshot of the PyDis private AoC leaderboard",      )      @whitelist_override(channels=AOC_WHITELIST_RESTRICTED) -    async def aoc_leaderboard(self, ctx: commands.Context) -> None: -        """Get the current top scorers of the Python Discord Leaderboard.""" +    async def aoc_leaderboard( +            self, +            ctx: commands.Context, +            self_placement_name: Optional[str] = None, +    ) -> None: +        """ +        Get the current top scorers of the Python Discord Leaderboard. + +        Additionally you can specify a `self_placement_name` +        that will append the specified profile's personal stats to the top of the leaderboard +        """          async with ctx.typing():              try: -                leaderboard = await _helpers.fetch_leaderboard() -            except _helpers.FetchingLeaderboardFailed: +                leaderboard = await _helpers.fetch_leaderboard(self_placement_name=self_placement_name) +            except _helpers.FetchingLeaderboardFailedError:                  await ctx.send(":x: Unable to fetch leaderboard!")                  return -            number_of_participants = leaderboard["number_of_participants"] +        number_of_participants = leaderboard["number_of_participants"] -            top_count = min(AocConfig.leaderboard_displayed_members, number_of_participants) -            header = f"Here's our current top {top_count}! {Emojis.christmas_tree * 3}" - -            table = f"```\n{leaderboard['top_leaderboard']}\n```" -            info_embed = _helpers.get_summary_embed(leaderboard) +        top_count = min(AocConfig.leaderboard_displayed_members, number_of_participants) +        self_placement_header = "(and your personal stats compared to the top 10)" if self_placement_name else "" +        header = f"Here's our current top {top_count}{self_placement_header}! {Emojis.christmas_tree * 3}" +        table = "```\n" \ +                f"{leaderboard['placement_leaderboard'] if self_placement_name else leaderboard['top_leaderboard']}" \ +                "\n```" +        info_embed = _helpers.get_summary_embed(leaderboard) -            await ctx.send(content=f"{header}\n\n{table}", embed=info_embed) +        await ctx.send(content=f"{header}\n\n{table}", embed=info_embed) +        return      @in_month(Month.DECEMBER)      @adventofcode_group.command( @@ -234,7 +284,7 @@ class AdventOfCode(commands.Cog):          """Send an embed with daily completion statistics for the Python Discord leaderboard."""          try:              leaderboard = await _helpers.fetch_leaderboard() -        except _helpers.FetchingLeaderboardFailed: +        except _helpers.FetchingLeaderboardFailedError:              await ctx.send(":x: Can't fetch leaderboard for stats right now!")              return @@ -270,7 +320,7 @@ class AdventOfCode(commands.Cog):          async with ctx.typing():              try:                  await _helpers.fetch_leaderboard(invalidate_cache=True) -            except _helpers.FetchingLeaderboardFailed: +            except _helpers.FetchingLeaderboardFailedError:                  await ctx.send(":x: Something went wrong while trying to refresh the cache!")              else:                  await ctx.send("\N{OK Hand Sign} Refreshed leaderboard cache!") diff --git a/bot/exts/events/advent_of_code/_helpers.py b/bot/exts/events/advent_of_code/_helpers.py index 5fedb60f..35258544 100644 --- a/bot/exts/events/advent_of_code/_helpers.py +++ b/bot/exts/events/advent_of_code/_helpers.py @@ -10,6 +10,7 @@ from typing import Any, Optional  import aiohttp  import arrow  import discord +from discord.ext import commands  from bot.bot import Bot  from bot.constants import AdventOfCode, Channels, Colours @@ -70,6 +71,33 @@ class FetchingLeaderboardFailedError(Exception):      """Raised when one or more leaderboards could not be fetched at all.""" +def _format_leaderboard_line(rank: int, data: dict[str, Any], *, is_author: bool) -> str: +    """ +    Build a string representing a line of the leaderboard. + +    Parameters: +        rank: +            Rank in the leaderboard of this entry. + +        data: +            Mapping with entry information. + +    Keyword arguments: +        is_author: +            Whether to address the name displayed in the returned line +            personally. + +    Returns: +        A formatted line for the leaderboard. +    """ +    return AOC_TABLE_TEMPLATE.format( +        rank=rank, +        name=data['name'] if not is_author else f"(You) {data['name']}", +        score=str(data['score']), +        stars=f"({data['star_1']}, {data['star_2']})" +    ) + +  def leaderboard_sorting_function(entry: tuple[str, dict]) -> tuple[int, int]:      """      Provide a sorting value for our leaderboard. @@ -105,6 +133,7 @@ def _parse_raw_leaderboard_data(raw_leaderboard_data: dict) -> dict:      # The data we get from the AoC website is structured by member, not by day/star,      # which means we need to iterate over the members to transpose the data to a per      # star view. We need that per star view to compute rank scores per star. +    per_day_star_stats = collections.defaultdict(list)      for member in raw_leaderboard_data.values():          name = member["name"] if member["name"] else f"Anonymous #{member['id']}"          member_id = member["id"] @@ -122,6 +151,11 @@ def _parse_raw_leaderboard_data(raw_leaderboard_data: dict) -> dict:                  star_results[(day, star)].append(                      StarResult(member_id=member_id, completion_time=completion_time)                  ) +                per_day_star_stats[f"{day}-{star}"].append( +                    {'completion_time': int(data["get_star_ts"]), 'member_name': name} +                ) +    for key in per_day_star_stats: +        per_day_star_stats[key] = sorted(per_day_star_stats[key], key=operator.itemgetter('completion_time'))      # Now that we have a transposed dataset that holds the completion time of all      # participants per star, we can compute the rank-based scores each participant @@ -151,13 +185,26 @@ def _parse_raw_leaderboard_data(raw_leaderboard_data: dict) -> dict:          # this data to JSON in order to cache it in Redis.          daily_stats[day] = {"star_one": star_one, "star_two": star_two} -    return {"daily_stats": daily_stats, "leaderboard": sorted_leaderboard} +    return {"daily_stats": daily_stats, "leaderboard": sorted_leaderboard, 'per_day_and_star': per_day_star_stats} -def _format_leaderboard(leaderboard: dict[str, dict]) -> str: +def _format_leaderboard(leaderboard: dict[str, dict], self_placement_name: str = None) -> str:      """Format the leaderboard using the AOC_TABLE_TEMPLATE."""      leaderboard_lines = [HEADER] +    self_placement_exists = False      for rank, data in enumerate(leaderboard.values(), start=1): +        if self_placement_name and data["name"].lower() == self_placement_name.lower(): +            leaderboard_lines.insert( +                1, +                AOC_TABLE_TEMPLATE.format( +                    rank=rank, +                    name=f"(You) {data['name']}", +                    score=str(data["score"]), +                    stars=f"({data['star_1']}, {data['star_2']})" +                ) +            ) +            self_placement_exists = True +            continue          leaderboard_lines.append(              AOC_TABLE_TEMPLATE.format(                  rank=rank, @@ -166,7 +213,10 @@ def _format_leaderboard(leaderboard: dict[str, dict]) -> str:                  stars=f"({data['star_1']}, {data['star_2']})"              )          ) - +    if self_placement_name and not self_placement_exists: +        raise commands.BadArgument( +            "Sorry, your profile does not exist in this leaderboard." +        )      return "\n".join(leaderboard_lines) @@ -254,7 +304,7 @@ def _get_top_leaderboard(full_leaderboard: str) -> str:  @_caches.leaderboard_cache.atomic_transaction -async def fetch_leaderboard(invalidate_cache: bool = False) -> dict: +async def fetch_leaderboard(invalidate_cache: bool = False, self_placement_name: str = None) -> dict:      """      Get the current Python Discord combined leaderboard. @@ -264,7 +314,6 @@ async def fetch_leaderboard(invalidate_cache: bool = False) -> dict:      miss, this function is locked to one call at a time using a decorator.      """      cached_leaderboard = await _caches.leaderboard_cache.to_dict() -      # Check if the cached leaderboard contains everything we expect it to. If it      # does not, this probably means the cache has not been created yet or has      # expired in Redis. This check also accounts for a malformed cache. @@ -283,12 +332,14 @@ async def fetch_leaderboard(invalidate_cache: bool = False) -> dict:          leaderboard_fetched_at = datetime.datetime.utcnow().isoformat()          cached_leaderboard = { +            "placement_leaderboard": json.dumps(raw_leaderboard_data),              "full_leaderboard": formatted_leaderboard,              "top_leaderboard": _get_top_leaderboard(formatted_leaderboard),              "full_leaderboard_url": full_leaderboard_url,              "leaderboard_fetched_at": leaderboard_fetched_at,              "number_of_participants": number_of_participants,              "daily_stats": json.dumps(parsed_leaderboard_data["daily_stats"]), +            "leaderboard_per_day_and_star": json.dumps(parsed_leaderboard_data["per_day_and_star"])          }          # Store the new values in Redis @@ -300,7 +351,13 @@ async def fetch_leaderboard(invalidate_cache: bool = False) -> dict:                  _caches.leaderboard_cache.namespace,                  AdventOfCode.leaderboard_cache_expiry_seconds              ) - +    if self_placement_name: +        formatted_placement_leaderboard = _parse_raw_leaderboard_data( +            json.loads(cached_leaderboard["placement_leaderboard"]) +        )["leaderboard"] +        cached_leaderboard["placement_leaderboard"] = _get_top_leaderboard( +            _format_leaderboard(formatted_placement_leaderboard, self_placement_name=self_placement_name) +        )      return cached_leaderboard diff --git a/bot/exts/events/advent_of_code/views/dayandstarview.py b/bot/exts/events/advent_of_code/views/dayandstarview.py new file mode 100644 index 00000000..a0bfa316 --- /dev/null +++ b/bot/exts/events/advent_of_code/views/dayandstarview.py @@ -0,0 +1,76 @@ +from datetime import datetime + +import discord + +AOC_DAY_AND_STAR_TEMPLATE = "{rank: >4} | {name:25.25} | {completion_time: >10}" + + +class AoCDropdownView(discord.ui.View): +    """Interactive view to filter AoC stats by Day and Star.""" + +    def __init__(self, original_author: discord.Member, day_and_star_data: dict[str: dict], maximum_scorers: int): +        super().__init__() +        self.day = 0 +        self.star = 0 +        self.data = day_and_star_data +        self.maximum_scorers = maximum_scorers +        self.original_author = original_author + +    def generate_output(self) -> str: +        """ +        Generates a formatted codeblock with AoC statistics based on the currently selected day and star. + +        Optionally, when the requested day and star data does not exist yet it returns an error message. +        """ +        header = AOC_DAY_AND_STAR_TEMPLATE.format( +            rank="Rank", +            name="Name", completion_time="Completion time (UTC)" +        ) +        lines = [f"{header}\n{'-' * (len(header) + 2)}"] +        if not (day_and_star_data := self.data.get(f"{self.day}-{self.star}")): +            return ":x: The requested data for the specified day and star does not exist yet." +        for rank, scorer in enumerate(day_and_star_data[:self.maximum_scorers]): +            time_data = datetime.fromtimestamp(scorer['completion_time']).strftime("%I:%M:%S %p") +            lines.append(AOC_DAY_AND_STAR_TEMPLATE.format( +                datastamp="", +                rank=rank + 1, +                name=scorer['member_name'], +                completion_time=time_data) +            ) +        joined_lines = "\n".join(lines) +        return f"Statistics for Day: {self.day}, Star: {self.star}.\n ```\n{joined_lines}\n```" + +    async def interaction_check(self, interaction: discord.Interaction) -> bool: +        """Global check to ensure that the interacting user is the user who invoked the command originally.""" +        return interaction.user == self.original_author + +    @discord.ui.select( +        placeholder="Day", +        options=[discord.SelectOption(label=str(i)) for i in range(1, 26)], +        custom_id="day_select" +    ) +    async def day_select(self, select: discord.ui.Select, interaction: discord.Interaction) -> None: +        """Dropdown to choose a Day of the AoC.""" +        self.day = select.values[0] + +    @discord.ui.select( +        placeholder="Star", +        options=[discord.SelectOption(label=str(i)) for i in range(1, 3)], +        custom_id="star_select" +    ) +    async def star_select(self, select: discord.ui.Select, interaction: discord.Interaction) -> None: +        """Dropdown to choose either the first or the second star.""" +        self.star = select.values[0] + +    @discord.ui.button(label="Fetch", style=discord.ButtonStyle.blurple) +    async def fetch(self, button: discord.ui.Button, interaction: discord.Interaction) -> None: +        """Button that fetches the statistics based on the dropdown values.""" +        if self.day == 0 or self.star == 0: +            await interaction.response.send_message( +                "You have to select a value from both of the dropdowns!", +                ephemeral=True +            ) +        else: +            await interaction.response.edit_message(content=self.generate_output()) +            self.day = 0 +            self.star = 0 diff --git a/bot/exts/events/hacktoberfest/hacktober-issue-finder.py b/bot/exts/events/hacktoberfest/hacktober-issue-finder.py index 088e7e43..1774564b 100644 --- a/bot/exts/events/hacktoberfest/hacktober-issue-finder.py +++ b/bot/exts/events/hacktoberfest/hacktober-issue-finder.py @@ -100,7 +100,8 @@ class HacktoberIssues(commands.Cog):          """Format the issue data into a embed."""          title = issue["title"]          issue_url = issue["url"].replace("api.", "").replace("/repos/", "/") -        body = issue["body"] +        # issues can have empty bodies, which in that case GitHub doesn't include the key in the API response +        body = issue.get("body", "")          labels = [label["name"] for label in issue["labels"]]          embed = discord.Embed(title=title) diff --git a/bot/exts/fun/tic_tac_toe.py b/bot/exts/fun/tic_tac_toe.py index 5c4f8051..946b6f7b 100644 --- a/bot/exts/fun/tic_tac_toe.py +++ b/bot/exts/fun/tic_tac_toe.py @@ -72,10 +72,12 @@ class Player:  class AI:      """Tic Tac Toe AI class for against computer gaming.""" -    def __init__(self, symbol: str): +    def __init__(self, bot_user: discord.Member, symbol: str): +        self.user = bot_user          self.symbol = symbol -    async def get_move(self, board: dict[int, str], _: discord.Message) -> tuple[bool, int]: +    @staticmethod +    async def get_move(board: dict[int, str], _: discord.Message) -> tuple[bool, int]:          """Get move from AI. AI use Minimax strategy."""          possible_moves = [i for i, emoji in board.items() if emoji in list(Emojis.number_emojis.values())] @@ -97,8 +99,8 @@ class AI:          return False, random.choice(open_edges)      def __str__(self) -> str: -        """Return `AI` as user name.""" -        return "AI" +        """Return mention of @Sir Lancebot.""" +        return self.user.mention  class Game: @@ -107,6 +109,7 @@ class Game:      def __init__(self, players: list[Union[Player, AI]], ctx: Context):          self.players = players          self.ctx = ctx +        self.channel = ctx.channel          self.board = {              1: Emojis.number_emojis[1],              2: Emojis.number_emojis[2], @@ -173,7 +176,8 @@ class Game:              self.canceled = True              return False, "User declined" -    async def add_reactions(self, msg: discord.Message) -> None: +    @staticmethod +    async def add_reactions(msg: discord.Message) -> None:          """Add number emojis to message."""          for nr in Emojis.number_emojis.values():              await msg.add_reaction(nr) @@ -265,7 +269,7 @@ class TicTacToe(Cog):              return          if opponent is None:              game = Game( -                [Player(ctx.author, ctx, Emojis.x_square), AI(Emojis.o_square)], +                [Player(ctx.author, ctx, Emojis.x_square), AI(ctx.me, Emojis.o_square)],                  ctx              )          else: diff --git a/bot/exts/holidays/easter/earth_photos.py b/bot/exts/holidays/easter/earth_photos.py index f65790af..27442f1c 100644 --- a/bot/exts/holidays/easter/earth_photos.py +++ b/bot/exts/holidays/easter/earth_photos.py @@ -4,8 +4,7 @@ import discord  from discord.ext import commands  from bot.bot import Bot -from bot.constants import Colours -from bot.constants import Tokens +from bot.constants import Colours, Tokens  log = logging.getLogger(__name__) diff --git a/bot/exts/holidays/halloween/candy_collection.py b/bot/exts/holidays/halloween/candy_collection.py index 09bd0e59..bb9c93be 100644 --- a/bot/exts/holidays/halloween/candy_collection.py +++ b/bot/exts/holidays/halloween/candy_collection.py @@ -83,6 +83,11 @@ class CandyCollection(commands.Cog):          # if its not a candy or skull, and it is one of 10 most recent messages,          # proceed to add a skull/candy with higher chance          if str(reaction.emoji) not in (EMOJIS["SKULL"], EMOJIS["CANDY"]): +            # Ensure the reaction is not for a bot's message so users can't spam +            # reaction buttons like in .help to get candies. +            if message.author.bot: +                return +              recent_message_ids = map(                  lambda m: m.id,                  await self.hacktober_channel.history(limit=10).flatten() @@ -182,7 +187,7 @@ class CandyCollection(commands.Cog):                  for index, record in enumerate(top_five)              ) if top_five else "No Candies" -        e = discord.Embed(colour=discord.Colour.blurple()) +        e = discord.Embed(colour=discord.Colour.og_blurple())          e.add_field(              name="Top Candy Records",              value=generate_leaderboard(), diff --git a/bot/exts/holidays/halloween/scarymovie.py b/bot/exts/holidays/halloween/scarymovie.py index 33659fd8..89310b97 100644 --- a/bot/exts/holidays/halloween/scarymovie.py +++ b/bot/exts/holidays/halloween/scarymovie.py @@ -6,6 +6,7 @@ from discord.ext import commands  from bot.bot import Bot  from bot.constants import Tokens +  log = logging.getLogger(__name__) diff --git a/bot/exts/utilities/challenges.py b/bot/exts/utilities/challenges.py new file mode 100644 index 00000000..ab7ae442 --- /dev/null +++ b/bot/exts/utilities/challenges.py @@ -0,0 +1,341 @@ +import logging +from asyncio import to_thread +from random import choice +from typing import Union + +from bs4 import BeautifulSoup +from discord import Embed, Interaction, SelectOption, ui +from discord.ext import commands + +from bot.bot import Bot +from bot.constants import Colours, Emojis, NEGATIVE_REPLIES + +log = logging.getLogger(__name__) +API_ROOT = "https://www.codewars.com/api/v1/code-challenges/{kata_id}" + +# Map difficulty for the kata to color we want to display in the embed. +# These colors are representative of the colors that each kyu's level represents on codewars.com +MAPPING_OF_KYU = { +    8: 0xdddbda, 7: 0xdddbda, 6: 0xecb613, 5: 0xecb613, +    4: 0x3c7ebb, 3: 0x3c7ebb, 2: 0x866cc7, 1: 0x866cc7 +} + +# Supported languages for a kata on codewars.com +SUPPORTED_LANGUAGES = { +    "stable": [ +        "c", "c#", "c++", "clojure", "coffeescript", "coq", "crystal", "dart", "elixir", +        "f#", "go", "groovy", "haskell", "java", "javascript", "kotlin", "lean", "lua", "nasm", +        "php", "python", "racket", "ruby", "rust", "scala", "shell", "sql", "swift", "typescript" +    ], +    "beta": [ +        "agda", "bf", "cfml", "cobol", "commonlisp", "elm", "erlang", "factor", +        "forth", "fortran", "haxe", "idris", "julia", "nim", "objective-c", "ocaml", +        "pascal", "perl", "powershell", "prolog", "purescript", "r", "raku", "reason", "solidity", "vb.net" +    ] +} + + +class InformationDropdown(ui.Select): +    """A dropdown inheriting from ui.Select that allows finding out other information about the kata.""" + +    def __init__(self, language_embed: Embed, tags_embed: Embed, other_info_embed: Embed, main_embed: Embed): +        options = [ +            SelectOption( +                label="Main Information", +                description="See the kata's difficulty, description, etc.", +                emoji="🌎" +            ), +            SelectOption( +                label="Languages", +                description="See what languages this kata supports!", +                emoji=Emojis.reddit_post_text +            ), +            SelectOption( +                label="Tags", +                description="See what categories this kata falls under!", +                emoji=Emojis.stackoverflow_tag +            ), +            SelectOption( +                label="Other Information", +                description="See how other people performed on this kata and more!", +                emoji="ℹ" +            ) +        ] + +        # We map the option label to the embed instance so that it can be easily looked up later in O(1) +        self.mapping_of_embeds = { +            "Main Information": main_embed, +            "Languages": language_embed, +            "Tags": tags_embed, +            "Other Information": other_info_embed, +        } + +        super().__init__( +            placeholder="See more information regarding this kata", +            min_values=1, +            max_values=1, +            options=options +        ) + +    async def callback(self, interaction: Interaction) -> None: +        """Callback for when someone clicks on a dropdown.""" +        # Edit the message to the embed selected in the option +        # The `original_message` attribute is set just after the message is sent with the view. +        # The attribute is not set during initialization. +        result_embed = self.mapping_of_embeds[self.values[0]] +        await self.original_message.edit(embed=result_embed) + + +class Challenges(commands.Cog): +    """ +    Cog for the challenge command. + +    The challenge command pulls a random kata from codewars.com. +    A kata is the name for a challenge, specific to codewars.com. + +    The challenge command also has filters to customize the kata that is given. +    You can specify the language the kata should be from, difficulty and topic of the kata. +    """ + +    def __init__(self, bot: Bot): +        self.bot = bot + +    async def kata_id(self, search_link: str, params: dict) -> Union[str, Embed]: +        """ +        Uses bs4 to get the HTML code for the page of katas, where the page is the link of the formatted `search_link`. + +        This will webscrape the search page with `search_link` and then get the ID of a kata for the +        codewars.com API to use. +        """ +        async with self.bot.http_session.get(search_link, params=params) as response: +            if response.status != 200: +                error_embed = Embed( +                    title=choice(NEGATIVE_REPLIES), +                    description="We ran into an error when getting the kata from codewars.com, try again later.", +                    color=Colours.soft_red +                ) +                log.error(f"Unexpected response from codewars.com, status code: {response.status}") +                return error_embed + +            soup = BeautifulSoup(await response.text(), features="lxml") +            first_kata_div = await to_thread(soup.find_all, "div", class_="item-title px-0") + +            if not first_kata_div: +                raise commands.BadArgument("No katas could be found with the filters provided.") +            elif len(first_kata_div) >= 3: +                first_kata_div = choice(first_kata_div[:3]) +            elif "q=" not in search_link: +                first_kata_div = choice(first_kata_div) +            else: +                first_kata_div = first_kata_div[0] + +            # There are numerous divs before arriving at the id of the kata, which can be used for the link. +            first_kata_id = first_kata_div.a["href"].split("/")[-1] +            return first_kata_id + +    async def kata_information(self, kata_id: str) -> Union[dict, Embed]: +        """ +        Returns the information about the Kata. + +        Uses the codewars.com API to get information about the kata using `kata_id`. +        """ +        async with self.bot.http_session.get(API_ROOT.format(kata_id=kata_id)) as response: +            if response.status != 200: +                error_embed = Embed( +                    title=choice(NEGATIVE_REPLIES), +                    description="We ran into an error when getting the kata information, try again later.", +                    color=Colours.soft_red +                ) +                log.error(f"Unexpected response from codewars.com/api/v1, status code: {response.status}") +                return error_embed + +            return await response.json() + +    @staticmethod +    def main_embed(kata_information: dict) -> Embed: +        """Creates the main embed which displays the name, difficulty and description of the kata.""" +        kata_description = kata_information["description"] +        kata_url = f"https://codewars.com/kata/{kata_information['id']}" + +        # Ensuring it isn't over the length 1024 +        if len(kata_description) > 1024: +            kata_description = "\n".join(kata_description[:1000].split("\n")[:-1]) + "..." +            kata_description += f" [continue reading]({kata_url})" + +        if kata_information["rank"]["name"] is None: +            embed_color = 8 +            kata_difficulty = "Unable to retrieve difficulty for beta languages." +        else: +            embed_color = int(kata_information["rank"]["name"].replace(" kyu", "")) +            kata_difficulty = kata_information["rank"]["name"] + +        kata_embed = Embed( +            title=kata_information["name"], +            description=kata_description, +            color=MAPPING_OF_KYU[embed_color], +            url=kata_url +        ) +        kata_embed.add_field(name="Difficulty", value=kata_difficulty, inline=False) +        return kata_embed + +    @staticmethod +    def language_embed(kata_information: dict) -> Embed: +        """Creates the 'language embed' which displays all languages the kata supports.""" +        kata_url = f"https://codewars.com/kata/{kata_information['id']}" + +        languages = "\n".join(map(str.title, kata_information["languages"])) +        language_embed = Embed( +            title=kata_information["name"], +            description=f"```yaml\nSupported Languages:\n{languages}\n```", +            color=Colours.python_blue, +            url=kata_url +        ) +        return language_embed + +    @staticmethod +    def tags_embed(kata_information: dict) -> Embed: +        """ +        Creates the 'tags embed' which displays all the tags of the Kata. + +        Tags explain what the kata is about, this is what codewars.com calls categories. +        """ +        kata_url = f"https://codewars.com/kata/{kata_information['id']}" + +        tags = "\n".join(kata_information["tags"]) +        tags_embed = Embed( +            title=kata_information["name"], +            description=f"```yaml\nTags:\n{tags}\n```", +            color=Colours.grass_green, +            url=kata_url +        ) +        return tags_embed + +    @staticmethod +    def miscellaneous_embed(kata_information: dict) -> Embed: +        """ +        Creates the 'other information embed' which displays miscellaneous information about the kata. + +        This embed shows statistics such as the total number of people who completed the kata, +        the total number of stars of the kata, etc. +        """ +        kata_url = f"https://codewars.com/kata/{kata_information['id']}" + +        embed = Embed( +            title=kata_information["name"], +            description="```nim\nOther Information\n```", +            color=Colours.grass_green, +            url=kata_url +        ) +        embed.add_field( +            name="`Total Score`", +            value=f"```css\n{kata_information['voteScore']}\n```", +            inline=False +        ) +        embed.add_field( +            name="`Total Stars`", +            value=f"```css\n{kata_information['totalStars']}\n```", +            inline=False +        ) +        embed.add_field( +            name="`Total Completed`", +            value=f"```css\n{kata_information['totalCompleted']}\n```", +            inline=False +        ) +        embed.add_field( +            name="`Total Attempts`", +            value=f"```css\n{kata_information['totalAttempts']}\n```", +            inline=False +        ) +        return embed + +    @staticmethod +    def create_view(dropdown: InformationDropdown, link: str) -> ui.View: +        """ +        Creates the discord.py View for the Discord message components (dropdowns and buttons). + +        The discord UI is implemented onto the embed, where the user can choose what information about the kata they +        want, along with a link button to the kata itself. +        """ +        view = ui.View() +        view.add_item(dropdown) +        view.add_item(ui.Button(label="View the Kata", url=link)) +        return view + +    @commands.command(aliases=["kata"]) +    @commands.cooldown(1, 5, commands.BucketType.user) +    async def challenge(self, ctx: commands.Context, language: str = "python", *, query: str = None) -> None: +        """ +        The challenge command pulls a random kata (challenge) from codewars.com. + +        The different ways to use this command are: +        `.challenge <language>` - Pulls a random challenge within that language's scope. +        `.challenge <language> <difficulty>` - The difficulty can be from 1-8, +        1 being the hardest, 8 being the easiest. This pulls a random challenge within that difficulty & language. +        `.challenge <language> <query>` - Pulls a random challenge with the query provided under the language +        `.challenge <language> <query>, <difficulty>` - Pulls a random challenge with the query provided, +        under that difficulty within the language's scope. +        """ +        language = language.lower() +        if language not in SUPPORTED_LANGUAGES["stable"] + SUPPORTED_LANGUAGES["beta"]: +            raise commands.BadArgument("This is not a recognized language on codewars.com!") + +        get_kata_link = f"https://codewars.com/kata/search/{language}" +        params = {} + +        if query is not None: +            if "," in query: +                query_splitted = query.split("," if ", " not in query else ", ") + +                if len(query_splitted) > 2: +                    raise commands.BadArgument( +                        "There can only be one comma within the query, separating the difficulty and the query itself." +                    ) + +                query, level = query_splitted +                params["q"] = query +                params["r[]"] = f"-{level}" +            elif query.isnumeric(): +                params["r[]"] = f"-{query}" +            else: +                params["q"] = query + +        params["beta"] = str(language in SUPPORTED_LANGUAGES["beta"]).lower() + +        first_kata_id = await self.kata_id(get_kata_link, params) +        if isinstance(first_kata_id, Embed): +            # We ran into an error when retrieving the website link +            await ctx.send(embed=first_kata_id) +            return + +        kata_information = await self.kata_information(first_kata_id) +        if isinstance(kata_information, Embed): +            # Something went wrong when trying to fetch the kata information +            await ctx.d(embed=kata_information) +            return + +        kata_embed = self.main_embed(kata_information) +        language_embed = self.language_embed(kata_information) +        tags_embed = self.tags_embed(kata_information) +        miscellaneous_embed = self.miscellaneous_embed(kata_information) + +        dropdown = InformationDropdown( +            main_embed=kata_embed, +            language_embed=language_embed, +            tags_embed=tags_embed, +            other_info_embed=miscellaneous_embed +        ) +        kata_view = self.create_view(dropdown, f"https://codewars.com/kata/{first_kata_id}") +        original_message = await ctx.send( +            embed=kata_embed, +            view=kata_view +        ) +        dropdown.original_message = original_message + +        wait_for_kata = await kata_view.wait() +        if wait_for_kata: +            await original_message.edit(embed=kata_embed, view=None) + + +def setup(bot: Bot) -> None: +    """Load the Challenges cog.""" +    bot.add_cog(Challenges(bot)) diff --git a/bot/exts/utilities/colour.py b/bot/exts/utilities/colour.py new file mode 100644 index 00000000..7c83fc66 --- /dev/null +++ b/bot/exts/utilities/colour.py @@ -0,0 +1,259 @@ +import colorsys +import json +import pathlib +import random +import string +from io import BytesIO +from typing import Optional + +import discord +import rapidfuzz +from PIL import Image, ImageColor +from discord.ext import commands + +from bot.bot import Bot +from bot.exts.core.extensions import invoke_help_command + +THUMBNAIL_SIZE = (80, 80) + + +class Colour(commands.Cog): +    """Cog for the Colour command.""" + +    def __init__(self, bot: Bot): +        self.bot = bot +        with open(pathlib.Path("bot/resources/utilities/ryanzec_colours.json")) as f: +            self.colour_mapping = json.load(f) +            del self.colour_mapping['_']  # Delete source credit entry + +    async def send_colour_response(self, ctx: commands.Context, rgb: tuple[int, int, int]) -> None: +        """Create and send embed from user given colour information.""" +        name = self._rgb_to_name(rgb) +        try: +            colour_or_color = ctx.invoked_parents[0] +        except IndexError: +            colour_or_color = "colour" + +        colour_mode = ctx.invoked_with +        if colour_mode == "random": +            colour_mode = colour_or_color +            input_colour = name +        elif colour_mode in ("colour", "color"): +            input_colour = ctx.kwargs["colour_input"] +        elif colour_mode == "name": +            input_colour = ctx.kwargs["user_colour_name"] +        elif colour_mode == "hex": +            input_colour = ctx.args[2:][0] +            if len(input_colour) > 7: +                input_colour = input_colour[0:-2] +        else: +            input_colour = tuple(ctx.args[2:]) + +        if colour_mode not in ("name", "hex", "random", "color", "colour"): +            colour_mode = colour_mode.upper() +        else: +            colour_mode = colour_mode.title() + +        colour_embed = discord.Embed( +            title=f"{name or input_colour}", +            description=f"{colour_or_color.title()} information for {colour_mode} `{input_colour or name}`.", +            colour=discord.Color.from_rgb(*rgb) +        ) +        colour_conversions = self.get_colour_conversions(rgb) +        for colour_space, value in colour_conversions.items(): +            colour_embed.add_field( +                name=colour_space, +                value=f"`{value}`", +                inline=True +            ) + +        thumbnail = Image.new("RGB", THUMBNAIL_SIZE, color=rgb) +        buffer = BytesIO() +        thumbnail.save(buffer, "PNG") +        buffer.seek(0) +        thumbnail_file = discord.File(buffer, filename="colour.png") + +        colour_embed.set_thumbnail(url="attachment://colour.png") + +        await ctx.send(file=thumbnail_file, embed=colour_embed) + +    @commands.group(aliases=("color",), invoke_without_command=True) +    async def colour(self, ctx: commands.Context, *, colour_input: Optional[str] = None) -> None: +        """ +        Create an embed that displays colour information. + +        If no subcommand is called, a randomly selected colour will be shown. +        """ +        if colour_input is None: +            await self.random(ctx) +            return + +        try: +            extra_colour = ImageColor.getrgb(colour_input) +            await self.send_colour_response(ctx, extra_colour) +        except ValueError: +            await invoke_help_command(ctx) + +    @colour.command() +    async def rgb(self, ctx: commands.Context, red: int, green: int, blue: int) -> None: +        """Create an embed from an RGB input.""" +        if any(c not in range(256) for c in (red, green, blue)): +            raise commands.BadArgument( +                message=f"RGB values can only be from 0 to 255. User input was: `{red, green, blue}`." +            ) +        rgb_tuple = (red, green, blue) +        await self.send_colour_response(ctx, rgb_tuple) + +    @colour.command() +    async def hsv(self, ctx: commands.Context, hue: int, saturation: int, value: int) -> None: +        """Create an embed from an HSV input.""" +        if (hue not in range(361)) or any(c not in range(101) for c in (saturation, value)): +            raise commands.BadArgument( +                message="Hue can only be from 0 to 360. Saturation and Value can only be from 0 to 100. " +                f"User input was: `{hue, saturation, value}`." +            ) +        hsv_tuple = ImageColor.getrgb(f"hsv({hue}, {saturation}%, {value}%)") +        await self.send_colour_response(ctx, hsv_tuple) + +    @colour.command() +    async def hsl(self, ctx: commands.Context, hue: int, saturation: int, lightness: int) -> None: +        """Create an embed from an HSL input.""" +        if (hue not in range(361)) or any(c not in range(101) for c in (saturation, lightness)): +            raise commands.BadArgument( +                message="Hue can only be from 0 to 360. Saturation and Lightness can only be from 0 to 100. " +                f"User input was: `{hue, saturation, lightness}`." +            ) +        hsl_tuple = ImageColor.getrgb(f"hsl({hue}, {saturation}%, {lightness}%)") +        await self.send_colour_response(ctx, hsl_tuple) + +    @colour.command() +    async def cmyk(self, ctx: commands.Context, cyan: int, magenta: int, yellow: int, key: int) -> None: +        """Create an embed from a CMYK input.""" +        if any(c not in range(101) for c in (cyan, magenta, yellow, key)): +            raise commands.BadArgument( +                message=f"CMYK values can only be from 0 to 100. User input was: `{cyan, magenta, yellow, key}`." +            ) +        r = round(255 * (1 - (cyan / 100)) * (1 - (key / 100))) +        g = round(255 * (1 - (magenta / 100)) * (1 - (key / 100))) +        b = round(255 * (1 - (yellow / 100)) * (1 - (key / 100))) +        await self.send_colour_response(ctx, (r, g, b)) + +    @colour.command() +    async def hex(self, ctx: commands.Context, hex_code: str) -> None: +        """Create an embed from a HEX input.""" +        if hex_code[0] != "#": +            hex_code = f"#{hex_code}" + +        if len(hex_code) not in (4, 5, 7, 9) or any(digit not in string.hexdigits for digit in hex_code[1:]): +            raise commands.BadArgument( +                message=f"Cannot convert `{hex_code}` to a recognizable Hex format. " +                "Hex values must be hexadecimal and take the form *#RRGGBB* or *#RGB*." +            ) + +        hex_tuple = ImageColor.getrgb(hex_code) +        if len(hex_tuple) == 4: +            hex_tuple = hex_tuple[:-1]  # Colour must be RGB. If RGBA, we remove the alpha value +        await self.send_colour_response(ctx, hex_tuple) + +    @colour.command() +    async def name(self, ctx: commands.Context, *, user_colour_name: str) -> None: +        """Create an embed from a name input.""" +        hex_colour = self.match_colour_name(ctx, user_colour_name) +        if hex_colour is None: +            name_error_embed = discord.Embed( +                title="No colour match found.", +                description=f"No colour found for: `{user_colour_name}`", +                colour=discord.Color.dark_red() +            ) +            await ctx.send(embed=name_error_embed) +            return +        hex_tuple = ImageColor.getrgb(hex_colour) +        await self.send_colour_response(ctx, hex_tuple) + +    @colour.command() +    async def random(self, ctx: commands.Context) -> None: +        """Create an embed from a randomly chosen colour.""" +        hex_colour = random.choice(list(self.colour_mapping.values())) +        hex_tuple = ImageColor.getrgb(f"#{hex_colour}") +        await self.send_colour_response(ctx, hex_tuple) + +    def get_colour_conversions(self, rgb: tuple[int, int, int]) -> dict[str, str]: +        """Create a dictionary mapping of colour types and their values.""" +        colour_name = self._rgb_to_name(rgb) +        if colour_name is None: +            colour_name = "No match found" +        return { +            "RGB": rgb, +            "HSV": self._rgb_to_hsv(rgb), +            "HSL": self._rgb_to_hsl(rgb), +            "CMYK": self._rgb_to_cmyk(rgb), +            "Hex": self._rgb_to_hex(rgb), +            "Name": colour_name +        } + +    @staticmethod +    def _rgb_to_hsv(rgb: tuple[int, int, int]) -> tuple[int, int, int]: +        """Convert RGB values to HSV values.""" +        rgb_list = [val / 255 for val in rgb] +        h, s, v = colorsys.rgb_to_hsv(*rgb_list) +        hsv = (round(h * 360), round(s * 100), round(v * 100)) +        return hsv + +    @staticmethod +    def _rgb_to_hsl(rgb: tuple[int, int, int]) -> tuple[int, int, int]: +        """Convert RGB values to HSL values.""" +        rgb_list = [val / 255.0 for val in rgb] +        h, l, s = colorsys.rgb_to_hls(*rgb_list) +        hsl = (round(h * 360), round(s * 100), round(l * 100)) +        return hsl + +    @staticmethod +    def _rgb_to_cmyk(rgb: tuple[int, int, int]) -> tuple[int, int, int, int]: +        """Convert RGB values to CMYK values.""" +        rgb_list = [val / 255.0 for val in rgb] +        if not any(rgb_list): +            return 0, 0, 0, 100 +        k = 1 - max(rgb_list) +        c = round((1 - rgb_list[0] - k) * 100 / (1 - k)) +        m = round((1 - rgb_list[1] - k) * 100 / (1 - k)) +        y = round((1 - rgb_list[2] - k) * 100 / (1 - k)) +        cmyk = (c, m, y, round(k * 100)) +        return cmyk + +    @staticmethod +    def _rgb_to_hex(rgb: tuple[int, int, int]) -> str: +        """Convert RGB values to HEX code.""" +        hex_ = "".join([hex(val)[2:].zfill(2) for val in rgb]) +        hex_code = f"#{hex_}".upper() +        return hex_code + +    def _rgb_to_name(self, rgb: tuple[int, int, int]) -> Optional[str]: +        """Convert RGB values to a fuzzy matched name.""" +        input_hex_colour = self._rgb_to_hex(rgb) +        try: +            match, certainty, _ = rapidfuzz.process.extractOne( +                query=input_hex_colour, +                choices=self.colour_mapping.values(), +                score_cutoff=80 +            ) +            colour_name = [name for name, hex_code in self.colour_mapping.items() if hex_code == match][0] +        except TypeError: +            colour_name = None +        return colour_name + +    def match_colour_name(self, ctx: commands.Context, input_colour_name: str) -> Optional[str]: +        """Convert a colour name to HEX code.""" +        try: +            match, certainty, _ = rapidfuzz.process.extractOne( +                query=input_colour_name, +                choices=self.colour_mapping.keys(), +                score_cutoff=80 +            ) +        except (ValueError, TypeError): +            return +        return f"#{self.colour_mapping[match]}" + + +def setup(bot: Bot) -> None: +    """Load the Colour cog.""" +    bot.add_cog(Colour(bot)) diff --git a/bot/exts/utilities/conversationstarters.py b/bot/exts/utilities/conversationstarters.py index dcbfe4d5..8bf2abfd 100644 --- a/bot/exts/utilities/conversationstarters.py +++ b/bot/exts/utilities/conversationstarters.py @@ -53,7 +53,7 @@ class ConvoStarters(commands.Cog):          # No matter what, the form will be shown.          embed = discord.Embed(              description=f"Suggest more topics [here]({SUGGESTION_FORM})!", -            color=discord.Color.blurple() +            color=discord.Colour.og_blurple()          )          try: diff --git a/bot/exts/utilities/emoji.py b/bot/exts/utilities/emoji.py index 83df39cc..fa438d7f 100644 --- a/bot/exts/utilities/emoji.py +++ b/bot/exts/utilities/emoji.py @@ -111,7 +111,7 @@ class Emojis(commands.Cog):                  **Date:** {datetime.strftime(emoji.created_at.replace(tzinfo=None), "%d/%m/%Y")}                  **ID:** {emoji.id}              """), -            color=Color.blurple(), +            color=Color.og_blurple(),              url=str(emoji.url),          ).set_thumbnail(url=emoji.url) diff --git a/bot/exts/utilities/githubinfo.py b/bot/exts/utilities/githubinfo.py index d00b408d..539e388b 100644 --- a/bot/exts/utilities/githubinfo.py +++ b/bot/exts/utilities/githubinfo.py @@ -67,7 +67,7 @@ class GithubInfo(commands.Cog):              embed = discord.Embed(                  title=f"`{user_data['login']}`'s GitHub profile info",                  description=f"```\n{user_data['bio']}\n```\n" if user_data["bio"] else "", -                colour=discord.Colour.blurple(), +                colour=discord.Colour.og_blurple(),                  url=user_data["html_url"],                  timestamp=datetime.strptime(user_data["created_at"], "%Y-%m-%dT%H:%M:%SZ")              ) @@ -139,7 +139,7 @@ class GithubInfo(commands.Cog):          embed = discord.Embed(              title=repo_data["name"],              description=repo_data["description"], -            colour=discord.Colour.blurple(), +            colour=discord.Colour.og_blurple(),              url=repo_data["html_url"]          ) diff --git a/bot/exts/utilities/issues.py b/bot/exts/utilities/issues.py index 8a7ebed0..b6d5a43e 100644 --- a/bot/exts/utilities/issues.py +++ b/bot/exts/utilities/issues.py @@ -9,14 +9,7 @@ from discord.ext import commands  from bot.bot import Bot  from bot.constants import ( -    Categories, -    Channels, -    Colours, -    ERROR_REPLIES, -    Emojis, -    NEGATIVE_REPLIES, -    Tokens, -    WHITELISTED_CHANNELS +    Categories, Channels, Colours, ERROR_REPLIES, Emojis, NEGATIVE_REPLIES, Tokens, WHITELISTED_CHANNELS  )  from bot.utils.decorators import whitelist_override  from bot.utils.extensions import invoke_help_command @@ -185,7 +178,7 @@ class Issues(commands.Cog):          return resp      @whitelist_override(channels=WHITELISTED_CHANNELS, categories=WHITELISTED_CATEGORIES) -    @commands.command(aliases=("pr",)) +    @commands.command(aliases=("issues", "pr", "prs"))      async def issue(          self,          ctx: commands.Context, @@ -197,14 +190,23 @@ class Issues(commands.Cog):          # Remove duplicates          numbers = set(numbers) -        if len(numbers) > MAXIMUM_ISSUES: -            embed = discord.Embed( +        err_message = None +        if not numbers: +            err_message = "You must have at least one issue/PR!" + +        elif len(numbers) > MAXIMUM_ISSUES: +            err_message = f"Too many issues/PRs! (maximum of {MAXIMUM_ISSUES})" + +        # If there's an error with command invocation then send an error embed +        if err_message is not None: +            err_embed = discord.Embed(                  title=random.choice(ERROR_REPLIES),                  color=Colours.soft_red, -                description=f"Too many issues/PRs! (maximum of {MAXIMUM_ISSUES})" +                description=err_message              ) -            await ctx.send(embed=embed) +            await ctx.send(embed=err_embed)              await invoke_help_command(ctx) +            return          results = [await self.fetch_issues(number, repository, user) for number in numbers]          await ctx.send(embed=self.format_embed(results, user, repository)) diff --git a/bot/exts/utilities/realpython.py b/bot/exts/utilities/realpython.py index ef8b2638..bf8f1341 100644 --- a/bot/exts/utilities/realpython.py +++ b/bot/exts/utilities/realpython.py @@ -1,5 +1,6 @@  import logging  from html import unescape +from typing import Optional  from urllib.parse import quote_plus  from discord import Embed @@ -31,9 +32,18 @@ class RealPython(commands.Cog):      @commands.command(aliases=["rp"])      @commands.cooldown(1, 10, commands.cooldowns.BucketType.user) -    async def realpython(self, ctx: commands.Context, *, user_search: str) -> None: -        """Send 5 articles that match the user's search terms.""" -        params = {"q": user_search, "limit": 5, "kind": "article"} +    async def realpython(self, ctx: commands.Context, amount: Optional[int] = 5, *, user_search: str) -> None: +        """ +        Send some articles from RealPython that match the search terms. + +        By default the top 5 matches are sent, this can be overwritten to +        a number between 1 and 5 by specifying an amount before the search query. +        """ +        if not 1 <= amount <= 5: +            await ctx.send("`amount` must be between 1 and 5 (inclusive).") +            return + +        params = {"q": user_search, "limit": amount, "kind": "article"}          async with self.bot.http_session.get(url=API_ROOT, params=params) as response:              if response.status != 200:                  logger.error( diff --git a/bot/exts/utilities/reddit.py b/bot/exts/utilities/reddit.py index e6cb5337..782583d2 100644 --- a/bot/exts/utilities/reddit.py +++ b/bot/exts/utilities/reddit.py @@ -244,7 +244,7 @@ class Reddit(Cog):          # Use only starting summary page for #reddit channel posts.          embed.description = self.build_pagination_pages(posts, paginate=False) -        embed.colour = Colour.blurple() +        embed.colour = Colour.og_blurple()          return embed      @loop() @@ -312,7 +312,7 @@ class Reddit(Cog):          await ctx.send(f"Here are the top {subreddit} posts of all time!")          embed = Embed( -            color=Colour.blurple() +            color=Colour.og_blurple()          )          await ImagePaginator.paginate(pages, ctx, embed) @@ -325,7 +325,7 @@ class Reddit(Cog):          await ctx.send(f"Here are today's top {subreddit} posts!")          embed = Embed( -            color=Colour.blurple() +            color=Colour.og_blurple()          )          await ImagePaginator.paginate(pages, ctx, embed) @@ -338,7 +338,7 @@ class Reddit(Cog):          await ctx.send(f"Here are this week's top {subreddit} posts!")          embed = Embed( -            color=Colour.blurple() +            color=Colour.og_blurple()          )          await ImagePaginator.paginate(pages, ctx, embed) @@ -349,7 +349,7 @@ class Reddit(Cog):          """Send a paginated embed of all the subreddits we're relaying."""          embed = Embed()          embed.title = "Relayed subreddits." -        embed.colour = Colour.blurple() +        embed.colour = Colour.og_blurple()          await LinePaginator.paginate(              RedditConfig.subreddits, diff --git a/bot/exts/utilities/wikipedia.py b/bot/exts/utilities/wikipedia.py index eccc1f8c..e5e8e289 100644 --- a/bot/exts/utilities/wikipedia.py +++ b/bot/exts/utilities/wikipedia.py @@ -82,13 +82,11 @@ class WikipediaSearch(commands.Cog):          if contents:              embed = Embed(                  title="Wikipedia Search Results", -                colour=Color.blurple() +                colour=Color.og_blurple()              )              embed.set_thumbnail(url=WIKI_THUMBNAIL)              embed.timestamp = datetime.utcnow() -            await LinePaginator.paginate( -                contents, ctx, embed -            ) +            await LinePaginator.paginate(contents, ctx, embed, restrict_to_user=ctx.author)          else:              await ctx.send(                  "Sorry, we could not find a wikipedia article using that search term." diff --git a/bot/exts/utilities/wtf_python.py b/bot/exts/utilities/wtf_python.py new file mode 100644 index 00000000..980b3dba --- /dev/null +++ b/bot/exts/utilities/wtf_python.py @@ -0,0 +1,138 @@ +import logging +import random +import re +from typing import Optional + +import rapidfuzz +from discord import Embed, File +from discord.ext import commands, tasks + +from bot import constants +from bot.bot import Bot + +log = logging.getLogger(__name__) + +WTF_PYTHON_RAW_URL = "http://raw.githubusercontent.com/satwikkansal/wtfpython/master/" +BASE_URL = "https://github.com/satwikkansal/wtfpython" +LOGO_PATH = "./bot/resources/utilities/wtf_python_logo.jpg" + +ERROR_MESSAGE = f""" +Unknown WTF Python Query. Please try to reformulate your query. + +**Examples**: +```md +{constants.Client.prefix}wtf wild imports +{constants.Client.prefix}wtf subclass +{constants.Client.prefix}wtf del +``` +If the problem persists send a message in <#{constants.Channels.dev_contrib}> +""" + +MINIMUM_CERTAINTY = 55 + + +class WTFPython(commands.Cog): +    """Cog that allows getting WTF Python entries from the WTF Python repository.""" + +    def __init__(self, bot: Bot): +        self.bot = bot +        self.headers: dict[str, str] = {} +        self.fetch_readme.start() + +    @tasks.loop(minutes=60) +    async def fetch_readme(self) -> None: +        """Gets the content of README.md from the WTF Python Repository.""" +        async with self.bot.http_session.get(f"{WTF_PYTHON_RAW_URL}README.md") as resp: +            log.trace("Fetching the latest WTF Python README.md") +            if resp.status == 200: +                raw = await resp.text() +                self.parse_readme(raw) + +    def parse_readme(self, data: str) -> None: +        """ +        Parses the README.md into a dict. + +        It parses the readme into the `self.headers` dict, +        where the key is the heading and the value is the +        link to the heading. +        """ +        # Match the start of examples, until the end of the table of contents (toc) +        table_of_contents = re.search( +            r"\[👀 Examples\]\(#-examples\)\n([\w\W]*)<!-- tocstop -->", data +        )[0].split("\n") + +        for header in list(map(str.strip, table_of_contents)): +            match = re.search(r"\[▶ (.*)\]\((.*)\)", header) +            if match: +                hyper_link = match[0].split("(")[1].replace(")", "") +                self.headers[match[0]] = f"{BASE_URL}/{hyper_link}" + +    def fuzzy_match_header(self, query: str) -> Optional[str]: +        """ +        Returns the fuzzy match of a query if its ratio is above "MINIMUM_CERTAINTY" else returns None. + +        "MINIMUM_CERTAINTY" is the lowest score at which the fuzzy match will return a result. +        The certainty returned by rapidfuzz.process.extractOne is a score between 0 and 100, +        with 100 being a perfect match. +        """ +        match, certainty, _ = rapidfuzz.process.extractOne(query, self.headers.keys()) +        return match if certainty > MINIMUM_CERTAINTY else None + +    @commands.command(aliases=("wtf", "WTF")) +    async def wtf_python(self, ctx: commands.Context, *, query: Optional[str] = None) -> None: +        """ +        Search WTF Python repository. + +        Gets the link of the fuzzy matched query from https://github.com/satwikkansal/wtfpython. +        Usage: +            --> .wtf wild imports +        """ +        if query is None: +            no_query_embed = Embed( +                title="WTF Python?!", +                colour=constants.Colours.dark_green, +                description="A repository filled with suprising snippets that can make you say WTF?!\n\n" +                f"[Go to the Repository]({BASE_URL})" +            ) +            logo = File(LOGO_PATH, filename="wtf_logo.jpg") +            no_query_embed.set_thumbnail(url="attachment://wtf_logo.jpg") +            await ctx.send(embed=no_query_embed, file=logo) +            return + +        if len(query) > 50: +            embed = Embed( +                title=random.choice(constants.ERROR_REPLIES), +                description=ERROR_MESSAGE, +                colour=constants.Colours.soft_red, +            ) +            match = None +        else: +            match = self.fuzzy_match_header(query) + +        if not match: +            embed = Embed( +                title=random.choice(constants.ERROR_REPLIES), +                description=ERROR_MESSAGE, +                colour=constants.Colours.soft_red, +            ) +            await ctx.send(embed=embed) +            return + +        embed = Embed( +            title="WTF Python?!", +            colour=constants.Colours.dark_green, +            description=f"""Search result for '{query}': {match.split("]")[0].replace("[", "")} +            [Go to Repository Section]({self.headers[match]})""", +        ) +        logo = File(LOGO_PATH, filename="wtf_logo.jpg") +        embed.set_thumbnail(url="attachment://wtf_logo.jpg") +        await ctx.send(embed=embed, file=logo) + +    def cog_unload(self) -> None: +        """Unload the cog and cancel the task.""" +        self.fetch_readme.cancel() + + +def setup(bot: Bot) -> None: +    """Load the WTFPython Cog.""" +    bot.add_cog(WTFPython(bot)) diff --git a/bot/log.py b/bot/log.py new file mode 100644 index 00000000..29e696e0 --- /dev/null +++ b/bot/log.py @@ -0,0 +1,96 @@ +import logging +import logging.handlers +import os +import sys +from pathlib import Path + +import coloredlogs + +from bot.constants import Client + + +def setup() -> None: +    """Set up loggers.""" +    # Configure the "TRACE" logging level (e.g. "log.trace(message)") +    logging.TRACE = 5 +    logging.addLevelName(logging.TRACE, "TRACE") +    logging.Logger.trace = _monkeypatch_trace + +    format_string = "%(asctime)s | %(name)s | %(levelname)s | %(message)s" +    log_format = logging.Formatter(format_string) +    root_logger = logging.getLogger() + +    if Client.file_logs: +        # Set up file logging +        log_file = Path("logs/sir-lancebot.log") +        log_file.parent.mkdir(exist_ok=True) + +        # File handler rotates logs every 5 MB +        file_handler = logging.handlers.RotatingFileHandler( +            log_file, maxBytes=5 * (2 ** 20), backupCount=10, encoding="utf-8", +        ) +        file_handler.setFormatter(log_format) +        root_logger.addHandler(file_handler) + +    if "COLOREDLOGS_LEVEL_STYLES" not in os.environ: +        coloredlogs.DEFAULT_LEVEL_STYLES = { +            **coloredlogs.DEFAULT_LEVEL_STYLES, +            "trace": {"color": 246}, +            "critical": {"background": "red"}, +            "debug": coloredlogs.DEFAULT_LEVEL_STYLES["info"], +        } + +    if "COLOREDLOGS_LOG_FORMAT" not in os.environ: +        coloredlogs.DEFAULT_LOG_FORMAT = format_string + +    coloredlogs.install(level=logging.TRACE, stream=sys.stdout) + +    root_logger.setLevel(logging.DEBUG if Client.debug else logging.INFO) +    # Silence irrelevant loggers +    logging.getLogger("discord").setLevel(logging.ERROR) +    logging.getLogger("websockets").setLevel(logging.ERROR) +    logging.getLogger("PIL").setLevel(logging.ERROR) +    logging.getLogger("matplotlib").setLevel(logging.ERROR) +    logging.getLogger("async_rediscache").setLevel(logging.WARNING) + +    _set_trace_loggers() + +    root_logger.info("Logging initialization complete") + + +def _monkeypatch_trace(self: logging.Logger, msg: str, *args, **kwargs) -> None: +    """ +    Log 'msg % args' with severity 'TRACE'. + +    To pass exception information, use the keyword argument exc_info with a true value, e.g. +    logger.trace("Houston, we have an %s", "interesting problem", exc_info=1) +    """ +    if self.isEnabledFor(logging.TRACE): +        self._log(logging.TRACE, msg, args, **kwargs) + + +def _set_trace_loggers() -> None: +    """ +    Set loggers to the trace level according to the value from the BOT_TRACE_LOGGERS env var. + +    When the env var is a list of logger names delimited by a comma, +    each of the listed loggers will be set to the trace level. + +    If this list is prefixed with a "!", all of the loggers except the listed ones will be set to the trace level. + +    Otherwise if the env var begins with a "*", +    the root logger is set to the trace level and other contents are ignored. +    """ +    level_filter = Client.trace_loggers +    if level_filter: +        if level_filter.startswith("*"): +            logging.getLogger().setLevel(logging.TRACE) + +        elif level_filter.startswith("!"): +            logging.getLogger().setLevel(logging.TRACE) +            for logger_name in level_filter.strip("!,").split(","): +                logging.getLogger(logger_name).setLevel(logging.DEBUG) + +        else: +            for logger_name in level_filter.strip(",").split(","): +                logging.getLogger(logger_name).setLevel(logging.TRACE) diff --git a/bot/monkey_patches.py b/bot/monkey_patches.py index fe81f2e3..fa6627d1 100644 --- a/bot/monkey_patches.py +++ b/bot/monkey_patches.py @@ -7,17 +7,6 @@ from discord.ext import commands  log = logging.getLogger(__name__) -def trace_log(self: logging.Logger, msg: str, *args, **kwargs) -> None: -    """ -    Log 'msg % args' with severity 'TRACE'. - -    To pass exception information, use the keyword argument exc_info with a true value, e.g. -    logger.trace("Houston, we have an %s", "interesting problem", exc_info=1) -    """ -    if self.isEnabledFor(logging.TRACE): -        self._log(logging.TRACE, msg, args, **kwargs) - -  class Command(commands.Command):      """      A `discord.ext.commands.Command` subclass which supports root aliases. diff --git a/bot/resources/fun/trivia_quiz.json b/bot/resources/fun/trivia_quiz.json index 0b3e6802..99aa5f42 100644 --- a/bot/resources/fun/trivia_quiz.json +++ b/bot/resources/fun/trivia_quiz.json @@ -440,7 +440,7 @@      {        "id": 229,        "question": "What is this triangle called?", -      "img_url": "https://cdn.askpython.com/wp-content/uploads/2020/07/Pascals-triangle.png", +      "img_url": "https://wikimedia.org/api/rest_v1/media/math/render/png/23050fcb53d6083d9e42043bebf2863fa9746043",        "answer": ["Pascal's triangle", "Pascal"]      },      { diff --git a/bot/resources/holidays/halloween/bat-clipart.png b/bot/resources/holidays/halloween/bat-clipart.pngBinary files differ index 7df26ba9..fc2f77b0 100644 --- a/bot/resources/holidays/halloween/bat-clipart.png +++ b/bot/resources/holidays/halloween/bat-clipart.png diff --git a/bot/resources/utilities/py_topics.yaml b/bot/resources/utilities/py_topics.yaml index a3fb2ccc..1cd2c325 100644 --- a/bot/resources/utilities/py_topics.yaml +++ b/bot/resources/utilities/py_topics.yaml @@ -33,7 +33,6 @@      - How often do you program in Python?      - How would you learn a new library if needed to do so?      - Have you ever worked with a microcontroller or anything physical with Python before? -    - How good would you say you are at Python so far? Beginner, intermediate, or advanced?      - Have you ever tried making your own programming language?      - Has a recently discovered Python module changed your general use of Python? diff --git a/bot/resources/utilities/ryanzec_colours.json b/bot/resources/utilities/ryanzec_colours.json new file mode 100644 index 00000000..552d5a3f --- /dev/null +++ b/bot/resources/utilities/ryanzec_colours.json @@ -0,0 +1,1569 @@ +{ +    "_": "Source: https://github.com/ryanzec/name-that-color/blob/0bb5ec7f37e4f6e7f2c164f39f7f08cca7c8e621/lib/ntc.js#L116-L1681\n\nCopyright (c) 2014 Ryan Zec\n\nPermission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the \"Software\"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.", +    "Abbey": "4C4F56", +    "Acadia": "1B1404", +    "Acapulco": "7CB0A1", +    "Aero Blue": "C9FFE5", +    "Affair": "714693", +    "Akaroa": "D4C4A8", +    "Alabaster": "FAFAFA", +    "Albescent White": "F5E9D3", +    "Algae Green": "93DFB8", +    "Alice Blue": "F0F8FF", +    "Alizarin Crimson": "E32636", +    "Allports": "0076A3", +    "Almond": "EED9C4", +    "Almond Frost": "907B71", +    "Alpine": "AF8F2C", +    "Alto": "DBDBDB", +    "Aluminium": "A9ACB6", +    "Amaranth": "E52B50", +    "Amazon": "3B7A57", +    "Amber": "FFBF00", +    "Americano": "87756E", +    "Amethyst": "9966CC", +    "Amethyst Smoke": "A397B4", +    "Amour": "F9EAF3", +    "Amulet": "7B9F80", +    "Anakiwa": "9DE5FF", +    "Antique Brass": "C88A65", +    "Antique Bronze": "704A07", +    "Anzac": "E0B646", +    "Apache": "DFBE6F", +    "Apple": "4FA83D", +    "Apple Blossom": "AF4D43", +    "Apple Green": "E2F3EC", +    "Apricot": "EB9373", +    "Apricot Peach": "FBCEB1", +    "Apricot White": "FFFEEC", +    "Aqua Deep": "014B43", +    "Aqua Forest": "5FA777", +    "Aqua Haze": "EDF5F5", +    "Aqua Island": "A1DAD7", +    "Aqua Spring": "EAF9F5", +    "Aqua Squeeze": "E8F5F2", +    "Aquamarine": "7FFFD4", +    "Aquamarine Blue": "71D9E2", +    "Arapawa": "110C6C", +    "Armadillo": "433E37", +    "Arrowtown": "948771", +    "Ash": "C6C3B5", +    "Asparagus": "7BA05B", +    "Asphalt": "130A06", +    "Astra": "FAEAB9", +    "Astral": "327DA0", +    "Astronaut": "283A77", +    "Astronaut Blue": "013E62", +    "Athens Gray": "EEF0F3", +    "Aths Special": "ECEBCE", +    "Atlantis": "97CD2D", +    "Atoll": "0A6F75", +    "Atomic Tangerine": "FF9966", +    "Au Chico": "97605D", +    "Aubergine": "3B0910", +    "Australian Mint": "F5FFBE", +    "Avocado": "888D65", +    "Axolotl": "4E6649", +    "Azalea": "F7C8DA", +    "Aztec": "0D1C19", +    "Azure": "315BA1", +    "Azure Radiance": "007FFF", +    "Baby Blue": "E0FFFF", +    "Bahama Blue": "026395", +    "Bahia": "A5CB0C", +    "Baja White": "FFF8D1", +    "Bali Hai": "859FAF", +    "Baltic Sea": "2A2630", +    "Bamboo": "DA6304", +    "Banana Mania": "FBE7B2", +    "Bandicoot": "858470", +    "Barberry": "DED717", +    "Barley Corn": "A68B5B", +    "Barley White": "FFF4CE", +    "Barossa": "44012D", +    "Bastille": "292130", +    "Battleship Gray": "828F72", +    "Bay Leaf": "7DA98D", +    "Bay of Many": "273A81", +    "Bazaar": "98777B", +    "Bean  ": "3D0C02", +    "Beauty Bush": "EEC1BE", +    "Beaver": "926F5B", +    "Beeswax": "FEF2C7", +    "Beige": "F5F5DC", +    "Bermuda": "7DD8C6", +    "Bermuda Gray": "6B8BA2", +    "Beryl Green": "DEE5C0", +    "Bianca": "FCFBF3", +    "Big Stone": "162A40", +    "Bilbao": "327C14", +    "Biloba Flower": "B2A1EA", +    "Birch": "373021", +    "Bird Flower": "D4CD16", +    "Biscay": "1B3162", +    "Bismark": "497183", +    "Bison Hide": "C1B7A4", +    "Bistre": "3D2B1F", +    "Bitter": "868974", +    "Bitter Lemon": "CAE00D", +    "Bittersweet": "FE6F5E", +    "Bizarre": "EEDEDA", +    "Black": "000000", +    "Black Bean": "081910", +    "Black Forest": "0B1304", +    "Black Haze": "F6F7F7", +    "Black Marlin": "3E2C1C", +    "Black Olive": "242E16", +    "Black Pearl": "041322", +    "Black Rock": "0D0332", +    "Black Rose": "67032D", +    "Black Russian": "0A001C", +    "Black Squeeze": "F2FAFA", +    "Black White": "FFFEF6", +    "Blackberry": "4D0135", +    "Blackcurrant": "32293A", +    "Blaze Orange": "FF6600", +    "Bleach White": "FEF3D8", +    "Bleached Cedar": "2C2133", +    "Blizzard Blue": "A3E3ED", +    "Blossom": "DCB4BC", +    "Blue": "0000FF", +    "Blue Bayoux": "496679", +    "Blue Bell": "9999CC", +    "Blue Chalk": "F1E9FF", +    "Blue Charcoal": "010D1A", +    "Blue Chill": "0C8990", +    "Blue Diamond": "380474", +    "Blue Dianne": "204852", +    "Blue Gem": "2C0E8C", +    "Blue Haze": "BFBED8", +    "Blue Lagoon": "017987", +    "Blue Marguerite": "7666C6", +    "Blue Ribbon": "0066FF", +    "Blue Romance": "D2F6DE", +    "Blue Smoke": "748881", +    "Blue Stone": "016162", +    "Blue Violet": "6456B7", +    "Blue Whale": "042E4C", +    "Blue Zodiac": "13264D", +    "Blumine": "18587A", +    "Blush": "B44668", +    "Blush Pink": "FF6FFF", +    "Bombay": "AFB1B8", +    "Bon Jour": "E5E0E1", +    "Bondi Blue": "0095B6", +    "Bone": "E4D1C0", +    "Bordeaux": "5C0120", +    "Bossanova": "4E2A5A", +    "Boston Blue": "3B91B4", +    "Botticelli": "C7DDE5", +    "Bottle Green": "093624", +    "Boulder": "7A7A7A", +    "Bouquet": "AE809E", +    "Bourbon": "BA6F1E", +    "Bracken": "4A2A04", +    "Brandy": "DEC196", +    "Brandy Punch": "CD8429", +    "Brandy Rose": "BB8983", +    "Breaker Bay": "5DA19F", +    "Brick Red": "C62D42", +    "Bridal Heath": "FFFAF4", +    "Bridesmaid": "FEF0EC", +    "Bright Gray": "3C4151", +    "Bright Green": "66FF00", +    "Bright Red": "B10000", +    "Bright Sun": "FED33C", +    "Bright Turquoise": "08E8DE", +    "Brilliant Rose": "F653A6", +    "Brink Pink": "FB607F", +    "Bronco": "ABA196", +    "Bronze": "3F2109", +    "Bronze Olive": "4E420C", +    "Bronzetone": "4D400F", +    "Broom": "FFEC13", +    "Brown": "964B00", +    "Brown Bramble": "592804", +    "Brown Derby": "492615", +    "Brown Pod": "401801", +    "Brown Rust": "AF593E", +    "Brown Tumbleweed": "37290E", +    "Bubbles": "E7FEFF", +    "Buccaneer": "622F30", +    "Bud": "A8AE9C", +    "Buddha Gold": "C1A004", +    "Buff": "F0DC82", +    "Bulgarian Rose": "480607", +    "Bull Shot": "864D1E", +    "Bunker": "0D1117", +    "Bunting": "151F4C", +    "Burgundy": "900020", +    "Burnham": "002E20", +    "Burning Orange": "FF7034", +    "Burning Sand": "D99376", +    "Burnt Maroon": "420303", +    "Burnt Orange": "CC5500", +    "Burnt Sienna": "E97451", +    "Burnt Umber": "8A3324", +    "Bush": "0D2E1C", +    "Buttercup": "F3AD16", +    "Buttered Rum": "A1750D", +    "Butterfly Bush": "624E9A", +    "Buttermilk": "FFF1B5", +    "Buttery White": "FFFCEA", +    "Cab Sav": "4D0A18", +    "Cabaret": "D94972", +    "Cabbage Pont": "3F4C3A", +    "Cactus": "587156", +    "Cadet Blue": "A9B2C3", +    "Cadillac": "B04C6A", +    "Cafe Royale": "6F440C", +    "Calico": "E0C095", +    "California": "FE9D04", +    "Calypso": "31728D", +    "Camarone": "00581A", +    "Camelot": "893456", +    "Cameo": "D9B99B", +    "Camouflage": "3C3910", +    "Camouflage Green": "78866B", +    "Can Can": "D591A4", +    "Canary": "F3FB62", +    "Candlelight": "FCD917", +    "Candy Corn": "FBEC5D", +    "Cannon Black": "251706", +    "Cannon Pink": "894367", +    "Cape Cod": "3C4443", +    "Cape Honey": "FEE5AC", +    "Cape Palliser": "A26645", +    "Caper": "DCEDB4", +    "Caramel": "FFDDAF", +    "Cararra": "EEEEE8", +    "Cardin Green": "01361C", +    "Cardinal": "C41E3A", +    "Cardinal Pink": "8C055E", +    "Careys Pink": "D29EAA", +    "Caribbean Green": "00CC99", +    "Carissma": "EA88A8", +    "Carla": "F3FFD8", +    "Carmine": "960018", +    "Carnaby Tan": "5C2E01", +    "Carnation": "F95A61", +    "Carnation Pink": "FFA6C9", +    "Carousel Pink": "F9E0ED", +    "Carrot Orange": "ED9121", +    "Casablanca": "F8B853", +    "Casal": "2F6168", +    "Cascade": "8BA9A5", +    "Cashmere": "E6BEA5", +    "Casper": "ADBED1", +    "Castro": "52001F", +    "Catalina Blue": "062A78", +    "Catskill White": "EEF6F7", +    "Cavern Pink": "E3BEBE", +    "Cedar": "3E1C14", +    "Cedar Wood Finish": "711A00", +    "Celadon": "ACE1AF", +    "Celery": "B8C25D", +    "Celeste": "D1D2CA", +    "Cello": "1E385B", +    "Celtic": "163222", +    "Cement": "8D7662", +    "Ceramic": "FCFFF9", +    "Cerise": "DA3287", +    "Cerise Red": "DE3163", +    "Cerulean": "02A4D3", +    "Cerulean Blue": "2A52BE", +    "Chablis": "FFF4F3", +    "Chalet Green": "516E3D", +    "Chalky": "EED794", +    "Chambray": "354E8C", +    "Chamois": "EDDCB1", +    "Champagne": "FAECCC", +    "Chantilly": "F8C3DF", +    "Charade": "292937", +    "Chardon": "FFF3F1", +    "Chardonnay": "FFCD8C", +    "Charlotte": "BAEEF9", +    "Charm": "D47494", +    "Chartreuse": "7FFF00", +    "Chartreuse Yellow": "DFFF00", +    "Chateau Green": "40A860", +    "Chatelle": "BDB3C7", +    "Chathams Blue": "175579", +    "Chelsea Cucumber": "83AA5D", +    "Chelsea Gem": "9E5302", +    "Chenin": "DFCD6F", +    "Cherokee": "FCDA98", +    "Cherry Pie": "2A0359", +    "Cherrywood": "651A14", +    "Cherub": "F8D9E9", +    "Chestnut": "B94E48", +    "Chestnut Rose": "CD5C5C", +    "Chetwode Blue": "8581D9", +    "Chicago": "5D5C58", +    "Chiffon": "F1FFC8", +    "Chilean Fire": "F77703", +    "Chilean Heath": "FFFDE6", +    "China Ivory": "FCFFE7", +    "Chino": "CEC7A7", +    "Chinook": "A8E3BD", +    "Chocolate": "370202", +    "Christalle": "33036B", +    "Christi": "67A712", +    "Christine": "E7730A", +    "Chrome White": "E8F1D4", +    "Cinder": "0E0E18", +    "Cinderella": "FDE1DC", +    "Cinnabar": "E34234", +    "Cinnamon": "7B3F00", +    "Cioccolato": "55280C", +    "Citrine White": "FAF7D6", +    "Citron": "9EA91F", +    "Citrus": "A1C50A", +    "Clairvoyant": "480656", +    "Clam Shell": "D4B6AF", +    "Claret": "7F1734", +    "Classic Rose": "FBCCE7", +    "Clay Ash": "BDC8B3", +    "Clay Creek": "8A8360", +    "Clear Day": "E9FFFD", +    "Clementine": "E96E00", +    "Clinker": "371D09", +    "Cloud": "C7C4BF", +    "Cloud Burst": "202E54", +    "Cloudy": "ACA59F", +    "Clover": "384910", +    "Cobalt": "0047AB", +    "Cocoa Bean": "481C1C", +    "Cocoa Brown": "301F1E", +    "Coconut Cream": "F8F7DC", +    "Cod Gray": "0B0B0B", +    "Coffee": "706555", +    "Coffee Bean": "2A140E", +    "Cognac": "9F381D", +    "Cola": "3F2500", +    "Cold Purple": "ABA0D9", +    "Cold Turkey": "CEBABA", +    "Colonial White": "FFEDBC", +    "Comet": "5C5D75", +    "Como": "517C66", +    "Conch": "C9D9D2", +    "Concord": "7C7B7A", +    "Concrete": "F2F2F2", +    "Confetti": "E9D75A", +    "Congo Brown": "593737", +    "Congress Blue": "02478E", +    "Conifer": "ACDD4D", +    "Contessa": "C6726B", +    "Copper": "B87333", +    "Copper Canyon": "7E3A15", +    "Copper Rose": "996666", +    "Copper Rust": "944747", +    "Copperfield": "DA8A67", +    "Coral": "FF7F50", +    "Coral Red": "FF4040", +    "Coral Reef": "C7BCA2", +    "Coral Tree": "A86B6B", +    "Corduroy": "606E68", +    "Coriander": "C4D0B0", +    "Cork": "40291D", +    "Corn": "E7BF05", +    "Corn Field": "F8FACD", +    "Corn Harvest": "8B6B0B", +    "Cornflower": "93CCEA", +    "Cornflower Blue": "6495ED", +    "Cornflower Lilac": "FFB0AC", +    "Corvette": "FAD3A2", +    "Cosmic": "76395D", +    "Cosmos": "FFD8D9", +    "Costa Del Sol": "615D30", +    "Cotton Candy": "FFB7D5", +    "Cotton Seed": "C2BDB6", +    "County Green": "01371A", +    "Cowboy": "4D282D", +    "Crail": "B95140", +    "Cranberry": "DB5079", +    "Crater Brown": "462425", +    "Cream": "FFFDD0", +    "Cream Brulee": "FFE5A0", +    "Cream Can": "F5C85C", +    "Creole": "1E0F04", +    "Crete": "737829", +    "Crimson": "DC143C", +    "Crocodile": "736D58", +    "Crown of Thorns": "771F1F", +    "Crowshead": "1C1208", +    "Cruise": "B5ECDF", +    "Crusoe": "004816", +    "Crusta": "FD7B33", +    "Cumin": "924321", +    "Cumulus": "FDFFD5", +    "Cupid": "FBBEDA", +    "Curious Blue": "2596D1", +    "Cutty Sark": "507672", +    "Cyan / Aqua": "00FFFF", +    "Cyprus": "003E40", +    "Daintree": "012731", +    "Dairy Cream": "F9E4BC", +    "Daisy Bush": "4F2398", +    "Dallas": "6E4B26", +    "Dandelion": "FED85D", +    "Danube": "6093D1", +    "Dark Blue": "0000C8", +    "Dark Burgundy": "770F05", +    "Dark Ebony": "3C2005", +    "Dark Fern": "0A480D", +    "Dark Tan": "661010", +    "Dawn": "A6A29A", +    "Dawn Pink": "F3E9E5", +    "De York": "7AC488", +    "Deco": "D2DA97", +    "Deep Blue": "220878", +    "Deep Blush": "E47698", +    "Deep Bronze": "4A3004", +    "Deep Cerulean": "007BA7", +    "Deep Cove": "051040", +    "Deep Fir": "002900", +    "Deep Forest Green": "182D09", +    "Deep Koamaru": "1B127B", +    "Deep Oak": "412010", +    "Deep Sapphire": "082567", +    "Deep Sea": "01826B", +    "Deep Sea Green": "095859", +    "Deep Teal": "003532", +    "Del Rio": "B09A95", +    "Dell": "396413", +    "Delta": "A4A49D", +    "Deluge": "7563A8", +    "Denim": "1560BD", +    "Derby": "FFEED8", +    "Desert": "AE6020", +    "Desert Sand": "EDC9AF", +    "Desert Storm": "F8F8F7", +    "Dew": "EAFFFE", +    "Di Serria": "DB995E", +    "Diesel": "130000", +    "Dingley": "5D7747", +    "Disco": "871550", +    "Dixie": "E29418", +    "Dodger Blue": "1E90FF", +    "Dolly": "F9FF8B", +    "Dolphin": "646077", +    "Domino": "8E775E", +    "Don Juan": "5D4C51", +    "Donkey Brown": "A69279", +    "Dorado": "6B5755", +    "Double Colonial White": "EEE3AD", +    "Double Pearl Lusta": "FCF4D0", +    "Double Spanish White": "E6D7B9", +    "Dove Gray": "6D6C6C", +    "Downriver": "092256", +    "Downy": "6FD0C5", +    "Driftwood": "AF8751", +    "Drover": "FDF7AD", +    "Dull Lavender": "A899E6", +    "Dune": "383533", +    "Dust Storm": "E5CCC9", +    "Dusty Gray": "A8989B", +    "Eagle": "B6BAA4", +    "Earls Green": "C9B93B", +    "Early Dawn": "FFF9E6", +    "East Bay": "414C7D", +    "East Side": "AC91CE", +    "Eastern Blue": "1E9AB0", +    "Ebb": "E9E3E3", +    "Ebony": "0C0B1D", +    "Ebony Clay": "26283B", +    "Eclipse": "311C17", +    "Ecru White": "F5F3E5", +    "Ecstasy": "FA7814", +    "Eden": "105852", +    "Edgewater": "C8E3D7", +    "Edward": "A2AEAB", +    "Egg Sour": "FFF4DD", +    "Egg White": "FFEFC1", +    "Eggplant": "614051", +    "El Paso": "1E1708", +    "El Salva": "8F3E33", +    "Electric Lime": "CCFF00", +    "Electric Violet": "8B00FF", +    "Elephant": "123447", +    "Elf Green": "088370", +    "Elm": "1C7C7D", +    "Emerald": "50C878", +    "Eminence": "6C3082", +    "Emperor": "514649", +    "Empress": "817377", +    "Endeavour": "0056A7", +    "Energy Yellow": "F8DD5C", +    "English Holly": "022D15", +    "English Walnut": "3E2B23", +    "Envy": "8BA690", +    "Equator": "E1BC64", +    "Espresso": "612718", +    "Eternity": "211A0E", +    "Eucalyptus": "278A5B", +    "Eunry": "CFA39D", +    "Evening Sea": "024E46", +    "Everglade": "1C402E", +    "Faded Jade": "427977", +    "Fair Pink": "FFEFEC", +    "Falcon": "7F626D", +    "Fall Green": "ECEBBD", +    "Falu Red": "801818", +    "Fantasy": "FAF3F0", +    "Fedora": "796A78", +    "Feijoa": "9FDD8C", +    "Fern": "63B76C", +    "Fern Frond": "657220", +    "Fern Green": "4F7942", +    "Ferra": "704F50", +    "Festival": "FBE96C", +    "Feta": "F0FCEA", +    "Fiery Orange": "B35213", +    "Finch": "626649", +    "Finlandia": "556D56", +    "Finn": "692D54", +    "Fiord": "405169", +    "Fire": "AA4203", +    "Fire Bush": "E89928", +    "Firefly": "0E2A30", +    "Flame Pea": "DA5B38", +    "Flamenco": "FF7D07", +    "Flamingo": "F2552A", +    "Flax": "EEDC82", +    "Flax Smoke": "7B8265", +    "Flesh": "FFCBA4", +    "Flint": "6F6A61", +    "Flirt": "A2006D", +    "Flush Mahogany": "CA3435", +    "Flush Orange": "FF7F00", +    "Foam": "D8FCFA", +    "Fog": "D7D0FF", +    "Foggy Gray": "CBCAB6", +    "Forest Green": "228B22", +    "Forget Me Not": "FFF1EE", +    "Fountain Blue": "56B4BE", +    "Frangipani": "FFDEB3", +    "French Gray": "BDBDC6", +    "French Lilac": "ECC7EE", +    "French Pass": "BDEDFD", +    "French Rose": "F64A8A", +    "Fresh Eggplant": "990066", +    "Friar Gray": "807E79", +    "Fringy Flower": "B1E2C1", +    "Froly": "F57584", +    "Frost": "EDF5DD", +    "Frosted Mint": "DBFFF8", +    "Frostee": "E4F6E7", +    "Fruit Salad": "4F9D5D", +    "Fuchsia Blue": "7A58C1", +    "Fuchsia Pink": "C154C1", +    "Fuego": "BEDE0D", +    "Fuel Yellow": "ECA927", +    "Fun Blue": "1959A8", +    "Fun Green": "016D39", +    "Fuscous Gray": "54534D", +    "Fuzzy Wuzzy Brown": "C45655", +    "Gable Green": "163531", +    "Gallery": "EFEFEF", +    "Galliano": "DCB20C", +    "Gamboge": "E49B0F", +    "Geebung": "D18F1B", +    "Genoa": "15736B", +    "Geraldine": "FB8989", +    "Geyser": "D4DFE2", +    "Ghost": "C7C9D5", +    "Gigas": "523C94", +    "Gimblet": "B8B56A", +    "Gin": "E8F2EB", +    "Gin Fizz": "FFF9E2", +    "Givry": "F8E4BF", +    "Glacier": "80B3C4", +    "Glade Green": "61845F", +    "Go Ben": "726D4E", +    "Goblin": "3D7D52", +    "Gold": "FFD700", +    "Gold Drop": "F18200", +    "Gold Sand": "E6BE8A", +    "Gold Tips": "DEBA13", +    "Golden Bell": "E28913", +    "Golden Dream": "F0D52D", +    "Golden Fizz": "F5FB3D", +    "Golden Glow": "FDE295", +    "Golden Grass": "DAA520", +    "Golden Sand": "F0DB7D", +    "Golden Tainoi": "FFCC5C", +    "Goldenrod": "FCD667", +    "Gondola": "261414", +    "Gordons Green": "0B1107", +    "Gorse": "FFF14F", +    "Gossamer": "069B81", +    "Gossip": "D2F8B0", +    "Gothic": "6D92A1", +    "Governor Bay": "2F3CB3", +    "Grain Brown": "E4D5B7", +    "Grandis": "FFD38C", +    "Granite Green": "8D8974", +    "Granny Apple": "D5F6E3", +    "Granny Smith": "84A0A0", +    "Granny Smith Apple": "9DE093", +    "Grape": "381A51", +    "Graphite": "251607", +    "Gravel": "4A444B", +    "Gray": "808080", +    "Gray Asparagus": "465945", +    "Gray Chateau": "A2AAB3", +    "Gray Nickel": "C3C3BD", +    "Gray Nurse": "E7ECE6", +    "Gray Olive": "A9A491", +    "Gray Suit": "C1BECD", +    "Green": "00FF00", +    "Green Haze": "01A368", +    "Green House": "24500F", +    "Green Kelp": "25311C", +    "Green Leaf": "436A0D", +    "Green Mist": "CBD3B0", +    "Green Pea": "1D6142", +    "Green Smoke": "A4AF6E", +    "Green Spring": "B8C1B1", +    "Green Vogue": "032B52", +    "Green Waterloo": "101405", +    "Green White": "E8EBE0", +    "Green Yellow": "ADFF2F", +    "Grenadier": "D54600", +    "Guardsman Red": "BA0101", +    "Gulf Blue": "051657", +    "Gulf Stream": "80B3AE", +    "Gull Gray": "9DACB7", +    "Gum Leaf": "B6D3BF", +    "Gumbo": "7CA1A6", +    "Gun Powder": "414257", +    "Gunsmoke": "828685", +    "Gurkha": "9A9577", +    "Hacienda": "98811B", +    "Hairy Heath": "6B2A14", +    "Haiti": "1B1035", +    "Half Baked": "85C4CC", +    "Half Colonial White": "FDF6D3", +    "Half Dutch White": "FEF7DE", +    "Half Spanish White": "FEF4DB", +    "Half and Half": "FFFEE1", +    "Hampton": "E5D8AF", +    "Harlequin": "3FFF00", +    "Harp": "E6F2EA", +    "Harvest Gold": "E0B974", +    "Havelock Blue": "5590D9", +    "Hawaiian Tan": "9D5616", +    "Hawkes Blue": "D4E2FC", +    "Heath": "541012", +    "Heather": "B7C3D0", +    "Heathered Gray": "B6B095", +    "Heavy Metal": "2B3228", +    "Heliotrope": "DF73FF", +    "Hemlock": "5E5D3B", +    "Hemp": "907874", +    "Hibiscus": "B6316C", +    "Highland": "6F8E63", +    "Hillary": "ACA586", +    "Himalaya": "6A5D1B", +    "Hint of Green": "E6FFE9", +    "Hint of Red": "FBF9F9", +    "Hint of Yellow": "FAFDE4", +    "Hippie Blue": "589AAF", +    "Hippie Green": "53824B", +    "Hippie Pink": "AE4560", +    "Hit Gray": "A1ADB5", +    "Hit Pink": "FFAB81", +    "Hokey Pokey": "C8A528", +    "Hoki": "65869F", +    "Holly": "011D13", +    "Hollywood Cerise": "F400A1", +    "Honey Flower": "4F1C70", +    "Honeysuckle": "EDFC84", +    "Hopbush": "D06DA1", +    "Horizon": "5A87A0", +    "Horses Neck": "604913", +    "Hot Cinnamon": "D2691E", +    "Hot Pink": "FF69B4", +    "Hot Toddy": "B38007", +    "Humming Bird": "CFF9F3", +    "Hunter Green": "161D10", +    "Hurricane": "877C7B", +    "Husk": "B7A458", +    "Ice Cold": "B1F4E7", +    "Iceberg": "DAF4F0", +    "Illusion": "F6A4C9", +    "Inch Worm": "B0E313", +    "Indian Khaki": "C3B091", +    "Indian Tan": "4D1E01", +    "Indigo": "4F69C6", +    "Indochine": "C26B03", +    "International Klein Blue": "002FA7", +    "International Orange": "FF4F00", +    "Irish Coffee": "5F3D26", +    "Iroko": "433120", +    "Iron": "D4D7D9", +    "Ironside Gray": "676662", +    "Ironstone": "86483C", +    "Island Spice": "FFFCEE", +    "Ivory": "FFFFF0", +    "Jacaranda": "2E0329", +    "Jacarta": "3A2A6A", +    "Jacko Bean": "2E1905", +    "Jacksons Purple": "20208D", +    "Jade": "00A86B", +    "Jaffa": "EF863F", +    "Jagged Ice": "C2E8E5", +    "Jagger": "350E57", +    "Jaguar": "080110", +    "Jambalaya": "5B3013", +    "Janna": "F4EBD3", +    "Japanese Laurel": "0A6906", +    "Japanese Maple": "780109", +    "Japonica": "D87C63", +    "Java": "1FC2C2", +    "Jazzberry Jam": "A50B5E", +    "Jelly Bean": "297B9A", +    "Jet Stream": "B5D2CE", +    "Jewel": "126B40", +    "Jon": "3B1F1F", +    "Jonquil": "EEFF9A", +    "Jordy Blue": "8AB9F1", +    "Judge Gray": "544333", +    "Jumbo": "7C7B82", +    "Jungle Green": "29AB87", +    "Jungle Mist": "B4CFD3", +    "Juniper": "6D9292", +    "Just Right": "ECCDB9", +    "Kabul": "5E483E", +    "Kaitoke Green": "004620", +    "Kangaroo": "C6C8BD", +    "Karaka": "1E1609", +    "Karry": "FFEAD4", +    "Kashmir Blue": "507096", +    "Kelp": "454936", +    "Kenyan Copper": "7C1C05", +    "Keppel": "3AB09E", +    "Key Lime Pie": "BFC921", +    "Khaki": "F0E68C", +    "Kidnapper": "E1EAD4", +    "Kilamanjaro": "240C02", +    "Killarney": "3A6A47", +    "Kimberly": "736C9F", +    "Kingfisher Daisy": "3E0480", +    "Kobi": "E79FC4", +    "Kokoda": "6E6D57", +    "Korma": "8F4B0E", +    "Koromiko": "FFBD5F", +    "Kournikova": "FFE772", +    "Kumera": "886221", +    "La Palma": "368716", +    "La Rioja": "B3C110", +    "Las Palmas": "C6E610", +    "Laser": "C8B568", +    "Laser Lemon": "FFFF66", +    "Laurel": "749378", +    "Lavender": "B57EDC", +    "Lavender Gray": "BDBBD7", +    "Lavender Magenta": "EE82EE", +    "Lavender Pink": "FBAED2", +    "Lavender Purple": "967BB6", +    "Lavender Rose": "FBA0E3", +    "Lavender blush": "FFF0F5", +    "Leather": "967059", +    "Lemon": "FDE910", +    "Lemon Chiffon": "FFFACD", +    "Lemon Ginger": "AC9E22", +    "Lemon Grass": "9B9E8F", +    "Light Apricot": "FDD5B1", +    "Light Orchid": "E29CD2", +    "Light Wisteria": "C9A0DC", +    "Lightning Yellow": "FCC01E", +    "Lilac": "C8A2C8", +    "Lilac Bush": "9874D3", +    "Lily": "C8AABF", +    "Lily White": "E7F8FF", +    "Lima": "76BD17", +    "Lime": "BFFF00", +    "Limeade": "6F9D02", +    "Limed Ash": "747D63", +    "Limed Oak": "AC8A56", +    "Limed Spruce": "394851", +    "Linen": "FAF0E6", +    "Link Water": "D9E4F5", +    "Lipstick": "AB0563", +    "Lisbon Brown": "423921", +    "Livid Brown": "4D282E", +    "Loafer": "EEF4DE", +    "Loblolly": "BDC9CE", +    "Lochinvar": "2C8C84", +    "Lochmara": "007EC7", +    "Locust": "A8AF8E", +    "Log Cabin": "242A1D", +    "Logan": "AAA9CD", +    "Lola": "DFCFDB", +    "London Hue": "BEA6C3", +    "Lonestar": "6D0101", +    "Lotus": "863C3C", +    "Loulou": "460B41", +    "Lucky": "AF9F1C", +    "Lucky Point": "1A1A68", +    "Lunar Green": "3C493A", +    "Luxor Gold": "A7882C", +    "Lynch": "697E9A", +    "Mabel": "D9F7FF", +    "Macaroni and Cheese": "FFB97B", +    "Madang": "B7F0BE", +    "Madison": "09255D", +    "Madras": "3F3002", +    "Magenta / Fuchsia": "FF00FF", +    "Magic Mint": "AAF0D1", +    "Magnolia": "F8F4FF", +    "Mahogany": "4E0606", +    "Mai Tai": "B06608", +    "Maize": "F5D5A0", +    "Makara": "897D6D", +    "Mako": "444954", +    "Malachite": "0BDA51", +    "Malibu": "7DC8F7", +    "Mallard": "233418", +    "Malta": "BDB2A1", +    "Mamba": "8E8190", +    "Manatee": "8D90A1", +    "Mandalay": "AD781B", +    "Mandy": "E25465", +    "Mandys Pink": "F2C3B2", +    "Mango Tango": "E77200", +    "Manhattan": "F5C999", +    "Mantis": "74C365", +    "Mantle": "8B9C90", +    "Manz": "EEEF78", +    "Mardi Gras": "350036", +    "Marigold": "B98D28", +    "Marigold Yellow": "FBE870", +    "Mariner": "286ACD", +    "Maroon": "800000", +    "Maroon Flush": "C32148", +    "Maroon Oak": "520C17", +    "Marshland": "0B0F08", +    "Martini": "AFA09E", +    "Martinique": "363050", +    "Marzipan": "F8DB9D", +    "Masala": "403B38", +    "Matisse": "1B659D", +    "Matrix": "B05D54", +    "Matterhorn": "4E3B41", +    "Mauve": "E0B0FF", +    "Mauvelous": "F091A9", +    "Maverick": "D8C2D5", +    "Medium Carmine": "AF4035", +    "Medium Purple": "9370DB", +    "Medium Red Violet": "BB3385", +    "Melanie": "E4C2D5", +    "Melanzane": "300529", +    "Melon": "FEBAAD", +    "Melrose": "C7C1FF", +    "Mercury": "E5E5E5", +    "Merino": "F6F0E6", +    "Merlin": "413C37", +    "Merlot": "831923", +    "Metallic Bronze": "49371B", +    "Metallic Copper": "71291D", +    "Meteor": "D07D12", +    "Meteorite": "3C1F76", +    "Mexican Red": "A72525", +    "Mid Gray": "5F5F6E", +    "Midnight": "011635", +    "Midnight Blue": "003366", +    "Midnight Moss": "041004", +    "Mikado": "2D2510", +    "Milan": "FAFFA4", +    "Milano Red": "B81104", +    "Milk Punch": "FFF6D4", +    "Millbrook": "594433", +    "Mimosa": "F8FDD3", +    "Mindaro": "E3F988", +    "Mine Shaft": "323232", +    "Mineral Green": "3F5D53", +    "Ming": "36747D", +    "Minsk": "3F307F", +    "Mint Green": "98FF98", +    "Mint Julep": "F1EEC1", +    "Mint Tulip": "C4F4EB", +    "Mirage": "161928", +    "Mischka": "D1D2DD", +    "Mist Gray": "C4C4BC", +    "Mobster": "7F7589", +    "Moccaccino": "6E1D14", +    "Mocha": "782D19", +    "Mojo": "C04737", +    "Mona Lisa": "FFA194", +    "Monarch": "8B0723", +    "Mondo": "4A3C30", +    "Mongoose": "B5A27F", +    "Monsoon": "8A8389", +    "Monte Carlo": "83D0C6", +    "Monza": "C7031E", +    "Moody Blue": "7F76D3", +    "Moon Glow": "FCFEDA", +    "Moon Mist": "DCDDCC", +    "Moon Raker": "D6CEF6", +    "Morning Glory": "9EDEE0", +    "Morocco Brown": "441D00", +    "Mortar": "504351", +    "Mosque": "036A6E", +    "Moss Green": "ADDFAD", +    "Mountain Meadow": "1AB385", +    "Mountain Mist": "959396", +    "Mountbatten Pink": "997A8D", +    "Muddy Waters": "B78E5C", +    "Muesli": "AA8B5B", +    "Mulberry": "C54B8C", +    "Mulberry Wood": "5C0536", +    "Mule Fawn": "8C472F", +    "Mulled Wine": "4E4562", +    "Mustard": "FFDB58", +    "My Pink": "D69188", +    "My Sin": "FFB31F", +    "Mystic": "E2EBED", +    "Nandor": "4B5D52", +    "Napa": "ACA494", +    "Narvik": "EDF9F1", +    "Natural Gray": "8B8680", +    "Navajo White": "FFDEAD", +    "Navy Blue": "000080", +    "Nebula": "CBDBD6", +    "Negroni": "FFE2C5", +    "Neon Carrot": "FF9933", +    "Nepal": "8EABC1", +    "Neptune": "7CB7BB", +    "Nero": "140600", +    "Nevada": "646E75", +    "New Orleans": "F3D69D", +    "New York Pink": "D7837F", +    "Niagara": "06A189", +    "Night Rider": "1F120F", +    "Night Shadz": "AA375A", +    "Nile Blue": "193751", +    "Nobel": "B7B1B1", +    "Nomad": "BAB1A2", +    "Norway": "A8BD9F", +    "Nugget": "C59922", +    "Nutmeg": "81422C", +    "Nutmeg Wood Finish": "683600", +    "Oasis": "FEEFCE", +    "Observatory": "02866F", +    "Ocean Green": "41AA78", +    "Ochre": "CC7722", +    "Off Green": "E6F8F3", +    "Off Yellow": "FEF9E3", +    "Oil": "281E15", +    "Old Brick": "901E1E", +    "Old Copper": "724A2F", +    "Old Gold": "CFB53B", +    "Old Lace": "FDF5E6", +    "Old Lavender": "796878", +    "Old Rose": "C08081", +    "Olive": "808000", +    "Olive Drab": "6B8E23", +    "Olive Green": "B5B35C", +    "Olive Haze": "8B8470", +    "Olivetone": "716E10", +    "Olivine": "9AB973", +    "Onahau": "CDF4FF", +    "Onion": "2F270E", +    "Opal": "A9C6C2", +    "Opium": "8E6F70", +    "Oracle": "377475", +    "Orange": "FF681F", +    "Orange Peel": "FFA000", +    "Orange Roughy": "C45719", +    "Orange White": "FEFCED", +    "Orchid": "DA70D6", +    "Orchid White": "FFFDF3", +    "Oregon": "9B4703", +    "Orient": "015E85", +    "Oriental Pink": "C69191", +    "Orinoco": "F3FBD4", +    "Oslo Gray": "878D91", +    "Ottoman": "E9F8ED", +    "Outer Space": "2D383A", +    "Outrageous Orange": "FF6037", +    "Oxford Blue": "384555", +    "Oxley": "779E86", +    "Oyster Bay": "DAFAFF", +    "Oyster Pink": "E9CECD", +    "Paarl": "A65529", +    "Pablo": "776F61", +    "Pacific Blue": "009DC4", +    "Pacifika": "778120", +    "Paco": "411F10", +    "Padua": "ADE6C4", +    "Pale Canary": "FFFF99", +    "Pale Leaf": "C0D3B9", +    "Pale Oyster": "988D77", +    "Pale Prim": "FDFEB8", +    "Pale Rose": "FFE1F2", +    "Pale Sky": "6E7783", +    "Pale Slate": "C3BFC1", +    "Palm Green": "09230F", +    "Palm Leaf": "19330E", +    "Pampas": "F4F2EE", +    "Panache": "EAF6EE", +    "Pancho": "EDCDAB", +    "Papaya Whip": "FFEFD5", +    "Paprika": "8D0226", +    "Paradiso": "317D82", +    "Parchment": "F1E9D2", +    "Paris Daisy": "FFF46E", +    "Paris M": "26056A", +    "Paris White": "CADCD4", +    "Parsley": "134F19", +    "Pastel Green": "77DD77", +    "Pastel Pink": "FFD1DC", +    "Patina": "639A8F", +    "Pattens Blue": "DEF5FF", +    "Paua": "260368", +    "Pavlova": "D7C498", +    "Peach": "FFE5B4", +    "Peach Cream": "FFF0DB", +    "Peach Orange": "FFCC99", +    "Peach Schnapps": "FFDCD6", +    "Peach Yellow": "FADFAD", +    "Peanut": "782F16", +    "Pear": "D1E231", +    "Pearl Bush": "E8E0D5", +    "Pearl Lusta": "FCF4DC", +    "Peat": "716B56", +    "Pelorous": "3EABBF", +    "Peppermint": "E3F5E1", +    "Perano": "A9BEF2", +    "Perfume": "D0BEF8", +    "Periglacial Blue": "E1E6D6", +    "Periwinkle": "CCCCFF", +    "Periwinkle Gray": "C3CDE6", +    "Persian Blue": "1C39BB", +    "Persian Green": "00A693", +    "Persian Indigo": "32127A", +    "Persian Pink": "F77FBE", +    "Persian Plum": "701C1C", +    "Persian Red": "CC3333", +    "Persian Rose": "FE28A2", +    "Persimmon": "FF6B53", +    "Peru Tan": "7F3A02", +    "Pesto": "7C7631", +    "Petite Orchid": "DB9690", +    "Pewter": "96A8A1", +    "Pharlap": "A3807B", +    "Picasso": "FFF39D", +    "Pickled Bean": "6E4826", +    "Pickled Bluewood": "314459", +    "Picton Blue": "45B1E8", +    "Pig Pink": "FDD7E4", +    "Pigeon Post": "AFBDD9", +    "Pigment Indigo": "4B0082", +    "Pine Cone": "6D5E54", +    "Pine Glade": "C7CD90", +    "Pine Green": "01796F", +    "Pine Tree": "171F04", +    "Pink": "FFC0CB", +    "Pink Flamingo": "FF66FF", +    "Pink Flare": "E1C0C8", +    "Pink Lace": "FFDDF4", +    "Pink Lady": "FFF1D8", +    "Pink Salmon": "FF91A4", +    "Pink Swan": "BEB5B7", +    "Piper": "C96323", +    "Pipi": "FEF4CC", +    "Pippin": "FFE1DF", +    "Pirate Gold": "BA7F03", +    "Pistachio": "9DC209", +    "Pixie Green": "C0D8B6", +    "Pizazz": "FF9000", +    "Pizza": "C99415", +    "Plantation": "27504B", +    "Plum": "843179", +    "Pohutukawa": "8F021C", +    "Polar": "E5F9F6", +    "Polo Blue": "8DA8CC", +    "Pomegranate": "F34723", +    "Pompadour": "660045", +    "Porcelain": "EFF2F3", +    "Porsche": "EAAE69", +    "Port Gore": "251F4F", +    "Portafino": "FFFFB4", +    "Portage": "8B9FEE", +    "Portica": "F9E663", +    "Pot Pourri": "F5E7E2", +    "Potters Clay": "8C5738", +    "Powder Ash": "BCC9C2", +    "Powder Blue": "B0E0E6", +    "Prairie Sand": "9A3820", +    "Prelude": "D0C0E5", +    "Prim": "F0E2EC", +    "Primrose": "EDEA99", +    "Provincial Pink": "FEF5F1", +    "Prussian Blue": "003153", +    "Puce": "CC8899", +    "Pueblo": "7D2C14", +    "Puerto Rico": "3FC1AA", +    "Pumice": "C2CAC4", +    "Pumpkin": "FF7518", +    "Pumpkin Skin": "B1610B", +    "Punch": "DC4333", +    "Punga": "4D3D14", +    "Purple": "660099", +    "Purple Heart": "652DC1", +    "Purple Mountain's Majesty": "9678B6", +    "Purple Pizzazz": "FF00CC", +    "Putty": "E7CD8C", +    "Quarter Pearl Lusta": "FFFDF4", +    "Quarter Spanish White": "F7F2E1", +    "Quicksand": "BD978E", +    "Quill Gray": "D6D6D1", +    "Quincy": "623F2D", +    "Racing Green": "0C1911", +    "Radical Red": "FF355E", +    "Raffia": "EADAB8", +    "Rainee": "B9C8AC", +    "Rajah": "F7B668", +    "Rangitoto": "2E3222", +    "Rangoon Green": "1C1E13", +    "Raven": "727B89", +    "Raw Sienna": "D27D46", +    "Raw Umber": "734A12", +    "Razzle Dazzle Rose": "FF33CC", +    "Razzmatazz": "E30B5C", +    "Rebel": "3C1206", +    "Red": "FF0000", +    "Red Beech": "7B3801", +    "Red Berry": "8E0000", +    "Red Damask": "DA6A41", +    "Red Devil": "860111", +    "Red Orange": "FF3F34", +    "Red Oxide": "6E0902", +    "Red Ribbon": "ED0A3F", +    "Red Robin": "80341F", +    "Red Stage": "D05F04", +    "Red Violet": "C71585", +    "Redwood": "5D1E0F", +    "Reef": "C9FFA2", +    "Reef Gold": "9F821C", +    "Regal Blue": "013F6A", +    "Regent Gray": "86949F", +    "Regent St Blue": "AAD6E6", +    "Remy": "FEEBF3", +    "Reno Sand": "A86515", +    "Resolution Blue": "002387", +    "Revolver": "2C1632", +    "Rhino": "2E3F62", +    "Rice Cake": "FFFEF0", +    "Rice Flower": "EEFFE2", +    "Rich Gold": "A85307", +    "Rio Grande": "BBD009", +    "Ripe Lemon": "F4D81C", +    "Ripe Plum": "410056", +    "Riptide": "8BE6D8", +    "River Bed": "434C59", +    "Rob Roy": "EAC674", +    "Robin's Egg Blue": "00CCCC", +    "Rock": "4D3833", +    "Rock Blue": "9EB1CD", +    "Rock Spray": "BA450C", +    "Rodeo Dust": "C9B29B", +    "Rolling Stone": "747D83", +    "Roman": "DE6360", +    "Roman Coffee": "795D4C", +    "Romance": "FFFEFD", +    "Romantic": "FFD2B7", +    "Ronchi": "ECC54E", +    "Roof Terracotta": "A62F20", +    "Rope": "8E4D1E", +    "Rose": "FF007F", +    "Rose Bud": "FBB2A3", +    "Rose Bud Cherry": "800B47", +    "Rose Fog": "E7BCB4", +    "Rose White": "FFF6F5", +    "Rose of Sharon": "BF5500", +    "Rosewood": "65000B", +    "Roti": "C6A84B", +    "Rouge": "A23B6C", +    "Royal Blue": "4169E1", +    "Royal Heath": "AB3472", +    "Royal Purple": "6B3FA0", +    "Rum": "796989", +    "Rum Swizzle": "F9F8E4", +    "Russet": "80461B", +    "Russett": "755A57", +    "Rust": "B7410E", +    "Rustic Red": "480404", +    "Rusty Nail": "86560A", +    "Saddle": "4C3024", +    "Saddle Brown": "583401", +    "Saffron": "F4C430", +    "Saffron Mango": "F9BF58", +    "Sage": "9EA587", +    "Sahara": "B7A214", +    "Sahara Sand": "F1E788", +    "Sail": "B8E0F9", +    "Salem": "097F4B", +    "Salmon": "FF8C69", +    "Salomie": "FEDB8D", +    "Salt Box": "685E6E", +    "Saltpan": "F1F7F2", +    "Sambuca": "3A2010", +    "San Felix": "0B6207", +    "San Juan": "304B6A", +    "San Marino": "456CAC", +    "Sand Dune": "826F65", +    "Sandal": "AA8D6F", +    "Sandrift": "AB917A", +    "Sandstone": "796D62", +    "Sandwisp": "F5E7A2", +    "Sandy Beach": "FFEAC8", +    "Sandy brown": "F4A460", +    "Sangria": "92000A", +    "Sanguine Brown": "8D3D38", +    "Santa Fe": "B16D52", +    "Santas Gray": "9FA0B1", +    "Sapling": "DED4A4", +    "Sapphire": "2F519E", +    "Saratoga": "555B10", +    "Satin Linen": "E6E4D4", +    "Sauvignon": "FFF5F3", +    "Sazerac": "FFF4E0", +    "Scampi": "675FA6", +    "Scandal": "CFFAF4", +    "Scarlet": "FF2400", +    "Scarlet Gum": "431560", +    "Scarlett": "950015", +    "Scarpa Flow": "585562", +    "Schist": "A9B497", +    "School bus Yellow": "FFD800", +    "Schooner": "8B847E", +    "Science Blue": "0066CC", +    "Scooter": "2EBFD4", +    "Scorpion": "695F62", +    "Scotch Mist": "FFFBDC", +    "Screamin' Green": "66FF66", +    "Sea Buckthorn": "FBA129", +    "Sea Green": "2E8B57", +    "Sea Mist": "C5DBCA", +    "Sea Nymph": "78A39C", +    "Sea Pink": "ED989E", +    "Seagull": "80CCEA", +    "Seance": "731E8F", +    "Seashell": "F1F1F1", +    "Seashell Peach": "FFF5EE", +    "Seaweed": "1B2F11", +    "Selago": "F0EEFD", +    "Selective Yellow": "FFBA00", +    "Sepia": "704214", +    "Sepia Black": "2B0202", +    "Sepia Skin": "9E5B40", +    "Serenade": "FFF4E8", +    "Shadow": "837050", +    "Shadow Green": "9AC2B8", +    "Shady Lady": "AAA5A9", +    "Shakespeare": "4EABD1", +    "Shalimar": "FBFFBA", +    "Shamrock": "33CC99", +    "Shark": "25272C", +    "Sherpa Blue": "004950", +    "Sherwood Green": "02402C", +    "Shilo": "E8B9B3", +    "Shingle Fawn": "6B4E31", +    "Ship Cove": "788BBA", +    "Ship Gray": "3E3A44", +    "Shiraz": "B20931", +    "Shocking": "E292C0", +    "Shocking Pink": "FC0FC0", +    "Shuttle Gray": "5F6672", +    "Siam": "646A54", +    "Sidecar": "F3E7BB", +    "Silk": "BDB1A8", +    "Silver": "C0C0C0", +    "Silver Chalice": "ACACAC", +    "Silver Rust": "C9C0BB", +    "Silver Sand": "BFC1C2", +    "Silver Tree": "66B58F", +    "Sinbad": "9FD7D3", +    "Siren": "7A013A", +    "Sirocco": "718080", +    "Sisal": "D3CBBA", +    "Skeptic": "CAE6DA", +    "Sky Blue": "76D7EA", +    "Slate Gray": "708090", +    "Smalt": "003399", +    "Smalt Blue": "51808F", +    "Smoky": "605B73", +    "Snow Drift": "F7FAF7", +    "Snow Flurry": "E4FFD1", +    "Snowy Mint": "D6FFDB", +    "Snuff": "E2D8ED", +    "Soapstone": "FFFBF9", +    "Soft Amber": "D1C6B4", +    "Soft Peach": "F5EDEF", +    "Solid Pink": "893843", +    "Solitaire": "FEF8E2", +    "Solitude": "EAF6FF", +    "Sorbus": "FD7C07", +    "Sorrell Brown": "CEB98F", +    "Soya Bean": "6A6051", +    "Spanish Green": "819885", +    "Spectra": "2F5A57", +    "Spice": "6A442E", +    "Spicy Mix": "885342", +    "Spicy Mustard": "74640D", +    "Spicy Pink": "816E71", +    "Spindle": "B6D1EA", +    "Spray": "79DEEC", +    "Spring Green": "00FF7F", +    "Spring Leaves": "578363", +    "Spring Rain": "ACCBB1", +    "Spring Sun": "F6FFDC", +    "Spring Wood": "F8F6F1", +    "Sprout": "C1D7B0", +    "Spun Pearl": "AAABB7", +    "Squirrel": "8F8176", +    "St Tropaz": "2D569B", +    "Stack": "8A8F8A", +    "Star Dust": "9F9F9C", +    "Stark White": "E5D7BD", +    "Starship": "ECF245", +    "Steel Blue": "4682B4", +    "Steel Gray": "262335", +    "Stiletto": "9C3336", +    "Stonewall": "928573", +    "Storm Dust": "646463", +    "Storm Gray": "717486", +    "Stratos": "000741", +    "Straw": "D4BF8D", +    "Strikemaster": "956387", +    "Stromboli": "325D52", +    "Studio": "714AB2", +    "Submarine": "BAC7C9", +    "Sugar Cane": "F9FFF6", +    "Sulu": "C1F07C", +    "Summer Green": "96BBAB", +    "Sun": "FBAC13", +    "Sundance": "C9B35B", +    "Sundown": "FFB1B3", +    "Sunflower": "E4D422", +    "Sunglo": "E16865", +    "Sunglow": "FFCC33", +    "Sunset Orange": "FE4C40", +    "Sunshade": "FF9E2C", +    "Supernova": "FFC901", +    "Surf": "BBD7C1", +    "Surf Crest": "CFE5D2", +    "Surfie Green": "0C7A79", +    "Sushi": "87AB39", +    "Suva Gray": "888387", +    "Swamp": "001B1C", +    "Swamp Green": "ACB78E", +    "Swans Down": "DCF0EA", +    "Sweet Corn": "FBEA8C", +    "Sweet Pink": "FD9FA2", +    "Swirl": "D3CDC5", +    "Swiss Coffee": "DDD6D5", +    "Sycamore": "908D39", +    "Tabasco": "A02712", +    "Tacao": "EDB381", +    "Tacha": "D6C562", +    "Tahiti Gold": "E97C07", +    "Tahuna Sands": "EEF0C8", +    "Tall Poppy": "B32D29", +    "Tallow": "A8A589", +    "Tamarillo": "991613", +    "Tamarind": "341515", +    "Tan": "D2B48C", +    "Tan Hide": "FA9D5A", +    "Tana": "D9DCC1", +    "Tangaroa": "03163C", +    "Tangerine": "F28500", +    "Tango": "ED7A1C", +    "Tapa": "7B7874", +    "Tapestry": "B05E81", +    "Tara": "E1F6E8", +    "Tarawera": "073A50", +    "Tasman": "CFDCCF", +    "Taupe": "483C32", +    "Taupe Gray": "B3AF95", +    "Tawny Port": "692545", +    "Te Papa Green": "1E433C", +    "Tea": "C1BAB0", +    "Tea Green": "D0F0C0", +    "Teak": "B19461", +    "Teal": "008080", +    "Teal Blue": "044259", +    "Temptress": "3B000B", +    "Tenn": "CD5700", +    "Tequila": "FFE6C7", +    "Terracotta": "E2725B", +    "Texas": "F8F99C", +    "Texas Rose": "FFB555", +    "Thatch": "B69D98", +    "Thatch Green": "403D19", +    "Thistle": "D8BFD8", +    "Thistle Green": "CCCAA8", +    "Thunder": "33292F", +    "Thunderbird": "C02B18", +    "Tia Maria": "C1440E", +    "Tiara": "C3D1D1", +    "Tiber": "063537", +    "Tickle Me Pink": "FC80A5", +    "Tidal": "F1FFAD", +    "Tide": "BFB8B0", +    "Timber Green": "16322C", +    "Timberwolf": "D9D6CF", +    "Titan White": "F0EEFF", +    "Toast": "9A6E61", +    "Tobacco Brown": "715D47", +    "Toledo": "3A0020", +    "Tolopea": "1B0245", +    "Tom Thumb": "3F583B", +    "Tonys Pink": "E79F8C", +    "Topaz": "7C778A", +    "Torch Red": "FD0E35", +    "Torea Bay": "0F2D9E", +    "Tory Blue": "1450AA", +    "Tosca": "8D3F3F", +    "Totem Pole": "991B07", +    "Tower Gray": "A9BDBF", +    "Tradewind": "5FB3AC", +    "Tranquil": "E6FFFF", +    "Travertine": "FFFDE8", +    "Tree Poppy": "FC9C1D", +    "Treehouse": "3B2820", +    "Trendy Green": "7C881A", +    "Trendy Pink": "8C6495", +    "Trinidad": "E64E03", +    "Tropical Blue": "C3DDF9", +    "Tropical Rain Forest": "00755E", +    "Trout": "4A4E5A", +    "True V": "8A73D6", +    "Tuatara": "363534", +    "Tuft Bush": "FFDDCD", +    "Tulip Tree": "EAB33B", +    "Tumbleweed": "DEA681", +    "Tuna": "353542", +    "Tundora": "4A4244", +    "Turbo": "FAE600", +    "Turkish Rose": "B57281", +    "Turmeric": "CABB48", +    "Turquoise": "30D5C8", +    "Turquoise Blue": "6CDAE7", +    "Turtle Green": "2A380B", +    "Tuscany": "BD5E2E", +    "Tusk": "EEF3C3", +    "Tussock": "C5994B", +    "Tutu": "FFF1F9", +    "Twilight": "E4CFDE", +    "Twilight Blue": "EEFDFF", +    "Twine": "C2955D", +    "Tyrian Purple": "66023C", +    "Ultramarine": "120A8F", +    "Valencia": "D84437", +    "Valentino": "350E42", +    "Valhalla": "2B194F", +    "Van Cleef": "49170C", +    "Vanilla": "D1BEA8", +    "Vanilla Ice": "F3D9DF", +    "Varden": "FFF6DF", +    "Venetian Red": "72010F", +    "Venice Blue": "055989", +    "Venus": "928590", +    "Verdigris": "5D5E37", +    "Verdun Green": "495400", +    "Vermilion": "FF4D00", +    "Vesuvius": "B14A0B", +    "Victoria": "534491", +    "Vida Loca": "549019", +    "Viking": "64CCDB", +    "Vin Rouge": "983D61", +    "Viola": "CB8FA9", +    "Violent Violet": "290C5E", +    "Violet": "240A40", +    "Violet Eggplant": "991199", +    "Violet Red": "F7468A", +    "Viridian": "40826D", +    "Viridian Green": "678975", +    "Vis Vis": "FFEFA1", +    "Vista Blue": "8FD6B4", +    "Vista White": "FCF8F7", +    "Vivid Tangerine": "FF9980", +    "Vivid Violet": "803790", +    "Voodoo": "533455", +    "Vulcan": "10121D", +    "Wafer": "DECBC6", +    "Waikawa Gray": "5A6E9C", +    "Waiouru": "363C0D", +    "Walnut": "773F1A", +    "Wasabi": "788A25", +    "Water Leaf": "A1E9DE", +    "Watercourse": "056F57", +    "Waterloo ": "7B7C94", +    "Wattle": "DCD747", +    "Watusi": "FFDDCF", +    "Wax Flower": "FFC0A8", +    "We Peep": "F7DBE6", +    "Web Orange": "FFA500", +    "Wedgewood": "4E7F9E", +    "Well Read": "B43332", +    "West Coast": "625119", +    "West Side": "FF910F", +    "Westar": "DCD9D2", +    "Wewak": "F19BAB", +    "Wheat": "F5DEB3", +    "Wheatfield": "F3EDCF", +    "Whiskey": "D59A6F", +    "Whisper": "F7F5FA", +    "White": "FFFFFF", +    "White Ice": "DDF9F1", +    "White Lilac": "F8F7FC", +    "White Linen": "F8F0E8", +    "White Pointer": "FEF8FF", +    "White Rock": "EAE8D4", +    "Wild Blue Yonder": "7A89B8", +    "Wild Rice": "ECE090", +    "Wild Sand": "F4F4F4", +    "Wild Strawberry": "FF3399", +    "Wild Watermelon": "FD5B78", +    "Wild Willow": "B9C46A", +    "William": "3A686C", +    "Willow Brook": "DFECDA", +    "Willow Grove": "65745D", +    "Windsor": "3C0878", +    "Wine Berry": "591D35", +    "Winter Hazel": "D5D195", +    "Wisp Pink": "FEF4F8", +    "Wisteria": "9771B5", +    "Wistful": "A4A6D3", +    "Witch Haze": "FFFC99", +    "Wood Bark": "261105", +    "Woodland": "4D5328", +    "Woodrush": "302A0F", +    "Woodsmoke": "0C0D0F", +    "Woody Brown": "483131", +    "Xanadu": "738678", +    "Yellow": "FFFF00", +    "Yellow Green": "C5E17A", +    "Yellow Metal": "716338", +    "Yellow Orange": "FFAE42", +    "Yellow Sea": "FEA904", +    "Your Pink": "FFC3C0", +    "Yukon Gold": "7B6608", +    "Yuma": "CEC291", +    "Zambezi": "685558", +    "Zanah": "DAECD6", +    "Zest": "E5841B", +    "Zeus": "292319", +    "Ziggurat": "BFDBE2", +    "Zinnwaldite": "EBC2AF", +    "Zircon": "F4F8FF", +    "Zombie": "E4D69B", +    "Zorba": "A59B91", +    "Zuccini": "044022", +    "Zumthor": "EDF6FF" +} diff --git a/bot/resources/utilities/wtf_python_logo.jpg b/bot/resources/utilities/wtf_python_logo.jpgBinary files differ new file mode 100644 index 00000000..851d7f9a --- /dev/null +++ b/bot/resources/utilities/wtf_python_logo.jpg diff --git a/bot/utils/checks.py b/bot/utils/checks.py index 612d1ed6..8c426ed7 100644 --- a/bot/utils/checks.py +++ b/bot/utils/checks.py @@ -4,14 +4,7 @@ from collections.abc import Container, Iterable  from typing import Callable, Optional  from discord.ext.commands import ( -    BucketType, -    CheckFailure, -    Cog, -    Command, -    CommandOnCooldown, -    Context, -    Cooldown, -    CooldownMapping, +    BucketType, CheckFailure, Cog, Command, CommandOnCooldown, Context, Cooldown, CooldownMapping  )  from bot import constants diff --git a/bot/utils/halloween/spookifications.py b/bot/utils/halloween/spookifications.py index 93c5ddb9..c45ef8dc 100644 --- a/bot/utils/halloween/spookifications.py +++ b/bot/utils/halloween/spookifications.py @@ -1,8 +1,7 @@  import logging  from random import choice, randint -from PIL import Image -from PIL import ImageOps +from PIL import Image, ImageOps  log = logging.getLogger() diff --git a/docker-compose.yml b/docker-compose.yml index cef49213..34408e82 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -11,6 +11,7 @@ services:        dockerfile: Dockerfile      container_name: sir-lancebot      init: true +    tty: true      depends_on:        - redis diff --git a/poetry.lock b/poetry.lock index 289f2039..e02c2baa 100644 --- a/poetry.lock +++ b/poetry.lock @@ -90,7 +90,7 @@ tests_no_zope = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>  [[package]]  name = "backports.entry-points-selectable" -version = "1.1.0" +version = "1.1.1"  description = "Compatibility shim providing selectable entry points for older implementations"  category = "dev"  optional = false @@ -98,11 +98,26 @@ python-versions = ">=2.7"  [package.extras]  docs = ["sphinx", "jaraco.packaging (>=8.2)", "rst.linker (>=1.9)"] -testing = ["pytest (>=4.6)", "pytest-flake8", "pytest-cov", "pytest-black (>=0.3.7)", "pytest-mypy", "pytest-checkdocs (>=2.4)", "pytest-enabler (>=1.0.1)"] +testing = ["pytest", "pytest-flake8", "pytest-cov", "pytest-black (>=0.3.7)", "pytest-mypy", "pytest-checkdocs (>=2.4)", "pytest-enabler (>=1.0.1)"] + +[[package]] +name = "beautifulsoup4" +version = "4.10.0" +description = "Screen-scraping library" +category = "main" +optional = false +python-versions = ">3.0.0" + +[package.dependencies] +soupsieve = ">1.2" + +[package.extras] +html5lib = ["html5lib"] +lxml = ["lxml"]  [[package]]  name = "certifi" -version = "2021.5.30" +version = "2021.10.8"  description = "Python package for providing Mozilla's CA Bundle."  category = "main"  optional = false @@ -110,7 +125,7 @@ python-versions = "*"  [[package]]  name = "cffi" -version = "1.14.6" +version = "1.15.0"  description = "Foreign Function Interface for Python calling C code."  category = "main"  optional = false @@ -139,20 +154,31 @@ python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"  name = "colorama"  version = "0.4.4"  description = "Cross-platform colored terminal text." -category = "dev" +category = "main"  optional = false  python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"  [[package]] -name = "cycler" -version = "0.10.0" -description = "Composable style cycles" +name = "coloredlogs" +version = "15.0.1" +description = "Colored terminal output for Python's logging module"  category = "main"  optional = false -python-versions = "*" +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"  [package.dependencies] -six = "*" +humanfriendly = ">=9.1" + +[package.extras] +cron = ["capturer (>=2.4)"] + +[[package]] +name = "cycler" +version = "0.11.0" +description = "Composable style cycles" +category = "main" +optional = false +python-versions = ">=3.6"  [[package]]  name = "discord.py" @@ -175,7 +201,7 @@ type = "url"  url = "https://github.com/Rapptz/discord.py/archive/45d498c1b76deaf3b394d17ccf56112fa691d160.zip"  [[package]]  name = "distlib" -version = "0.3.2" +version = "0.3.3"  description = "Distribution utilities"  category = "dev"  optional = false @@ -191,13 +217,14 @@ python-versions = "*"  [[package]]  name = "fakeredis" -version = "1.6.0" +version = "1.6.1"  description = "Fake implementation of redis API for testing purposes."  category = "main"  optional = false  python-versions = ">=3.5"  [package.dependencies] +packaging = "*"  redis = "<3.6.0"  six = ">=1.12"  sortedcontainers = "*" @@ -208,11 +235,15 @@ lua = ["lupa"]  [[package]]  name = "filelock" -version = "3.0.12" +version = "3.3.2"  description = "A platform independent file lock."  category = "dev"  optional = false -python-versions = "*" +python-versions = ">=3.6" + +[package.extras] +docs = ["furo (>=2021.8.17b43)", "sphinx (>=4.1)", "sphinx-autodoc-typehints (>=1.12)"] +testing = ["covdefaults (>=1.2.0)", "coverage (>=4)", "pytest (>=4)", "pytest-cov", "pytest-timeout (>=1.4.2)"]  [[package]]  name = "flake8" @@ -229,14 +260,14 @@ pyflakes = ">=2.3.0,<2.4.0"  [[package]]  name = "flake8-annotations" -version = "2.6.2" +version = "2.7.0"  description = "Flake8 Type Annotation Checks"  category = "dev"  optional = false -python-versions = ">=3.6.1,<4.0.0" +python-versions = ">=3.6.2,<4.0.0"  [package.dependencies] -flake8 = ">=3.7,<4.0" +flake8 = ">=3.7,<5.0"  [[package]]  name = "flake8-bugbear" @@ -266,15 +297,20 @@ flake8 = ">=3"  pydocstyle = ">=2.1"  [[package]] -name = "flake8-import-order" -version = "0.18.1" -description = "Flake8 and pylama plugin that checks the ordering of import statements." +name = "flake8-isort" +version = "4.1.1" +description = "flake8 plugin that integrates isort ."  category = "dev"  optional = false  python-versions = "*"  [package.dependencies] -pycodestyle = "*" +flake8 = ">=3.2.1,<5" +isort = ">=4.3.5,<6" +testfixtures = ">=6.8.0,<7" + +[package.extras] +test = ["pytest-cov"]  [[package]]  name = "flake8-polyfill" @@ -300,14 +336,14 @@ flake8 = "*"  [[package]]  name = "flake8-tidy-imports" -version = "4.4.1" +version = "4.5.0"  description = "A flake8 plugin that helps you write tidier imports."  category = "dev"  optional = false  python-versions = ">=3.6"  [package.dependencies] -flake8 = ">=3.8.0,<4" +flake8 = ">=3.8.0,<5"  [[package]]  name = "flake8-todo" @@ -329,8 +365,19 @@ optional = false  python-versions = ">=3.6"  [[package]] +name = "humanfriendly" +version = "10.0" +description = "Human friendly output for text interfaces using Python" +category = "main" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" + +[package.dependencies] +pyreadline3 = {version = "*", markers = "sys_platform == \"win32\" and python_version >= \"3.8\""} + +[[package]]  name = "identify" -version = "2.2.13" +version = "2.3.5"  description = "File identification library for Python"  category = "dev"  optional = false @@ -341,13 +388,27 @@ license = ["editdistance-s"]  [[package]]  name = "idna" -version = "3.2" +version = "3.3"  description = "Internationalized Domain Names in Applications (IDNA)"  category = "main"  optional = false  python-versions = ">=3.5"  [[package]] +name = "isort" +version = "5.10.1" +description = "A Python utility / library to sort Python imports." +category = "dev" +optional = false +python-versions = ">=3.6.1,<4.0" + +[package.extras] +pipfile_deprecated_finder = ["pipreqs", "requirementslib"] +requirements_deprecated_finder = ["pipreqs", "pip-api"] +colors = ["colorama (>=0.4.3,<0.5.0)"] +plugins = ["setuptools"] + +[[package]]  name = "kiwisolver"  version = "1.3.2"  description = "A fast implementation of the Cassowary constraint solver" @@ -356,6 +417,20 @@ optional = false  python-versions = ">=3.7"  [[package]] +name = "lxml" +version = "4.6.4" +description = "Powerful and Pythonic XML processing library combining libxml2/libxslt with the ElementTree API." +category = "main" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, != 3.4.*" + +[package.extras] +cssselect = ["cssselect (>=0.7)"] +html5 = ["html5lib"] +htmlsoup = ["beautifulsoup4"] +source = ["Cython (>=0.29.7)"] + +[[package]]  name = "matplotlib"  version = "3.4.3"  description = "Python plotting package" @@ -389,7 +464,7 @@ python-versions = ">=3.5"  [[package]]  name = "multidict" -version = "5.1.0" +version = "5.2.0"  description = "multidict implementation"  category = "main"  optional = false @@ -412,6 +487,17 @@ optional = false  python-versions = ">=3.7"  [[package]] +name = "packaging" +version = "21.0" +description = "Core utilities for Python packages" +category = "main" +optional = false +python-versions = ">=3.6" + +[package.dependencies] +pyparsing = ">=2.0.2" + +[[package]]  name = "pep8-naming"  version = "0.12.1"  description = "Check PEP-8 naming conventions, plugin for flake8" @@ -425,7 +511,7 @@ flake8-polyfill = ">=1.0.2,<2"  [[package]]  name = "pillow" -version = "8.3.2" +version = "8.4.0"  description = "Python Imaging Library (Fork)"  category = "main"  optional = false @@ -433,7 +519,7 @@ python-versions = ">=3.6"  [[package]]  name = "pip-licenses" -version = "3.5.2" +version = "3.5.3"  description = "Dump the software license list of Python packages installed with pip."  category = "dev"  optional = false @@ -447,7 +533,7 @@ test = ["docutils", "pytest-cov", "pytest-pycodestyle", "pytest-runner"]  [[package]]  name = "platformdirs" -version = "2.3.0" +version = "2.4.0"  description = "A small Python module for determining appropriate platform-specific dirs, e.g. a \"user data dir\"."  category = "dev"  optional = false @@ -494,7 +580,7 @@ python-versions = "*"  [[package]]  name = "pycares" -version = "4.0.0" +version = "4.1.2"  description = "Python interface for c-ares"  category = "main"  optional = false @@ -516,7 +602,7 @@ python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"  [[package]]  name = "pycparser" -version = "2.20" +version = "2.21"  description = "C parser in Python"  category = "main"  optional = false @@ -546,11 +632,22 @@ python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"  [[package]]  name = "pyparsing" -version = "2.4.7" +version = "3.0.5"  description = "Python parsing module"  category = "main"  optional = false -python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*" +python-versions = ">=3.6" + +[package.extras] +diagrams = ["jinja2", "railroad-diagrams"] + +[[package]] +name = "pyreadline3" +version = "3.3" +description = "A python implementation of GNU readline." +category = "main" +optional = false +python-versions = "*"  [[package]]  name = "python-dateutil" @@ -565,11 +662,11 @@ six = ">=1.5"  [[package]]  name = "python-dotenv" -version = "0.15.0" -description = "Add .env support to your django/flask apps in development and deployments" +version = "0.19.1" +description = "Read key-value pairs from a .env file and set them as environment variables"  category = "dev"  optional = false -python-versions = "*" +python-versions = ">=3.5"  [package.extras]  cli = ["click (>=5.0)"] @@ -584,11 +681,14 @@ python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*"  [[package]]  name = "rapidfuzz" -version = "1.5.1" +version = "1.8.2"  description = "rapid fuzzy string matching"  category = "main"  optional = false -python-versions = ">=3.5" +python-versions = ">=2.7" + +[package.extras] +full = ["numpy"]  [[package]]  name = "redis" @@ -654,8 +754,16 @@ optional = false  python-versions = "*"  [[package]] +name = "soupsieve" +version = "2.3" +description = "A modern CSS selector implementation for Beautiful Soup." +category = "main" +optional = false +python-versions = ">=3.6" + +[[package]]  name = "taskipy" -version = "1.8.1" +version = "1.9.0"  description = "tasks runner for python projects"  category = "dev"  optional = false @@ -663,11 +771,24 @@ python-versions = ">=3.6,<4.0"  [package.dependencies]  colorama = ">=0.4.4,<0.5.0" -mslex = ">=0.3.0,<0.4.0" +mslex = {version = ">=0.3.0,<0.4.0", markers = "sys_platform == \"win32\""}  psutil = ">=5.7.2,<6.0.0"  toml = ">=0.10.0,<0.11.0"  [[package]] +name = "testfixtures" +version = "6.18.3" +description = "A collection of helpers and mock objects for unit tests and doc tests." +category = "dev" +optional = false +python-versions = "*" + +[package.extras] +build = ["setuptools-git", "wheel", "twine"] +docs = ["sphinx", "zope.component", "sybil", "twisted", "mock", "django (<2)", "django"] +test = ["pytest (>=3.6)", "pytest-cov", "pytest-django", "zope.component", "sybil", "twisted", "mock", "django (<2)", "django"] + +[[package]]  name = "toml"  version = "0.10.2"  description = "Python Library for Tom's Obvious, Minimal Language" @@ -685,7 +806,7 @@ python-versions = "*"  [[package]]  name = "urllib3" -version = "1.26.6" +version = "1.26.7"  description = "HTTP library with thread-safe connection pooling, file post, and more."  category = "main"  optional = false @@ -698,7 +819,7 @@ socks = ["PySocks (>=1.5.6,!=1.5.7,<2.0)"]  [[package]]  name = "virtualenv" -version = "20.7.2" +version = "20.10.0"  description = "Virtual Python Environment builder"  category = "dev"  optional = false @@ -707,17 +828,17 @@ python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7"  [package.dependencies]  "backports.entry-points-selectable" = ">=1.0.4"  distlib = ">=0.3.1,<1" -filelock = ">=3.0.0,<4" +filelock = ">=3.2,<4"  platformdirs = ">=2,<3"  six = ">=1.9.0,<2"  [package.extras] -docs = ["proselint (>=0.10.2)", "sphinx (>=3)", "sphinx-argparse (>=0.2.5)", "sphinx-rtd-theme (>=0.4.3)", "towncrier (>=19.9.0rc1)"] +docs = ["proselint (>=0.10.2)", "sphinx (>=3)", "sphinx-argparse (>=0.2.5)", "sphinx-rtd-theme (>=0.4.3)", "towncrier (>=21.3)"]  testing = ["coverage (>=4)", "coverage-enable-subprocess (>=1)", "flaky (>=3)", "pytest (>=4)", "pytest-env (>=0.6.2)", "pytest-freezegun (>=0.4.1)", "pytest-mock (>=2)", "pytest-randomly (>=1)", "pytest-timeout (>=1)", "packaging (>=20.0)"]  [[package]]  name = "yarl" -version = "1.6.3" +version = "1.7.2"  description = "Yet another URL library"  category = "main"  optional = false @@ -730,7 +851,7 @@ multidict = ">=4.0"  [metadata]  lock-version = "1.1"  python-versions = "^3.9" -content-hash = "9efbf6be5298ab8ace2588e218be309e105987bfdfa8317453d584a1faac4934" +content-hash = "c370c87b0425b330845ef1aad9cead2c3c3f9db7be95895895ac96fc3a2d28a6"  [metadata.files]  aiodns = [ @@ -797,59 +918,68 @@ attrs = [      {file = "attrs-21.2.0.tar.gz", hash = "sha256:ef6aaac3ca6cd92904cdd0d83f629a15f18053ec84e6432106f7a4d04ae4f5fb"},  ]  "backports.entry-points-selectable" = [ -    {file = "backports.entry_points_selectable-1.1.0-py2.py3-none-any.whl", hash = "sha256:a6d9a871cde5e15b4c4a53e3d43ba890cc6861ec1332c9c2428c92f977192acc"}, -    {file = "backports.entry_points_selectable-1.1.0.tar.gz", hash = "sha256:988468260ec1c196dab6ae1149260e2f5472c9110334e5d51adcb77867361f6a"}, +    {file = "backports.entry_points_selectable-1.1.1-py2.py3-none-any.whl", hash = "sha256:7fceed9532a7aa2bd888654a7314f864a3c16a4e710b34a58cfc0f08114c663b"}, +    {file = "backports.entry_points_selectable-1.1.1.tar.gz", hash = "sha256:914b21a479fde881635f7af5adc7f6e38d6b274be32269070c53b698c60d5386"}, +] +beautifulsoup4 = [ +    {file = "beautifulsoup4-4.10.0-py3-none-any.whl", hash = "sha256:9a315ce70049920ea4572a4055bc4bd700c940521d36fc858205ad4fcde149bf"}, +    {file = "beautifulsoup4-4.10.0.tar.gz", hash = "sha256:c23ad23c521d818955a4151a67d81580319d4bf548d3d49f4223ae041ff98891"},  ]  certifi = [ -    {file = "certifi-2021.5.30-py2.py3-none-any.whl", hash = "sha256:50b1e4f8446b06f41be7dd6338db18e0990601dce795c2b1686458aa7e8fa7d8"}, -    {file = "certifi-2021.5.30.tar.gz", hash = "sha256:2bbf76fd432960138b3ef6dda3dde0544f27cbf8546c458e60baf371917ba9ee"}, +    {file = "certifi-2021.10.8-py2.py3-none-any.whl", hash = "sha256:d62a0163eb4c2344ac042ab2bdf75399a71a2d8c7d47eac2e2ee91b9d6339569"}, +    {file = "certifi-2021.10.8.tar.gz", hash = "sha256:78884e7c1d4b00ce3cea67b44566851c4343c120abd683433ce934a68ea58872"},  ]  cffi = [ -    {file = "cffi-1.14.6-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:22b9c3c320171c108e903d61a3723b51e37aaa8c81255b5e7ce102775bd01e2c"}, -    {file = "cffi-1.14.6-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:f0c5d1acbfca6ebdd6b1e3eded8d261affb6ddcf2186205518f1428b8569bb99"}, -    {file = "cffi-1.14.6-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:99f27fefe34c37ba9875f224a8f36e31d744d8083e00f520f133cab79ad5e819"}, -    {file = "cffi-1.14.6-cp27-cp27m-win32.whl", hash = "sha256:55af55e32ae468e9946f741a5d51f9896da6b9bf0bbdd326843fec05c730eb20"}, -    {file = "cffi-1.14.6-cp27-cp27m-win_amd64.whl", hash = "sha256:7bcac9a2b4fdbed2c16fa5681356d7121ecabf041f18d97ed5b8e0dd38a80224"}, -    {file = "cffi-1.14.6-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:ed38b924ce794e505647f7c331b22a693bee1538fdf46b0222c4717b42f744e7"}, -    {file = "cffi-1.14.6-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:e22dcb48709fc51a7b58a927391b23ab37eb3737a98ac4338e2448bef8559b33"}, -    {file = "cffi-1.14.6-cp35-cp35m-macosx_10_9_x86_64.whl", hash = "sha256:aedb15f0a5a5949ecb129a82b72b19df97bbbca024081ed2ef88bd5c0a610534"}, -    {file = "cffi-1.14.6-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:48916e459c54c4a70e52745639f1db524542140433599e13911b2f329834276a"}, -    {file = "cffi-1.14.6-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:f627688813d0a4140153ff532537fbe4afea5a3dffce1f9deb7f91f848a832b5"}, -    {file = "cffi-1.14.6-cp35-cp35m-win32.whl", hash = "sha256:f0010c6f9d1a4011e429109fda55a225921e3206e7f62a0c22a35344bfd13cca"}, -    {file = "cffi-1.14.6-cp35-cp35m-win_amd64.whl", hash = "sha256:57e555a9feb4a8460415f1aac331a2dc833b1115284f7ded7278b54afc5bd218"}, -    {file = "cffi-1.14.6-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:e8c6a99be100371dbb046880e7a282152aa5d6127ae01783e37662ef73850d8f"}, -    {file = "cffi-1.14.6-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:19ca0dbdeda3b2615421d54bef8985f72af6e0c47082a8d26122adac81a95872"}, -    {file = "cffi-1.14.6-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:d950695ae4381ecd856bcaf2b1e866720e4ab9a1498cba61c602e56630ca7195"}, -    {file = "cffi-1.14.6-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e9dc245e3ac69c92ee4c167fbdd7428ec1956d4e754223124991ef29eb57a09d"}, -    {file = "cffi-1.14.6-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a8661b2ce9694ca01c529bfa204dbb144b275a31685a075ce123f12331be790b"}, -    {file = "cffi-1.14.6-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b315d709717a99f4b27b59b021e6207c64620790ca3e0bde636a6c7f14618abb"}, -    {file = "cffi-1.14.6-cp36-cp36m-win32.whl", hash = "sha256:80b06212075346b5546b0417b9f2bf467fea3bfe7352f781ffc05a8ab24ba14a"}, -    {file = "cffi-1.14.6-cp36-cp36m-win_amd64.whl", hash = "sha256:a9da7010cec5a12193d1af9872a00888f396aba3dc79186604a09ea3ee7c029e"}, -    {file = "cffi-1.14.6-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:4373612d59c404baeb7cbd788a18b2b2a8331abcc84c3ba40051fcd18b17a4d5"}, -    {file = "cffi-1.14.6-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:f10afb1004f102c7868ebfe91c28f4a712227fe4cb24974350ace1f90e1febbf"}, -    {file = "cffi-1.14.6-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:fd4305f86f53dfd8cd3522269ed7fc34856a8ee3709a5e28b2836b2db9d4cd69"}, -    {file = "cffi-1.14.6-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6d6169cb3c6c2ad50db5b868db6491a790300ade1ed5d1da29289d73bbe40b56"}, -    {file = "cffi-1.14.6-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5d4b68e216fc65e9fe4f524c177b54964af043dde734807586cf5435af84045c"}, -    {file = "cffi-1.14.6-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:33791e8a2dc2953f28b8d8d300dde42dd929ac28f974c4b4c6272cb2955cb762"}, -    {file = "cffi-1.14.6-cp37-cp37m-win32.whl", hash = "sha256:0c0591bee64e438883b0c92a7bed78f6290d40bf02e54c5bf0978eaf36061771"}, -    {file = "cffi-1.14.6-cp37-cp37m-win_amd64.whl", hash = "sha256:8eb687582ed7cd8c4bdbff3df6c0da443eb89c3c72e6e5dcdd9c81729712791a"}, -    {file = "cffi-1.14.6-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:ba6f2b3f452e150945d58f4badd92310449876c4c954836cfb1803bdd7b422f0"}, -    {file = "cffi-1.14.6-cp38-cp38-manylinux1_i686.whl", hash = "sha256:64fda793737bc4037521d4899be780534b9aea552eb673b9833b01f945904c2e"}, -    {file = "cffi-1.14.6-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:9f3e33c28cd39d1b655ed1ba7247133b6f7fc16fa16887b120c0c670e35ce346"}, -    {file = "cffi-1.14.6-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:26bb2549b72708c833f5abe62b756176022a7b9a7f689b571e74c8478ead51dc"}, -    {file = "cffi-1.14.6-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:eb687a11f0a7a1839719edd80f41e459cc5366857ecbed383ff376c4e3cc6afd"}, -    {file = "cffi-1.14.6-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:d2ad4d668a5c0645d281dcd17aff2be3212bc109b33814bbb15c4939f44181cc"}, -    {file = "cffi-1.14.6-cp38-cp38-win32.whl", hash = "sha256:487d63e1454627c8e47dd230025780e91869cfba4c753a74fda196a1f6ad6548"}, -    {file = "cffi-1.14.6-cp38-cp38-win_amd64.whl", hash = "sha256:c33d18eb6e6bc36f09d793c0dc58b0211fccc6ae5149b808da4a62660678b156"}, -    {file = "cffi-1.14.6-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:06c54a68935738d206570b20da5ef2b6b6d92b38ef3ec45c5422c0ebaf338d4d"}, -    {file = "cffi-1.14.6-cp39-cp39-manylinux1_i686.whl", hash = "sha256:f174135f5609428cc6e1b9090f9268f5c8935fddb1b25ccb8255a2d50de6789e"}, -    {file = "cffi-1.14.6-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:f3ebe6e73c319340830a9b2825d32eb6d8475c1dac020b4f0aa774ee3b898d1c"}, -    {file = "cffi-1.14.6-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3c8d896becff2fa653dc4438b54a5a25a971d1f4110b32bd3068db3722c80202"}, -    {file = "cffi-1.14.6-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:4922cd707b25e623b902c86188aca466d3620892db76c0bdd7b99a3d5e61d35f"}, -    {file = "cffi-1.14.6-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:c9e005e9bd57bc987764c32a1bee4364c44fdc11a3cc20a40b93b444984f2b87"}, -    {file = "cffi-1.14.6-cp39-cp39-win32.whl", hash = "sha256:eb9e2a346c5238a30a746893f23a9535e700f8192a68c07c0258e7ece6ff3728"}, -    {file = "cffi-1.14.6-cp39-cp39-win_amd64.whl", hash = "sha256:818014c754cd3dba7229c0f5884396264d51ffb87ec86e927ef0be140bfdb0d2"}, -    {file = "cffi-1.14.6.tar.gz", hash = "sha256:c9a875ce9d7fe32887784274dd533c57909b7b1dcadcc128a2ac21331a9765dd"}, +    {file = "cffi-1.15.0-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:c2502a1a03b6312837279c8c1bd3ebedf6c12c4228ddbad40912d671ccc8a962"}, +    {file = "cffi-1.15.0-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:23cfe892bd5dd8941608f93348c0737e369e51c100d03718f108bf1add7bd6d0"}, +    {file = "cffi-1.15.0-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:41d45de54cd277a7878919867c0f08b0cf817605e4eb94093e7516505d3c8d14"}, +    {file = "cffi-1.15.0-cp27-cp27m-win32.whl", hash = "sha256:4a306fa632e8f0928956a41fa8e1d6243c71e7eb59ffbd165fc0b41e316b2474"}, +    {file = "cffi-1.15.0-cp27-cp27m-win_amd64.whl", hash = "sha256:e7022a66d9b55e93e1a845d8c9eba2a1bebd4966cd8bfc25d9cd07d515b33fa6"}, +    {file = "cffi-1.15.0-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:14cd121ea63ecdae71efa69c15c5543a4b5fbcd0bbe2aad864baca0063cecf27"}, +    {file = "cffi-1.15.0-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:d4d692a89c5cf08a8557fdeb329b82e7bf609aadfaed6c0d79f5a449a3c7c023"}, +    {file = "cffi-1.15.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:0104fb5ae2391d46a4cb082abdd5c69ea4eab79d8d44eaaf79f1b1fd806ee4c2"}, +    {file = "cffi-1.15.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:91ec59c33514b7c7559a6acda53bbfe1b283949c34fe7440bcf917f96ac0723e"}, +    {file = "cffi-1.15.0-cp310-cp310-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:f5c7150ad32ba43a07c4479f40241756145a1f03b43480e058cfd862bf5041c7"}, +    {file = "cffi-1.15.0-cp310-cp310-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:00c878c90cb53ccfaae6b8bc18ad05d2036553e6d9d1d9dbcf323bbe83854ca3"}, +    {file = "cffi-1.15.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:abb9a20a72ac4e0fdb50dae135ba5e77880518e742077ced47eb1499e29a443c"}, +    {file = "cffi-1.15.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a5263e363c27b653a90078143adb3d076c1a748ec9ecc78ea2fb916f9b861962"}, +    {file = "cffi-1.15.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f54a64f8b0c8ff0b64d18aa76675262e1700f3995182267998c31ae974fbc382"}, +    {file = "cffi-1.15.0-cp310-cp310-win32.whl", hash = "sha256:c21c9e3896c23007803a875460fb786118f0cdd4434359577ea25eb556e34c55"}, +    {file = "cffi-1.15.0-cp310-cp310-win_amd64.whl", hash = "sha256:5e069f72d497312b24fcc02073d70cb989045d1c91cbd53979366077959933e0"}, +    {file = "cffi-1.15.0-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:64d4ec9f448dfe041705426000cc13e34e6e5bb13736e9fd62e34a0b0c41566e"}, +    {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2756c88cbb94231c7a147402476be2c4df2f6078099a6f4a480d239a8817ae39"}, +    {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3b96a311ac60a3f6be21d2572e46ce67f09abcf4d09344c49274eb9e0bf345fc"}, +    {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:75e4024375654472cc27e91cbe9eaa08567f7fbdf822638be2814ce059f58032"}, +    {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:59888172256cac5629e60e72e86598027aca6bf01fa2465bdb676d37636573e8"}, +    {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:27c219baf94952ae9d50ec19651a687b826792055353d07648a5695413e0c605"}, +    {file = "cffi-1.15.0-cp36-cp36m-win32.whl", hash = "sha256:4958391dbd6249d7ad855b9ca88fae690783a6be9e86df65865058ed81fc860e"}, +    {file = "cffi-1.15.0-cp36-cp36m-win_amd64.whl", hash = "sha256:f6f824dc3bce0edab5f427efcfb1d63ee75b6fcb7282900ccaf925be84efb0fc"}, +    {file = "cffi-1.15.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:06c48159c1abed75c2e721b1715c379fa3200c7784271b3c46df01383b593636"}, +    {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:c2051981a968d7de9dd2d7b87bcb9c939c74a34626a6e2f8181455dd49ed69e4"}, +    {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:fd8a250edc26254fe5b33be00402e6d287f562b6a5b2152dec302fa15bb3e997"}, +    {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:91d77d2a782be4274da750752bb1650a97bfd8f291022b379bb8e01c66b4e96b"}, +    {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:45db3a33139e9c8f7c09234b5784a5e33d31fd6907800b316decad50af323ff2"}, +    {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:263cc3d821c4ab2213cbe8cd8b355a7f72a8324577dc865ef98487c1aeee2bc7"}, +    {file = "cffi-1.15.0-cp37-cp37m-win32.whl", hash = "sha256:17771976e82e9f94976180f76468546834d22a7cc404b17c22df2a2c81db0c66"}, +    {file = "cffi-1.15.0-cp37-cp37m-win_amd64.whl", hash = "sha256:3415c89f9204ee60cd09b235810be700e993e343a408693e80ce7f6a40108029"}, +    {file = "cffi-1.15.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:4238e6dab5d6a8ba812de994bbb0a79bddbdf80994e4ce802b6f6f3142fcc880"}, +    {file = "cffi-1.15.0-cp38-cp38-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:0808014eb713677ec1292301ea4c81ad277b6cdf2fdd90fd540af98c0b101d20"}, +    {file = "cffi-1.15.0-cp38-cp38-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:57e9ac9ccc3101fac9d6014fba037473e4358ef4e89f8e181f8951a2c0162024"}, +    {file = "cffi-1.15.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8b6c2ea03845c9f501ed1313e78de148cd3f6cad741a75d43a29b43da27f2e1e"}, +    {file = "cffi-1.15.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:10dffb601ccfb65262a27233ac273d552ddc4d8ae1bf93b21c94b8511bffe728"}, +    {file = "cffi-1.15.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:786902fb9ba7433aae840e0ed609f45c7bcd4e225ebb9c753aa39725bb3e6ad6"}, +    {file = "cffi-1.15.0-cp38-cp38-win32.whl", hash = "sha256:da5db4e883f1ce37f55c667e5c0de439df76ac4cb55964655906306918e7363c"}, +    {file = "cffi-1.15.0-cp38-cp38-win_amd64.whl", hash = "sha256:181dee03b1170ff1969489acf1c26533710231c58f95534e3edac87fff06c443"}, +    {file = "cffi-1.15.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:45e8636704eacc432a206ac7345a5d3d2c62d95a507ec70d62f23cd91770482a"}, +    {file = "cffi-1.15.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:31fb708d9d7c3f49a60f04cf5b119aeefe5644daba1cd2a0fe389b674fd1de37"}, +    {file = "cffi-1.15.0-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:6dc2737a3674b3e344847c8686cf29e500584ccad76204efea14f451d4cc669a"}, +    {file = "cffi-1.15.0-cp39-cp39-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:74fdfdbfdc48d3f47148976f49fab3251e550a8720bebc99bf1483f5bfb5db3e"}, +    {file = "cffi-1.15.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ffaa5c925128e29efbde7301d8ecaf35c8c60ffbcd6a1ffd3a552177c8e5e796"}, +    {file = "cffi-1.15.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3f7d084648d77af029acb79a0ff49a0ad7e9d09057a9bf46596dac9514dc07df"}, +    {file = "cffi-1.15.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:ef1f279350da2c586a69d32fc8733092fd32cc8ac95139a00377841f59a3f8d8"}, +    {file = "cffi-1.15.0-cp39-cp39-win32.whl", hash = "sha256:2a23af14f408d53d5e6cd4e3d9a24ff9e05906ad574822a10563efcef137979a"}, +    {file = "cffi-1.15.0-cp39-cp39-win_amd64.whl", hash = "sha256:3773c4d81e6e818df2efbc7dd77325ca0dcb688116050fb2b3011218eda36139"}, +    {file = "cffi-1.15.0.tar.gz", hash = "sha256:920f0d66a896c2d99f0adbb391f990a84091179542c205fa53ce5787aff87954"},  ]  cfgv = [      {file = "cfgv-3.3.1-py2.py3-none-any.whl", hash = "sha256:c6a0883f3917a037485059700b9e75da2464e6c27051014ad85ba6aaa5884426"}, @@ -863,34 +993,38 @@ colorama = [      {file = "colorama-0.4.4-py2.py3-none-any.whl", hash = "sha256:9f47eda37229f68eee03b24b9748937c7dc3868f906e8ba69fbcbdd3bc5dc3e2"},      {file = "colorama-0.4.4.tar.gz", hash = "sha256:5941b2b48a20143d2267e95b1c2a7603ce057ee39fd88e7329b0c292aa16869b"},  ] +coloredlogs = [ +    {file = "coloredlogs-15.0.1-py2.py3-none-any.whl", hash = "sha256:612ee75c546f53e92e70049c9dbfcc18c935a2b9a53b66085ce9ef6a6e5c0934"}, +    {file = "coloredlogs-15.0.1.tar.gz", hash = "sha256:7c991aa71a4577af2f82600d8f8f3a89f936baeaf9b50a9c197da014e5bf16b0"}, +]  cycler = [ -    {file = "cycler-0.10.0-py2.py3-none-any.whl", hash = "sha256:1d8a5ae1ff6c5cf9b93e8811e581232ad8920aeec647c37316ceac982b08cb2d"}, -    {file = "cycler-0.10.0.tar.gz", hash = "sha256:cd7b2d1018258d7247a71425e9f26463dfb444d411c39569972f4ce586b0c9d8"}, +    {file = "cycler-0.11.0-py3-none-any.whl", hash = "sha256:3a27e95f763a428a739d2add979fa7494c912a32c17c4c38c4d5f082cad165a3"}, +    {file = "cycler-0.11.0.tar.gz", hash = "sha256:9c87405839a19696e837b3b818fed3f5f69f16f1eec1a1ad77e043dcea9c772f"},  ]  "discord.py" = []  distlib = [ -    {file = "distlib-0.3.2-py2.py3-none-any.whl", hash = "sha256:23e223426b28491b1ced97dc3bbe183027419dfc7982b4fa2f05d5f3ff10711c"}, -    {file = "distlib-0.3.2.zip", hash = "sha256:106fef6dc37dd8c0e2c0a60d3fca3e77460a48907f335fa28420463a6f799736"}, +    {file = "distlib-0.3.3-py2.py3-none-any.whl", hash = "sha256:c8b54e8454e5bf6237cc84c20e8264c3e991e824ef27e8f1e81049867d861e31"}, +    {file = "distlib-0.3.3.zip", hash = "sha256:d982d0751ff6eaaab5e2ec8e691d949ee80eddf01a62eaa96ddb11531fe16b05"},  ]  emojis = [      {file = "emojis-0.6.0-py3-none-any.whl", hash = "sha256:7da34c8a78ae262fd68cef9e2c78a3c1feb59784489eeea0f54ba1d4b7111c7c"},      {file = "emojis-0.6.0.tar.gz", hash = "sha256:bf605d1f1a27a81cd37fe82eb65781c904467f569295a541c33710b97e4225ec"},  ]  fakeredis = [ -    {file = "fakeredis-1.6.0-py3-none-any.whl", hash = "sha256:3449b306f3a85102b28f8180c24722ef966fcb1e3c744758b6f635ec80321a5c"}, -    {file = "fakeredis-1.6.0.tar.gz", hash = "sha256:11ccfc9769d718d37e45b382e64a6ba02586b622afa0371a6bd85766d72255f3"}, +    {file = "fakeredis-1.6.1-py3-none-any.whl", hash = "sha256:5eb1516f1fe1813e9da8f6c482178fc067af09f53de587ae03887ef5d9d13024"}, +    {file = "fakeredis-1.6.1.tar.gz", hash = "sha256:0d06a9384fb79da9f2164ce96e34eb9d4e2ea46215070805ea6fd3c174590b47"},  ]  filelock = [ -    {file = "filelock-3.0.12-py3-none-any.whl", hash = "sha256:929b7d63ec5b7d6b71b0fa5ac14e030b3f70b75747cef1b10da9b879fef15836"}, -    {file = "filelock-3.0.12.tar.gz", hash = "sha256:18d82244ee114f543149c66a6e0c14e9c4f8a1044b5cdaadd0f82159d6a6ff59"}, +    {file = "filelock-3.3.2-py3-none-any.whl", hash = "sha256:bb2a1c717df74c48a2d00ed625e5a66f8572a3a30baacb7657add1d7bac4097b"}, +    {file = "filelock-3.3.2.tar.gz", hash = "sha256:7afc856f74fa7006a289fd10fa840e1eebd8bbff6bffb69c26c54a0512ea8cf8"},  ]  flake8 = [      {file = "flake8-3.9.2-py2.py3-none-any.whl", hash = "sha256:bf8fd333346d844f616e8d47905ef3a3384edae6b4e9beb0c5101e25e3110907"},      {file = "flake8-3.9.2.tar.gz", hash = "sha256:07528381786f2a6237b061f6e96610a4167b226cb926e2aa2b6b1d78057c576b"},  ]  flake8-annotations = [ -    {file = "flake8-annotations-2.6.2.tar.gz", hash = "sha256:0d6cd2e770b5095f09689c9d84cc054c51b929c41a68969ea1beb4b825cac515"}, -    {file = "flake8_annotations-2.6.2-py3-none-any.whl", hash = "sha256:d10c4638231f8a50c0a597c4efce42bd7b7d85df4f620a0ddaca526138936a4f"}, +    {file = "flake8-annotations-2.7.0.tar.gz", hash = "sha256:52e53c05b0c06cac1c2dec192ea2c36e85081238add3bd99421d56f574b9479b"}, +    {file = "flake8_annotations-2.7.0-py3-none-any.whl", hash = "sha256:3edfbbfb58e404868834fe6ec3eaf49c139f64f0701259f707d043185545151e"},  ]  flake8-bugbear = [      {file = "flake8-bugbear-20.11.1.tar.gz", hash = "sha256:528020129fea2dea33a466b9d64ab650aa3e5f9ffc788b70ea4bc6cf18283538"}, @@ -900,9 +1034,9 @@ flake8-docstrings = [      {file = "flake8-docstrings-1.6.0.tar.gz", hash = "sha256:9fe7c6a306064af8e62a055c2f61e9eb1da55f84bb39caef2b84ce53708ac34b"},      {file = "flake8_docstrings-1.6.0-py2.py3-none-any.whl", hash = "sha256:99cac583d6c7e32dd28bbfbef120a7c0d1b6dde4adb5a9fd441c4227a6534bde"},  ] -flake8-import-order = [ -    {file = "flake8-import-order-0.18.1.tar.gz", hash = "sha256:a28dc39545ea4606c1ac3c24e9d05c849c6e5444a50fb7e9cdd430fc94de6e92"}, -    {file = "flake8_import_order-0.18.1-py2.py3-none-any.whl", hash = "sha256:90a80e46886259b9c396b578d75c749801a41ee969a235e163cfe1be7afd2543"}, +flake8-isort = [ +    {file = "flake8-isort-4.1.1.tar.gz", hash = "sha256:d814304ab70e6e58859bc5c3e221e2e6e71c958e7005239202fee19c24f82717"}, +    {file = "flake8_isort-4.1.1-py3-none-any.whl", hash = "sha256:c4e8b6dcb7be9b71a02e6e5d4196cefcef0f3447be51e82730fb336fff164949"},  ]  flake8-polyfill = [      {file = "flake8-polyfill-1.0.2.tar.gz", hash = "sha256:e44b087597f6da52ec6393a709e7108b2905317d0c0b744cdca6208e670d8eda"}, @@ -913,8 +1047,8 @@ flake8-string-format = [      {file = "flake8_string_format-0.3.0-py2.py3-none-any.whl", hash = "sha256:812ff431f10576a74c89be4e85b8e075a705be39bc40c4b4278b5b13e2afa9af"},  ]  flake8-tidy-imports = [ -    {file = "flake8-tidy-imports-4.4.1.tar.gz", hash = "sha256:c18b3351b998787db071e766e318da1f0bd9d5cecc69c4022a69e7aa2efb2c51"}, -    {file = "flake8_tidy_imports-4.4.1-py3-none-any.whl", hash = "sha256:631a1ba9daaedbe8bb53f6086c5a92b390e98371205259e0e311a378df8c3dc8"}, +    {file = "flake8-tidy-imports-4.5.0.tar.gz", hash = "sha256:ac637961d0f319012d099e49619f8c928e3221f74e00fe6eb89513bc64c40adb"}, +    {file = "flake8_tidy_imports-4.5.0-py3-none-any.whl", hash = "sha256:87eed94ae6a2fda6a5918d109746feadf1311e0eb8274ab7a7920f6db00a41c9"},  ]  flake8-todo = [      {file = "flake8-todo-0.7.tar.gz", hash = "sha256:6e4c5491ff838c06fe5a771b0e95ee15fc005ca57196011011280fc834a85915"}, @@ -962,13 +1096,21 @@ hiredis = [      {file = "hiredis-2.0.0-pp37-pypy37_pp73-win32.whl", hash = "sha256:f52010e0a44e3d8530437e7da38d11fb822acfb0d5b12e9cd5ba655509937ca0"},      {file = "hiredis-2.0.0.tar.gz", hash = "sha256:81d6d8e39695f2c37954d1011c0480ef7cf444d4e3ae24bc5e89ee5de360139a"},  ] +humanfriendly = [ +    {file = "humanfriendly-10.0-py2.py3-none-any.whl", hash = "sha256:1697e1a8a8f550fd43c2865cd84542fc175a61dcb779b6fee18cf6b6ccba1477"}, +    {file = "humanfriendly-10.0.tar.gz", hash = "sha256:6b0b831ce8f15f7300721aa49829fc4e83921a9a301cc7f606be6686a2288ddc"}, +]  identify = [ -    {file = "identify-2.2.13-py2.py3-none-any.whl", hash = "sha256:7199679b5be13a6b40e6e19ea473e789b11b4e3b60986499b1f589ffb03c217c"}, -    {file = "identify-2.2.13.tar.gz", hash = "sha256:7bc6e829392bd017236531963d2d937d66fc27cadc643ac0aba2ce9f26157c79"}, +    {file = "identify-2.3.5-py2.py3-none-any.whl", hash = "sha256:ba945bddb4322394afcf3f703fa68eda08a6acc0f99d9573eb2be940aa7b9bba"}, +    {file = "identify-2.3.5.tar.gz", hash = "sha256:6f0368ba0f21c199645a331beb7425d5374376e71bc149e9cb55e45cb45f832d"},  ]  idna = [ -    {file = "idna-3.2-py3-none-any.whl", hash = "sha256:14475042e284991034cb48e06f6851428fb14c4dc953acd9be9a5e95c7b6dd7a"}, -    {file = "idna-3.2.tar.gz", hash = "sha256:467fbad99067910785144ce333826c71fb0e63a425657295239737f7ecd125f3"}, +    {file = "idna-3.3-py3-none-any.whl", hash = "sha256:84d9dd047ffa80596e0f246e2eab0b391788b0503584e8945f2368256d2735ff"}, +    {file = "idna-3.3.tar.gz", hash = "sha256:9d643ff0a55b762d5cdb124b8eaa99c66322e2157b69160bc32796e824360e6d"}, +] +isort = [ +    {file = "isort-5.10.1-py3-none-any.whl", hash = "sha256:6f62d78e2f89b4500b080fe3a81690850cd254227f27f75c3a0c491a1f351ba7"}, +    {file = "isort-5.10.1.tar.gz", hash = "sha256:e8443a5e7a020e9d7f97f1d7d9cd17c88bcb3bc7e218bf9cf5095fe550be2951"},  ]  kiwisolver = [      {file = "kiwisolver-1.3.2-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:1d819553730d3c2724582124aee8a03c846ec4362ded1034c16fb3ef309264e6"}, @@ -1016,6 +1158,68 @@ kiwisolver = [      {file = "kiwisolver-1.3.2-pp37-pypy37_pp73-win_amd64.whl", hash = "sha256:bcadb05c3d4794eb9eee1dddf1c24215c92fb7b55a80beae7a60530a91060560"},      {file = "kiwisolver-1.3.2.tar.gz", hash = "sha256:fc4453705b81d03568d5b808ad8f09c77c47534f6ac2e72e733f9ca4714aa75c"},  ] +lxml = [ +    {file = "lxml-4.6.4-cp27-cp27m-macosx_10_14_x86_64.whl", hash = "sha256:bbf2dc330bd44bfc0254ab37677ec60f7c7ecea55ad8ba1b8b2ea7bf20c265f5"}, +    {file = "lxml-4.6.4-cp27-cp27m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:b667c51682fe9b9788c69465956baa8b6999531876ccedcafc895c74ad716cd8"}, +    {file = "lxml-4.6.4-cp27-cp27m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:72e730d33fe2e302fd07285f14624fca5e5e2fb2bb4fb2c3941e318c41c443d1"}, +    {file = "lxml-4.6.4-cp27-cp27m-win32.whl", hash = "sha256:433df8c7dde0f9e41cbf4f36b0829d50a378116ef5e962ba3881f2f5f025c7be"}, +    {file = "lxml-4.6.4-cp27-cp27m-win_amd64.whl", hash = "sha256:35752ee40f7bbf6adc9ff4e1f4b84794a3593736dcce80db32e3c2aa85e294ac"}, +    {file = "lxml-4.6.4-cp27-cp27mu-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:5ff5bb2a198ea67403bb6818705e9a4f90e0313f2215428ec51001ce56d939fb"}, +    {file = "lxml-4.6.4-cp27-cp27mu-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:9b87727561c1150c0cc91c5d9d389448b37a7d15f0ba939ed3d1acb2f11bf6c5"}, +    {file = "lxml-4.6.4-cp310-cp310-macosx_10_14_x86_64.whl", hash = "sha256:45fdb2899c755138722797161547a40b3e2a06feda620cc41195ee7e97806d81"}, +    {file = "lxml-4.6.4-cp310-cp310-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:38b9de0de3aa689fe9fb9877ae1be1e83b8cf9621f7e62049d0436b9ecf4ad64"}, +    {file = "lxml-4.6.4-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_24_aarch64.whl", hash = "sha256:662523cd2a0246740225c7e32531f2e766544122e58bee70e700a024cfc0cf81"}, +    {file = "lxml-4.6.4-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:4aa349c5567651f34d4eaae7de6ed5b523f6d70a288f9c6fbac22d13a0784e04"}, +    {file = "lxml-4.6.4-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:08eb9200d88b376a8ed5e50f1dc1d1a45b49305169674002a3b5929943390591"}, +    {file = "lxml-4.6.4-cp310-cp310-win32.whl", hash = "sha256:bdc224f216ead849e902151112efef6e96c41ee1322e15d4e5f7c8a826929aee"}, +    {file = "lxml-4.6.4-cp310-cp310-win_amd64.whl", hash = "sha256:ab6db93a2b6b66cbf62b4e4a7135f476e708e8c5c990d186584142c77d7f975a"}, +    {file = "lxml-4.6.4-cp35-cp35m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:50790313df028aa05cf22be9a8da033b86c42fa32523e4fd944827b482b17bf0"}, +    {file = "lxml-4.6.4-cp35-cp35m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:6764998345552b1dfc9326a932d2bad6367c6b37a176bb73ada6b9486bf602f7"}, +    {file = "lxml-4.6.4-cp35-cp35m-win32.whl", hash = "sha256:543b239b191bb3b6d9bef5f09f1fb2be5b7eb09ab4d386aa655e4d53fbe9ff47"}, +    {file = "lxml-4.6.4-cp35-cp35m-win_amd64.whl", hash = "sha256:a75c1ad05eedb1a3ff2a34a52a4f0836cfaa892e12796ba39a7732c82701eff4"}, +    {file = "lxml-4.6.4-cp36-cp36m-macosx_10_14_x86_64.whl", hash = "sha256:47e955112ce64241fdb357acf0216081f9f3255b3ac9c502ca4b3323ec1ca558"}, +    {file = "lxml-4.6.4-cp36-cp36m-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:20d7c8d90d449c6a353b15ee0459abae8395dbe59ad01e406ccbf30cd81c6f98"}, +    {file = "lxml-4.6.4-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:240db6f3228d26e3c6f4fad914b9ddaaf8707254e8b3efd564dc680c8ec3c264"}, +    {file = "lxml-4.6.4-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:351482da8dd028834028537f08724b1de22d40dcf3bb723b469446564f409074"}, +    {file = "lxml-4.6.4-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:e678a643177c0e5ec947b645fa7bc84260dfb9b6bf8fb1fdd83008dfc2ca5928"}, +    {file = "lxml-4.6.4-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:15d0381feb56f08f78c5cc4fc385ddfe0bde1456e37f54a9322833371aec4060"}, +    {file = "lxml-4.6.4-cp36-cp36m-win32.whl", hash = "sha256:4ba74afe5ee5cb5e28d83b513a6e8f0875fda1dc1a9aea42cc0065f029160d2a"}, +    {file = "lxml-4.6.4-cp36-cp36m-win_amd64.whl", hash = "sha256:9c91a73971a922c13070fd8fa5a114c858251791ba2122a941e6aa781c713e44"}, +    {file = "lxml-4.6.4-cp37-cp37m-macosx_10_14_x86_64.whl", hash = "sha256:6020c70ff695106bf80651953a23e37718ef1fee9abd060dcad8e32ab2dc13f3"}, +    {file = "lxml-4.6.4-cp37-cp37m-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:f5dd358536b8a964bf6bd48de038754c1609e72e5f17f5d21efe2dda17594dbf"}, +    {file = "lxml-4.6.4-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_24_aarch64.whl", hash = "sha256:7ae7089d81fc502df4b217ad77f03c54039fe90dac0acbe70448d7e53bfbc57e"}, +    {file = "lxml-4.6.4-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:80d10d53d3184837445ff8562021bdd37f57c4cadacbf9d8726cc16220a00d54"}, +    {file = "lxml-4.6.4-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:e95da348d57eb448d226a44b868ff2ca5786fbcbe417ac99ff62d0a7d724b9c7"}, +    {file = "lxml-4.6.4-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:ffd65cfa33fed01735c82aca640fde4cc63f0414775cba11e06f84fae2085a6e"}, +    {file = "lxml-4.6.4-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:877666418598f6cb289546c77ff87590cfd212f903b522b0afa0b9fb73b3ccfb"}, +    {file = "lxml-4.6.4-cp37-cp37m-win32.whl", hash = "sha256:e91d24623e747eeb2d8121f4a94c6a7ad27dc48e747e2dc95bfe88632bd028a2"}, +    {file = "lxml-4.6.4-cp37-cp37m-win_amd64.whl", hash = "sha256:4ec9a80dd5704ecfde54319b6964368daf02848c8954d3bacb9b64d1c7659159"}, +    {file = "lxml-4.6.4-cp38-cp38-macosx_10_14_x86_64.whl", hash = "sha256:2901625f4a878a055d275beedc20ba9cb359cefc4386a967222fee29eb236038"}, +    {file = "lxml-4.6.4-cp38-cp38-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:b567178a74a2261345890eac66fbf394692a6e002709d329f28a673ca6042473"}, +    {file = "lxml-4.6.4-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_24_aarch64.whl", hash = "sha256:4717123f7c11c81e0da69989e5a64079c3f402b0efeb4c6241db6c369d657bd8"}, +    {file = "lxml-4.6.4-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:cf201bf5594d1aab139fe53e3fca457e4f8204a5bbd65d48ab3b82a16f517868"}, +    {file = "lxml-4.6.4-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:a77a3470ba37e11872c75ca95baf9b3312133a3d5a5dc720803b23098c653976"}, +    {file = "lxml-4.6.4-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:619c6d2b552bba00491e96c0518aad94002651c108a0f7364ff2d7798812c00e"}, +    {file = "lxml-4.6.4-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:601f0ab75538b280aaf1e720eb9d68d4fa104ac274e1e9e6971df488f4dcdb0f"}, +    {file = "lxml-4.6.4-cp38-cp38-win32.whl", hash = "sha256:75d3c5bbc0ddbad03bb68b9be638599f67e4b98ed3dcd0fec9f6f39e41ee96cb"}, +    {file = "lxml-4.6.4-cp38-cp38-win_amd64.whl", hash = "sha256:4341d135f5660db10184963d9c3418c3e28d7f868aaf8b11a323ebf85813f7f4"}, +    {file = "lxml-4.6.4-cp39-cp39-macosx_10_14_x86_64.whl", hash = "sha256:9db24803fa71e3305fe4a7812782b708da21a0b774b130dd1860cf40a6d7a3ee"}, +    {file = "lxml-4.6.4-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:afd60230ad9d8bcba005945ec3a343722f09e0b7f8ae804246e5d2cfc6bd71a6"}, +    {file = "lxml-4.6.4-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_24_aarch64.whl", hash = "sha256:0c15e1cd55055956e77b0732270f1c6005850696bc3ef3e03d01e78af84eaa42"}, +    {file = "lxml-4.6.4-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:6d422b3c729737d8a39279a25fa156c983a56458f8b2f97661ee6fb22b80b1d6"}, +    {file = "lxml-4.6.4-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:2eb90f6ec3c236ef2f1bb38aee7c0d23e77d423d395af6326e7cca637519a4cb"}, +    {file = "lxml-4.6.4-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:51a0e5d243687596f46e24e464121d4b232ad772e2d1785b2a2c0eb413c285d4"}, +    {file = "lxml-4.6.4-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:d43bd68714049c84e297c005456a15ecdec818f7b5aa5868c8b0a865cfb78a44"}, +    {file = "lxml-4.6.4-cp39-cp39-win32.whl", hash = "sha256:ee9e4b07b0eba4b6a521509e9e1877476729c1243246b6959de697ebea739643"}, +    {file = "lxml-4.6.4-cp39-cp39-win_amd64.whl", hash = "sha256:48eaac2991b3036175b42ee8d3c23f4cca13f2be8426bf29401a690ab58c88f4"}, +    {file = "lxml-4.6.4-pp37-pypy37_pp73-macosx_10_14_x86_64.whl", hash = "sha256:2b06a91cf7b8acea7793006e4ae50646cef0fe35ce5acd4f5cb1c77eb228e4a1"}, +    {file = "lxml-4.6.4-pp37-pypy37_pp73-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:523f195948a1ba4f9f5b7294d83c6cd876547dc741820750a7e5e893a24bbe38"}, +    {file = "lxml-4.6.4-pp37-pypy37_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:b0ca0ada9d3bc18bd6f611bd001a28abdd49ab9698bd6d717f7f5394c8e94628"}, +    {file = "lxml-4.6.4-pp38-pypy38_pp73-macosx_10_14_x86_64.whl", hash = "sha256:197b7cb7a753cf553a45115739afd8458464a28913da00f5c525063f94cd3f48"}, +    {file = "lxml-4.6.4-pp38-pypy38_pp73-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:6298f5b42a26581206ef63fffa97c754245d329414108707c525512a5197f2ba"}, +    {file = "lxml-4.6.4-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:0b12c95542f04d10cba46b3ff28ea52ea56995b78cf918f0b11b05e75812bb79"}, +    {file = "lxml-4.6.4.tar.gz", hash = "sha256:daf9bd1fee31f1c7a5928b3e1059e09a8d683ea58fb3ffc773b6c88cb8d1399c"}, +]  matplotlib = [      {file = "matplotlib-3.4.3-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:5c988bb43414c7c2b0a31bd5187b4d27fd625c080371b463a6d422047df78913"},      {file = "matplotlib-3.4.3-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:f1c5efc278d996af8a251b2ce0b07bbeccb821f25c8c9846bdcb00ffc7f158aa"}, @@ -1048,43 +1252,78 @@ mslex = [      {file = "mslex-0.3.0.tar.gz", hash = "sha256:4a1ac3f25025cad78ad2fe499dd16d42759f7a3801645399cce5c404415daa97"},  ]  multidict = [ -    {file = "multidict-5.1.0-cp36-cp36m-macosx_10_14_x86_64.whl", hash = "sha256:b7993704f1a4b204e71debe6095150d43b2ee6150fa4f44d6d966ec356a8d61f"}, -    {file = "multidict-5.1.0-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:9dd6e9b1a913d096ac95d0399bd737e00f2af1e1594a787e00f7975778c8b2bf"}, -    {file = "multidict-5.1.0-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:f21756997ad8ef815d8ef3d34edd98804ab5ea337feedcd62fb52d22bf531281"}, -    {file = "multidict-5.1.0-cp36-cp36m-manylinux2014_i686.whl", hash = "sha256:1ab820665e67373de5802acae069a6a05567ae234ddb129f31d290fc3d1aa56d"}, -    {file = "multidict-5.1.0-cp36-cp36m-manylinux2014_ppc64le.whl", hash = "sha256:9436dc58c123f07b230383083855593550c4d301d2532045a17ccf6eca505f6d"}, -    {file = "multidict-5.1.0-cp36-cp36m-manylinux2014_s390x.whl", hash = "sha256:830f57206cc96ed0ccf68304141fec9481a096c4d2e2831f311bde1c404401da"}, -    {file = "multidict-5.1.0-cp36-cp36m-manylinux2014_x86_64.whl", hash = "sha256:2e68965192c4ea61fff1b81c14ff712fc7dc15d2bd120602e4a3494ea6584224"}, -    {file = "multidict-5.1.0-cp36-cp36m-win32.whl", hash = "sha256:2f1a132f1c88724674271d636e6b7351477c27722f2ed789f719f9e3545a3d26"}, -    {file = "multidict-5.1.0-cp36-cp36m-win_amd64.whl", hash = "sha256:3a4f32116f8f72ecf2a29dabfb27b23ab7cdc0ba807e8459e59a93a9be9506f6"}, -    {file = "multidict-5.1.0-cp37-cp37m-macosx_10_14_x86_64.whl", hash = "sha256:46c73e09ad374a6d876c599f2328161bcd95e280f84d2060cf57991dec5cfe76"}, -    {file = "multidict-5.1.0-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:018132dbd8688c7a69ad89c4a3f39ea2f9f33302ebe567a879da8f4ca73f0d0a"}, -    {file = "multidict-5.1.0-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:4b186eb7d6ae7c06eb4392411189469e6a820da81447f46c0072a41c748ab73f"}, -    {file = "multidict-5.1.0-cp37-cp37m-manylinux2014_i686.whl", hash = "sha256:3a041b76d13706b7fff23b9fc83117c7b8fe8d5fe9e6be45eee72b9baa75f348"}, -    {file = "multidict-5.1.0-cp37-cp37m-manylinux2014_ppc64le.whl", hash = "sha256:051012ccee979b2b06be928a6150d237aec75dd6bf2d1eeeb190baf2b05abc93"}, -    {file = "multidict-5.1.0-cp37-cp37m-manylinux2014_s390x.whl", hash = "sha256:6a4d5ce640e37b0efcc8441caeea8f43a06addace2335bd11151bc02d2ee31f9"}, -    {file = "multidict-5.1.0-cp37-cp37m-manylinux2014_x86_64.whl", hash = "sha256:5cf3443199b83ed9e955f511b5b241fd3ae004e3cb81c58ec10f4fe47c7dce37"}, -    {file = "multidict-5.1.0-cp37-cp37m-win32.whl", hash = "sha256:f200755768dc19c6f4e2b672421e0ebb3dd54c38d5a4f262b872d8cfcc9e93b5"}, -    {file = "multidict-5.1.0-cp37-cp37m-win_amd64.whl", hash = "sha256:05c20b68e512166fddba59a918773ba002fdd77800cad9f55b59790030bab632"}, -    {file = "multidict-5.1.0-cp38-cp38-macosx_10_14_x86_64.whl", hash = "sha256:54fd1e83a184e19c598d5e70ba508196fd0bbdd676ce159feb412a4a6664f952"}, -    {file = "multidict-5.1.0-cp38-cp38-manylinux1_i686.whl", hash = "sha256:0e3c84e6c67eba89c2dbcee08504ba8644ab4284863452450520dad8f1e89b79"}, -    {file = "multidict-5.1.0-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:dc862056f76443a0db4509116c5cd480fe1b6a2d45512a653f9a855cc0517456"}, -    {file = "multidict-5.1.0-cp38-cp38-manylinux2014_i686.whl", hash = "sha256:0e929169f9c090dae0646a011c8b058e5e5fb391466016b39d21745b48817fd7"}, -    {file = "multidict-5.1.0-cp38-cp38-manylinux2014_ppc64le.whl", hash = "sha256:d81eddcb12d608cc08081fa88d046c78afb1bf8107e6feab5d43503fea74a635"}, -    {file = "multidict-5.1.0-cp38-cp38-manylinux2014_s390x.whl", hash = "sha256:585fd452dd7782130d112f7ddf3473ffdd521414674c33876187e101b588738a"}, -    {file = "multidict-5.1.0-cp38-cp38-manylinux2014_x86_64.whl", hash = "sha256:37e5438e1c78931df5d3c0c78ae049092877e5e9c02dd1ff5abb9cf27a5914ea"}, -    {file = "multidict-5.1.0-cp38-cp38-win32.whl", hash = "sha256:07b42215124aedecc6083f1ce6b7e5ec5b50047afa701f3442054373a6deb656"}, -    {file = "multidict-5.1.0-cp38-cp38-win_amd64.whl", hash = "sha256:929006d3c2d923788ba153ad0de8ed2e5ed39fdbe8e7be21e2f22ed06c6783d3"}, -    {file = "multidict-5.1.0-cp39-cp39-macosx_10_14_x86_64.whl", hash = "sha256:b797515be8743b771aa868f83563f789bbd4b236659ba52243b735d80b29ed93"}, -    {file = "multidict-5.1.0-cp39-cp39-manylinux1_i686.whl", hash = "sha256:d5c65bdf4484872c4af3150aeebe101ba560dcfb34488d9a8ff8dbcd21079647"}, -    {file = "multidict-5.1.0-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:b47a43177a5e65b771b80db71e7be76c0ba23cc8aa73eeeb089ed5219cdbe27d"}, -    {file = "multidict-5.1.0-cp39-cp39-manylinux2014_i686.whl", hash = "sha256:806068d4f86cb06af37cd65821554f98240a19ce646d3cd24e1c33587f313eb8"}, -    {file = "multidict-5.1.0-cp39-cp39-manylinux2014_ppc64le.whl", hash = "sha256:46dd362c2f045095c920162e9307de5ffd0a1bfbba0a6e990b344366f55a30c1"}, -    {file = "multidict-5.1.0-cp39-cp39-manylinux2014_s390x.whl", hash = "sha256:ace010325c787c378afd7f7c1ac66b26313b3344628652eacd149bdd23c68841"}, -    {file = "multidict-5.1.0-cp39-cp39-manylinux2014_x86_64.whl", hash = "sha256:ecc771ab628ea281517e24fd2c52e8f31c41e66652d07599ad8818abaad38cda"}, -    {file = "multidict-5.1.0-cp39-cp39-win32.whl", hash = "sha256:fc13a9524bc18b6fb6e0dbec3533ba0496bbed167c56d0aabefd965584557d80"}, -    {file = "multidict-5.1.0-cp39-cp39-win_amd64.whl", hash = "sha256:7df80d07818b385f3129180369079bd6934cf70469f99daaebfac89dca288359"}, -    {file = "multidict-5.1.0.tar.gz", hash = "sha256:25b4e5f22d3a37ddf3effc0710ba692cfc792c2b9edfb9c05aefe823256e84d5"}, +    {file = "multidict-5.2.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:3822c5894c72e3b35aae9909bef66ec83e44522faf767c0ad39e0e2de11d3b55"}, +    {file = "multidict-5.2.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:28e6d883acd8674887d7edc896b91751dc2d8e87fbdca8359591a13872799e4e"}, +    {file = "multidict-5.2.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:b61f85101ef08cbbc37846ac0e43f027f7844f3fade9b7f6dd087178caedeee7"}, +    {file = "multidict-5.2.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d9b668c065968c5979fe6b6fa6760bb6ab9aeb94b75b73c0a9c1acf6393ac3bf"}, +    {file = "multidict-5.2.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:517d75522b7b18a3385726b54a081afd425d4f41144a5399e5abd97ccafdf36b"}, +    {file = "multidict-5.2.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1b4ac3ba7a97b35a5ccf34f41b5a8642a01d1e55454b699e5e8e7a99b5a3acf5"}, +    {file = "multidict-5.2.0-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:df23c83398715b26ab09574217ca21e14694917a0c857e356fd39e1c64f8283f"}, +    {file = "multidict-5.2.0-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:e58a9b5cc96e014ddf93c2227cbdeca94b56a7eb77300205d6e4001805391747"}, +    {file = "multidict-5.2.0-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:f76440e480c3b2ca7f843ff8a48dc82446b86ed4930552d736c0bac507498a52"}, +    {file = "multidict-5.2.0-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:cfde464ca4af42a629648c0b0d79b8f295cf5b695412451716531d6916461628"}, +    {file = "multidict-5.2.0-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:0fed465af2e0eb6357ba95795d003ac0bdb546305cc2366b1fc8f0ad67cc3fda"}, +    {file = "multidict-5.2.0-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:b70913cbf2e14275013be98a06ef4b412329fe7b4f83d64eb70dce8269ed1e1a"}, +    {file = "multidict-5.2.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:a5635bcf1b75f0f6ef3c8a1ad07b500104a971e38d3683167b9454cb6465ac86"}, +    {file = "multidict-5.2.0-cp310-cp310-win32.whl", hash = "sha256:77f0fb7200cc7dedda7a60912f2059086e29ff67cefbc58d2506638c1a9132d7"}, +    {file = "multidict-5.2.0-cp310-cp310-win_amd64.whl", hash = "sha256:9416cf11bcd73c861267e88aea71e9fcc35302b3943e45e1dbb4317f91a4b34f"}, +    {file = "multidict-5.2.0-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:fd77c8f3cba815aa69cb97ee2b2ef385c7c12ada9c734b0f3b32e26bb88bbf1d"}, +    {file = "multidict-5.2.0-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:98ec9aea6223adf46999f22e2c0ab6cf33f5914be604a404f658386a8f1fba37"}, +    {file = "multidict-5.2.0-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e5283c0a00f48e8cafcecadebfa0ed1dac8b39e295c7248c44c665c16dc1138b"}, +    {file = "multidict-5.2.0-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:5f79c19c6420962eb17c7e48878a03053b7ccd7b69f389d5831c0a4a7f1ac0a1"}, +    {file = "multidict-5.2.0-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:e4a67f1080123de76e4e97a18d10350df6a7182e243312426d508712e99988d4"}, +    {file = "multidict-5.2.0-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:94b117e27efd8e08b4046c57461d5a114d26b40824995a2eb58372b94f9fca02"}, +    {file = "multidict-5.2.0-cp36-cp36m-musllinux_1_1_aarch64.whl", hash = "sha256:2e77282fd1d677c313ffcaddfec236bf23f273c4fba7cdf198108f5940ae10f5"}, +    {file = "multidict-5.2.0-cp36-cp36m-musllinux_1_1_i686.whl", hash = "sha256:116347c63ba049c1ea56e157fa8aa6edaf5e92925c9b64f3da7769bdfa012858"}, +    {file = "multidict-5.2.0-cp36-cp36m-musllinux_1_1_ppc64le.whl", hash = "sha256:dc3a866cf6c13d59a01878cd806f219340f3e82eed514485e094321f24900677"}, +    {file = "multidict-5.2.0-cp36-cp36m-musllinux_1_1_s390x.whl", hash = "sha256:ac42181292099d91217a82e3fa3ce0e0ddf3a74fd891b7c2b347a7f5aa0edded"}, +    {file = "multidict-5.2.0-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:f0bb0973f42ffcb5e3537548e0767079420aefd94ba990b61cf7bb8d47f4916d"}, +    {file = "multidict-5.2.0-cp36-cp36m-win32.whl", hash = "sha256:ea21d4d5104b4f840b91d9dc8cbc832aba9612121eaba503e54eaab1ad140eb9"}, +    {file = "multidict-5.2.0-cp36-cp36m-win_amd64.whl", hash = "sha256:e6453f3cbeb78440747096f239d282cc57a2997a16b5197c9bc839099e1633d0"}, +    {file = "multidict-5.2.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:d3def943bfd5f1c47d51fd324df1e806d8da1f8e105cc7f1c76a1daf0f7e17b0"}, +    {file = "multidict-5.2.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:35591729668a303a02b06e8dba0eb8140c4a1bfd4c4b3209a436a02a5ac1de11"}, +    {file = "multidict-5.2.0-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ce8cacda0b679ebc25624d5de66c705bc53dcc7c6f02a7fb0f3ca5e227d80422"}, +    {file = "multidict-5.2.0-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:baf1856fab8212bf35230c019cde7c641887e3fc08cadd39d32a421a30151ea3"}, +    {file = "multidict-5.2.0-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:a43616aec0f0d53c411582c451f5d3e1123a68cc7b3475d6f7d97a626f8ff90d"}, +    {file = "multidict-5.2.0-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:25cbd39a9029b409167aa0a20d8a17f502d43f2efebfe9e3ac019fe6796c59ac"}, +    {file = "multidict-5.2.0-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:0a2cbcfbea6dc776782a444db819c8b78afe4db597211298dd8b2222f73e9cd0"}, +    {file = "multidict-5.2.0-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:3d2d7d1fff8e09d99354c04c3fd5b560fb04639fd45926b34e27cfdec678a704"}, +    {file = "multidict-5.2.0-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:a37e9a68349f6abe24130846e2f1d2e38f7ddab30b81b754e5a1fde32f782b23"}, +    {file = "multidict-5.2.0-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:637c1896497ff19e1ee27c1c2c2ddaa9f2d134bbb5e0c52254361ea20486418d"}, +    {file = "multidict-5.2.0-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:9815765f9dcda04921ba467957be543423e5ec6a1136135d84f2ae092c50d87b"}, +    {file = "multidict-5.2.0-cp37-cp37m-win32.whl", hash = "sha256:8b911d74acdc1fe2941e59b4f1a278a330e9c34c6c8ca1ee21264c51ec9b67ef"}, +    {file = "multidict-5.2.0-cp37-cp37m-win_amd64.whl", hash = "sha256:380b868f55f63d048a25931a1632818f90e4be71d2081c2338fcf656d299949a"}, +    {file = "multidict-5.2.0-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:e7d81ce5744757d2f05fc41896e3b2ae0458464b14b5a2c1e87a6a9d69aefaa8"}, +    {file = "multidict-5.2.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:2d1d55cdf706ddc62822d394d1df53573d32a7a07d4f099470d3cb9323b721b6"}, +    {file = "multidict-5.2.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:a4771d0d0ac9d9fe9e24e33bed482a13dfc1256d008d101485fe460359476065"}, +    {file = "multidict-5.2.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:da7d57ea65744d249427793c042094c4016789eb2562576fb831870f9c878d9e"}, +    {file = "multidict-5.2.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:cdd68778f96216596218b4e8882944d24a634d984ee1a5a049b300377878fa7c"}, +    {file = "multidict-5.2.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:ecc99bce8ee42dcad15848c7885197d26841cb24fa2ee6e89d23b8993c871c64"}, +    {file = "multidict-5.2.0-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:067150fad08e6f2dd91a650c7a49ba65085303fcc3decbd64a57dc13a2733031"}, +    {file = "multidict-5.2.0-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:78c106b2b506b4d895ddc801ff509f941119394b89c9115580014127414e6c2d"}, +    {file = "multidict-5.2.0-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:e6c4fa1ec16e01e292315ba76eb1d012c025b99d22896bd14a66628b245e3e01"}, +    {file = "multidict-5.2.0-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:b227345e4186809d31f22087d0265655114af7cda442ecaf72246275865bebe4"}, +    {file = "multidict-5.2.0-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:06560fbdcf22c9387100979e65b26fba0816c162b888cb65b845d3def7a54c9b"}, +    {file = "multidict-5.2.0-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:7878b61c867fb2df7a95e44b316f88d5a3742390c99dfba6c557a21b30180cac"}, +    {file = "multidict-5.2.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:246145bff76cc4b19310f0ad28bd0769b940c2a49fc601b86bfd150cbd72bb22"}, +    {file = "multidict-5.2.0-cp38-cp38-win32.whl", hash = "sha256:c30ac9f562106cd9e8071c23949a067b10211917fdcb75b4718cf5775356a940"}, +    {file = "multidict-5.2.0-cp38-cp38-win_amd64.whl", hash = "sha256:f19001e790013ed580abfde2a4465388950728861b52f0da73e8e8a9418533c0"}, +    {file = "multidict-5.2.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:c1ff762e2ee126e6f1258650ac641e2b8e1f3d927a925aafcfde943b77a36d24"}, +    {file = "multidict-5.2.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:bd6c9c50bf2ad3f0448edaa1a3b55b2e6866ef8feca5d8dbec10ec7c94371d21"}, +    {file = "multidict-5.2.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:fc66d4016f6e50ed36fb39cd287a3878ffcebfa90008535c62e0e90a7ab713ae"}, +    {file = "multidict-5.2.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a9acb76d5f3dd9421874923da2ed1e76041cb51b9337fd7f507edde1d86535d6"}, +    {file = "multidict-5.2.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:dfc924a7e946dd3c6360e50e8f750d51e3ef5395c95dc054bc9eab0f70df4f9c"}, +    {file = "multidict-5.2.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:32fdba7333eb2351fee2596b756d730d62b5827d5e1ab2f84e6cbb287cc67fe0"}, +    {file = "multidict-5.2.0-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:b9aad49466b8d828b96b9e3630006234879c8d3e2b0a9d99219b3121bc5cdb17"}, +    {file = "multidict-5.2.0-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:93de39267c4c676c9ebb2057e98a8138bade0d806aad4d864322eee0803140a0"}, +    {file = "multidict-5.2.0-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:f9bef5cff994ca3026fcc90680e326d1a19df9841c5e3d224076407cc21471a1"}, +    {file = "multidict-5.2.0-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:5f841c4f14331fd1e36cbf3336ed7be2cb2a8f110ce40ea253e5573387db7621"}, +    {file = "multidict-5.2.0-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:38ba256ee9b310da6a1a0f013ef4e422fca30a685bcbec86a969bd520504e341"}, +    {file = "multidict-5.2.0-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:3bc3b1621b979621cee9f7b09f024ec76ec03cc365e638126a056317470bde1b"}, +    {file = "multidict-5.2.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:6ee908c070020d682e9b42c8f621e8bb10c767d04416e2ebe44e37d0f44d9ad5"}, +    {file = "multidict-5.2.0-cp39-cp39-win32.whl", hash = "sha256:1c7976cd1c157fa7ba5456ae5d31ccdf1479680dc9b8d8aa28afabc370df42b8"}, +    {file = "multidict-5.2.0-cp39-cp39-win_amd64.whl", hash = "sha256:c9631c642e08b9fff1c6255487e62971d8b8e821808ddd013d8ac058087591ac"}, +    {file = "multidict-5.2.0.tar.gz", hash = "sha256:0dd1c93edb444b33ba2274b66f63def8a327d607c6c790772f448a53b6ea59ce"},  ]  nodeenv = [      {file = "nodeenv-1.6.0-py2.py3-none-any.whl", hash = "sha256:621e6b7076565ddcacd2db0294c0381e01fd28945ab36bcf00f41c5daf63bef7"}, @@ -1120,72 +1359,64 @@ numpy = [      {file = "numpy-1.21.1-pp37-pypy37_pp73-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:2d4d1de6e6fb3d28781c73fbde702ac97f03d79e4ffd6598b880b2d95d62ead4"},      {file = "numpy-1.21.1.zip", hash = "sha256:dff4af63638afcc57a3dfb9e4b26d434a7a602d225b42d746ea7fe2edf1342fd"},  ] +packaging = [ +    {file = "packaging-21.0-py3-none-any.whl", hash = "sha256:c86254f9220d55e31cc94d69bade760f0847da8000def4dfe1c6b872fd14ff14"}, +    {file = "packaging-21.0.tar.gz", hash = "sha256:7dc96269f53a4ccec5c0670940a4281106dd0bb343f47b7471f779df49c2fbe7"}, +]  pep8-naming = [      {file = "pep8-naming-0.12.1.tar.gz", hash = "sha256:bb2455947757d162aa4cad55dba4ce029005cd1692f2899a21d51d8630ca7841"},      {file = "pep8_naming-0.12.1-py2.py3-none-any.whl", hash = "sha256:4a8daeaeb33cfcde779309fc0c9c0a68a3bbe2ad8a8308b763c5068f86eb9f37"},  ]  pillow = [ -    {file = "Pillow-8.3.2-cp310-cp310-macosx_10_10_universal2.whl", hash = "sha256:c691b26283c3a31594683217d746f1dad59a7ae1d4cfc24626d7a064a11197d4"}, -    {file = "Pillow-8.3.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:f514c2717012859ccb349c97862568fdc0479aad85b0270d6b5a6509dbc142e2"}, -    {file = "Pillow-8.3.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:be25cb93442c6d2f8702c599b51184bd3ccd83adebd08886b682173e09ef0c3f"}, -    {file = "Pillow-8.3.2-cp310-cp310-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d675a876b295afa114ca8bf42d7f86b5fb1298e1b6bb9a24405a3f6c8338811c"}, -    {file = "Pillow-8.3.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:59697568a0455764a094585b2551fd76bfd6b959c9f92d4bdec9d0e14616303a"}, -    {file = "Pillow-8.3.2-cp310-cp310-win32.whl", hash = "sha256:2d5e9dc0bf1b5d9048a94c48d0813b6c96fccfa4ccf276d9c36308840f40c228"}, -    {file = "Pillow-8.3.2-cp310-cp310-win_amd64.whl", hash = "sha256:11c27e74bab423eb3c9232d97553111cc0be81b74b47165f07ebfdd29d825875"}, -    {file = "Pillow-8.3.2-cp36-cp36m-macosx_10_10_x86_64.whl", hash = "sha256:11eb7f98165d56042545c9e6db3ce394ed8b45089a67124298f0473b29cb60b2"}, -    {file = "Pillow-8.3.2-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2f23b2d3079522fdf3c09de6517f625f7a964f916c956527bed805ac043799b8"}, -    {file = "Pillow-8.3.2-cp36-cp36m-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:19ec4cfe4b961edc249b0e04b5618666c23a83bc35842dea2bfd5dfa0157f81b"}, -    {file = "Pillow-8.3.2-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e5a31c07cea5edbaeb4bdba6f2b87db7d3dc0f446f379d907e51cc70ea375629"}, -    {file = "Pillow-8.3.2-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:15ccb81a6ffc57ea0137f9f3ac2737ffa1d11f786244d719639df17476d399a7"}, -    {file = "Pillow-8.3.2-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:8f284dc1695caf71a74f24993b7c7473d77bc760be45f776a2c2f4e04c170550"}, -    {file = "Pillow-8.3.2-cp36-cp36m-win32.whl", hash = "sha256:4abc247b31a98f29e5224f2d31ef15f86a71f79c7f4d2ac345a5d551d6393073"}, -    {file = "Pillow-8.3.2-cp36-cp36m-win_amd64.whl", hash = "sha256:a048dad5ed6ad1fad338c02c609b862dfaa921fcd065d747194a6805f91f2196"}, -    {file = "Pillow-8.3.2-cp37-cp37m-macosx_10_10_x86_64.whl", hash = "sha256:06d1adaa284696785375fa80a6a8eb309be722cf4ef8949518beb34487a3df71"}, -    {file = "Pillow-8.3.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bd24054aaf21e70a51e2a2a5ed1183560d3a69e6f9594a4bfe360a46f94eba83"}, -    {file = "Pillow-8.3.2-cp37-cp37m-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:27a330bf7014ee034046db43ccbb05c766aa9e70b8d6c5260bfc38d73103b0ba"}, -    {file = "Pillow-8.3.2-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:13654b521fb98abdecec105ea3fb5ba863d1548c9b58831dd5105bb3873569f1"}, -    {file = "Pillow-8.3.2-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:a1bd983c565f92779be456ece2479840ec39d386007cd4ae83382646293d681b"}, -    {file = "Pillow-8.3.2-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:4326ea1e2722f3dc00ed77c36d3b5354b8fb7399fb59230249ea6d59cbed90da"}, -    {file = "Pillow-8.3.2-cp37-cp37m-win32.whl", hash = "sha256:085a90a99404b859a4b6c3daa42afde17cb3ad3115e44a75f0d7b4a32f06a6c9"}, -    {file = "Pillow-8.3.2-cp37-cp37m-win_amd64.whl", hash = "sha256:18a07a683805d32826c09acfce44a90bf474e6a66ce482b1c7fcd3757d588df3"}, -    {file = "Pillow-8.3.2-cp38-cp38-macosx_10_10_x86_64.whl", hash = "sha256:4e59e99fd680e2b8b11bbd463f3c9450ab799305d5f2bafb74fefba6ac058616"}, -    {file = "Pillow-8.3.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:4d89a2e9219a526401015153c0e9dd48319ea6ab9fe3b066a20aa9aee23d9fd3"}, -    {file = "Pillow-8.3.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:56fd98c8294f57636084f4b076b75f86c57b2a63a8410c0cd172bc93695ee979"}, -    {file = "Pillow-8.3.2-cp38-cp38-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2b11c9d310a3522b0fd3c35667914271f570576a0e387701f370eb39d45f08a4"}, -    {file = "Pillow-8.3.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0412516dcc9de9b0a1e0ae25a280015809de8270f134cc2c1e32c4eeb397cf30"}, -    {file = "Pillow-8.3.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:bcb04ff12e79b28be6c9988f275e7ab69f01cc2ba319fb3114f87817bb7c74b6"}, -    {file = "Pillow-8.3.2-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:0b9911ec70731711c3b6ebcde26caea620cbdd9dcb73c67b0730c8817f24711b"}, -    {file = "Pillow-8.3.2-cp38-cp38-win32.whl", hash = "sha256:ce2e5e04bb86da6187f96d7bab3f93a7877830981b37f0287dd6479e27a10341"}, -    {file = "Pillow-8.3.2-cp38-cp38-win_amd64.whl", hash = "sha256:35d27687f027ad25a8d0ef45dd5208ef044c588003cdcedf05afb00dbc5c2deb"}, -    {file = "Pillow-8.3.2-cp39-cp39-macosx_10_10_x86_64.whl", hash = "sha256:04835e68ef12904bc3e1fd002b33eea0779320d4346082bd5b24bec12ad9c3e9"}, -    {file = "Pillow-8.3.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:10e00f7336780ca7d3653cf3ac26f068fa11b5a96894ea29a64d3dc4b810d630"}, -    {file = "Pillow-8.3.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2cde7a4d3687f21cffdf5bb171172070bb95e02af448c4c8b2f223d783214056"}, -    {file = "Pillow-8.3.2-cp39-cp39-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1c3ff00110835bdda2b1e2b07f4a2548a39744bb7de5946dc8e95517c4fb2ca6"}, -    {file = "Pillow-8.3.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:35d409030bf3bd05fa66fb5fdedc39c521b397f61ad04309c90444e893d05f7d"}, -    {file = "Pillow-8.3.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:6bff50ba9891be0a004ef48828e012babaaf7da204d81ab9be37480b9020a82b"}, -    {file = "Pillow-8.3.2-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:7dbfbc0020aa1d9bc1b0b8bcf255a7d73f4ad0336f8fd2533fcc54a4ccfb9441"}, -    {file = "Pillow-8.3.2-cp39-cp39-win32.whl", hash = "sha256:963ebdc5365d748185fdb06daf2ac758116deecb2277ec5ae98139f93844bc09"}, -    {file = "Pillow-8.3.2-cp39-cp39-win_amd64.whl", hash = "sha256:cc9d0dec711c914ed500f1d0d3822868760954dce98dfb0b7382a854aee55d19"}, -    {file = "Pillow-8.3.2-pp36-pypy36_pp73-macosx_10_10_x86_64.whl", hash = "sha256:2c661542c6f71dfd9dc82d9d29a8386287e82813b0375b3a02983feac69ef864"}, -    {file = "Pillow-8.3.2-pp36-pypy36_pp73-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:548794f99ff52a73a156771a0402f5e1c35285bd981046a502d7e4793e8facaa"}, -    {file = "Pillow-8.3.2-pp36-pypy36_pp73-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:8b68f565a4175e12e68ca900af8910e8fe48aaa48fd3ca853494f384e11c8bcd"}, -    {file = "Pillow-8.3.2-pp36-pypy36_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:838eb85de6d9307c19c655c726f8d13b8b646f144ca6b3771fa62b711ebf7624"}, -    {file = "Pillow-8.3.2-pp36-pypy36_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:feb5db446e96bfecfec078b943cc07744cc759893cef045aa8b8b6d6aaa8274e"}, -    {file = "Pillow-8.3.2-pp37-pypy37_pp73-macosx_10_10_x86_64.whl", hash = "sha256:fc0db32f7223b094964e71729c0361f93db43664dd1ec86d3df217853cedda87"}, -    {file = "Pillow-8.3.2-pp37-pypy37_pp73-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:fd4fd83aa912d7b89b4b4a1580d30e2a4242f3936882a3f433586e5ab97ed0d5"}, -    {file = "Pillow-8.3.2-pp37-pypy37_pp73-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:d0c8ebbfd439c37624db98f3877d9ed12c137cadd99dde2d2eae0dab0bbfc355"}, -    {file = "Pillow-8.3.2-pp37-pypy37_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6cb3dd7f23b044b0737317f892d399f9e2f0b3a02b22b2c692851fb8120d82c6"}, -    {file = "Pillow-8.3.2-pp37-pypy37_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a66566f8a22561fc1a88dc87606c69b84fa9ce724f99522cf922c801ec68f5c1"}, -    {file = "Pillow-8.3.2-pp37-pypy37_pp73-win_amd64.whl", hash = "sha256:ce651ca46d0202c302a535d3047c55a0131a720cf554a578fc1b8a2aff0e7d96"}, -    {file = "Pillow-8.3.2.tar.gz", hash = "sha256:dde3f3ed8d00c72631bc19cbfff8ad3b6215062a5eed402381ad365f82f0c18c"}, +    {file = "Pillow-8.4.0-cp310-cp310-macosx_10_10_universal2.whl", hash = "sha256:81f8d5c81e483a9442d72d182e1fb6dcb9723f289a57e8030811bac9ea3fef8d"}, +    {file = "Pillow-8.4.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:3f97cfb1e5a392d75dd8b9fd274d205404729923840ca94ca45a0af57e13dbe6"}, +    {file = "Pillow-8.4.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:eb9fc393f3c61f9054e1ed26e6fe912c7321af2f41ff49d3f83d05bacf22cc78"}, +    {file = "Pillow-8.4.0-cp310-cp310-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d82cdb63100ef5eedb8391732375e6d05993b765f72cb34311fab92103314649"}, +    {file = "Pillow-8.4.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:62cc1afda735a8d109007164714e73771b499768b9bb5afcbbee9d0ff374b43f"}, +    {file = "Pillow-8.4.0-cp310-cp310-win32.whl", hash = "sha256:e3dacecfbeec9a33e932f00c6cd7996e62f53ad46fbe677577394aaa90ee419a"}, +    {file = "Pillow-8.4.0-cp310-cp310-win_amd64.whl", hash = "sha256:620582db2a85b2df5f8a82ddeb52116560d7e5e6b055095f04ad828d1b0baa39"}, +    {file = "Pillow-8.4.0-cp36-cp36m-macosx_10_10_x86_64.whl", hash = "sha256:1bc723b434fbc4ab50bb68e11e93ce5fb69866ad621e3c2c9bdb0cd70e345f55"}, +    {file = "Pillow-8.4.0-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:72cbcfd54df6caf85cc35264c77ede902452d6df41166010262374155947460c"}, +    {file = "Pillow-8.4.0-cp36-cp36m-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:70ad9e5c6cb9b8487280a02c0ad8a51581dcbbe8484ce058477692a27c151c0a"}, +    {file = "Pillow-8.4.0-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:25a49dc2e2f74e65efaa32b153527fc5ac98508d502fa46e74fa4fd678ed6645"}, +    {file = "Pillow-8.4.0-cp36-cp36m-win32.whl", hash = "sha256:93ce9e955cc95959df98505e4608ad98281fff037350d8c2671c9aa86bcf10a9"}, +    {file = "Pillow-8.4.0-cp36-cp36m-win_amd64.whl", hash = "sha256:2e4440b8f00f504ee4b53fe30f4e381aae30b0568193be305256b1462216feff"}, +    {file = "Pillow-8.4.0-cp37-cp37m-macosx_10_10_x86_64.whl", hash = "sha256:8c803ac3c28bbc53763e6825746f05cc407b20e4a69d0122e526a582e3b5e153"}, +    {file = "Pillow-8.4.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c8a17b5d948f4ceeceb66384727dde11b240736fddeda54ca740b9b8b1556b29"}, +    {file = "Pillow-8.4.0-cp37-cp37m-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1394a6ad5abc838c5cd8a92c5a07535648cdf6d09e8e2d6df916dfa9ea86ead8"}, +    {file = "Pillow-8.4.0-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:792e5c12376594bfcb986ebf3855aa4b7c225754e9a9521298e460e92fb4a488"}, +    {file = "Pillow-8.4.0-cp37-cp37m-win32.whl", hash = "sha256:d99ec152570e4196772e7a8e4ba5320d2d27bf22fdf11743dd882936ed64305b"}, +    {file = "Pillow-8.4.0-cp37-cp37m-win_amd64.whl", hash = "sha256:7b7017b61bbcdd7f6363aeceb881e23c46583739cb69a3ab39cb384f6ec82e5b"}, +    {file = "Pillow-8.4.0-cp38-cp38-macosx_10_10_x86_64.whl", hash = "sha256:d89363f02658e253dbd171f7c3716a5d340a24ee82d38aab9183f7fdf0cdca49"}, +    {file = "Pillow-8.4.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:0a0956fdc5defc34462bb1c765ee88d933239f9a94bc37d132004775241a7585"}, +    {file = "Pillow-8.4.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5b7bb9de00197fb4261825c15551adf7605cf14a80badf1761d61e59da347779"}, +    {file = "Pillow-8.4.0-cp38-cp38-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:72b9e656e340447f827885b8d7a15fc8c4e68d410dc2297ef6787eec0f0ea409"}, +    {file = "Pillow-8.4.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a5a4532a12314149d8b4e4ad8ff09dde7427731fcfa5917ff16d0291f13609df"}, +    {file = "Pillow-8.4.0-cp38-cp38-win32.whl", hash = "sha256:82aafa8d5eb68c8463b6e9baeb4f19043bb31fefc03eb7b216b51e6a9981ae09"}, +    {file = "Pillow-8.4.0-cp38-cp38-win_amd64.whl", hash = "sha256:066f3999cb3b070a95c3652712cffa1a748cd02d60ad7b4e485c3748a04d9d76"}, +    {file = "Pillow-8.4.0-cp39-cp39-macosx_10_10_x86_64.whl", hash = "sha256:5503c86916d27c2e101b7f71c2ae2cddba01a2cf55b8395b0255fd33fa4d1f1a"}, +    {file = "Pillow-8.4.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:4acc0985ddf39d1bc969a9220b51d94ed51695d455c228d8ac29fcdb25810e6e"}, +    {file = "Pillow-8.4.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0b052a619a8bfcf26bd8b3f48f45283f9e977890263e4571f2393ed8898d331b"}, +    {file = "Pillow-8.4.0-cp39-cp39-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:493cb4e415f44cd601fcec11c99836f707bb714ab03f5ed46ac25713baf0ff20"}, +    {file = "Pillow-8.4.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b8831cb7332eda5dc89b21a7bce7ef6ad305548820595033a4b03cf3091235ed"}, +    {file = "Pillow-8.4.0-cp39-cp39-win32.whl", hash = "sha256:5e9ac5f66616b87d4da618a20ab0a38324dbe88d8a39b55be8964eb520021e02"}, +    {file = "Pillow-8.4.0-cp39-cp39-win_amd64.whl", hash = "sha256:3eb1ce5f65908556c2d8685a8f0a6e989d887ec4057326f6c22b24e8a172c66b"}, +    {file = "Pillow-8.4.0-pp36-pypy36_pp73-macosx_10_10_x86_64.whl", hash = "sha256:ddc4d832a0f0b4c52fff973a0d44b6c99839a9d016fe4e6a1cb8f3eea96479c2"}, +    {file = "Pillow-8.4.0-pp36-pypy36_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:9a3e5ddc44c14042f0844b8cf7d2cd455f6cc80fd7f5eefbe657292cf601d9ad"}, +    {file = "Pillow-8.4.0-pp36-pypy36_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c70e94281588ef053ae8998039610dbd71bc509e4acbc77ab59d7d2937b10698"}, +    {file = "Pillow-8.4.0-pp37-pypy37_pp73-macosx_10_10_x86_64.whl", hash = "sha256:3862b7256046fcd950618ed22d1d60b842e3a40a48236a5498746f21189afbbc"}, +    {file = "Pillow-8.4.0-pp37-pypy37_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a4901622493f88b1a29bd30ec1a2f683782e57c3c16a2dbc7f2595ba01f639df"}, +    {file = "Pillow-8.4.0-pp37-pypy37_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:84c471a734240653a0ec91dec0996696eea227eafe72a33bd06c92697728046b"}, +    {file = "Pillow-8.4.0-pp37-pypy37_pp73-win_amd64.whl", hash = "sha256:244cf3b97802c34c41905d22810846802a3329ddcb93ccc432870243211c79fc"}, +    {file = "Pillow-8.4.0.tar.gz", hash = "sha256:b8e2f83c56e141920c39464b852de3719dfbfb6e3c99a2d8da0edf4fb33176ed"},  ]  pip-licenses = [ -    {file = "pip-licenses-3.5.2.tar.gz", hash = "sha256:c5e984f461b34ad04dafa151d0048eb9d049e3d6439966c6440bb6b53ad077b6"}, -    {file = "pip_licenses-3.5.2-py3-none-any.whl", hash = "sha256:62deafc82d5dccea1a4cab55172706e02f228abcd67f4d53e382fcb1497e9b62"}, +    {file = "pip-licenses-3.5.3.tar.gz", hash = "sha256:f44860e00957b791c6c6005a3328f2d5eaeee96ddb8e7d87d4b0aa25b02252e4"}, +    {file = "pip_licenses-3.5.3-py3-none-any.whl", hash = "sha256:59c148d6a03784bf945d232c0dc0e9de4272a3675acaa0361ad7712398ca86ba"},  ]  platformdirs = [ -    {file = "platformdirs-2.3.0-py3-none-any.whl", hash = "sha256:8003ac87717ae2c7ee1ea5a84a1a61e87f3fbd16eb5aadba194ea30a9019f648"}, -    {file = "platformdirs-2.3.0.tar.gz", hash = "sha256:15b056538719b1c94bdaccb29e5f81879c7f7f0f4a153f46086d155dffcd4f0f"}, +    {file = "platformdirs-2.4.0-py3-none-any.whl", hash = "sha256:8868bbe3c3c80d42f20156f22e7131d2fb321f5bc86a2a345375c6481a67021d"}, +    {file = "platformdirs-2.4.0.tar.gz", hash = "sha256:367a5e80b3d04d2428ffa76d33f124cf11e8fff2acdaa9b43d545f5c7d661ef2"},  ]  pre-commit = [      {file = "pre_commit-2.15.0-py2.py3-none-any.whl", hash = "sha256:a4ed01000afcb484d9eb8d504272e642c4c4099bbad3a6b27e519bd6a3e928a6"}, @@ -1225,47 +1456,45 @@ ptable = [      {file = "PTable-0.9.2.tar.gz", hash = "sha256:aa7fc151cb40f2dabcd2275ba6f7fd0ff8577a86be3365cd3fb297cbe09cc292"},  ]  pycares = [ -    {file = "pycares-4.0.0-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:db5a533111a3cfd481e7e4fb2bf8bef69f4fa100339803e0504dd5aecafb96a5"}, -    {file = "pycares-4.0.0-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:fdff88393c25016f417770d82678423fc7a56995abb2df3d2a1e55725db6977d"}, -    {file = "pycares-4.0.0-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:0aa97f900a7ffb259be77d640006585e2a907b0cd4edeee0e85cf16605995d5a"}, -    {file = "pycares-4.0.0-cp36-cp36m-manylinux2010_i686.whl", hash = "sha256:a34b0e3e693dceb60b8a1169668d606c75cb100ceba0a2df53c234a0eb067fbc"}, -    {file = "pycares-4.0.0-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:7661d6bbd51a337e7373cb356efa8be9b4655fda484e068f9455e939aec8d54e"}, -    {file = "pycares-4.0.0-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:57315b8eb8fdbc56b3ad4932bc4b17132bb7c7fd2bd590f7fb84b6b522098aa9"}, -    {file = "pycares-4.0.0-cp36-cp36m-win32.whl", hash = "sha256:dca9dc58845a9d083f302732a3130c68ded845ad5d463865d464e53c75a3dd45"}, -    {file = "pycares-4.0.0-cp36-cp36m-win_amd64.whl", hash = "sha256:c95c964d5dd307e104b44b193095c67bb6b10c9eda1ffe7d44ab7a9e84c476d9"}, -    {file = "pycares-4.0.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:26e67e4f81c80a5955dcf6193f3d9bee3c491fc0056299b383b84d792252fba4"}, -    {file = "pycares-4.0.0-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:cd3011ffd5e1ad55880f7256791dbab9c43ebeda260474a968f19cd0319e1aef"}, -    {file = "pycares-4.0.0-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:1b959dd5921d207d759d421eece1b60416df33a7f862465739d5f2c363c2f523"}, -    {file = "pycares-4.0.0-cp37-cp37m-manylinux2010_i686.whl", hash = "sha256:6f258c1b74c048a9501a25f732f11b401564005e5e3c18f1ca6cad0c3dc0fb19"}, -    {file = "pycares-4.0.0-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:b17ef48729786e62b574c6431f675f4cb02b27691b49e7428a605a50cd59c072"}, -    {file = "pycares-4.0.0-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:82b3259cb590ddd107a6d2dc52da2a2e9a986bf242e893d58c786af2f8191047"}, -    {file = "pycares-4.0.0-cp37-cp37m-win32.whl", hash = "sha256:4876fc790ae32832ae270c4a010a1a77e12ddf8d8e6ad70ad0b0a9d506c985f7"}, -    {file = "pycares-4.0.0-cp37-cp37m-win_amd64.whl", hash = "sha256:f60c04c5561b1ddf85ca4e626943cc09d7fb684e1adb22abb632095415a40fd7"}, -    {file = "pycares-4.0.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:615406013cdcd1b445e5d1a551d276c6200b3abe77e534f8a7f7e1551208d14f"}, -    {file = "pycares-4.0.0-cp38-cp38-manylinux1_i686.whl", hash = "sha256:6580aef5d1b29a88c3d72fe73c691eacfd454f86e74d3fdd18f4bad8e8def98b"}, -    {file = "pycares-4.0.0-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:8ebb3ba0485f66cae8eed7ce3e9ed6f2c0bfd5e7319d5d0fbbb511064f17e1d4"}, -    {file = "pycares-4.0.0-cp38-cp38-manylinux2010_i686.whl", hash = "sha256:c5362b7690ca481440f6b98395ac6df06aa50518ccb183c560464d1e5e2ab5d4"}, -    {file = "pycares-4.0.0-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:eb60be66accc9a9ea1018b591a1f5800cba83491d07e9acc8c56bc6e6607ab54"}, -    {file = "pycares-4.0.0-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:44896d6e191a6b5a914dbe3aa7c748481bf6ad19a9df33c1e76f8f2dc33fc8f0"}, -    {file = "pycares-4.0.0-cp38-cp38-win32.whl", hash = "sha256:09b28fc7bc2cc05f7f69bf1636ddf46086e0a1837b62961e2092fcb40477320d"}, -    {file = "pycares-4.0.0-cp38-cp38-win_amd64.whl", hash = "sha256:d4a5081e232c1d181883dcac4675807f3a6cf33911c4173fbea00c0523687ed4"}, -    {file = "pycares-4.0.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:103353577a6266a53e71bfee4cf83825f1401fefa60f0fb8bdec35f13be6a5f2"}, -    {file = "pycares-4.0.0-cp39-cp39-manylinux1_i686.whl", hash = "sha256:ad6caf580ee69806fc6534be93ddbb6e99bf94296d79ab351c37b2992b17abfd"}, -    {file = "pycares-4.0.0-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:3d5e50c95849f6905d2a9dbf02ed03f82580173e3c5604a39e2ad054185631f1"}, -    {file = "pycares-4.0.0-cp39-cp39-manylinux2010_i686.whl", hash = "sha256:53bc4f181b19576499b02cea4b45391e8dcbe30abd4cd01492f66bfc15615a13"}, -    {file = "pycares-4.0.0-cp39-cp39-manylinux2010_x86_64.whl", hash = "sha256:d52f9c725d2a826d5ffa37681eb07ffb996bfe21788590ef257664a3898fc0b5"}, -    {file = "pycares-4.0.0-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:3c7fb8d34ee11971c39acfaf98d0fac66725385ccef3bfe1b174c92b210e1aa4"}, -    {file = "pycares-4.0.0-cp39-cp39-win32.whl", hash = "sha256:e9773e07684a55f54657df05237267611a77b294ec3bacb5f851c4ffca38a465"}, -    {file = "pycares-4.0.0-cp39-cp39-win_amd64.whl", hash = "sha256:38e54037f36c149146ff15f17a4a963fbdd0f9871d4a21cd94ff9f368140f57e"}, -    {file = "pycares-4.0.0.tar.gz", hash = "sha256:d0154fc5753b088758fbec9bc137e1b24bb84fc0c6a09725c8bac25a342311cd"}, +    {file = "pycares-4.1.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:71b99b9e041ae3356b859822c511f286f84c8889ec9ed1fbf6ac30fb4da13e4c"}, +    {file = "pycares-4.1.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c000942f5fc64e6e046aa61aa53b629b576ba11607d108909727c3c8f211a157"}, +    {file = "pycares-4.1.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:b0e50ddc78252f2e2b6b5f2c73e5b2449dfb6bea7a5a0e21dfd1e2bcc9e17382"}, +    {file = "pycares-4.1.2-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:6831e963a910b0a8cbdd2750ffcdf5f2bb0edb3f53ca69ff18484de2cc3807c4"}, +    {file = "pycares-4.1.2-cp310-cp310-win32.whl", hash = "sha256:ad7b28e1b6bc68edd3d678373fa3af84e39d287090434f25055d21b4716b2fc6"}, +    {file = "pycares-4.1.2-cp310-cp310-win_amd64.whl", hash = "sha256:27a6f09dbfb69bb79609724c0f90dfaa7c215876a7cd9f12d585574d1f922112"}, +    {file = "pycares-4.1.2-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:e5a060f5fa90ae245aa99a4a8ad13ec39c2340400de037c7e8d27b081e1a3c64"}, +    {file = "pycares-4.1.2-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:056330275dea42b7199494047a745e1d9785d39fb8c4cd469dca043532240b80"}, +    {file = "pycares-4.1.2-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:0aa897543a786daba74ec5e19638bd38b2b432d179a0e248eac1e62de5756207"}, +    {file = "pycares-4.1.2-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:cbceaa9b2c416aa931627466d3240aecfc905c292c842252e3d77b8630072505"}, +    {file = "pycares-4.1.2-cp36-cp36m-win32.whl", hash = "sha256:112e1385c451069112d6b5ea1f9c378544f3c6b89882ff964e9a64be3336d7e4"}, +    {file = "pycares-4.1.2-cp36-cp36m-win_amd64.whl", hash = "sha256:c6680f7fdc0f1163e8f6c2a11d11b9a0b524a61000d2a71f9ccd410f154fb171"}, +    {file = "pycares-4.1.2-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:58a41a2baabcd95266db776c510d349d417919407f03510fc87ac7488730d913"}, +    {file = "pycares-4.1.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a810d01c9a426ee8b0f36969c2aef5fb966712be9d7e466920beb328cd9cefa3"}, +    {file = "pycares-4.1.2-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:b266cec81dcea2c3efbbd3dda00af8d7eb0693ae9e47e8706518334b21f27d4a"}, +    {file = "pycares-4.1.2-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:8319afe4838e09df267c421ca93da408f770b945ec6217dda72f1f6a493e37e4"}, +    {file = "pycares-4.1.2-cp37-cp37m-win32.whl", hash = "sha256:4d5da840aa0d9b15fa51107f09270c563a348cb77b14ae9653d0bbdbe326fcc2"}, +    {file = "pycares-4.1.2-cp37-cp37m-win_amd64.whl", hash = "sha256:5632f21d92cc0225ba5ff906e4e5dec415ef0b3df322c461d138190681cd5d89"}, +    {file = "pycares-4.1.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:8fd1ff17a26bb004f0f6bb902ba7dddd810059096ae0cc3b45e4f5be46315d19"}, +    {file = "pycares-4.1.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:439799be4b7576e907139a7f9b3c8a01b90d3e38af4af9cd1fc6c1ee9a42b9e6"}, +    {file = "pycares-4.1.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:40079ed58efa91747c50aac4edf8ecc7e570132ab57dc0a4030eb0d016a6cab8"}, +    {file = "pycares-4.1.2-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:4e190471a015f8225fa38069617192e06122771cce2b169ac7a60bfdbd3d4ab2"}, +    {file = "pycares-4.1.2-cp38-cp38-win32.whl", hash = "sha256:2b837315ed08c7df009b67725fe1f50489e99de9089f58ec1b243dc612f172aa"}, +    {file = "pycares-4.1.2-cp38-cp38-win_amd64.whl", hash = "sha256:c7eba3c8354b730a54d23237d0b6445a2f68570fa68d0848887da23a3f3b71f3"}, +    {file = "pycares-4.1.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:2f5f84fe9f83eab9cd68544b165b74ba6e3412d029cc9ab20098d9c332869fc5"}, +    {file = "pycares-4.1.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:569eef8597b5e02b1bc4644b9f272160304d8c9985357d7ecfcd054da97c0771"}, +    {file = "pycares-4.1.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:e1489aa25d14dbf7176110ead937c01176ed5a0ebefd3b092bbd6b202241814c"}, +    {file = "pycares-4.1.2-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:dc942692fca0e27081b7bb414bb971d34609c80df5e953f6d0c62ecc8019acd9"}, +    {file = "pycares-4.1.2-cp39-cp39-win32.whl", hash = "sha256:ed71dc4290d9c3353945965604ef1f6a4de631733e9819a7ebc747220b27e641"}, +    {file = "pycares-4.1.2-cp39-cp39-win_amd64.whl", hash = "sha256:ec00f3594ee775665167b1a1630edceefb1b1283af9ac57480dba2fb6fd6c360"}, +    {file = "pycares-4.1.2.tar.gz", hash = "sha256:03490be0e7b51a0c8073f877bec347eff31003f64f57d9518d419d9369452837"},  ]  pycodestyle = [      {file = "pycodestyle-2.7.0-py2.py3-none-any.whl", hash = "sha256:514f76d918fcc0b55c6680472f0a37970994e07bbb80725808c17089be302068"},      {file = "pycodestyle-2.7.0.tar.gz", hash = "sha256:c389c1d06bf7904078ca03399a4816f974a1d590090fecea0c63ec26ebaf1cef"},  ]  pycparser = [ -    {file = "pycparser-2.20-py2.py3-none-any.whl", hash = "sha256:7582ad22678f0fcd81102833f60ef8d0e57288b6b5fb00323d101be910e35705"}, -    {file = "pycparser-2.20.tar.gz", hash = "sha256:2d475327684562c3a96cc71adf7dc8c4f0565175cf86b6d7a404ff4c771f15f0"}, +    {file = "pycparser-2.21-py2.py3-none-any.whl", hash = "sha256:8ee45429555515e1f6b185e78100aea234072576aa43ab53aefcae078162fca9"}, +    {file = "pycparser-2.21.tar.gz", hash = "sha256:e644fdec12f7872f86c58ff790da456218b10f863970249516d60a5eaca77206"},  ]  pydocstyle = [      {file = "pydocstyle-6.1.1-py3-none-any.whl", hash = "sha256:6987826d6775056839940041beef5c08cc7e3d71d63149b48e36727f70144dc4"}, @@ -1276,16 +1505,20 @@ pyflakes = [      {file = "pyflakes-2.3.1.tar.gz", hash = "sha256:f5bc8ecabc05bb9d291eb5203d6810b49040f6ff446a756326104746cc00c1db"},  ]  pyparsing = [ -    {file = "pyparsing-2.4.7-py2.py3-none-any.whl", hash = "sha256:ef9d7589ef3c200abe66653d3f1ab1033c3c419ae9b9bdb1240a85b024efc88b"}, -    {file = "pyparsing-2.4.7.tar.gz", hash = "sha256:c203ec8783bf771a155b207279b9bccb8dea02d8f0c9e5f8ead507bc3246ecc1"}, +    {file = "pyparsing-3.0.5-py3-none-any.whl", hash = "sha256:4881e3d2979f27b41a3a2421b10be9cbfa7ce2baa6c7117952222f8bbea6650c"}, +    {file = "pyparsing-3.0.5.tar.gz", hash = "sha256:9329d1c1b51f0f76371c4ded42c5ec4cc0be18456b22193e0570c2da98ed288b"}, +] +pyreadline3 = [ +    {file = "pyreadline3-3.3-py3-none-any.whl", hash = "sha256:0003fd0079d152ecbd8111202c5a7dfa6a5569ffd65b235e45f3c2ecbee337b4"}, +    {file = "pyreadline3-3.3.tar.gz", hash = "sha256:ff3b5a1ac0010d0967869f723e687d42cabc7dccf33b14934c92aa5168d260b3"},  ]  python-dateutil = [      {file = "python-dateutil-2.8.2.tar.gz", hash = "sha256:0123cacc1627ae19ddf3c27a5de5bd67ee4586fbdd6440d9748f8abb483d3e86"},      {file = "python_dateutil-2.8.2-py2.py3-none-any.whl", hash = "sha256:961d03dc3453ebbc59dbdea9e4e11c5651520a876d0f4db161e8674aae935da9"},  ]  python-dotenv = [ -    {file = "python-dotenv-0.15.0.tar.gz", hash = "sha256:587825ed60b1711daea4832cf37524dfd404325b7db5e25ebe88c495c9f807a0"}, -    {file = "python_dotenv-0.15.0-py2.py3-none-any.whl", hash = "sha256:0c8d1b80d1a1e91717ea7d526178e3882732420b03f08afea0406db6402e220e"}, +    {file = "python-dotenv-0.19.1.tar.gz", hash = "sha256:14f8185cc8d494662683e6914addcb7e95374771e707601dfc70166946b4c4b8"}, +    {file = "python_dotenv-0.19.1-py2.py3-none-any.whl", hash = "sha256:bbd3da593fc49c249397cbfbcc449cf36cb02e75afc8157fcc6a81df6fb7750a"},  ]  pyyaml = [      {file = "PyYAML-5.4.1-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:3b2b1824fe7112845700f815ff6a489360226a5609b96ec2190a45e62a9fc922"}, @@ -1319,67 +1552,57 @@ pyyaml = [      {file = "PyYAML-5.4.1.tar.gz", hash = "sha256:607774cbba28732bfa802b54baa7484215f530991055bb562efbed5b2f20a45e"},  ]  rapidfuzz = [ -    {file = "rapidfuzz-1.5.1-cp35-cp35m-macosx_10_9_x86_64.whl", hash = "sha256:6a951ad31ef121bacf40bbe6fbd387740d5038400ec2bfb41d037e9fd2754ef5"}, -    {file = "rapidfuzz-1.5.1-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:b64acce4f6e745417218fc0bb6ff31b26fac0d723506b82ee4b9cad448b85ebb"}, -    {file = "rapidfuzz-1.5.1-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:a834061e6d4dfb9672e89e28583486f60821796cf0d7cc559643a0d597ce33a9"}, -    {file = "rapidfuzz-1.5.1-cp35-cp35m-manylinux2010_i686.whl", hash = "sha256:50548b919bc7608f7b9b4780415ddad135cfc3a54135bdb4bd0bb7ff2cdf9fdf"}, -    {file = "rapidfuzz-1.5.1-cp35-cp35m-manylinux2010_x86_64.whl", hash = "sha256:c39c7f200eef49f4f9d6b808950709334e6f1c22262d570f1f77d6d3d373ad81"}, -    {file = "rapidfuzz-1.5.1-cp35-cp35m-manylinux2014_aarch64.whl", hash = "sha256:347973ddf12d66d4d06daf1aca3a096a1bffe12306bcf13b832bdfc8db6d9f4a"}, -    {file = "rapidfuzz-1.5.1-cp35-cp35m-manylinux2014_ppc64le.whl", hash = "sha256:e6cd6717d87d02dde2088c080b0851bdba970b77085b68e213a7b786dee4be88"}, -    {file = "rapidfuzz-1.5.1-cp35-cp35m-manylinux2014_s390x.whl", hash = "sha256:ba956add4c34da019fb5e8f5e1768604b05569dd68055382795ad9062b9ca55e"}, -    {file = "rapidfuzz-1.5.1-cp35-cp35m-win32.whl", hash = "sha256:95164076d8e0433f9f93e218270f19e3020a3a9b8db28a3d74143810d4243600"}, -    {file = "rapidfuzz-1.5.1-cp35-cp35m-win_amd64.whl", hash = "sha256:274878c59440d6ad3efca833da61594836306af7dcdd914cc1b6ceb2d4cea23b"}, -    {file = "rapidfuzz-1.5.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:3b3df0c1a307a64273f6fd64c0f28218e002768eda1d94b9fffdab9371e38a6a"}, -    {file = "rapidfuzz-1.5.1-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:55472932a8bcf008855b2cc8e5bc47d60066b504ef02dbf8d8fd43ddd8f20a6e"}, -    {file = "rapidfuzz-1.5.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:170e71f2ec36a086ce5d2667331721cc9b779370d0ef7248ef6979819cd8fb09"}, -    {file = "rapidfuzz-1.5.1-cp36-cp36m-manylinux2010_i686.whl", hash = "sha256:00734857b412afc35b29f0ea2f1d9ee26ff93d4cd3fa5f47bb90f6aef385f2a1"}, -    {file = "rapidfuzz-1.5.1-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:ef0cf038c910a3ed626a3224effde8eb49dd7dcda87af59fcd37bc63b78a9bd1"}, -    {file = "rapidfuzz-1.5.1-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:9f454f79bc463e3de08c5d5c0f438fce1b1736cd4da1a1f47f72dc37da156552"}, -    {file = "rapidfuzz-1.5.1-cp36-cp36m-manylinux2014_ppc64le.whl", hash = "sha256:05e8dede642af1b38ebcf8fb5e9bbfdcdf8debba660ae1aafb5c1b0e6ca3e4de"}, -    {file = "rapidfuzz-1.5.1-cp36-cp36m-manylinux2014_s390x.whl", hash = "sha256:f08773adb7f21e1f530bad2c6ababaf472f80283650bc265a7e8f614480cd49c"}, -    {file = "rapidfuzz-1.5.1-cp36-cp36m-win32.whl", hash = "sha256:4e1a7fb18d4a6c3d471a3ad8f820f179216de61bef74663342340cf9c685a31e"}, -    {file = "rapidfuzz-1.5.1-cp36-cp36m-win_amd64.whl", hash = "sha256:09e992579c4aae59310e44db99ed848a8437ed1e8810a513d3bbab7ac7a8f215"}, -    {file = "rapidfuzz-1.5.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:38e2cd05869bd50f25b6d384e0cc73f8cfd6ebb8f1e7bdf1315384e21611f091"}, -    {file = "rapidfuzz-1.5.1-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:b173f8d4c9360b8b32b5ab7a669623f239cb85013e1749bdca03e1b3c297faa7"}, -    {file = "rapidfuzz-1.5.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:6ab75111e2216a48c7e01d47d8903fc2d0c1df398e7262a6df544d86812e40c7"}, -    {file = "rapidfuzz-1.5.1-cp37-cp37m-manylinux2010_i686.whl", hash = "sha256:eb560622d9970eb0c615d5dff26af8a8647ba541a89a927fca1eb0862898f997"}, -    {file = "rapidfuzz-1.5.1-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:59246615b819e4aff685aa57359f5bbaf02441cccc83e8899608037542a3fe36"}, -    {file = "rapidfuzz-1.5.1-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:7a960dfedcf1acdb8435b5b00aebfc2ee8fd53b7b4f7acf613915b4c24fc0ef7"}, -    {file = "rapidfuzz-1.5.1-cp37-cp37m-manylinux2014_ppc64le.whl", hash = "sha256:3d727a7e09f1a01b61452c86d687d0564bad923d5d209224549ae790408d6449"}, -    {file = "rapidfuzz-1.5.1-cp37-cp37m-manylinux2014_s390x.whl", hash = "sha256:eb5e43dbef900367b91fb73a4c447efde034656b25b397844c8cf622dae84ac3"}, -    {file = "rapidfuzz-1.5.1-cp37-cp37m-win32.whl", hash = "sha256:03ea226734cca3f86bc402fc04b8a38b795795e99dbf02dd834845b80bcf7588"}, -    {file = "rapidfuzz-1.5.1-cp37-cp37m-win_amd64.whl", hash = "sha256:5e9beeb6643d663c410ad8ccf88eafbe59ba7aa9b34eea5b51c6312976789803"}, -    {file = "rapidfuzz-1.5.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:0d3cdb6ced024ed1567ba0be4b0909b17f691bd6e9e9f29626e4953ecf7cba9e"}, -    {file = "rapidfuzz-1.5.1-cp38-cp38-manylinux1_i686.whl", hash = "sha256:c06f98bb94fbad9b773c38a3e2cf28a315466b41f862917ba4d228052bcc0966"}, -    {file = "rapidfuzz-1.5.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:7bf51eb2ff342c4a0d77ab22b3d7de461ef9d2c480fd863c57fb139e7578fa7b"}, -    {file = "rapidfuzz-1.5.1-cp38-cp38-manylinux2010_i686.whl", hash = "sha256:6b0c74f60c03eed4b5d19f866df79c1d1bffc4c61f9bf31b114402c47680997f"}, -    {file = "rapidfuzz-1.5.1-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:bbba42e244a0ebc1639c62ab44e4a172767d3721d2af48f2764ca00de7721479"}, -    {file = "rapidfuzz-1.5.1-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:0b9f76f47b6df8c6aaa02a27fdff52e6aaf64d39296683ed06d0ec9acf2515d2"}, -    {file = "rapidfuzz-1.5.1-cp38-cp38-manylinux2014_ppc64le.whl", hash = "sha256:c52ce4b4bfe8e0c2cf102f7b71cca00fc3228113e71712597940c9c340ae31b1"}, -    {file = "rapidfuzz-1.5.1-cp38-cp38-manylinux2014_s390x.whl", hash = "sha256:0d3ae040c91f500814df6557276462c0c869b16168ef51d01c8a7da150f54513"}, -    {file = "rapidfuzz-1.5.1-cp38-cp38-win32.whl", hash = "sha256:112ecc4825c5d362298d1e3c512d3f942c1a74f26ca69dc4b19a4f2cd95cb764"}, -    {file = "rapidfuzz-1.5.1-cp38-cp38-win_amd64.whl", hash = "sha256:b1feec7407df54045bc9d4dce3431ce20855c1ff4dd170480fbace62164f8f9c"}, -    {file = "rapidfuzz-1.5.1-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:8b352fe56b92bd2aa4ceae550543a923996c16efecf8f981c955dd5f522d2002"}, -    {file = "rapidfuzz-1.5.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:9454f46bc4007be9148f18143bb1b615a740a99737a38cf7b9baf3c495d5d17c"}, -    {file = "rapidfuzz-1.5.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:4c1a75c87a2f4c9709c6e3ecdbb2317f0964ac96f845f6a331d8a437a2944d24"}, -    {file = "rapidfuzz-1.5.1-cp39-cp39-manylinux1_i686.whl", hash = "sha256:02f8b282a940cb749b1c51196baab7abb5590fcc8c065ce540c5d8414366036d"}, -    {file = "rapidfuzz-1.5.1-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:2def32a4228a717c5e6a699f0742546aee4091eb1e59e79781ceacabfc54452c"}, -    {file = "rapidfuzz-1.5.1-cp39-cp39-manylinux2010_i686.whl", hash = "sha256:2e9d494ff51b942ed1504f84c13476319c89fc9bcc6379cc0816b776a7994199"}, -    {file = "rapidfuzz-1.5.1-cp39-cp39-manylinux2010_x86_64.whl", hash = "sha256:e94a6af7f0cc8ff49ab22842af255d8d927ca3b168b1a7e8e0784f1a2f49bc38"}, -    {file = "rapidfuzz-1.5.1-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:c4ee8b9baaf46447dcaed925ad1d3606d3a375dfc5c67d1f3e33c46a3008cb5a"}, -    {file = "rapidfuzz-1.5.1-cp39-cp39-manylinux2014_ppc64le.whl", hash = "sha256:10e236b3ce5851f584bbf178e7fb04ae5d0fbb008f3bc580ef6185bbbb346cd1"}, -    {file = "rapidfuzz-1.5.1-cp39-cp39-manylinux2014_s390x.whl", hash = "sha256:068404913619182739fa3fde3079c17e3402744a1117df7f60055db331095a01"}, -    {file = "rapidfuzz-1.5.1-cp39-cp39-win32.whl", hash = "sha256:e933e3ce2d88b7584248493abcba2cd27240f42bf73ca040babfd1ce8036750e"}, -    {file = "rapidfuzz-1.5.1-cp39-cp39-win_amd64.whl", hash = "sha256:46f5931f7441e13574d0fe33e897212d00ff63f69c0db1d449afbc5e87bafd7f"}, -    {file = "rapidfuzz-1.5.1-pp36-pypy36_pp73-macosx_10_9_x86_64.whl", hash = "sha256:7fb25ee0340cc26dad0bb4a97019bf61b4cefaec67a1be64ac9dac2f98c697cd"}, -    {file = "rapidfuzz-1.5.1-pp36-pypy36_pp73-manylinux1_x86_64.whl", hash = "sha256:e738ec4e680bebe4442befda5cdd18020c3721d4cd75f9bfe2fb94e78ef55618"}, -    {file = "rapidfuzz-1.5.1-pp36-pypy36_pp73-manylinux2010_x86_64.whl", hash = "sha256:cb083609923bc4ac602e6f1d61be61a25b35cccfb5ee208d2aa89eb0be357c69"}, -    {file = "rapidfuzz-1.5.1-pp36-pypy36_pp73-win32.whl", hash = "sha256:08ecef2995b6ed1187b375d8f28ba4557522f098a1515b6afb0e3b452997a3a4"}, -    {file = "rapidfuzz-1.5.1-pp37-pypy37_pp73-macosx_10_9_x86_64.whl", hash = "sha256:b6ff10d856fce55e2b1c681e4e7cd7da9b9eb6854571df60d6ed8904c777e64b"}, -    {file = "rapidfuzz-1.5.1-pp37-pypy37_pp73-manylinux1_x86_64.whl", hash = "sha256:b41c346f16cd1ee71b259106d3cfad3347bd8fff4ff20f334a12738df6736c01"}, -    {file = "rapidfuzz-1.5.1-pp37-pypy37_pp73-manylinux2010_x86_64.whl", hash = "sha256:f7148a53a0fd3466b82b81d94ad91aee7ce7947f37f16f9fb54319ea7df7f4af"}, -    {file = "rapidfuzz-1.5.1-pp37-pypy37_pp73-win32.whl", hash = "sha256:31d83af9ac39f47f47ce4830ee118e6fa53964cccd8161e9a478a326f2a994cf"}, -    {file = "rapidfuzz-1.5.1.tar.gz", hash = "sha256:4ebbd071425ee812548c301c60661a4f8faa5e5bcc97a6f0bef5b562585a8025"}, +    {file = "rapidfuzz-1.8.2-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:d218a4aac1488cc7d63f1a597a33863aa304f6ac590d70057e708ec6865a4118"}, +    {file = "rapidfuzz-1.8.2-cp27-cp27m-manylinux2010_i686.whl", hash = "sha256:8e47b2f9f49edbfd582f6703cde54d22ffa98d83c8393ccd07073852b33832ea"}, +    {file = "rapidfuzz-1.8.2-cp27-cp27m-manylinux2010_x86_64.whl", hash = "sha256:17e456d2ba6bee54b5c83bb403c33e02c3873958e153c0413021a2a042b0940d"}, +    {file = "rapidfuzz-1.8.2-cp27-cp27m-win32.whl", hash = "sha256:ab067c4f04f037686d6cad1a7fce4c3998548f38778f0edb351280b902b8b3e1"}, +    {file = "rapidfuzz-1.8.2-cp27-cp27m-win_amd64.whl", hash = "sha256:6c3e298aa955b164c85e7e0e2372805da1d6bae7399dad256211caafdab46e7f"}, +    {file = "rapidfuzz-1.8.2-cp27-cp27mu-manylinux2010_i686.whl", hash = "sha256:f62471a08d2c46c79a8e0129b74c2997e62692c36bef47a7392b542d6dafc6bf"}, +    {file = "rapidfuzz-1.8.2-cp27-cp27mu-manylinux2010_x86_64.whl", hash = "sha256:271ce29c63eb87d76bcd384753cdfbfb8f2a0aeb3c7d0891787b1f19b007a0e8"}, +    {file = "rapidfuzz-1.8.2-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:22f49eb345759602cc002200217a62564d947f65a568723f99b80c74027ce77b"}, +    {file = "rapidfuzz-1.8.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:4ec6015b74cdbd4ac8bbbf280522e913f11c4656e334559a8454713def4f3b33"}, +    {file = "rapidfuzz-1.8.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:d53a474e3b7592a2ff3d4c1e545324314845d3ccbbc80f6dd89d044bceaf8a59"}, +    {file = "rapidfuzz-1.8.2-cp310-cp310-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:242ebe941991a086b9e455dadcdaf624bb895457e3ce254b0b51a7b9adc68fed"}, +    {file = "rapidfuzz-1.8.2-cp310-cp310-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:dab9822666819350aa32008fe35a7c7da422289d6961a01a2e00e99c2581e7fe"}, +    {file = "rapidfuzz-1.8.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8107e175577f200a2426c4896b8e17a97b86f513122cba4155c7462f0fcc18ac"}, +    {file = "rapidfuzz-1.8.2-cp310-cp310-win32.whl", hash = "sha256:526d92f1ff354303cba0a4cbf184d11b94395841d00eaecf8963c6dc89deec21"}, +    {file = "rapidfuzz-1.8.2-cp310-cp310-win_amd64.whl", hash = "sha256:55168678ae7df1ad8099ec2f0ce54024c924b375b18a6f5d3237c930083fcfca"}, +    {file = "rapidfuzz-1.8.2-cp35-cp35m-macosx_10_9_x86_64.whl", hash = "sha256:23c0632047bf8fac891ef73dfd397e951514d11fb5f168b630e3569ffcd51d61"}, +    {file = "rapidfuzz-1.8.2-cp35-cp35m-manylinux2010_i686.whl", hash = "sha256:90d0e10c7075f761b17e590cf9254b32e7084de7e2b4cd11201031c61386714e"}, +    {file = "rapidfuzz-1.8.2-cp35-cp35m-manylinux2010_x86_64.whl", hash = "sha256:c9537e7f2a489654671c4e3bdc3e01d1116052b3889217da7fe2f0b0f0b27e10"}, +    {file = "rapidfuzz-1.8.2-cp35-cp35m-win32.whl", hash = "sha256:8ded55e5395af1705bbc89ab94bea3c73218e1f71ae2b72cd905a827ab131fa1"}, +    {file = "rapidfuzz-1.8.2-cp35-cp35m-win_amd64.whl", hash = "sha256:2d4240fce22e74c6f3d381201288cea6cc5af8d310ec271b573b26558c2eaec8"}, +    {file = "rapidfuzz-1.8.2-cp36-cp36m-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:90901b7be174b1f14455c5601eff46b03b174cf1d61cec0fd6d37c74dd727c88"}, +    {file = "rapidfuzz-1.8.2-cp36-cp36m-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:7dbd1524e0f2d4290cd60fe1474068a8d8a27a41e1dfa14b1e4d53735529a11c"}, +    {file = "rapidfuzz-1.8.2-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4b8eee29e35ce51adf1cbfcc5b64d1a89d71401a5101b8fe9b87abbe6bf4f893"}, +    {file = "rapidfuzz-1.8.2-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:648fd2e561b6beb208e6cf46eb9f20ba952add69c2f90fb4e46898a821bca4c9"}, +    {file = "rapidfuzz-1.8.2-cp37-cp37m-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:40dacd641d93708818ff2c9921eb6e5b9463e9b3b7f410dde3c96e0e0f80c414"}, +    {file = "rapidfuzz-1.8.2-cp37-cp37m-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:28528d34fb73f991e2822fdf1aa28eff7832c2189c8d2d0b598f15dd1b1b7d88"}, +    {file = "rapidfuzz-1.8.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1f46137937dfa2ec04fbb9f0225ff3160b9f1ed993dc5abf126ccdeae38726c5"}, +    {file = "rapidfuzz-1.8.2-cp37-cp37m-win32.whl", hash = "sha256:709ea3151a7448bb15adc3212cbdd4766eca557f1ae73cdff8ae656b6899e71a"}, +    {file = "rapidfuzz-1.8.2-cp37-cp37m-win_amd64.whl", hash = "sha256:c426eb833d16ddeec95c966efa78492803d741c85cf6553febf78979fc267692"}, +    {file = "rapidfuzz-1.8.2-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:91e20676b562c972de8af6a153df6eaaea27da96ade8321e51da7bab32285242"}, +    {file = "rapidfuzz-1.8.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:ea3c3c1a5401affeb7d279c2b7cf418d5e7d7e170eb010fcb1b45bb28de5e4d1"}, +    {file = "rapidfuzz-1.8.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:ab21f86812a7da111a17c118fc8e827b4c289d02f39536f7a6acc6baf479dcc0"}, +    {file = "rapidfuzz-1.8.2-cp38-cp38-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:0eef79b223e239af5cc9c0b89ae8096a38bc8a9109474ff610151ea0115b931c"}, +    {file = "rapidfuzz-1.8.2-cp38-cp38-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:20dfda3bd9c14e6d7951360af66792a15e81051fa017db00c71bad4b1e25688d"}, +    {file = "rapidfuzz-1.8.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ec24328d855b88e8a472379c61af9972f0d7992edfe2ebaca03afed2c5282f94"}, +    {file = "rapidfuzz-1.8.2-cp38-cp38-win32.whl", hash = "sha256:69ab203a4b59c7e9ddec96044bd684d6165eab3362f84677822936fea2e8c4f1"}, +    {file = "rapidfuzz-1.8.2-cp38-cp38-win_amd64.whl", hash = "sha256:bd039be9798a3deba25301827b86b3ff084d69e2b4b16ae8da6a34638235680c"}, +    {file = "rapidfuzz-1.8.2-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:7219ca078df644aa0694af9174b7abb09a75302021f1e062f39fcf183cb9f087"}, +    {file = "rapidfuzz-1.8.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:d240b7cd8dc7890c103adf4666a43fb3b3d1e5b1231e59aa3986e18dba1d086f"}, +    {file = "rapidfuzz-1.8.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:06f639a65f1269a52679942437228c804a354618dda79b68a70612c5c0a36f4e"}, +    {file = "rapidfuzz-1.8.2-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:750ea6977e68a17f16308ac044017f8ce35c8a553698696f5a4fbb77e3110c8c"}, +    {file = "rapidfuzz-1.8.2-cp39-cp39-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:517f26474012dbce37358a447165ca3b100b9b1b83ee8b3d6877accfdb209d46"}, +    {file = "rapidfuzz-1.8.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5b2bf70a24e129af11e217d94cb34f96a1487606bf4a9d8c8f16bc49c9465b56"}, +    {file = "rapidfuzz-1.8.2-cp39-cp39-win32.whl", hash = "sha256:02ec1154e62493c0cb66c8b058d7196de48acbee7e884b757cff7bf075252232"}, +    {file = "rapidfuzz-1.8.2-cp39-cp39-win_amd64.whl", hash = "sha256:be564aea09ec8da0478b67f35be73f8c129054d45da7f4b02cd7e035d1aea96d"}, +    {file = "rapidfuzz-1.8.2-pp27-pypy_73-macosx_10_9_x86_64.whl", hash = "sha256:54406d07788ca85aff12134425a2b931e842ee2c3b1ae915523cdd3d104c1136"}, +    {file = "rapidfuzz-1.8.2-pp27-pypy_73-manylinux2010_x86_64.whl", hash = "sha256:5648fcf1cbeacb05ea4b441ddf99c7f5e15dd53b7d3192e7449581b638e8b3f7"}, +    {file = "rapidfuzz-1.8.2-pp37-pypy37_pp73-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:7d1d6ab126f960aa0df10454a06d17e303680df6370cc3c44c1f0697c4587c5c"}, +    {file = "rapidfuzz-1.8.2-pp37-pypy37_pp73-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:1412ec4d43f0d1854ed3dd220dacbb68b977519553e80a2c4a250d3569793f92"}, +    {file = "rapidfuzz-1.8.2-pp37-pypy37_pp73-win_amd64.whl", hash = "sha256:9dd34ad050ccb680d9dbf9871a53e1f01b084488d5310349c3e9cedcb4561eb2"}, +    {file = "rapidfuzz-1.8.2.tar.gz", hash = "sha256:d6efbb2b6b18b3a67d7bdfbcd9bb72732f55736852bbef823bdf210f9e0c6c90"},  ]  redis = [      {file = "redis-3.5.3-py2.py3-none-any.whl", hash = "sha256:432b788c4530cfe16d8d943a09d40ca6c16149727e4afe8c2c9d5580c59d9f24"}, @@ -1401,9 +1624,17 @@ sortedcontainers = [      {file = "sortedcontainers-2.4.0-py2.py3-none-any.whl", hash = "sha256:a163dcaede0f1c021485e957a39245190e74249897e2ae4b2aa38595db237ee0"},      {file = "sortedcontainers-2.4.0.tar.gz", hash = "sha256:25caa5a06cc30b6b83d11423433f65d1f9d76c4c6a0c90e3379eaa43b9bfdb88"},  ] +soupsieve = [ +    {file = "soupsieve-2.3-py3-none-any.whl", hash = "sha256:617ffc4d0dfd39c66f4d1413a6e165663a34eca86be9b54f97b91756300ff6df"}, +    {file = "soupsieve-2.3.tar.gz", hash = "sha256:e4860f889dfa88774c07da0b276b70c073b6470fa1a4a8350800bb7bce3dcc76"}, +]  taskipy = [ -    {file = "taskipy-1.8.1-py3-none-any.whl", hash = "sha256:2b98f499966e40175d1f1306a64587f49dfa41b90d0d86c8f28b067cc58d0a56"}, -    {file = "taskipy-1.8.1.tar.gz", hash = "sha256:7a2404125817e45d80e13fa663cae35da6e8ba590230094e815633653e25f98f"}, +    {file = "taskipy-1.9.0-py3-none-any.whl", hash = "sha256:02bd2c51c7356ed3f7f8853210ada1cd2ab273e68359ee865021c3057eec6615"}, +    {file = "taskipy-1.9.0.tar.gz", hash = "sha256:449c160b557cdb1d9c17097a5ea4aa0cd5223723ddbaaa5d5032dd16274fb8f0"}, +] +testfixtures = [ +    {file = "testfixtures-6.18.3-py2.py3-none-any.whl", hash = "sha256:6ddb7f56a123e1a9339f130a200359092bd0a6455e31838d6c477e8729bb7763"}, +    {file = "testfixtures-6.18.3.tar.gz", hash = "sha256:2600100ae96ffd082334b378e355550fef8b4a529a6fa4c34f47130905c7426d"},  ]  toml = [      {file = "toml-0.10.2-py2.py3-none-any.whl", hash = "sha256:806143ae5bfb6a3c6e736a764057db0e6a0e05e338b5630894a5f779cabb4f9b"}, @@ -1415,49 +1646,84 @@ typing-extensions = [      {file = "typing_extensions-3.10.0.2.tar.gz", hash = "sha256:49f75d16ff11f1cd258e1b988ccff82a3ca5570217d7ad8c5f48205dd99a677e"},  ]  urllib3 = [ -    {file = "urllib3-1.26.6-py2.py3-none-any.whl", hash = "sha256:39fb8672126159acb139a7718dd10806104dec1e2f0f6c88aab05d17df10c8d4"}, -    {file = "urllib3-1.26.6.tar.gz", hash = "sha256:f57b4c16c62fa2760b7e3d97c35b255512fb6b59a259730f36ba32ce9f8e342f"}, +    {file = "urllib3-1.26.7-py2.py3-none-any.whl", hash = "sha256:c4fdf4019605b6e5423637e01bc9fe4daef873709a7973e195ceba0a62bbc844"}, +    {file = "urllib3-1.26.7.tar.gz", hash = "sha256:4987c65554f7a2dbf30c18fd48778ef124af6fab771a377103da0585e2336ece"},  ]  virtualenv = [ -    {file = "virtualenv-20.7.2-py2.py3-none-any.whl", hash = "sha256:e4670891b3a03eb071748c569a87cceaefbf643c5bac46d996c5a45c34aa0f06"}, -    {file = "virtualenv-20.7.2.tar.gz", hash = "sha256:9ef4e8ee4710826e98ff3075c9a4739e2cb1040de6a2a8d35db0055840dc96a0"}, +    {file = "virtualenv-20.10.0-py2.py3-none-any.whl", hash = "sha256:4b02e52a624336eece99c96e3ab7111f469c24ba226a53ec474e8e787b365814"}, +    {file = "virtualenv-20.10.0.tar.gz", hash = "sha256:576d05b46eace16a9c348085f7d0dc8ef28713a2cabaa1cf0aea41e8f12c9218"},  ]  yarl = [ -    {file = "yarl-1.6.3-cp36-cp36m-macosx_10_14_x86_64.whl", hash = "sha256:0355a701b3998dcd832d0dc47cc5dedf3874f966ac7f870e0f3a6788d802d434"}, -    {file = "yarl-1.6.3-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:bafb450deef6861815ed579c7a6113a879a6ef58aed4c3a4be54400ae8871478"}, -    {file = "yarl-1.6.3-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:547f7665ad50fa8563150ed079f8e805e63dd85def6674c97efd78eed6c224a6"}, -    {file = "yarl-1.6.3-cp36-cp36m-manylinux2014_i686.whl", hash = "sha256:63f90b20ca654b3ecc7a8d62c03ffa46999595f0167d6450fa8383bab252987e"}, -    {file = "yarl-1.6.3-cp36-cp36m-manylinux2014_ppc64le.whl", hash = "sha256:97b5bdc450d63c3ba30a127d018b866ea94e65655efaf889ebeabc20f7d12406"}, -    {file = "yarl-1.6.3-cp36-cp36m-manylinux2014_s390x.whl", hash = "sha256:d8d07d102f17b68966e2de0e07bfd6e139c7c02ef06d3a0f8d2f0f055e13bb76"}, -    {file = "yarl-1.6.3-cp36-cp36m-manylinux2014_x86_64.whl", hash = "sha256:15263c3b0b47968c1d90daa89f21fcc889bb4b1aac5555580d74565de6836366"}, -    {file = "yarl-1.6.3-cp36-cp36m-win32.whl", hash = "sha256:b5dfc9a40c198334f4f3f55880ecf910adebdcb2a0b9a9c23c9345faa9185721"}, -    {file = "yarl-1.6.3-cp36-cp36m-win_amd64.whl", hash = "sha256:b2e9a456c121e26d13c29251f8267541bd75e6a1ccf9e859179701c36a078643"}, -    {file = "yarl-1.6.3-cp37-cp37m-macosx_10_14_x86_64.whl", hash = "sha256:ce3beb46a72d9f2190f9e1027886bfc513702d748047b548b05dab7dfb584d2e"}, -    {file = "yarl-1.6.3-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:2ce4c621d21326a4a5500c25031e102af589edb50c09b321049e388b3934eec3"}, -    {file = "yarl-1.6.3-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:d26608cf178efb8faa5ff0f2d2e77c208f471c5a3709e577a7b3fd0445703ac8"}, -    {file = "yarl-1.6.3-cp37-cp37m-manylinux2014_i686.whl", hash = "sha256:4c5bcfc3ed226bf6419f7a33982fb4b8ec2e45785a0561eb99274ebbf09fdd6a"}, -    {file = "yarl-1.6.3-cp37-cp37m-manylinux2014_ppc64le.whl", hash = "sha256:4736eaee5626db8d9cda9eb5282028cc834e2aeb194e0d8b50217d707e98bb5c"}, -    {file = "yarl-1.6.3-cp37-cp37m-manylinux2014_s390x.whl", hash = "sha256:68dc568889b1c13f1e4745c96b931cc94fdd0defe92a72c2b8ce01091b22e35f"}, -    {file = "yarl-1.6.3-cp37-cp37m-manylinux2014_x86_64.whl", hash = "sha256:7356644cbed76119d0b6bd32ffba704d30d747e0c217109d7979a7bc36c4d970"}, -    {file = "yarl-1.6.3-cp37-cp37m-win32.whl", hash = "sha256:00d7ad91b6583602eb9c1d085a2cf281ada267e9a197e8b7cae487dadbfa293e"}, -    {file = "yarl-1.6.3-cp37-cp37m-win_amd64.whl", hash = "sha256:69ee97c71fee1f63d04c945f56d5d726483c4762845400a6795a3b75d56b6c50"}, -    {file = "yarl-1.6.3-cp38-cp38-macosx_10_14_x86_64.whl", hash = "sha256:e46fba844f4895b36f4c398c5af062a9808d1f26b2999c58909517384d5deda2"}, -    {file = "yarl-1.6.3-cp38-cp38-manylinux1_i686.whl", hash = "sha256:31ede6e8c4329fb81c86706ba8f6bf661a924b53ba191b27aa5fcee5714d18ec"}, -    {file = "yarl-1.6.3-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:fcbb48a93e8699eae920f8d92f7160c03567b421bc17362a9ffbbd706a816f71"}, -    {file = "yarl-1.6.3-cp38-cp38-manylinux2014_i686.whl", hash = "sha256:72a660bdd24497e3e84f5519e57a9ee9220b6f3ac4d45056961bf22838ce20cc"}, -    {file = "yarl-1.6.3-cp38-cp38-manylinux2014_ppc64le.whl", hash = "sha256:324ba3d3c6fee56e2e0b0d09bf5c73824b9f08234339d2b788af65e60040c959"}, -    {file = "yarl-1.6.3-cp38-cp38-manylinux2014_s390x.whl", hash = "sha256:e6b5460dc5ad42ad2b36cca524491dfcaffbfd9c8df50508bddc354e787b8dc2"}, -    {file = "yarl-1.6.3-cp38-cp38-manylinux2014_x86_64.whl", hash = "sha256:6d6283d8e0631b617edf0fd726353cb76630b83a089a40933043894e7f6721e2"}, -    {file = "yarl-1.6.3-cp38-cp38-win32.whl", hash = "sha256:9ede61b0854e267fd565e7527e2f2eb3ef8858b301319be0604177690e1a3896"}, -    {file = "yarl-1.6.3-cp38-cp38-win_amd64.whl", hash = "sha256:f0b059678fd549c66b89bed03efcabb009075bd131c248ecdf087bdb6faba24a"}, -    {file = "yarl-1.6.3-cp39-cp39-macosx_10_14_x86_64.whl", hash = "sha256:329412812ecfc94a57cd37c9d547579510a9e83c516bc069470db5f75684629e"}, -    {file = "yarl-1.6.3-cp39-cp39-manylinux1_i686.whl", hash = "sha256:c49ff66d479d38ab863c50f7bb27dee97c6627c5fe60697de15529da9c3de724"}, -    {file = "yarl-1.6.3-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:f040bcc6725c821a4c0665f3aa96a4d0805a7aaf2caf266d256b8ed71b9f041c"}, -    {file = "yarl-1.6.3-cp39-cp39-manylinux2014_i686.whl", hash = "sha256:d5c32c82990e4ac4d8150fd7652b972216b204de4e83a122546dce571c1bdf25"}, -    {file = "yarl-1.6.3-cp39-cp39-manylinux2014_ppc64le.whl", hash = "sha256:d597767fcd2c3dc49d6eea360c458b65643d1e4dbed91361cf5e36e53c1f8c96"}, -    {file = "yarl-1.6.3-cp39-cp39-manylinux2014_s390x.whl", hash = "sha256:8aa3decd5e0e852dc68335abf5478a518b41bf2ab2f330fe44916399efedfae0"}, -    {file = "yarl-1.6.3-cp39-cp39-manylinux2014_x86_64.whl", hash = "sha256:73494d5b71099ae8cb8754f1df131c11d433b387efab7b51849e7e1e851f07a4"}, -    {file = "yarl-1.6.3-cp39-cp39-win32.whl", hash = "sha256:5b883e458058f8d6099e4420f0cc2567989032b5f34b271c0827de9f1079a424"}, -    {file = "yarl-1.6.3-cp39-cp39-win_amd64.whl", hash = "sha256:4953fb0b4fdb7e08b2f3b3be80a00d28c5c8a2056bb066169de00e6501b986b6"}, -    {file = "yarl-1.6.3.tar.gz", hash = "sha256:8a9066529240171b68893d60dca86a763eae2139dd42f42106b03cf4b426bf10"}, +    {file = "yarl-1.7.2-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:f2a8508f7350512434e41065684076f640ecce176d262a7d54f0da41d99c5a95"}, +    {file = "yarl-1.7.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:da6df107b9ccfe52d3a48165e48d72db0eca3e3029b5b8cb4fe6ee3cb870ba8b"}, +    {file = "yarl-1.7.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:a1d0894f238763717bdcfea74558c94e3bc34aeacd3351d769460c1a586a8b05"}, +    {file = "yarl-1.7.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:dfe4b95b7e00c6635a72e2d00b478e8a28bfb122dc76349a06e20792eb53a523"}, +    {file = "yarl-1.7.2-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c145ab54702334c42237a6c6c4cc08703b6aa9b94e2f227ceb3d477d20c36c63"}, +    {file = "yarl-1.7.2-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1ca56f002eaf7998b5fcf73b2421790da9d2586331805f38acd9997743114e98"}, +    {file = "yarl-1.7.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:1d3d5ad8ea96bd6d643d80c7b8d5977b4e2fb1bab6c9da7322616fd26203d125"}, +    {file = "yarl-1.7.2-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:167ab7f64e409e9bdd99333fe8c67b5574a1f0495dcfd905bc7454e766729b9e"}, +    {file = "yarl-1.7.2-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:95a1873b6c0dd1c437fb3bb4a4aaa699a48c218ac7ca1e74b0bee0ab16c7d60d"}, +    {file = "yarl-1.7.2-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:6152224d0a1eb254f97df3997d79dadd8bb2c1a02ef283dbb34b97d4f8492d23"}, +    {file = "yarl-1.7.2-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:5bb7d54b8f61ba6eee541fba4b83d22b8a046b4ef4d8eb7f15a7e35db2e1e245"}, +    {file = "yarl-1.7.2-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:9c1f083e7e71b2dd01f7cd7434a5f88c15213194df38bc29b388ccdf1492b739"}, +    {file = "yarl-1.7.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:f44477ae29025d8ea87ec308539f95963ffdc31a82f42ca9deecf2d505242e72"}, +    {file = "yarl-1.7.2-cp310-cp310-win32.whl", hash = "sha256:cff3ba513db55cc6a35076f32c4cdc27032bd075c9faef31fec749e64b45d26c"}, +    {file = "yarl-1.7.2-cp310-cp310-win_amd64.whl", hash = "sha256:c9c6d927e098c2d360695f2e9d38870b2e92e0919be07dbe339aefa32a090265"}, +    {file = "yarl-1.7.2-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:9b4c77d92d56a4c5027572752aa35082e40c561eec776048330d2907aead891d"}, +    {file = "yarl-1.7.2-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c01a89a44bb672c38f42b49cdb0ad667b116d731b3f4c896f72302ff77d71656"}, +    {file = "yarl-1.7.2-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c19324a1c5399b602f3b6e7db9478e5b1adf5cf58901996fc973fe4fccd73eed"}, +    {file = "yarl-1.7.2-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3abddf0b8e41445426d29f955b24aeecc83fa1072be1be4e0d194134a7d9baee"}, +    {file = "yarl-1.7.2-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:6a1a9fe17621af43e9b9fcea8bd088ba682c8192d744b386ee3c47b56eaabb2c"}, +    {file = "yarl-1.7.2-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:8b0915ee85150963a9504c10de4e4729ae700af11df0dc5550e6587ed7891e92"}, +    {file = "yarl-1.7.2-cp36-cp36m-musllinux_1_1_aarch64.whl", hash = "sha256:29e0656d5497733dcddc21797da5a2ab990c0cb9719f1f969e58a4abac66234d"}, +    {file = "yarl-1.7.2-cp36-cp36m-musllinux_1_1_i686.whl", hash = "sha256:bf19725fec28452474d9887a128e98dd67eee7b7d52e932e6949c532d820dc3b"}, +    {file = "yarl-1.7.2-cp36-cp36m-musllinux_1_1_ppc64le.whl", hash = "sha256:d6f3d62e16c10e88d2168ba2d065aa374e3c538998ed04996cd373ff2036d64c"}, +    {file = "yarl-1.7.2-cp36-cp36m-musllinux_1_1_s390x.whl", hash = "sha256:ac10bbac36cd89eac19f4e51c032ba6b412b3892b685076f4acd2de18ca990aa"}, +    {file = "yarl-1.7.2-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:aa32aaa97d8b2ed4e54dc65d241a0da1c627454950f7d7b1f95b13985afd6c5d"}, +    {file = "yarl-1.7.2-cp36-cp36m-win32.whl", hash = "sha256:87f6e082bce21464857ba58b569370e7b547d239ca22248be68ea5d6b51464a1"}, +    {file = "yarl-1.7.2-cp36-cp36m-win_amd64.whl", hash = "sha256:ac35ccde589ab6a1870a484ed136d49a26bcd06b6a1c6397b1967ca13ceb3913"}, +    {file = "yarl-1.7.2-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:a467a431a0817a292121c13cbe637348b546e6ef47ca14a790aa2fa8cc93df63"}, +    {file = "yarl-1.7.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6ab0c3274d0a846840bf6c27d2c60ba771a12e4d7586bf550eefc2df0b56b3b4"}, +    {file = "yarl-1.7.2-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d260d4dc495c05d6600264a197d9d6f7fc9347f21d2594926202fd08cf89a8ba"}, +    {file = "yarl-1.7.2-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:fc4dd8b01a8112809e6b636b00f487846956402834a7fd59d46d4f4267181c41"}, +    {file = "yarl-1.7.2-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:c1164a2eac148d85bbdd23e07dfcc930f2e633220f3eb3c3e2a25f6148c2819e"}, +    {file = "yarl-1.7.2-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:67e94028817defe5e705079b10a8438b8cb56e7115fa01640e9c0bb3edf67332"}, +    {file = "yarl-1.7.2-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:89ccbf58e6a0ab89d487c92a490cb5660d06c3a47ca08872859672f9c511fc52"}, +    {file = "yarl-1.7.2-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:8cce6f9fa3df25f55521fbb5c7e4a736683148bcc0c75b21863789e5185f9185"}, +    {file = "yarl-1.7.2-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:211fcd65c58bf250fb994b53bc45a442ddc9f441f6fec53e65de8cba48ded986"}, +    {file = "yarl-1.7.2-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:c10ea1e80a697cf7d80d1ed414b5cb8f1eec07d618f54637067ae3c0334133c4"}, +    {file = "yarl-1.7.2-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:52690eb521d690ab041c3919666bea13ab9fbff80d615ec16fa81a297131276b"}, +    {file = "yarl-1.7.2-cp37-cp37m-win32.whl", hash = "sha256:695ba021a9e04418507fa930d5f0704edbce47076bdcfeeaba1c83683e5649d1"}, +    {file = "yarl-1.7.2-cp37-cp37m-win_amd64.whl", hash = "sha256:c17965ff3706beedafd458c452bf15bac693ecd146a60a06a214614dc097a271"}, +    {file = "yarl-1.7.2-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:fce78593346c014d0d986b7ebc80d782b7f5e19843ca798ed62f8e3ba8728576"}, +    {file = "yarl-1.7.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:c2a1ac41a6aa980db03d098a5531f13985edcb451bcd9d00670b03129922cd0d"}, +    {file = "yarl-1.7.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:39d5493c5ecd75c8093fa7700a2fb5c94fe28c839c8e40144b7ab7ccba6938c8"}, +    {file = "yarl-1.7.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1eb6480ef366d75b54c68164094a6a560c247370a68c02dddb11f20c4c6d3c9d"}, +    {file = "yarl-1.7.2-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5ba63585a89c9885f18331a55d25fe81dc2d82b71311ff8bd378fc8004202ff6"}, +    {file = "yarl-1.7.2-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:e39378894ee6ae9f555ae2de332d513a5763276a9265f8e7cbaeb1b1ee74623a"}, +    {file = "yarl-1.7.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:c0910c6b6c31359d2f6184828888c983d54d09d581a4a23547a35f1d0b9484b1"}, +    {file = "yarl-1.7.2-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:6feca8b6bfb9eef6ee057628e71e1734caf520a907b6ec0d62839e8293e945c0"}, +    {file = "yarl-1.7.2-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:8300401dc88cad23f5b4e4c1226f44a5aa696436a4026e456fe0e5d2f7f486e6"}, +    {file = "yarl-1.7.2-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:788713c2896f426a4e166b11f4ec538b5736294ebf7d5f654ae445fd44270832"}, +    {file = "yarl-1.7.2-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:fd547ec596d90c8676e369dd8a581a21227fe9b4ad37d0dc7feb4ccf544c2d59"}, +    {file = "yarl-1.7.2-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:737e401cd0c493f7e3dd4db72aca11cfe069531c9761b8ea474926936b3c57c8"}, +    {file = "yarl-1.7.2-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:baf81561f2972fb895e7844882898bda1eef4b07b5b385bcd308d2098f1a767b"}, +    {file = "yarl-1.7.2-cp38-cp38-win32.whl", hash = "sha256:ede3b46cdb719c794427dcce9d8beb4abe8b9aa1e97526cc20de9bd6583ad1ef"}, +    {file = "yarl-1.7.2-cp38-cp38-win_amd64.whl", hash = "sha256:cc8b7a7254c0fc3187d43d6cb54b5032d2365efd1df0cd1749c0c4df5f0ad45f"}, +    {file = "yarl-1.7.2-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:580c1f15500e137a8c37053e4cbf6058944d4c114701fa59944607505c2fe3a0"}, +    {file = "yarl-1.7.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:3ec1d9a0d7780416e657f1e405ba35ec1ba453a4f1511eb8b9fbab81cb8b3ce1"}, +    {file = "yarl-1.7.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:3bf8cfe8856708ede6a73907bf0501f2dc4e104085e070a41f5d88e7faf237f3"}, +    {file = "yarl-1.7.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1be4bbb3d27a4e9aa5f3df2ab61e3701ce8fcbd3e9846dbce7c033a7e8136746"}, +    {file = "yarl-1.7.2-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:534b047277a9a19d858cde163aba93f3e1677d5acd92f7d10ace419d478540de"}, +    {file = "yarl-1.7.2-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:c6ddcd80d79c96eb19c354d9dca95291589c5954099836b7c8d29278a7ec0bda"}, +    {file = "yarl-1.7.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:9bfcd43c65fbb339dc7086b5315750efa42a34eefad0256ba114cd8ad3896f4b"}, +    {file = "yarl-1.7.2-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:f64394bd7ceef1237cc604b5a89bf748c95982a84bcd3c4bbeb40f685c810794"}, +    {file = "yarl-1.7.2-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:044daf3012e43d4b3538562da94a88fb12a6490652dbc29fb19adfa02cf72eac"}, +    {file = "yarl-1.7.2-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:368bcf400247318382cc150aaa632582d0780b28ee6053cd80268c7e72796dec"}, +    {file = "yarl-1.7.2-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:bab827163113177aee910adb1f48ff7af31ee0289f434f7e22d10baf624a6dfe"}, +    {file = "yarl-1.7.2-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:0cba38120db72123db7c58322fa69e3c0efa933040ffb586c3a87c063ec7cae8"}, +    {file = "yarl-1.7.2-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:59218fef177296451b23214c91ea3aba7858b4ae3306dde120224cfe0f7a6ee8"}, +    {file = "yarl-1.7.2-cp39-cp39-win32.whl", hash = "sha256:1edc172dcca3f11b38a9d5c7505c83c1913c0addc99cd28e993efeaafdfaa18d"}, +    {file = "yarl-1.7.2-cp39-cp39-win_amd64.whl", hash = "sha256:797c2c412b04403d2da075fb93c123df35239cd7b4cc4e0cd9e5839b73f52c58"}, +    {file = "yarl-1.7.2.tar.gz", hash = "sha256:45399b46d60c253327a460e99856752009fcee5f5d3c80b2f7c0cae1c38d56dd"},  ] diff --git a/pyproject.toml b/pyproject.toml index 7848f593..71c79b13 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -12,24 +12,28 @@ aiodns = "~=2.0"  aioredis = "~1.3"  rapidfuzz = "~=1.4"  arrow = "~=1.1.0" +beautifulsoup4 = "~=4.9"  pillow = "~=8.1"  sentry-sdk = "~=0.19"  PyYAML = "~=5.4"  async-rediscache = {extras = ["fakeredis"], version = "~=0.1.4"}  emojis = "~=0.6.0"  matplotlib = "~=3.4.1" +coloredlogs = "~=15.0" +colorama = { version = "~=0.4.3", markers = "sys_platform == 'win32'" } +lxml = "~=4.4"  [tool.poetry.dev-dependencies]  flake8 = "~=3.8"  flake8-annotations = "~=2.3"  flake8-bugbear = "~=20.1"  flake8-docstrings = "~=1.5" -flake8-import-order = "~=0.18"  flake8-string-format = "~=0.3"  flake8-tidy-imports = "~=4.1"  flake8-todo = "~=0.7" +flake8-isort = "~=4.0"  pep8-naming = "~=0.11" -pip-licenses = "~=3.5.2" +pip-licenses = "~=3.5"  pre-commit = "~=2.1"  python-dotenv = "~=0.15"  taskipy = "~=1.6" @@ -38,7 +42,17 @@ taskipy = "~=1.6"  start = "python -m bot"  lint = "pre-commit run --all-files"  precommit = "pre-commit install" +isort = "isort ."  [build-system]  requires = ["poetry-core>=1.0.0"]  build-backend = "poetry.core.masonry.api" + +[tool.isort] +multi_line_output = 6 +order_by_type = false +case_sensitive = true +combine_as_imports = true +line_length = 120 +atomic = true +known_first_party = ["bot"] | 
