2022-06-28 07:21:35 -07:00
|
|
|
# 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
|
2022-11-25 15:47:05 -08:00
|
|
|
import pathlib
|
2023-01-29 17:59:23 -08:00
|
|
|
from typing import Any, Callable
|
2023-01-01 17:04:15 -08:00
|
|
|
|
|
|
|
import settngs
|
2022-06-28 07:21:35 -07:00
|
|
|
|
|
|
|
from comicapi.genericmetadata import GenericMetadata
|
2022-12-21 17:00:59 -08:00
|
|
|
from comictalker.resulttypes import ComicIssue, ComicSeries
|
2022-06-28 07:21:35 -07:00
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
class SourceDetails:
|
|
|
|
def __init__(
|
|
|
|
self,
|
2022-10-27 15:36:57 -07:00
|
|
|
name: str = "",
|
|
|
|
ident: str = "",
|
2022-12-22 10:43:00 -08:00
|
|
|
logo: str = "",
|
2022-06-28 07:21:35 -07:00
|
|
|
):
|
|
|
|
self.name = name
|
|
|
|
self.id = ident
|
2022-11-11 17:09:17 -08:00
|
|
|
self.logo = logo
|
2022-06-28 07:21:35 -07:00
|
|
|
|
|
|
|
|
|
|
|
class SourceStaticOptions:
|
|
|
|
def __init__(
|
|
|
|
self,
|
2022-11-11 17:09:17 -08:00
|
|
|
website: str = "",
|
2023-02-01 07:39:24 -08:00
|
|
|
attribution_string: str = "", # Full string including web link, example: Metadata provided by <a href='http://website'>Example</a>
|
2022-06-28 07:21:35 -07:00
|
|
|
has_issues: bool = False,
|
|
|
|
has_alt_covers: bool = False,
|
|
|
|
requires_apikey: bool = False,
|
|
|
|
has_nsfw: bool = False,
|
|
|
|
has_censored_covers: bool = False,
|
|
|
|
) -> None:
|
2022-11-11 17:09:17 -08:00
|
|
|
self.website = website
|
2023-01-19 16:29:02 -08:00
|
|
|
self.attribution_string = attribution_string
|
2022-06-28 07:21:35 -07:00
|
|
|
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 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
|
|
|
|
"""
|
|
|
|
|
2022-12-15 20:21:53 -08:00
|
|
|
codes = {1: "General", 2: "Network", 3: "Data", 4: "Other"}
|
2022-06-28 07:21:35 -07:00
|
|
|
|
2022-11-25 15:47:05 -08:00
|
|
|
def __init__(self, source: str, desc: str, code: int = 4, sub_code: int = 0) -> None:
|
2022-06-28 07:21:35 -07:00
|
|
|
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
|
|
|
|
|
2022-11-25 15:47:05 -08:00
|
|
|
def __str__(self) -> str:
|
2022-06-28 07:21:35 -07:00
|
|
|
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]
|
|
|
|
|
2022-11-25 15:47:05 -08:00
|
|
|
super().__init__(source, desc, 2, sub_code)
|
2022-06-28 07:21:35 -07:00
|
|
|
|
|
|
|
|
|
|
|
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]
|
|
|
|
|
2022-11-25 15:47:05 -08:00
|
|
|
super().__init__(source, desc, 3, sub_code)
|
2022-06-28 07:21:35 -07:00
|
|
|
|
|
|
|
|
|
|
|
# Class talkers instance
|
2022-10-27 15:36:57 -07:00
|
|
|
class ComicTalker:
|
2022-11-24 15:26:48 -08:00
|
|
|
"""The base class for all comic source talkers"""
|
2022-06-28 07:21:35 -07:00
|
|
|
|
2023-01-25 11:10:58 -08:00
|
|
|
def __init__(self, version: str, cache_folder: pathlib.Path) -> None:
|
2022-06-28 07:21:35 -07:00
|
|
|
# Identity name for the information source etc.
|
2022-12-22 10:43:00 -08:00
|
|
|
self.source_details = SourceDetails()
|
|
|
|
self.static_options = SourceStaticOptions()
|
2022-11-25 15:47:05 -08:00
|
|
|
self.cache_folder = cache_folder
|
|
|
|
self.version = version
|
2023-01-25 11:10:58 -08:00
|
|
|
self.api_key: str = ""
|
|
|
|
self.api_url: str = ""
|
2022-11-25 15:47:05 -08:00
|
|
|
|
2023-01-29 17:59:23 -08:00
|
|
|
def register_settings(self, parser: settngs.Manager) -> None:
|
|
|
|
"""Allows registering settings using the settngs package with an argparse like interface"""
|
2023-02-01 16:53:13 -08:00
|
|
|
return None
|
2022-11-25 15:47:05 -08:00
|
|
|
|
2023-01-29 17:59:23 -08:00
|
|
|
def parse_settings(self, settings: dict[str, Any]) -> None:
|
|
|
|
"""settings is a dictionary of options defined in register_settings.
|
|
|
|
It is only guaranteed that the settings defined in register_settings will be present."""
|
2023-02-01 16:53:13 -08:00
|
|
|
return None
|
2022-06-28 07:21:35 -07:00
|
|
|
|
2022-11-24 15:26:48 -08:00
|
|
|
def check_api_key(self, key: str, url: str) -> bool:
|
2022-12-22 10:43:00 -08:00
|
|
|
"""
|
|
|
|
This function should return true if the given api key and url are valid.
|
|
|
|
If the Talker does not use an api key it should validate that the url works.
|
|
|
|
"""
|
2022-06-28 07:21:35 -07:00
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
def search_for_series(
|
|
|
|
self,
|
|
|
|
series_name: str,
|
|
|
|
callback: Callable[[int, int], None] | None = None,
|
|
|
|
refresh_cache: bool = False,
|
|
|
|
literal: bool = False,
|
2022-12-21 17:00:59 -08:00
|
|
|
) -> list[ComicSeries]:
|
2022-12-22 10:43:00 -08:00
|
|
|
"""
|
|
|
|
This function should return a list of series that match the given series name
|
|
|
|
according to the source the Talker uses.
|
|
|
|
Sanitizing the series name is the responsibility of the talker.
|
|
|
|
If `literal` == True then it is requested that no filtering or
|
|
|
|
transformation/sanitizing of the title or results be performed by the talker.
|
|
|
|
A sensible amount of results should be returned.
|
|
|
|
For example the `ComicVineTalker` stops requesting new pages after the results
|
|
|
|
become too different from the `series_name` by use of the `titles_match` function
|
|
|
|
provided by the `comicapi.utils` module, and only allows a maximum of 5 pages
|
|
|
|
"""
|
2022-06-28 07:21:35 -07:00
|
|
|
raise NotImplementedError
|
|
|
|
|
2022-12-23 00:09:19 -08:00
|
|
|
def fetch_issues_by_series(self, series_id: str) -> list[ComicIssue]:
|
2022-12-21 17:00:59 -08:00
|
|
|
"""Expected to return a list of issues with a given series ID"""
|
2022-06-28 07:21:35 -07:00
|
|
|
raise NotImplementedError
|
|
|
|
|
2022-12-22 10:43:00 -08:00
|
|
|
def fetch_comic_data(
|
2022-12-23 00:09:19 -08:00
|
|
|
self, issue_id: str | None = None, series_id: str | None = None, issue_number: str = ""
|
2022-12-22 10:43:00 -08:00
|
|
|
) -> GenericMetadata:
|
|
|
|
"""
|
|
|
|
This function should return an instance of GenericMetadata for a single issue.
|
|
|
|
It is guaranteed that either `issue_id` or (`series_id` and `issue_number` is set).
|
|
|
|
Below is an example of how this function might be implemented:
|
|
|
|
|
|
|
|
if issue_number and series_id:
|
|
|
|
return self.fetch_issue_data(series_id, issue_number)
|
|
|
|
elif issue_id:
|
|
|
|
return self.fetch_issue_data_by_issue_id(issue_id)
|
|
|
|
else:
|
|
|
|
return GenericMetadata()
|
|
|
|
"""
|
2022-06-28 07:21:35 -07:00
|
|
|
raise NotImplementedError
|
|
|
|
|
2022-12-21 17:00:59 -08:00
|
|
|
def fetch_issues_by_series_issue_num_and_year(
|
2022-12-23 00:09:19 -08:00
|
|
|
self, series_id_list: list[str], issue_number: str, year: int | None
|
2022-06-28 07:21:35 -07:00
|
|
|
) -> list[ComicIssue]:
|
2022-12-22 10:43:00 -08:00
|
|
|
"""
|
|
|
|
This function should return a single issue for each series id in
|
|
|
|
the `series_id_list` and it should match the issue_number.
|
|
|
|
Preferably it should also only return issues published in the given `year`.
|
|
|
|
If there is no year given (`year` == None) or the Talker does not have issue publication info
|
|
|
|
return the results unfiltered.
|
|
|
|
"""
|
2022-06-28 07:21:35 -07:00
|
|
|
raise NotImplementedError
|