Skip to content

Commit

Permalink
Factorize http error handling
Browse files Browse the repository at this point in the history
  • Loading branch information
frankie567 committed Jul 12, 2024
1 parent 1a92d0d commit 4dcc42d
Show file tree
Hide file tree
Showing 4 changed files with 144 additions and 171 deletions.
50 changes: 17 additions & 33 deletions httpx_oauth/clients/facebook.py
Original file line number Diff line number Diff line change
@@ -1,10 +1,7 @@
import json
from typing import Any, Dict, List, Optional, Tuple, Union, cast

import httpx
from typing import Any, Dict, List, Optional, Tuple, cast

from httpx_oauth.errors import GetIdEmailError
from httpx_oauth.oauth2 import BaseOAuth2, OAuth2Error, OAuth2Token
from httpx_oauth.oauth2 import BaseOAuth2, OAuth2RequestError, OAuth2Token

AUTHORIZE_ENDPOINT = "https://www.facebook.com/v5.0/dialog/oauth"
ACCESS_TOKEN_ENDPOINT = "https://graph.facebook.com/v5.0/oauth/access_token"
Expand All @@ -23,12 +20,7 @@
"""


class GetLongLivedAccessTokenError(OAuth2Error):
def __init__(
self, message: str, response: Union[httpx.Response, None] = None
) -> None:
self.response = response
super().__init__(message)
class GetLongLivedAccessTokenError(OAuth2RequestError): ...


class FacebookOAuth2(BaseOAuth2[Dict[str, Any]]):
Expand All @@ -53,28 +45,20 @@ def __init__(

async def get_long_lived_access_token(self, token: str):
async with self.get_httpx_client() as client:
try:
response = await client.post(
self.access_token_endpoint,
data={
"grant_type": "fb_exchange_token",
"fb_exchange_token": token,
"client_id": self.client_id,
"client_secret": self.client_secret,
},
)
response.raise_for_status()
except httpx.HTTPStatusError as e:
raise GetLongLivedAccessTokenError(str(e), e.response) from e
except httpx.HTTPError as e:
raise GetLongLivedAccessTokenError(str(e)) from e

try:
data = cast(Dict[str, Any], response.json())
except json.decoder.JSONDecodeError as e:
message = "Invalid JSON content"
raise GetLongLivedAccessTokenError(message, response) from e

request, auth = self.build_request(
client,
"POST",
self.access_token_endpoint,
auth_method=self.token_endpoint_auth_method,
data={
"grant_type": "fb_exchange_token",
"fb_exchange_token": token,
},
)
response = await self.send_request(
client, request, auth, exc_class=GetLongLivedAccessTokenError
)
data = self.get_json(response, exc_class=GetLongLivedAccessTokenError)
return OAuth2Token(data)

async def get_id_email(self, token: str) -> Tuple[str, Optional[str]]:
Expand Down
36 changes: 14 additions & 22 deletions httpx_oauth/clients/github.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,3 @@
import json
import urllib.parse
from typing import Any, Dict, List, Optional, Tuple, TypedDict, cast

Expand Down Expand Up @@ -53,22 +52,19 @@ def __init__(
async def refresh_token(self, refresh_token: str):
assert self.refresh_token_endpoint is not None
async with self.get_httpx_client() as client:
try:
response = await client.post(
self.refresh_token_endpoint,
data={
"grant_type": "refresh_token",
"refresh_token": refresh_token,
"client_id": self.client_id,
"client_secret": self.client_secret,
},
headers=self.request_headers,
)
response.raise_for_status()
except httpx.HTTPStatusError as e:
raise RefreshTokenError(str(e), e.response) from e
except httpx.HTTPError as e:
raise RefreshTokenError(str(e)) from e
request, auth = self.build_request(
client,
"POST",
self.refresh_token_endpoint,
auth_method=self.token_endpoint_auth_method,
data={
"grant_type": "refresh_token",
"refresh_token": refresh_token,
},
)
response = await self.send_request(
client, request, auth, exc_class=RefreshTokenError
)

content_type = response.headers.get("content-type", "")

Expand All @@ -78,11 +74,7 @@ async def refresh_token(self, refresh_token: str):
data = urllib.parse.parse_qs(response.text)
raise RefreshTokenError(cast(str, data["error"]), response)

try:
data = cast(Dict[str, Any], response.json())
except json.decoder.JSONDecodeError as e:
message = "Invalid JSON content"
raise RefreshTokenError(message, response) from e
data = self.get_json(response, exc_class=RefreshTokenError)

return OAuth2Token(data)

Expand Down
23 changes: 8 additions & 15 deletions httpx_oauth/clients/naver.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,5 @@
from typing import Any, Dict, List, Optional, Tuple, cast

import httpx

from httpx_oauth.errors import GetIdEmailError
from httpx_oauth.oauth2 import BaseOAuth2, RevokeTokenError

Expand Down Expand Up @@ -55,26 +53,21 @@ async def revoke_token(
async with self.get_httpx_client() as client:
data = {
"grant_type": "delete",
"client_id": self.client_id,
"client_secret": self.client_secret,
"access_token": token,
"service_provider": "NAVER",
}

if token_type_hint is not None:
data["token_type_hint"] = token_type_hint

try:
response = await client.post(
self.revoke_token_endpoint,
data=data,
headers=self.request_headers,
)
response.raise_for_status()
except httpx.HTTPStatusError as e:
raise RevokeTokenError(str(e), e.response) from e
except httpx.HTTPError as e:
raise RevokeTokenError(str(e)) from e
request, auth = self.build_request(
client,
"POST",
self.revoke_token_endpoint,
auth_method=self.token_endpoint_auth_method,
data=data,
)
await self.send_request(client, request, auth, exc_class=RevokeTokenError)

return None

Expand Down
Loading

0 comments on commit 4dcc42d

Please sign in to comment.