2022-06-28 07:21:35 -07:00
|
|
|
"""A template for an information source
|
|
|
|
"""
|
|
|
|
# Copyright 2012-2014 Anthony Beville
|
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
import logging
|
|
|
|
from typing import Any, Callable
|
|
|
|
|
2022-10-05 16:34:32 -07:00
|
|
|
from typing_extensions import Required, TypedDict
|
2022-06-28 07:21:35 -07:00
|
|
|
|
|
|
|
from comicapi.genericmetadata import GenericMetadata
|
|
|
|
from comictalker.resulttypes import ComicIssue, ComicVolume
|
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
# NOTE: Series and Volume are synonymous. Some sources (ComicVine) use "volume" and others (MangaUpdates) use "series".
|
|
|
|
|
|
|
|
|
|
|
|
class SourceDetails:
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
name: str,
|
|
|
|
ident: str,
|
|
|
|
static_options: SourceStaticOptions,
|
|
|
|
settings_options: dict[str, SourceSettingsOptions],
|
|
|
|
):
|
|
|
|
self.name = name
|
|
|
|
self.id = ident
|
|
|
|
self.static_options = static_options
|
|
|
|
self.settings_options = settings_options
|
|
|
|
|
|
|
|
|
|
|
|
class SourceStaticOptions:
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
logo_url: str = "",
|
|
|
|
has_issues: bool = False,
|
|
|
|
has_alt_covers: bool = False,
|
|
|
|
requires_apikey: bool = False,
|
|
|
|
has_nsfw: bool = False,
|
|
|
|
has_censored_covers: bool = False,
|
|
|
|
) -> None:
|
|
|
|
self.logo_url = logo_url
|
|
|
|
self.has_issues = has_issues
|
|
|
|
self.has_alt_covers = has_alt_covers
|
|
|
|
self.requires_apikey = requires_apikey
|
|
|
|
self.has_nsfw = has_nsfw
|
|
|
|
self.has_censored_covers = has_censored_covers
|
|
|
|
|
|
|
|
|
|
|
|
class SourceSettingsOptions(TypedDict):
|
|
|
|
# Source settings options and used to generate settings options in panel
|
|
|
|
name: Required[str] # Internal name for setting i.e "remove_html_tables"
|
|
|
|
text: Required[str] # Display text i.e "Remove HTML tables"
|
|
|
|
help_text: str # Tooltip text i.e "Enabling this will remove HTML tables from the description."
|
|
|
|
hidden: Required[bool] # To hide an option from the settings menu.
|
2022-10-19 11:33:51 -07:00
|
|
|
type: Required[type[bool] | type[int] | type[str] | type[float]]
|
2022-06-28 07:21:35 -07:00
|
|
|
value: Any
|
|
|
|
|
|
|
|
|
|
|
|
class TalkerError(Exception):
|
|
|
|
"""Base class exception for information sources.
|
|
|
|
|
|
|
|
Attributes:
|
|
|
|
code -- a numerical code
|
|
|
|
1 - General
|
|
|
|
2 - Network
|
|
|
|
3 - Data
|
|
|
|
desc -- description of the error
|
|
|
|
source -- the name of the source producing the error
|
|
|
|
"""
|
|
|
|
|
|
|
|
codes = {
|
|
|
|
1: "General",
|
|
|
|
2: "Network",
|
|
|
|
3: "Data",
|
|
|
|
4: "Other",
|
|
|
|
}
|
|
|
|
|
|
|
|
def __init__(self, source: str = "", code: int = 4, desc: str = "", sub_code: int = 0) -> None:
|
|
|
|
super().__init__()
|
|
|
|
if desc == "":
|
|
|
|
desc = "Unknown"
|
|
|
|
self.desc = desc
|
|
|
|
self.code = code
|
|
|
|
self.code_name = self.codes[code]
|
|
|
|
self.sub_code = sub_code
|
|
|
|
self.source = source
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return f"{self.source} encountered a {self.code_name} error. {self.desc}"
|
|
|
|
|
|
|
|
|
|
|
|
class TalkerNetworkError(TalkerError):
|
|
|
|
"""Network class exception for information sources
|
|
|
|
|
|
|
|
Attributes:
|
|
|
|
sub_code -- numerical code for finer detail
|
|
|
|
1 -- connected refused
|
|
|
|
2 -- api key
|
|
|
|
3 -- rate limit
|
|
|
|
4 -- timeout
|
|
|
|
"""
|
|
|
|
|
|
|
|
net_codes = {
|
|
|
|
0: "General network error.",
|
|
|
|
1: "The connection was refused.",
|
|
|
|
2: "An API key error occurred.",
|
|
|
|
3: "Rate limit exceeded. Please wait a bit or enter a personal key if using the default.",
|
|
|
|
4: "The connection timed out.",
|
|
|
|
5: "Number of retries exceeded.",
|
|
|
|
}
|
|
|
|
|
|
|
|
def __init__(self, source: str = "", sub_code: int = 0, desc: str = "") -> None:
|
|
|
|
if desc == "":
|
|
|
|
desc = self.net_codes[sub_code]
|
|
|
|
|
|
|
|
super().__init__(source, 2, desc, sub_code)
|
|
|
|
|
|
|
|
|
|
|
|
class TalkerDataError(TalkerError):
|
|
|
|
"""Data class exception for information sources
|
|
|
|
|
|
|
|
Attributes:
|
|
|
|
sub_code -- numerical code for finer detail
|
|
|
|
1 -- unexpected data
|
|
|
|
2 -- malformed data
|
|
|
|
3 -- missing data
|
|
|
|
"""
|
|
|
|
|
|
|
|
data_codes = {
|
|
|
|
0: "General data error.",
|
|
|
|
1: "Unexpected data encountered.",
|
|
|
|
2: "Malformed data encountered.",
|
|
|
|
3: "Missing data encountered.",
|
|
|
|
}
|
|
|
|
|
|
|
|
def __init__(self, source: str = "", sub_code: int = 0, desc: str = "") -> None:
|
|
|
|
if desc == "":
|
|
|
|
desc = self.data_codes[sub_code]
|
|
|
|
|
|
|
|
super().__init__(source, 3, desc, sub_code)
|
|
|
|
|
|
|
|
|
|
|
|
# Class talkers instance
|
|
|
|
class TalkerBase:
|
|
|
|
"""This is the class for mysource."""
|
|
|
|
|
|
|
|
def __init__(self) -> None:
|
|
|
|
# Identity name for the information source etc.
|
|
|
|
self.source_details: SourceDetails | None = None # Can use this to test if custom talker has been configured
|
|
|
|
|
|
|
|
self.log_func: Callable[[str], None] | None = None
|
|
|
|
|
|
|
|
def set_log_func(self, log_func: Callable[[str], None]) -> None:
|
|
|
|
self.log_func = log_func
|
|
|
|
|
|
|
|
# issueidentifier will set_log_func to print any write_log to console otherwise logger.info is not printed
|
|
|
|
def write_log(self, text: str) -> None:
|
|
|
|
if self.log_func is None:
|
|
|
|
logger.info(text)
|
|
|
|
else:
|
|
|
|
self.log_func(text)
|
|
|
|
|
|
|
|
def check_api_key(self, key: str, url: str):
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
# Search for series/volumes
|
|
|
|
def search_for_series(
|
|
|
|
self,
|
|
|
|
series_name: str,
|
|
|
|
callback: Callable[[int, int], None] | None = None,
|
|
|
|
refresh_cache: bool = False,
|
|
|
|
literal: bool = False,
|
|
|
|
) -> list[ComicVolume]:
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
# Get issues in a series/volume
|
|
|
|
def fetch_issues_by_volume(self, series_id: int) -> list[ComicIssue]:
|
|
|
|
raise NotImplementedError
|
|
|
|
|
2022-10-13 17:10:46 -07:00
|
|
|
# Get issue or volume information
|
|
|
|
def fetch_comic_data(self, series_id: int, issue_number: str = "") -> GenericMetadata:
|
2022-10-24 16:37:18 -07:00
|
|
|
"""This function is expected to handle a few possibilities:
|
|
|
|
1. Only series_id. Retrieve the SERIES/VOLUME information only.
|
|
|
|
2. series_id and issue_number. Retrieve the ISSUE information.
|
2022-10-25 16:29:30 -07:00
|
|
|
3. Only issue_id. Retrieve the ISSUE information."""
|
2022-06-28 07:21:35 -07:00
|
|
|
raise NotImplementedError
|
|
|
|
|
2022-10-24 08:30:58 -07:00
|
|
|
def fetch_alternate_cover_urls(self, issue_id: int) -> list[str]:
|
2022-06-28 07:21:35 -07:00
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
def fetch_issues_by_volume_issue_num_and_year(
|
|
|
|
self, volume_id_list: list[int], issue_number: str, year: str | int | None
|
|
|
|
) -> list[ComicIssue]:
|
|
|
|
raise NotImplementedError
|