diff --git a/comictaggerlib/ui/pyqttoast/tests/__init__.py b/comictaggerlib/ui/pyqttoast/tests/__init__.py
deleted file mode 100644
index 22147f5..0000000
--- a/comictaggerlib/ui/pyqttoast/tests/__init__.py
+++ /dev/null
@@ -1 +0,0 @@
-# This file is needed for the tests be able to run properly
diff --git a/comictalker/vendor/pyrate_limiter/LICENSE b/comictalker/vendor/pyrate_limiter/LICENSE
new file mode 100644
index 0000000..ad9c2c3
--- /dev/null
+++ b/comictalker/vendor/pyrate_limiter/LICENSE
@@ -0,0 +1,21 @@
+MIT License
+
+Copyright (c) 2021 vutran1710
+
+Permission 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:
+
+The above copyright notice and this permission notice shall be included in all
+copies or substantial portions of the Software.
+
+THE 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.
diff --git a/comictalker/vendor/pyrate_limiter/README.md b/comictalker/vendor/pyrate_limiter/README.md
new file mode 100644
index 0000000..abc3681
--- /dev/null
+++ b/comictalker/vendor/pyrate_limiter/README.md
@@ -0,0 +1,402 @@
+
+
+# PyrateLimiter
+The request rate limiter using Leaky-bucket algorithm.
+
+Full project documentation can be found at [pyratelimiter.readthedocs.io](https://pyratelimiter.readthedocs.io).
+
+[![PyPI version](https://badge.fury.io/py/pyrate-limiter.svg)](https://badge.fury.io/py/pyrate-limiter)
+[![PyPI - Python Versions](https://img.shields.io/pypi/pyversions/pyrate-limiter)](https://pypi.org/project/pyrate-limiter)
+[![codecov](https://codecov.io/gh/vutran1710/PyrateLimiter/branch/master/graph/badge.svg?token=E0Q0YBSINS)](https://codecov.io/gh/vutran1710/PyrateLimiter)
+[![Maintenance](https://img.shields.io/badge/Maintained%3F-yes-green.svg)](https://github.com/vutran1710/PyrateLimiter/graphs/commit-activity)
+[![PyPI license](https://img.shields.io/pypi/l/ansicolortags.svg)](https://pypi.python.org/pypi/pyrate-limiter/)
+
+
+
+## Contents
+- [PyrateLimiter](#pyratelimiter)
+ - [Contents](#contents)
+ - [Features](#features)
+ - [Installation](#installation)
+ - [Basic usage](#basic-usage)
+ - [Defining rate limits](#defining-rate-limits)
+ - [Applying rate limits](#applying-rate-limits)
+ - [Identities](#identities)
+ - [Handling exceeded limits](#handling-exceeded-limits)
+ - [Bucket analogy](#bucket-analogy)
+ - [Rate limit exceptions](#rate-limit-exceptions)
+ - [Rate limit delays](#rate-limit-delays)
+ - [Additional usage options](#additional-usage-options)
+ - [Decorator](#decorator)
+ - [Contextmanager](#contextmanager)
+ - [Async decorator/contextmanager](#async-decoratorcontextmanager)
+ - [Backends](#backends)
+ - [Memory](#memory)
+ - [SQLite](#sqlite)
+ - [Redis](#redis)
+ - [Custom backends](#custom-backends)
+ - [Additional features](#additional-features)
+ - [Time sources](#time-sources)
+ - [Examples](#examples)
+
+## Features
+* Tracks any number of rate limits and intervals you want to define
+* Independently tracks rate limits for multiple services or resources
+* Handles exceeded rate limits by either raising errors or adding delays
+* Several usage options including a normal function call, a decorator, or a contextmanager
+* Async support
+* Includes optional SQLite and Redis backends, which can be used to persist limit tracking across
+ multiple threads, processes, or application restarts
+
+## Installation
+Install using pip:
+```
+pip install pyrate-limiter
+```
+
+Or using conda:
+```
+conda install --channel conda-forge pyrate-limiter
+```
+
+## Basic usage
+
+### Defining rate limits
+Consider some public API (like LinkedIn, GitHub, etc.) that has rate limits like the following:
+```
+- 500 requests per hour
+- 1000 requests per day
+- 10000 requests per month
+```
+
+You can define these rates using the `RequestRate` class, and add them to a `Limiter`:
+``` python
+from pyrate_limiter import Duration, RequestRate, Limiter
+
+hourly_rate = RequestRate(500, Duration.HOUR) # 500 requests per hour
+daily_rate = RequestRate(1000, Duration.DAY) # 1000 requests per day
+monthly_rate = RequestRate(10000, Duration.MONTH) # 10000 requests per month
+
+limiter = Limiter(hourly_rate, daily_rate, monthly_rate)
+```
+
+or
+
+``` python
+from pyrate_limiter import Duration, RequestRate, Limiter
+
+rate_limits = (
+ RequestRate(500, Duration.HOUR), # 500 requests per hour
+ RequestRate(1000, Duration.DAY), # 1000 requests per day
+ RequestRate(10000, Duration.MONTH), # 10000 requests per month
+)
+
+limiter = Limiter(*rate_limits)
+```
+
+Note that these rates need to be ordered by interval length; in other words, an hourly rate must
+come before a daily rate, etc.
+
+### Applying rate limits
+Then, use `Limiter.try_acquire()` wherever you are making requests (or other rate-limited operations).
+This will raise an exception if the rate limit is exceeded.
+
+```python
+import requests
+
+def request_function():
+ limiter.try_acquire('identity')
+ requests.get('https://example.com')
+
+while True:
+ request_function()
+```
+
+Alternatively, you can use `Limiter.ratelimit()` as a function decorator:
+```python
+@limiter.ratelimit('identity')
+def request_function():
+ requests.get('https://example.com')
+```
+See [Additional usage options](#additional-usage-options) below for more details.
+
+### Identities
+Note that both `try_acquire()` and `ratelimit()` take one or more `identity` arguments. Typically this is
+the name of the service or resource that is being rate-limited. This allows you to track rate limits
+for these resources independently. For example, if you have a service that is rate-limited by user:
+```python
+def request_function(user_ids):
+ limiter.try_acquire(*user_ids)
+ for user_id in user_ids:
+ requests.get(f'https://example.com?user_id={user_id}')
+```
+
+## Handling exceeded limits
+When a rate limit is exceeded, you have two options: raise an exception, or add delays.
+
+### Bucket analogy
+
+
+At this point it's useful to introduce the analogy of "buckets" used for rate-limiting. Here is a
+quick summary:
+
+* This library implements the [Leaky Bucket algorithm](https://en.wikipedia.org/wiki/Leaky_bucket).
+* It is named after the idea of representing some kind of fixed capacity -- like a network or service -- as a bucket.
+* The bucket "leaks" at a constant rate. For web services, this represents the **ideal or permitted request rate**.
+* The bucket is "filled" at an intermittent, unpredicatble rate, representing the **actual rate of requests**.
+* When the bucket is "full", it will overflow, representing **canceled or delayed requests**.
+
+### Rate limit exceptions
+By default, a `BucketFullException` will be raised when a rate limit is exceeded.
+The error contains a `meta_info` attribute with the following information:
+* `identity`: The identity it received
+* `rate`: The specific rate that has been exceeded
+* `remaining_time`: The remaining time until the next request can be sent
+
+Here's an example that will raise an exception on the 4th request:
+```python
+from pyrate_limiter import (Duration, RequestRate,
+ Limiter, BucketFullException)
+
+rate = RequestRate(3, Duration.SECOND)
+limiter = Limiter(rate)
+
+for _ in range(4):
+ try:
+ limiter.try_acquire('vutran')
+ except BucketFullException as err:
+ print(err)
+ # Output: Bucket for vutran with Rate 3/1 is already full
+ print(err.meta_info)
+ # Output: {'identity': 'vutran', 'rate': '3/1', 'remaining_time': 2.9,
+ # 'error': 'Bucket for vutran with Rate 3/1 is already full'}
+```
+
+The rate part of the output is constructed as: `limit / interval`. On the above example, the limit
+is 3 and the interval is 1, hence the `Rate 3/1`.
+
+### Rate limit delays
+You may want to simply slow down your requests to stay within the rate limits instead of canceling
+them. In that case you can use the `delay` argument. Note that this is only available for
+`Limiter.ratelimit()`:
+```python
+@limiter.ratelimit('identity', delay=True)
+def my_function():
+ do_stuff()
+```
+
+If you exceed a rate limit with a long interval (daily, monthly, etc.), you may not want to delay
+that long. In this case, you can set a `max_delay` (in seconds) that you are willing to wait in
+between calls:
+```python
+@limiter.ratelimit('identity', delay=True, max_delay=360)
+def my_function():
+ do_stuff()
+```
+In this case, calls may be delayed by at most 360 seconds to stay within the rate limits; any longer
+than that, and a `BucketFullException` will be raised instead. Without specifying `max_delay`, calls
+will be delayed as long as necessary.
+
+## Additional usage options
+Besides `Limiter.try_acquire()`, some additional usage options are available using `Limiter.ratelimit()`:
+### Decorator
+`Limiter.ratelimit()` can be used as a decorator:
+```python
+@limiter.ratelimit('identity')
+def my_function():
+ do_stuff()
+```
+
+As with `Limiter.try_acquire()`, if calls to the wrapped function exceed the rate limits you
+defined, a `BucketFullException` will be raised.
+
+### Contextmanager
+`Limiter.ratelimit()` also works as a contextmanager:
+
+```python
+def my_function():
+ with limiter.ratelimit('identity', delay=True):
+ do_stuff()
+```
+
+### Async decorator/contextmanager
+`Limiter.ratelimit()` also support async functions, either as a decorator or contextmanager:
+```python
+@limiter.ratelimit('identity', delay=True)
+async def my_function():
+ await do_stuff()
+
+async def my_function():
+ async with limiter.ratelimit('identity'):
+ await do_stuff()
+```
+
+When delays are enabled for an async function, `asyncio.sleep()` will be used instead of `time.sleep()`.
+
+## Backends
+A few different bucket backends are available, which can be selected using the `bucket_class`
+argument for `Limiter`. Any additional backend-specific arguments can be passed
+via `bucket_kwargs`.
+
+### Memory
+The default bucket is stored in memory, backed by a `queue.Queue`. A list implementation is also available:
+```python
+from pyrate_limiter import Limiter, MemoryListBucket
+
+limiter = Limiter(bucket_class=MemoryListBucket)
+```
+
+### SQLite
+If you need to persist the bucket state, a SQLite backend is available.
+
+By default it will store the state in the system temp directory, and you can use
+the `path` argument to use a different location:
+```python
+from pyrate_limiter import Limiter, SQLiteBucket
+
+limiter = Limiter(bucket_class=SQLiteBucket)
+```
+
+By default, the database will be stored in the system temp directory. You can specify a different
+path via `bucket_kwargs`:
+```python
+limiter = Limiter(
+ bucket_class=SQLiteBucket,
+ bucket_kwargs={'path': '/path/to/db.sqlite'},
+)
+```
+
+#### Concurrency
+This backend is thread-safe.
+
+If you want to use SQLite with multiprocessing, some additional protections are needed. For
+these cases, a separate `FileLockSQLiteBucket` class is available. This requires installing the
+[py-filelock](https://py-filelock.readthedocs.io) library.
+```python
+limiter = Limiter(bucket_class=FileLockSQLiteBucket)
+```
+
+### Redis
+If you have a larger, distributed application, Redis is an ideal backend. This
+option requires [redis-py](https://github.com/andymccurdy/redis-py).
+
+Note that this backend requires a `bucket_name` argument, which will be used as a prefix for the
+Redis keys created. This can be used to disambiguate between multiple services using the same Redis
+instance with pyrate-limiter.
+
+**Important**: you might want to consider adding `expire_time` for each buckets. In a scenario where some `identity` produces a request rate that is too sparsed, it is a good practice to expire the bucket which holds such identity's info to save memory.
+
+```python
+from pyrate_limiter import Limiter, RedisBucket, Duration, RequestRate
+
+rates = [
+ RequestRate(5, 10 * Duration.SECOND),
+ RequestRate(8, 20 * Duration.SECOND),
+]
+
+limiter = Limiter(
+ *rates
+ bucket_class=RedisBucket,
+ bucket_kwargs={
+ 'bucket_name':
+ 'my_service',
+ 'expire_time': rates[-1].interval,
+ },
+)
+
+```
+
+#### Connection settings
+If you need to pass additional connection settings, you can use the `redis_pool` bucket argument:
+```python
+from redis import ConnectionPool
+
+redis_pool = ConnectionPool(host='localhost', port=6379, db=0)
+
+rate = RequestRate(5, 10 * Duration.SECOND)
+
+limiter = Limiter(
+ rate,
+ bucket_class=RedisBucket,
+ bucket_kwargs={'redis_pool': redis_pool, 'bucket_name': 'my_service'},
+)
+```
+
+#### Redis clusters
+Redis clusters are also supported, which requires
+[redis-py-cluster](https://github.com/Grokzen/redis-py-cluster):
+```python
+from pyrate_limiter import Limiter, RedisClusterBucket
+
+limiter = Limiter(bucket_class=RedisClusterBucket)
+```
+
+### Custom backends
+If these don't suit your needs, you can also create your own bucket backend by extending `pyrate_limiter.bucket.AbstractBucket`.
+
+
+## Additional features
+
+### Time sources
+By default, monotonic time is used, to ensure requests are always logged in the correct order.
+
+You can specify a custom time source with the `time_function` argument. For example, you may want to
+use the current UTC time for consistency across a distributed application using a Redis backend.
+```python
+from datetime import datetime
+from pyrate_limiter import Duration, Limiter, RequestRate
+
+rate = RequestRate(5, Duration.SECOND)
+limiter_datetime = Limiter(rate, time_function=lambda: datetime.utcnow().timestamp())
+```
+
+Or simply use the basic `time.time()` function:
+```python
+from time import time
+
+rate = RequestRate(5, Duration.SECOND)
+limiter_time = Limiter(rate, time_function=time)
+```
+
+## Examples
+To prove that pyrate-limiter is working as expected, here is a complete example to demonstrate
+rate-limiting with delays:
+```python
+from time import perf_counter as time
+from pyrate_limiter import Duration, Limiter, RequestRate
+
+limiter = Limiter(RequestRate(5, Duration.SECOND))
+n_requests = 27
+
+@limiter.ratelimit("test", delay=True)
+def limited_function(start_time):
+ print(f"t + {(time() - start_time):.5f}")
+
+start_time = time()
+for _ in range(n_requests):
+ limited_function(start_time)
+
+print(f"Ran {n_requests} requests in {time() - start_time:.5f} seconds")
+```
+
+And an equivalent example for async usage:
+```python
+import asyncio
+from time import perf_counter as time
+from pyrate_limiter import Duration, Limiter, RequestRate
+
+limiter = Limiter(RequestRate(5, Duration.SECOND))
+n_requests = 27
+
+@limiter.ratelimit("test", delay=True)
+async def limited_function(start_time):
+ print(f"t + {(time() - start_time):.5f}")
+
+async def test_ratelimit():
+ start_time = time()
+ tasks = [limited_function(start_time) for _ in range(n_requests)]
+ await asyncio.gather(*tasks)
+ print(f"Ran {n_requests} requests in {time() - start_time:.5f} seconds")
+
+asyncio.run(test_ratelimit())
+```
diff --git a/setup.cfg b/setup.cfg
index 0b6005c..58bab51 100644
--- a/setup.cfg
+++ b/setup.cfg
@@ -135,6 +135,7 @@ description = run the tests with pytest
package = wheel
deps =
pytest>=7
+ gui,all: pytest-qt
extras =
7z: 7Z
cbr: CBR
@@ -150,17 +151,16 @@ description = run the tests with pytest
package = wheel
deps =
pytest>=7
- icu,all: pyicu-binary
+ gui,all: pytest-qt
extras =
7z: 7Z
cbr: CBR
gui: GUI
- all: 7Z,CBR,GUI
+ icu: ICU
+ all: 7Z,CBR,GUI,ICU
commands =
python -m pytest {tty:--color=yes} {posargs}
-
-[testenv:py3.9-{icu,all}]
-base = {env:tox_env:testenv}
+ icu,all: python -c 'import importlib,platform; importlib.import_module("icu") if platform.system() != "Windows" else ...' # Sanity check for icu
[testenv:format]
labels =
@@ -247,7 +247,7 @@ description = Generate pyinstaller executable
labels =
build
release
-base = {env:tox_env:testenv}
+base = testenv
depends =
clean
deps =
@@ -263,7 +263,7 @@ commands =
description = Generate appimage executable
skip_install = true
platform = linux
-base = {env:tox_env:testenv}
+base = testenv
labels =
release
build
diff --git a/comictaggerlib/ui/pyqttoast/tests/toast_test.py b/tests/pyqttoast_test.py
similarity index 99%
rename from comictaggerlib/ui/pyqttoast/tests/toast_test.py
rename to tests/pyqttoast_test.py
index 40dd0c5..602aa27 100644
--- a/comictaggerlib/ui/pyqttoast/tests/toast_test.py
+++ b/tests/pyqttoast_test.py
@@ -1,18 +1,17 @@
from __future__ import annotations
-import os
from unittest.mock import patch
import pytest
+
+pytest.importorskip("PyQt5")
from PyQt5.QtCore import QMargins, QRect, QSize, Qt
from PyQt5.QtGui import QColor, QFont, QGuiApplication, QPixmap
from PyQt5.QtWidgets import QMainWindow
-from .. import Toast, ToastButtonAlignment, ToastIcon, ToastPosition, ToastPreset
-from ..constants import DROP_SHADOW_SIZE
-from ..icons import icon_path
-
-ROOT_PATH = os.path.abspath(os.curdir)
+from comictaggerlib.ui.pyqttoast import Toast, ToastButtonAlignment, ToastIcon, ToastPosition, ToastPreset
+from comictaggerlib.ui.pyqttoast.constants import DROP_SHADOW_SIZE
+from comictaggerlib.ui.pyqttoast.icons import icon_path
@pytest.fixture(autouse=True)