usso 0.27.2__py3-none-any.whl → 0.27.5__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
usso/exceptions.py CHANGED
@@ -14,3 +14,4 @@ class USSOException(Exception):
14
14
  if message is None:
15
15
  self.message = error_messages[error]
16
16
  super().__init__(message)
17
+
@@ -1,6 +1,7 @@
1
1
  import logging
2
2
 
3
3
  from fastapi import Request, WebSocket
4
+ from fastapi.responses import JSONResponse
4
5
  from starlette.status import HTTP_401_UNAUTHORIZED
5
6
 
6
7
  from usso.exceptions import USSOException
@@ -73,3 +74,10 @@ def jwt_access_security_ws(websocket: WebSocket, jwt_config=None) -> UserData |
73
74
  )
74
75
 
75
76
  return Usso(jwt_config=jwt_config).user_data_from_token(token)
77
+
78
+
79
+ async def usso_exception_handler(request: Request, exc: USSOException):
80
+ return JSONResponse(
81
+ status_code=exc.status_code,
82
+ content={"message": exc.message, "error": exc.error},
83
+ )
@@ -1,29 +1,33 @@
1
+ import os
2
+
1
3
  import httpx
2
4
 
3
5
  from ..core import is_expired
4
- from .session import BaseUssoSession
6
+ from .base_session import BaseUssoSession
5
7
 
6
8
 
7
9
  class AsyncUssoSession(httpx.AsyncClient, BaseUssoSession):
10
+
8
11
  def __init__(
9
12
  self,
10
- usso_base_url: str | None = None,
11
- api_key: str | None = None,
12
- usso_refresh_url: str | None = None,
13
- refresh_token: str | None = None,
14
- usso_api_key: str | None = None,
13
+ *,
14
+ usso_base_url: str | None = os.getenv("USSO_URL"),
15
+ api_key: str | None = os.getenv("USSO_API_KEY"),
16
+ usso_refresh_url: str | None = os.getenv("USSO_REFRESH_URL"),
17
+ refresh_token: str | None = os.getenv("USSO_REFRESH_TOKEN"),
18
+ usso_api_key: str | None = os.getenv("USSO_ADMIN_API_KEY"),
15
19
  user_id: str | None = None,
16
20
  ):
21
+ httpx.AsyncClient.__init__(self)
17
22
  BaseUssoSession.__init__(
18
23
  self,
19
- usso_base_url,
20
- api_key,
21
- usso_refresh_url,
22
- refresh_token,
23
- usso_api_key,
24
- user_id,
24
+ usso_base_url=usso_base_url,
25
+ api_key=api_key,
26
+ usso_refresh_url=usso_refresh_url,
27
+ refresh_token=refresh_token,
28
+ usso_api_key=usso_api_key,
29
+ user_id=user_id,
25
30
  )
26
- httpx.AsyncClient.__init__(self, headers=self.headers)
27
31
 
28
32
  async def _refresh_api(self):
29
33
  params = {"user_id": self.user_id} if self.user_id else {}
@@ -0,0 +1,69 @@
1
+ import os
2
+ from urllib.parse import urlparse
3
+ from usso.core import is_expired
4
+
5
+
6
+ class BaseUssoSession:
7
+
8
+ def __init__(
9
+ self,
10
+ usso_base_url: str | None = os.getenv("USSO_URL"),
11
+ api_key: str | None = os.getenv("USSO_API_KEY"),
12
+ usso_refresh_url: str | None = os.getenv("USSO_REFRESH_URL"),
13
+ refresh_token: str | None = os.getenv("USSO_REFRESH_TOKEN"),
14
+ usso_api_key: str | None = os.getenv("USSO_ADMIN_API_KEY"),
15
+ user_id: str | None = None,
16
+ ):
17
+ assert (
18
+ usso_base_url or usso_refresh_url
19
+ ), "usso_base_url or usso_refresh_url is required"
20
+ assert (
21
+ refresh_token or api_key or usso_api_key
22
+ ), "refresh_token or api_key or usso_api_key is required"
23
+
24
+ if not usso_base_url:
25
+ url_parts = urlparse(usso_refresh_url)
26
+ usso_base_url = f"{url_parts.scheme}://{url_parts.netloc}"
27
+ if usso_base_url.endswith("/"):
28
+ usso_base_url = usso_base_url[:-1]
29
+
30
+ self.usso_refresh_url = usso_refresh_url or f"{usso_base_url}/auth/refresh"
31
+ self._refresh_token = refresh_token
32
+ self.access_token = None
33
+ self.api_key = api_key
34
+ self.usso_api_key = usso_api_key
35
+ self.user_id = user_id
36
+ self.headers = getattr(self, "headers", {})
37
+ if api_key:
38
+ self.headers.update({"x-api-key": api_key})
39
+
40
+ @property
41
+ def refresh_token(self):
42
+ if self._refresh_token and is_expired(self._refresh_token):
43
+ self._refresh_token = None
44
+
45
+ return self._refresh_token
46
+
47
+ def request(self, method: str, url: str, **kwargs):
48
+ return self._request(method, url, **kwargs)
49
+
50
+ def get(self, url: str, **kwargs):
51
+ return self._request("GET", url, **kwargs)
52
+
53
+ def post(self, url: str, **kwargs):
54
+ return self._request("POST", url, **kwargs)
55
+
56
+ def put(self, url: str, **kwargs):
57
+ return self._request("PUT", url, **kwargs)
58
+
59
+ def patch(self, url: str, **kwargs):
60
+ return self._request("PATCH", url, **kwargs)
61
+
62
+ def delete(self, url: str, **kwargs):
63
+ return self._request("DELETE", url, **kwargs)
64
+
65
+ def head(self, url: str, **kwargs):
66
+ return self._request("HEAD", url, **kwargs)
67
+
68
+ def options(self, url: str, **kwargs):
69
+ return self._request("OPTIONS", url, **kwargs)
usso/session/session.py CHANGED
@@ -1,79 +1,37 @@
1
- from urllib.parse import urlparse
1
+ import os
2
2
 
3
3
  import requests
4
+ from usso.core import is_expired
4
5
 
5
- from ..core import is_expired
6
+ from .base_session import BaseUssoSession
6
7
 
7
8
 
8
- class BaseUssoSession:
9
+ class UssoSession(BaseUssoSession, requests.Session):
10
+
9
11
  def __init__(
10
12
  self,
11
- usso_base_url: str | None = None,
12
- api_key: str | None = None,
13
- usso_refresh_url: str | None = None,
14
- refresh_token: str | None = None,
15
- usso_api_key: str | None = None,
13
+ *,
14
+ usso_base_url: str | None = os.getenv("USSO_URL"),
15
+ api_key: str | None = os.getenv("USSO_API_KEY"),
16
+ usso_refresh_url: str | None = os.getenv("USSO_REFRESH_URL"),
17
+ refresh_token: str | None = os.getenv("USSO_REFRESH_TOKEN"),
18
+ usso_api_key: str | None = os.getenv("USSO_ADMIN_API_KEY"),
16
19
  user_id: str | None = None,
17
20
  ):
18
- assert (
19
- usso_base_url or usso_refresh_url
20
- ), "usso_base_url or usso_refresh_url is required"
21
- assert (
22
- refresh_token or api_key or usso_api_key
23
- ), "refresh_token or api_key or usso_api_key is required"
24
-
25
- if not usso_base_url:
26
- url_parts = urlparse(usso_refresh_url)
27
- usso_base_url = f"{url_parts.scheme}://{url_parts.netloc}"
28
- if usso_base_url.endswith("/"):
29
- usso_base_url = usso_base_url[:-1]
30
-
31
- self.usso_refresh_url = usso_refresh_url or f"{usso_base_url}/auth/refresh"
32
- self._refresh_token = refresh_token
33
- self.session = requests.Session()
34
- self.access_token = None
35
- self.api_key = api_key
36
- self.usso_api_key = usso_api_key
37
- self.user_id = user_id
38
- self.headers = {}
39
- if api_key:
40
- self.headers = {"x-api-key": api_key}
41
- self.session.headers.update(self.headers)
42
-
43
- @property
44
- def refresh_token(self):
45
- if self._refresh_token and is_expired(self._refresh_token):
46
- self._refresh_token = None
47
-
48
- return self._refresh_token
49
-
50
- def request(self, method: str, url: str, **kwargs):
51
- return self._request(method, url, **kwargs)
52
-
53
- def get(self, url: str, **kwargs):
54
- return self._request("GET", url, **kwargs)
55
-
56
- def post(self, url: str, **kwargs):
57
- return self._request("POST", url, **kwargs)
58
-
59
- def put(self, url: str, **kwargs):
60
- return self._request("PUT", url, **kwargs)
61
-
62
- def patch(self, url: str, **kwargs):
63
- return self._request("PATCH", url, **kwargs)
64
-
65
- def delete(self, url: str, **kwargs):
66
- return self._request("DELETE", url, **kwargs)
67
-
68
- def head(self, url: str, **kwargs):
69
- return self._request("HEAD", url, **kwargs)
70
-
71
- def options(self, url: str, **kwargs):
72
- return self._request("OPTIONS", url, **kwargs)
73
-
21
+ requests.Session.__init__(self)
22
+ BaseUssoSession.__init__(
23
+ self,
24
+ usso_base_url=usso_base_url,
25
+ api_key=api_key,
26
+ usso_refresh_url=usso_refresh_url,
27
+ refresh_token=refresh_token,
28
+ usso_api_key=usso_api_key,
29
+ user_id=user_id,
30
+ )
31
+ self.headers.update(self.headers)
74
32
 
75
- class UssoSession(BaseUssoSession):
76
33
  def _refresh_api(self):
34
+ assert self.usso_api_key, "usso_api_key is required"
77
35
  params = {"user_id": self.user_id} if self.user_id else {}
78
36
  response = requests.get(
79
37
  f"{self.usso_refresh_url}/api",
@@ -97,20 +55,17 @@ class UssoSession(BaseUssoSession):
97
55
  )
98
56
  response.raise_for_status()
99
57
  self.access_token = response.json().get("access_token")
100
- self.session.headers.update({"Authorization": f"Bearer {self.access_token}"})
58
+ self.headers.update({"Authorization": f"Bearer {self.access_token}"})
101
59
  return response.json()
102
60
 
103
61
  def get_session(self):
104
62
  if self.api_key:
105
- return self.session
63
+ return self
106
64
 
107
65
  if not self.access_token or is_expired(self.access_token):
108
66
  self._refresh()
109
- return self.session
67
+ return self
110
68
 
111
69
  def _request(self, method: str, url: str, **kwargs):
112
- session = self.get_session()
113
- return session.request(method, url, **kwargs)
114
-
115
- def close(self):
116
- self.session.close()
70
+ self.get_session()
71
+ return requests.Session.request(self, method, url, **kwargs)
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: usso
3
- Version: 0.27.2
3
+ Version: 0.27.5
4
4
  Summary: A plug-and-play client for integrating universal single sign-on (SSO) with Python frameworks, enabling secure and seamless authentication across microservices.
5
5
  Author-email: Mahdi Kiani <mahdikiany@gmail.com>
6
6
  Maintainer-email: Mahdi Kiani <mahdikiany@gmail.com>
@@ -81,7 +81,7 @@ The USSO-Client provides a universal single sign-on (SSO) integration for micros
81
81
  Install the USSO client using pip:
82
82
 
83
83
  ```bash
84
- pip install usso-client
84
+ pip install usso
85
85
  ```
86
86
 
87
87
  To add framework-specific support, use the following commands:
@@ -89,7 +89,13 @@ To add framework-specific support, use the following commands:
89
89
  For FastAPI:
90
90
 
91
91
  ```bash
92
- pip install "usso-client[fastapi]"
92
+ pip install "usso[fastapi]"
93
+ ```
94
+
95
+ For Django:
96
+
97
+ ```bash
98
+ pip install "usso[django]"
93
99
  ```
94
100
 
95
101
  ## Quick Start
@@ -1,7 +1,7 @@
1
1
  usso/__init__.py,sha256=NnOS_S1a-JKTOlGe1nw-kCL3m0y82mA2mDraus7BQ2o,120
2
2
  usso/b64tools.py,sha256=HGQ0E59vzjrQo2-4jrcY03ebtTaYwTtCZ7KgJaEmxO0,610
3
3
  usso/core.py,sha256=KI_61zIxZNDB1QeZsYN7wgKYy54Huhdm2T2WDFq69Sw,8043
4
- usso/exceptions.py,sha256=hawOAuVbvQtjgRfwp1KFZ4SmV7fh720y5Gom9JVA8W8,504
4
+ usso/exceptions.py,sha256=syb8V6ysKivFk52NyPAkeMSMQkNi0h8yxDnz_uI2VkA,505
5
5
  usso/schemas.py,sha256=nYFqBMtnGJw13cSKSIMIZdxKVz3AIbnETDuiENHdl5g,850
6
6
  usso/client/__init__.py,sha256=ilGFrugI7bhGXVIcETdbRAye8S7k2mVjkEeziToVzSs,100
7
7
  usso/client/api.py,sha256=xlDq2nZNpq3mhAvqIbGEfANHNjJpPquSeULBfS7iMJw,5094
@@ -9,13 +9,14 @@ usso/client/async_api.py,sha256=VBmuUsx9vBy-naeiVNhsGgJOTpD1z7VgH_23lazz3_4,5156
9
9
  usso/django/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
10
10
  usso/django/middleware.py,sha256=EEEpHvMQ6QiWw2HY8zQ2Aec0RCATcLWsCKeyiPWJKio,3245
11
11
  usso/fastapi/__init__.py,sha256=0EcdOzb4f3yu9nILIdGWnlyUz-0VaVX2az1e3f2BusI,201
12
- usso/fastapi/integration.py,sha256=O788Pkvyv3ncTs4ObUwL_iDvzvL83MT2qnp4Al0sxXo,2321
12
+ usso/fastapi/integration.py,sha256=aHNwGYaLeaVkHHk9qEJXWZ3iwOM19qWO1YQL9fslP0M,2568
13
13
  usso/session/__init__.py,sha256=tE4qWUdSI7iN_pywm47Mg8NKOTBa2nCNwCy3wCZWRmU,124
14
- usso/session/async_session.py,sha256=SSfB1tDcq5p0eXd2awfTsVYQWm5IRL-1GeMo1zsVFks,2081
15
- usso/session/session.py,sha256=R5i5ZYiNy5B76gHXDlr-uFT_OIyUIK72mXiw7xLLhn4,3738
16
- usso-0.27.2.dist-info/LICENSE.txt,sha256=ceC9ZJOV9H6CtQDcYmHOS46NA3dHJ_WD4J9blH513pc,1081
17
- usso-0.27.2.dist-info/METADATA,sha256=Nyx0DshLl4n6xPGG1maRjniligcgQHR_-fB6S1pSmus,4489
18
- usso-0.27.2.dist-info/WHEEL,sha256=PZUExdf71Ui_so67QXpySuHtCi3-J3wvF4ORK6k_S8U,91
19
- usso-0.27.2.dist-info/entry_points.txt,sha256=4Zgpm5ELaAWPf0jPGJFz1_X69H7un8ycT3WdGoJ0Vvk,35
20
- usso-0.27.2.dist-info/top_level.txt,sha256=g9Jf6h1Oyidh0vPiFni7UHInTJjSvu6cUalpLTIvthg,5
21
- usso-0.27.2.dist-info/RECORD,,
14
+ usso/session/async_session.py,sha256=_9NU_kds1IuMLmBGCDcBnbpGiLUMJJaoAvuzSxs7jZ8,2278
15
+ usso/session/base_session.py,sha256=1A12QulP6UHe1lqYEihGAcBaZ_7C09EWZt3fB6JvbOE,2347
16
+ usso/session/session.py,sha256=GpQtXNau_q-2JQAWjJZ_JEA9QrVJa_3yzW5_lROyzXA,2333
17
+ usso-0.27.5.dist-info/LICENSE.txt,sha256=ceC9ZJOV9H6CtQDcYmHOS46NA3dHJ_WD4J9blH513pc,1081
18
+ usso-0.27.5.dist-info/METADATA,sha256=hIVFwizPXBX16sSI-GpJuGcRFpr7YNsg2McGTvJt45Y,4528
19
+ usso-0.27.5.dist-info/WHEEL,sha256=PZUExdf71Ui_so67QXpySuHtCi3-J3wvF4ORK6k_S8U,91
20
+ usso-0.27.5.dist-info/entry_points.txt,sha256=4Zgpm5ELaAWPf0jPGJFz1_X69H7un8ycT3WdGoJ0Vvk,35
21
+ usso-0.27.5.dist-info/top_level.txt,sha256=g9Jf6h1Oyidh0vPiFni7UHInTJjSvu6cUalpLTIvthg,5
22
+ usso-0.27.5.dist-info/RECORD,,
File without changes