From 9d901ef9e07c5c2a8f0d2943b0dc1ada85383f9b Mon Sep 17 00:00:00 2001 From: LEv145 Date: Sat, 8 Jan 2022 14:40:31 +0200 Subject: [PATCH 1/5] Update errors --- aiontai/api.py | 30 ++++++++++++++---------------- 1 file changed, 14 insertions(+), 16 deletions(-) diff --git a/aiontai/api.py b/aiontai/api.py index a54f10f..ce93393 100644 --- a/aiontai/api.py +++ b/aiontai/api.py @@ -150,12 +150,12 @@ async def search( Doujins raw data result from responce. Raises: - WrongPageError: If number of page is invalid. + ValueError: If number of page is invalid. EmptyAPIResultError: If api result is empty. ClientResponseError: Error from response. """ if page < 1: - raise WrongPageError("Page can not be less than 1") + raise ValueError("Page can not be less than 1") url = "https://nhentai.net/api/galleries/search" params = { @@ -193,15 +193,14 @@ async def search_by_tag( Doujins raw data result from responce. Raises: - WrongPageError: If number of page is invalid. - WrongTagError: If tag ID is invalid. + ValueError: If number of page is invalid or tag ID is invalid. EmptyAPIResultError: If api result is empty. ClientResponseError: Error from response. """ if page < 1: - raise WrongPageError("Page can not be less than 1") + raise ValueError("Page can not be less than 1") elif tag_id < 1: - raise WrongTagError("Tag id can not be less than 1") + raise ValueError("Tag id can not be less than 1") url = "https://nhentai.net/api/galleries/tagged" @@ -265,6 +264,9 @@ async def _request( url, **kwargs, ) + if not response.ok: + response.release() + ... response.raise_for_status() try: @@ -320,21 +322,17 @@ async def _request( # return [doujin for page in pages for doujin in page] -class WrongPageError(Exception): - """Wrong page.""" +class NHentaiError(Exception): + """Base NHentai api error.""" -class WrongSearchError(Exception): - """Wrong search.""" +class HTTPError(NHentaiError): # TODO + """Error from responce.""" -class WrongTagError(Exception): - """Wrong tag.""" - - -class DoujinDoesNotExistError(Exception): +class DoujinDoesNotExistError(NHentaiError): """Doujin does noe exist.""" -class EmptyAPIResultError(Exception): +class EmptyAPIResultError(NHentaiError): """API result is empty.""" From 14e9993ce3bb809ad53faf22b031a3315980e22c Mon Sep 17 00:00:00 2001 From: LEv145 Date: Sat, 8 Jan 2022 16:40:46 +0200 Subject: [PATCH 2/5] Add `HttpError`, rewrite `api.NhentaiAPI_requests` --- aiontai/__init__.py | 6 --- aiontai/api.py | 92 +++++++++++++++------------------------------ aiontai/client.py | 17 ++++----- tests/test_api.py | 50 +++++++++--------------- 4 files changed, 57 insertions(+), 108 deletions(-) diff --git a/aiontai/__init__.py b/aiontai/__init__.py index 43ff9a6..60fd533 100644 --- a/aiontai/__init__.py +++ b/aiontai/__init__.py @@ -3,9 +3,6 @@ EmptyAPIResultError, NHentaiAPI, SortOptions, - WrongPageError, - WrongSearchError, - WrongTagError, ) from .client import ( NHentaiClient, @@ -41,7 +38,4 @@ "Tag", "TagType", "Title", - "WrongPageError", - "WrongSearchError", - "WrongTagError", ] diff --git a/aiontai/api.py b/aiontai/api.py index ce93393..c8ac6d1 100644 --- a/aiontai/api.py +++ b/aiontai/api.py @@ -8,13 +8,13 @@ Any, AsyncIterator, Dict, + Optional, Type, TypeVar, ) from aiohttp import ( ClientResponse, - ClientResponseError, ClientSession, ) @@ -73,16 +73,18 @@ async def get_doujin(self, doujin_id: int) -> Dict[str, Any]: Raises: DoujinDoesNotExistError: If doujin does not exit. - ClientResponseError: Error from response. + HTTPError: Error from response. """ url = f"https://nhentai.net/api/gallery/{doujin_id}" try: async with self._request("GET", url=url) as response: json: Dict[str, Any] = await response.json() - except ClientResponseError as error: - if error.status == 404: - raise DoujinDoesNotExistError("That doujin does not exist.") from error + except HTTPError as error: + if error.responce.status == 404: + raise DoujinDoesNotExistError( + "That doujin does not exist.", + ) from error else: raise error @@ -99,7 +101,7 @@ async def is_exist(self, doujin_id: int) -> bool: Doujin is exists. Raises: - ClientResponseError: Error from response. + HTTPError: Error from response. """ try: await self.get_doujin(doujin_id) @@ -115,7 +117,7 @@ async def get_random_doujin(self) -> Dict[str, Any]: Doujin raw data from responce. Raises: - ClientResponseError: Error from response. + HTTPError: Error from response. """ url = "https://nhentai.net/random/" @@ -152,7 +154,7 @@ async def search( Raises: ValueError: If number of page is invalid. EmptyAPIResultError: If api result is empty. - ClientResponseError: Error from response. + HTTPError: Error from response. """ if page < 1: raise ValueError("Page can not be less than 1") @@ -195,7 +197,7 @@ async def search_by_tag( Raises: ValueError: If number of page is invalid or tag ID is invalid. EmptyAPIResultError: If api result is empty. - ClientResponseError: Error from response. + HTTPError: Error from response. """ if page < 1: raise ValueError("Page can not be less than 1") @@ -235,7 +237,7 @@ async def get_homepage_doujins( Raises: EmptyAPIResultError: If api result is empty. - ClientResponseError: Error from response. + HTTPError: Error from response. """ url = "https://nhentai.net/api/galleries/all" @@ -264,10 +266,10 @@ async def _request( url, **kwargs, ) - if not response.ok: + + if response.status >= 400: response.release() - ... - response.raise_for_status() + raise HTTPError(response=response) try: yield response @@ -275,59 +277,27 @@ async def _request( await response.__aexit__(None, None, None) -# TODO -# async def search_all_by_tags(self, tag_ids: list) -> List[dict]: -# """Method for search doujins by tags. -# Args: -# :tag_ids list: List of tags - -# Returns: -# List of doujins JSON - -# Raises: -# IsNotValidSort if sort is not a member of SortOptions. -# WrongPage if page less than 1. -# """ - -# async def get_limit(tag_id: int) -> List[dict]: -# utils.is_valid_search_by_tag_parameters(tag_id, 1, "date") - -# url = f"{config.api_gallery_url}/tagged" -# params = { -# "tag_id": tag_id, -# "page": 1, -# "sort_by": "date" -# } - -# result = await self._get_requests(url, params=params) -# if result: -# return result["num_pages"] -# else: -# raise errors.WrongTag("There is no tag with given tag_id") - - -# limits = await asyncio.gather(*[get_limit(tag_id) for tag_id in tag_ids]) -# limits = zip(tag_ids, limits) - -# data = [] - -# for args in limits: -# limits = args[1] -# tag_ids = args[0] -# for i in range(1, limits+1): -# data.append((tag_ids, i)) - +class NHentaiError(Exception): + """Base NHentai api error.""" -# pages = await asyncio.gather(*[self.search_by_tag(*args) for args in data]) -# return [doujin for page in pages for doujin in page] +class HTTPError(NHentaiError): + """ + Error from responce. -class NHentaiError(Exception): - """Base NHentai api error.""" + Attributes: + responce: Responce object. + message: Message about error. + """ + def __init__( + self, + response: ClientResponse, + message: Optional[str] = None, + ) -> None: + super().__init__(message) -class HTTPError(NHentaiError): # TODO - """Error from responce.""" + self.responce = response class DoujinDoesNotExistError(NHentaiError): diff --git a/aiontai/client.py b/aiontai/client.py index 7035ba0..c9d1526 100644 --- a/aiontai/client.py +++ b/aiontai/client.py @@ -70,7 +70,7 @@ async def get_doujin(self, doujin_id: int) -> Doujin: Raises: DoujinDoesNotExistError: If doujin does not exit. - ClientResponseError: Error from response. + HTTPError: Error from response. """ raw_data = await self.api.get_doujin(doujin_id) @@ -87,7 +87,7 @@ async def is_exist(self, doujin_id: int) -> bool: Doujin is exists. Raises: - ClientResponseError: Error from response. + HTTPError: Error from response. """ raw_data = await self.api.is_exist(doujin_id) @@ -101,7 +101,7 @@ async def get_random_doujin(self) -> Doujin: Doujin model. Raises: - ClientResponseError: Error from response. + HTTPError: Error from response. """ raw_data = await self.api.get_random_doujin() @@ -128,9 +128,9 @@ async def search( Doujins result model. Raises: - WrongPageError: If number of page is invalid. + ValueError: If number of page is invalid. EmptyAPIResultError: If api result is empty. - ClientResponseError: Error from response. + HTTPError: Error from response. """ result = await self.api.search( query=query, @@ -161,10 +161,9 @@ async def search_by_tag( Doujins result model. Raises: - WrongPageError: If number of page is invalid. - WrongTagError: If tag ID is invalid. + ValueError: If number of page is invalid or tag ID is invalid. EmptyAPIResultError: If api result is empty. - ClientResponseError: Error from response. + HTTPError: Error from response. """ result = await self.api.search_by_tag( tag_id=tag_id, @@ -191,7 +190,7 @@ async def get_homepage_doujins( Raises: EmptyAPIResultError: If api result is empty. - ClientResponseError: Error from response. + HTTPError: Error from response. """ result = await self.api.get_homepage_doujins( page=page, diff --git a/tests/test_api.py b/tests/test_api.py index 5b44eab..b51f3fa 100644 --- a/tests/test_api.py +++ b/tests/test_api.py @@ -11,10 +11,9 @@ from aiontai.api import ( DoujinDoesNotExistError, EmptyAPIResultError, + HTTPError, NHentaiAPI, SortOptions, - WrongPageError, - WrongTagError, ) @@ -36,7 +35,7 @@ async def test__context_manager(self) -> None: async def test__request(self) -> None: # Normal test - self.response_mosk.raise_for_status = Mock() + self.response_mosk.status = 200 async with self.api._request("GET", "https://nhentai.net/"): ... @@ -47,7 +46,7 @@ async def test__get_doujin(self) -> None: raw_data = json.load(fp) self.response_mosk.json.return_value = raw_data - self.response_mosk.raise_for_status = Mock() + self.response_mosk.status = 200 self.assertEqual( await self.api.get_doujin(123), @@ -55,11 +54,7 @@ async def test__get_doujin(self) -> None: ) # Test error: DoujinDoesNotExistError - self.response_mosk.raise_for_status.side_effect = ClientResponseError( - request_info=Mock(), - history=Mock(), - status=404, - ) + self.response_mosk.status = 404 with self.assertRaises(DoujinDoesNotExistError): self.assertEqual( @@ -67,14 +62,10 @@ async def test__get_doujin(self) -> None: raw_data, ) - # Test error: ClientResponseError - self.response_mosk.raise_for_status.side_effect = ClientResponseError( - request_info=Mock(), - history=Mock(), - status=201, - ) + # Test error: HTTPError + self.response_mosk.status = 500 - with self.assertRaises(ClientResponseError): + with self.assertRaises(HTTPError): self.assertEqual( await self.api.get_doujin(123), raw_data, @@ -86,7 +77,7 @@ async def test__is_exist(self) -> None: raw_data = json.load(fp) self.response_mosk.json.return_value = raw_data - self.response_mosk.raise_for_status = Mock() + self.response_mosk.status = 200 self.assertEqual( await self.api.is_exist(123), @@ -94,11 +85,7 @@ async def test__is_exist(self) -> None: ) # Test error: DoujinDoesNotExistError - self.response_mosk.raise_for_status.side_effect = ClientResponseError( - request_info=Mock(), - history=Mock(), - status=404, - ) + self.response_mosk.status = 404 self.assertEqual( await self.api.is_exist(123), @@ -114,7 +101,7 @@ async def test__get_random_doujin(self) -> None: return_value="https://nhentai.net/g/123/", ) self.response_mosk.json.return_value = raw_data - self.response_mosk.raise_for_status = Mock() + self.response_mosk.status = 200 self.assertEqual( await self.api.get_random_doujin(), @@ -127,7 +114,7 @@ async def test__search(self) -> None: raw_data = json.load(fp) self.response_mosk.json.return_value = raw_data - self.response_mosk.raise_for_status = Mock() + self.response_mosk.status = 200 self.assertEqual( await self.api.search( @@ -138,8 +125,8 @@ async def test__search(self) -> None: raw_data, ) - # Test error: WrongPageError - with self.assertRaises(WrongPageError): + # Test error: ValueError + with self.assertRaises(ValueError): await self.api.search( query="Omakehon 2005", page=-1, @@ -166,7 +153,7 @@ async def test__search_by_tag(self) -> None: raw_data = json.load(fp) self.response_mosk.json.return_value = raw_data - self.response_mosk.raise_for_status = Mock() + self.response_mosk.status = 200 self.assertEqual( await self.api.search_by_tag( @@ -177,16 +164,15 @@ async def test__search_by_tag(self) -> None: raw_data, ) - # Test error: WrongPageError - with self.assertRaises(WrongPageError): + # Test error: ValueError + with self.assertRaises(ValueError): await self.api.search_by_tag( tag_id=7752, page=-1, sort_by=SortOptions.DATE, ) - # Test error: WrongTagError - with self.assertRaises(WrongTagError): + with self.assertRaises(ValueError): await self.api.search_by_tag( tag_id=-1, page=1, @@ -212,7 +198,7 @@ async def test__get_homepage_doujins(self) -> None: raw_data = json.load(fp) self.response_mosk.json.return_value = raw_data - self.response_mosk.raise_for_status = Mock() + self.response_mosk.status = 200 self.assertEqual( await self.api.get_homepage_doujins( From cf80d25bae1df7bda437ae874484d1f4d7adfb40 Mon Sep 17 00:00:00 2001 From: LEv145 Date: Sat, 8 Jan 2022 16:41:23 +0200 Subject: [PATCH 3/5] Remove unused import --- tests/test_api.py | 2 -- 1 file changed, 2 deletions(-) diff --git a/tests/test_api.py b/tests/test_api.py index b51f3fa..69c2823 100644 --- a/tests/test_api.py +++ b/tests/test_api.py @@ -6,8 +6,6 @@ Mock, ) -from aiohttp import ClientResponseError - from aiontai.api import ( DoujinDoesNotExistError, EmptyAPIResultError, From 469b7469f30b51b7ec2be4f54231689b01e6fc45 Mon Sep 17 00:00:00 2001 From: LEv145 Date: Sat, 8 Jan 2022 17:13:15 +0200 Subject: [PATCH 4/5] Fix `ID` naming --- aiontai/api.py | 2 +- aiontai/client.py | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/aiontai/api.py b/aiontai/api.py index c8ac6d1..56ba366 100644 --- a/aiontai/api.py +++ b/aiontai/api.py @@ -63,7 +63,7 @@ async def close(self) -> None: async def get_doujin(self, doujin_id: int) -> Dict[str, Any]: """ - Get doujin raw data by id. + Get doujin raw data by ID. Args: doujin_id: ID of doujin. diff --git a/aiontai/client.py b/aiontai/client.py index c9d1526..cf04e36 100644 --- a/aiontai/client.py +++ b/aiontai/client.py @@ -60,7 +60,7 @@ async def close(self) -> None: async def get_doujin(self, doujin_id: int) -> Doujin: """ - Get doujin model by id. + Get doujin model by ID. Args: doujin_id: ID of doujin. From 3b16b40144950bf6832a928ab318da8d00d99b4c Mon Sep 17 00:00:00 2001 From: LEv145 Date: Sat, 8 Jan 2022 17:13:53 +0200 Subject: [PATCH 5/5] Update version --- pyproject.toml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pyproject.toml b/pyproject.toml index 8dc230f..7a8b28a 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,6 +1,6 @@ [tool.poetry] name = "aiontai" -version = "1.0.10" +version = "1.0.11" description = "Async wrapper for nhentai API" authors = ["LEv145"] license = "MIT"