usso 0.27.7__tar.gz → 0.27.9__tar.gz

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.
Files changed (31) hide show
  1. {usso-0.27.7/src/usso.egg-info → usso-0.27.9}/PKG-INFO +1 -1
  2. {usso-0.27.7 → usso-0.27.9}/pyproject.toml +1 -1
  3. {usso-0.27.7 → usso-0.27.9}/src/usso/core.py +1 -1
  4. usso-0.27.9/src/usso/session/async_session.py +96 -0
  5. {usso-0.27.7 → usso-0.27.9}/src/usso/session/session.py +1 -0
  6. {usso-0.27.7 → usso-0.27.9/src/usso.egg-info}/PKG-INFO +1 -1
  7. usso-0.27.7/src/usso/session/async_session.py +0 -69
  8. {usso-0.27.7 → usso-0.27.9}/LICENSE.txt +0 -0
  9. {usso-0.27.7 → usso-0.27.9}/README.md +0 -0
  10. {usso-0.27.7 → usso-0.27.9}/setup.cfg +0 -0
  11. {usso-0.27.7 → usso-0.27.9}/src/usso/__init__.py +0 -0
  12. {usso-0.27.7 → usso-0.27.9}/src/usso/b64tools.py +0 -0
  13. {usso-0.27.7 → usso-0.27.9}/src/usso/client/__init__.py +0 -0
  14. {usso-0.27.7 → usso-0.27.9}/src/usso/client/api.py +0 -0
  15. {usso-0.27.7 → usso-0.27.9}/src/usso/client/async_api.py +0 -0
  16. {usso-0.27.7 → usso-0.27.9}/src/usso/django/__init__.py +0 -0
  17. {usso-0.27.7 → usso-0.27.9}/src/usso/django/middleware.py +0 -0
  18. {usso-0.27.7 → usso-0.27.9}/src/usso/exceptions.py +0 -0
  19. {usso-0.27.7 → usso-0.27.9}/src/usso/fastapi/__init__.py +0 -0
  20. {usso-0.27.7 → usso-0.27.9}/src/usso/fastapi/integration.py +0 -0
  21. {usso-0.27.7 → usso-0.27.9}/src/usso/schemas.py +0 -0
  22. {usso-0.27.7 → usso-0.27.9}/src/usso/session/__init__.py +0 -0
  23. {usso-0.27.7 → usso-0.27.9}/src/usso/session/base_session.py +0 -0
  24. {usso-0.27.7 → usso-0.27.9}/src/usso.egg-info/SOURCES.txt +0 -0
  25. {usso-0.27.7 → usso-0.27.9}/src/usso.egg-info/dependency_links.txt +0 -0
  26. {usso-0.27.7 → usso-0.27.9}/src/usso.egg-info/entry_points.txt +0 -0
  27. {usso-0.27.7 → usso-0.27.9}/src/usso.egg-info/requires.txt +0 -0
  28. {usso-0.27.7 → usso-0.27.9}/src/usso.egg-info/top_level.txt +0 -0
  29. {usso-0.27.7 → usso-0.27.9}/tests/test_api.py +0 -0
  30. {usso-0.27.7 → usso-0.27.9}/tests/test_core.py +0 -0
  31. {usso-0.27.7 → usso-0.27.9}/tests/test_simple.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: usso
3
- Version: 0.27.7
3
+ Version: 0.27.9
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>
@@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta"
4
4
 
5
5
  [project]
6
6
  name = "usso"
7
- version = "0.27.7"
7
+ version = "0.27.9"
8
8
  description = "A plug-and-play client for integrating universal single sign-on (SSO) with Python frameworks, enabling secure and seamless authentication across microservices."
9
9
  readme = "README.md"
10
10
  requires-python = ">=3.9"
@@ -63,7 +63,7 @@ def is_expired(token: str, **kwargs) -> bool:
63
63
  decoded_token: dict = jwt.decode(token, options={"verify_signature": False})
64
64
  exp = decoded_token.get("exp", (now + timedelta(days=1)).timestamp())
65
65
  exp = datetime.fromtimestamp(exp)
66
- return exp >= now
66
+ return exp < now
67
67
 
68
68
 
69
69
  @cached(TTLCache(maxsize=128, ttl=10 * 60))
@@ -0,0 +1,96 @@
1
+ import os
2
+ import httpx
3
+ from ..core import is_expired
4
+ from .base_session import BaseUssoSession
5
+
6
+
7
+ class AsyncUssoSession(httpx.AsyncClient, BaseUssoSession):
8
+ def __init__(
9
+ self,
10
+ *,
11
+ usso_base_url: str | None = os.getenv("USSO_URL"),
12
+ api_key: str | None = os.getenv("USSO_API_KEY"),
13
+ usso_refresh_url: str | None = os.getenv("USSO_REFRESH_URL"),
14
+ refresh_token: str | None = os.getenv("USSO_REFRESH_TOKEN"),
15
+ usso_api_key: str | None = os.getenv("USSO_ADMIN_API_KEY"),
16
+ user_id: str | None = None,
17
+ ):
18
+ httpx.AsyncClient.__init__(self)
19
+ BaseUssoSession.__init__(
20
+ self,
21
+ usso_base_url=usso_base_url,
22
+ api_key=api_key,
23
+ usso_refresh_url=usso_refresh_url,
24
+ refresh_token=refresh_token,
25
+ usso_api_key=usso_api_key,
26
+ user_id=user_id,
27
+ )
28
+ self._refresh_sync()
29
+
30
+ def _prepare_refresh_request(self) -> tuple[dict, dict]:
31
+ """
32
+ Helper function to prepare headers and parameters for refresh requests.
33
+ """
34
+ headers = {"x-api-key": self.usso_api_key} if self.usso_api_key else {}
35
+ params = {"user_id": self.user_id} if self.user_id else {}
36
+ return headers, params
37
+
38
+ def _handle_refresh_response(self, response: httpx.Response) -> dict:
39
+ """
40
+ Helper function to process the response from refresh requests.
41
+ """
42
+ response.raise_for_status()
43
+ data = response.json()
44
+ self.access_token = data.get("access_token")
45
+ self._refresh_token = data.get("token", {}).get("refresh_token")
46
+ if self.access_token:
47
+ self.headers.update({"Authorization": f"Bearer {self.access_token}"})
48
+ return data
49
+
50
+ def _refresh_sync(self) -> dict:
51
+ assert (
52
+ self.refresh_token or self.usso_api_key
53
+ ), "refresh_token or usso_api_key is required"
54
+
55
+ headers, params = self._prepare_refresh_request()
56
+
57
+ if self.usso_api_key and not self.refresh_token:
58
+ response = httpx.get(
59
+ f"{self.usso_refresh_url}/api", headers=headers, params=params
60
+ )
61
+ self._handle_refresh_response(response)
62
+
63
+ response = httpx.post(
64
+ self.usso_refresh_url, json={"refresh_token": self.refresh_token}
65
+ )
66
+ return self._handle_refresh_response(response)
67
+
68
+ async def _refresh(self) -> dict:
69
+ assert (
70
+ self.refresh_token or self.usso_api_key
71
+ ), "refresh_token or usso_api_key is required"
72
+
73
+ headers, params = self._prepare_refresh_request()
74
+
75
+ if self.usso_api_key and not self.refresh_token:
76
+ response = await self.get(
77
+ f"{self.usso_refresh_url}/api", headers=headers, params=params
78
+ )
79
+ self._handle_refresh_response(response)
80
+
81
+ response = await self.post(
82
+ self.usso_refresh_url, json={"refresh_token": self.refresh_token}
83
+ )
84
+ return self._handle_refresh_response(response)
85
+
86
+ async def get_session(self):
87
+ if self.api_key:
88
+ return self
89
+
90
+ if not self.access_token or is_expired(self.access_token):
91
+ await self._refresh()
92
+ return self
93
+
94
+ async def _request(self, method: str, url: str, **kwargs):
95
+ session = await self.get_session()
96
+ return await session.request(method, url, **kwargs)
@@ -28,6 +28,7 @@ class UssoSession(BaseUssoSession, requests.Session):
28
28
  usso_api_key=usso_api_key,
29
29
  user_id=user_id,
30
30
  )
31
+ self._refresh()
31
32
  self.headers.update(self.headers)
32
33
 
33
34
  def _refresh_api(self):
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: usso
3
- Version: 0.27.7
3
+ Version: 0.27.9
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>
@@ -1,69 +0,0 @@
1
- import os
2
-
3
- import httpx
4
-
5
- from ..core import is_expired
6
- from .base_session import BaseUssoSession
7
-
8
-
9
- class AsyncUssoSession(httpx.AsyncClient, BaseUssoSession):
10
-
11
- def __init__(
12
- self,
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"),
19
- user_id: str | None = None,
20
- ):
21
- httpx.AsyncClient.__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
-
32
- async def _refresh_api(self):
33
- params = {"user_id": self.user_id} if self.user_id else {}
34
- response = await self.get(
35
- f"{self.usso_refresh_url}/api",
36
- headers={"x-api-key": self.usso_api_key},
37
- params=params,
38
- )
39
- response.raise_for_status()
40
- data: dict = response.json()
41
- self._refresh_token = data.get("token", {}).get("refresh_token")
42
-
43
- async def _refresh(self):
44
- assert (
45
- self.refresh_token or self.usso_api_key
46
- ), "refresh_token or usso_api_key is required"
47
-
48
- if self.usso_api_key and not self.refresh_token:
49
- await self._refresh_api()
50
-
51
- response = await self.post(
52
- self.usso_refresh_url, json={"refresh_token": f"{self.refresh_token}"}
53
- )
54
- response.raise_for_status()
55
- self.access_token = response.json().get("access_token")
56
- self.headers.update({"Authorization": f"Bearer {self.access_token}"})
57
- return response.json()
58
-
59
- async def get_session(self):
60
- if self.api_key:
61
- return self
62
-
63
- if not self.access_token or is_expired(self.access_token):
64
- await self._refresh()
65
- return self
66
-
67
- async def _request(self, method: str, url: str, **kwargs):
68
- session = await self.get_session()
69
- return await session.request(method, url, **kwargs)
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes