Update typing

This commit is contained in:
Timmy Welch 2025-03-18 19:49:17 -07:00
parent 6a9d4bf648
commit 575d36b67f
8 changed files with 33 additions and 23 deletions

View File

@ -41,6 +41,6 @@ repos:
rev: v1.15.0
hooks:
- id: mypy
additional_dependencies: [types-setuptools, types-requests, settngs>=0.10.4]
additional_dependencies: [types-setuptools, types-requests, settngs>=0.10.4, pillow>=9.1.0]
ci:
skip: [mypy]

View File

@ -382,7 +382,7 @@ def lex_number(lex: Lexer) -> LexerFunc | None:
return lex_filename
def lex_issue_number(lex: Lexer) -> Callable[[Lexer], Callable | None] | None: # type: ignore[type-arg]
def lex_issue_number(lex: Lexer) -> LexerFunc:
# Only called when lex.input[lex.start] == "#"
original_start = lex.pos
lex.accept_run(str.isalpha)

View File

@ -3,7 +3,7 @@ from __future__ import annotations
import dataclasses
from collections.abc import Collection
from enum import auto
from typing import Any
from typing import Any, Callable
from comicapi.utils import DefaultDict, StrEnum, norm_fold
@ -54,7 +54,7 @@ def overlay(old: Any, new: Any) -> Any:
return new
attribute = DefaultDict(
attribute: DefaultDict[Mode, Callable[[Any, Any], Any]] = DefaultDict(
{
Mode.OVERLAY: overlay,
Mode.ADD_MISSING: lambda old, new: overlay(new, old),
@ -63,7 +63,7 @@ attribute = DefaultDict(
)
lists = DefaultDict(
lists: DefaultDict[Mode, Callable[[Collection[Any], Collection[Any]], list[Any] | set[Any]]] = DefaultDict(
{
Mode.OVERLAY: merge_lists,
Mode.ADD_MISSING: lambda old, new: merge_lists(new, old),

View File

@ -23,7 +23,7 @@ import pathlib
import platform
import sys
import unicodedata
from collections.abc import Iterable, Mapping
from collections.abc import Iterable, Mapping, Sequence
from enum import Enum, auto
from shutil import which # noqa: F401
from typing import Any, Callable, TypeVar, cast
@ -47,7 +47,7 @@ except ImportError:
if sys.version_info < (3, 11):
def file_digest(fileobj, digest, /, *, _bufsize=2**18):
def file_digest(fileobj, digest, /, *, _bufsize=2**18): # type: ignore[no-untyped-def]
"""Hash the contents of a file-like object. Returns a digest object.
*fileobj* must be a file-like object opened for reading in binary mode.
@ -147,12 +147,16 @@ else:
logger = logging.getLogger(__name__)
class DefaultDict(dict):
def __init__(self, *args, default: Callable[[Any], Any] | None = None) -> None:
super().__init__(*args)
_KT = TypeVar("_KT")
_VT = TypeVar("_VT")
class DefaultDict(dict[_KT, _VT]):
def __init__(self, *args, default: Callable[[_KT], _VT | _KT] | None = None, **kwargs) -> None: # type: ignore[no-untyped-def]
super().__init__(*args, **kwargs)
self.default = default
def __missing__(self, key: Any) -> Any:
def __missing__(self, key: _KT) -> _VT | _KT:
if self.default is None:
return key
return self.default(key)
@ -170,7 +174,7 @@ def _custom_key(tup: Any) -> Any:
lst = []
for x in natsort.os_sort_keygen()(tup):
ret = x
if len(x) > 1 and isinstance(x[1], int) and isinstance(x[0], str) and x[0] == "":
if isinstance(x, Sequence) and len(x) > 1 and isinstance(x[1], int) and isinstance(x[0], str) and x[0] == "":
ret = ("a", *x[1:])
lst.append(ret)
@ -623,7 +627,7 @@ def update_publishers(new_publishers: Mapping[str, Mapping[str, str]]) -> None:
publishers[publisher] = ImprintDict(publisher, new_publishers[publisher])
class ImprintDict(dict): # type: ignore
class ImprintDict(dict[str, str]):
"""
ImprintDict takes a publisher and a dict or mapping of lowercased
imprint names to the proper imprint name. Retrieving a value from an
@ -631,14 +635,14 @@ class ImprintDict(dict): # type: ignore
if the key does not exist the key is returned as the publisher unchanged
"""
def __init__(self, publisher: str, mapping: tuple | Mapping = (), **kwargs: dict) -> None: # type: ignore
def __init__(self, publisher: str, mapping: Mapping[str, str] = {}, **kwargs) -> None: # type: ignore[no-untyped-def]
super().__init__(mapping, **kwargs)
self.publisher = publisher
def __missing__(self, key: str) -> None:
return None
def __getitem__(self, k: str) -> tuple[str, str, bool]:
def __getitem__(self, k: str) -> tuple[str, str, bool]: # type: ignore[override]
item = super().__getitem__(k.casefold())
if k.casefold() == self.publisher.casefold():
return "", self.publisher, True

View File

@ -10,7 +10,7 @@ import pathlib
import platform
import re
import sys
from collections.abc import Generator, Iterable
from collections.abc import Generator, Iterable, Sequence
from typing import Any, NamedTuple, TypeVar
if sys.version_info < (3, 10):
@ -31,7 +31,7 @@ def _custom_key(tup: Any) -> Any:
lst = []
for x in natsort.os_sort_keygen()(tup):
ret = x
if len(x) > 1 and isinstance(x[1], int) and isinstance(x[0], str) and x[0] == "":
if isinstance(x, Sequence) and len(x) > 1 and isinstance(x[1], int) and isinstance(x[0], str) and x[0] == "":
ret = ("a", *x[1:])
lst.append(ret)

View File

@ -35,7 +35,12 @@ logger = logging.getLogger(__name__)
class ImageHasher:
def __init__(
self, path: str | None = None, image: Image | None = None, data: bytes = b"", width: int = 8, height: int = 8
self,
path: str | None = None,
image: Image.Image | None = None,
data: bytes = b"",
width: int = 8,
height: int = 8,
) -> None:
self.width = width
self.height = height
@ -141,6 +146,7 @@ class ImageHasher:
row = []
for x in range(width):
pixel = image.getpixel((x, y))
assert isinstance(pixel, float)
row.append(pixel)
pixels2.append(row)

View File

@ -132,7 +132,7 @@ class IssueIdentifier:
def set_cover_url_callback(self, cb_func: Callable[[bytes], None]) -> None:
self.cover_url_callback = cb_func
def calculate_hash(self, image_data: bytes = b"", image: Image = None) -> int:
def calculate_hash(self, image_data: bytes = b"", image: Image.Image | None = None) -> int:
if self.image_hasher == 3:
return ImageHasher(data=image_data, image=image).p_hash()
if self.image_hasher == 2:
@ -377,8 +377,8 @@ class IssueIdentifier:
def _process_cover(self, name: str, image_data: bytes) -> list[tuple[str, Image.Image]]:
assert Image
cover_image = Image.open(io.BytesIO(image_data))
images = [(name, cover_image)]
cover_image: Image.Image = Image.open(io.BytesIO(image_data))
images: list[tuple[str, Image.Image]] = [(name, cover_image)]
# check the aspect ratio
# if it's wider than it is high, it's probably a two page spread (back_cover, front_cover)
@ -413,7 +413,7 @@ class IssueIdentifier:
def _get_search_keys(self, md: GenericMetadata) -> Any:
search_keys = SearchKeys(
series=md.series,
series=md.series or "",
issue_number=IssueString(md.issue).as_string(),
alternate_number=IssueString(md.alternate_number).as_string(),
month=md.month,

View File

@ -258,7 +258,7 @@ all_seed_imprints = {
"Marvel": seed_imprints["Marvel"].copy(),
"DC Comics": additional_seed_imprints["DC Comics"].copy(),
}
all_seed_imprints["Marvel"].update(additional_seed_imprints["Marvel"])
all_seed_imprints["Marvel"].update(additional_seed_imprints["Marvel"].items())
conflicting_seed_imprints = {"Marvel": {"test": "Never"}}