usso 0.27.17__tar.gz → 0.27.19__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.
- {usso-0.27.17/src/usso.egg-info → usso-0.27.19}/PKG-INFO +2 -2
- {usso-0.27.17 → usso-0.27.19}/pyproject.toml +1 -1
- {usso-0.27.17 → usso-0.27.19}/src/usso/client/api.py +1 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso/client/async_api.py +1 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso/core.py +9 -5
- usso-0.27.19/src/usso/session/base_session.py +95 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso/session/session.py +7 -3
- {usso-0.27.17 → usso-0.27.19/src/usso.egg-info}/PKG-INFO +2 -2
- usso-0.27.17/src/usso/session/base_session.py +0 -86
- {usso-0.27.17 → usso-0.27.19}/LICENSE.txt +0 -0
- {usso-0.27.17 → usso-0.27.19}/README.md +0 -0
- {usso-0.27.17 → usso-0.27.19}/setup.cfg +0 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso/__init__.py +0 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso/b64tools.py +0 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso/client/__init__.py +0 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso/django/__init__.py +0 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso/django/middleware.py +0 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso/exceptions.py +0 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso/fastapi/__init__.py +0 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso/fastapi/integration.py +0 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso/schemas.py +0 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso/session/__init__.py +0 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso/session/async_session.py +0 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso.egg-info/SOURCES.txt +0 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso.egg-info/dependency_links.txt +0 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso.egg-info/entry_points.txt +0 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso.egg-info/requires.txt +0 -0
- {usso-0.27.17 → usso-0.27.19}/src/usso.egg-info/top_level.txt +0 -0
@@ -1,6 +1,6 @@
|
|
1
|
-
Metadata-Version: 2.
|
1
|
+
Metadata-Version: 2.2
|
2
2
|
Name: usso
|
3
|
-
Version: 0.27.
|
3
|
+
Version: 0.27.19
|
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
|
+
version = "0.27.19"
|
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"
|
@@ -79,11 +79,15 @@ def decode_token_with_jwk(jwk_url: str, token: str, **kwargs) -> UserData | None
|
|
79
79
|
|
80
80
|
@cachetools.func.ttl_cache(maxsize=128, ttl=10 * 60)
|
81
81
|
def fetch_api_key_data(jwk_url: str, api_key: str):
|
82
|
-
|
83
|
-
|
84
|
-
|
85
|
-
|
86
|
-
|
82
|
+
try:
|
83
|
+
parsed = urlparse(jwk_url)
|
84
|
+
url = f"{parsed.scheme}://{parsed.netloc}/api_key/verify"
|
85
|
+
response = httpx.post(url, json={"api_key": api_key})
|
86
|
+
response.raise_for_status()
|
87
|
+
return UserData(**response.json())
|
88
|
+
except Exception as e:
|
89
|
+
_handle_exception("error", message=str(e))
|
90
|
+
|
87
91
|
|
88
92
|
|
89
93
|
class Usso:
|
@@ -0,0 +1,95 @@
|
|
1
|
+
import os
|
2
|
+
from urllib.parse import urlparse
|
3
|
+
|
4
|
+
from usso.core import is_expired
|
5
|
+
from typing import Optional, Any, Callable
|
6
|
+
import inspect
|
7
|
+
|
8
|
+
|
9
|
+
|
10
|
+
class BaseUssoSession:
|
11
|
+
|
12
|
+
def __init__(
|
13
|
+
self,
|
14
|
+
api_key: str | None = None,
|
15
|
+
*,
|
16
|
+
usso_base_url: str | None = None,
|
17
|
+
usso_refresh_url: str | None = None,
|
18
|
+
refresh_token: str | None = None,
|
19
|
+
app_id: str | None = None,
|
20
|
+
app_secret: str | None = None,
|
21
|
+
access_token: str | None = None,
|
22
|
+
usso_admin_api_key: str | None = None,
|
23
|
+
user_id: str | None = None,
|
24
|
+
client: Optional["BaseUssoSession"] = None,
|
25
|
+
**kwargs,
|
26
|
+
):
|
27
|
+
if client:
|
28
|
+
self.copy_attributes_from(client)
|
29
|
+
return
|
30
|
+
|
31
|
+
if not (api_key or usso_base_url or usso_refresh_url):
|
32
|
+
if os.getenv("USSO_API_KEY"):
|
33
|
+
api_key = os.getenv("USSO_API_KEY")
|
34
|
+
elif os.getenv("USSO_URL"):
|
35
|
+
usso_base_url = os.getenv("USSO_URL")
|
36
|
+
elif os.getenv("USSO_REFRESH_URL"):
|
37
|
+
usso_refresh_url = os.getenv("USSO_REFRESH_URL")
|
38
|
+
else:
|
39
|
+
raise ValueError(
|
40
|
+
"one of api_key, usso_base_url or usso_refresh_url is required"
|
41
|
+
)
|
42
|
+
|
43
|
+
if not (
|
44
|
+
api_key
|
45
|
+
or refresh_token
|
46
|
+
or usso_admin_api_key
|
47
|
+
or (app_id and app_secret)
|
48
|
+
or access_token
|
49
|
+
):
|
50
|
+
if os.getenv("USSO_REFRESH_TOKEN"):
|
51
|
+
refresh_token = os.getenv("USSO_REFRESH_TOKEN")
|
52
|
+
elif os.getenv("USSO_ADMIN_API_KEY"):
|
53
|
+
usso_admin_api_key = os.getenv("USSO_ADMIN_API_KEY")
|
54
|
+
elif os.getenv("USSO_APP_ID") and os.getenv("USSO_APP_SECRET"):
|
55
|
+
app_id = os.getenv("USSO_APP_ID")
|
56
|
+
app_secret = os.getenv("USSO_APP_SECRET")
|
57
|
+
else:
|
58
|
+
raise ValueError(
|
59
|
+
"one of api_key, refresh_token, usso_admin_api_key, app_id and app_secret or access_token is required"
|
60
|
+
)
|
61
|
+
|
62
|
+
if api_key:
|
63
|
+
self.api_key = api_key
|
64
|
+
self.headers.update({"x-api-key": api_key})
|
65
|
+
|
66
|
+
if not usso_base_url:
|
67
|
+
url_parts = urlparse(usso_refresh_url)
|
68
|
+
usso_base_url = f"{url_parts.scheme}://{url_parts.netloc}"
|
69
|
+
elif usso_base_url.endswith("/"):
|
70
|
+
usso_base_url = usso_base_url[:-1]
|
71
|
+
|
72
|
+
self.usso_base_url = usso_base_url
|
73
|
+
self.usso_refresh_url = usso_refresh_url or f"{usso_base_url}/auth/refresh"
|
74
|
+
self._refresh_token = refresh_token
|
75
|
+
self.access_token = None
|
76
|
+
self.usso_admin_api_key = usso_admin_api_key
|
77
|
+
self.user_id = user_id
|
78
|
+
self.headers = getattr(self, "headers", {})
|
79
|
+
|
80
|
+
def copy_attributes_from(self, client: "BaseUssoSession"):
|
81
|
+
self.usso_base_url = client.usso_base_url
|
82
|
+
self.usso_refresh_url = client.usso_refresh_url
|
83
|
+
self._refresh_token = client._refresh_token
|
84
|
+
self.access_token = client.access_token
|
85
|
+
self.api_key = client.api_key
|
86
|
+
self.usso_admin_api_key = client.usso_admin_api_key
|
87
|
+
self.user_id = client.user_id
|
88
|
+
self.headers = client.headers.copy()
|
89
|
+
|
90
|
+
@property
|
91
|
+
def refresh_token(self):
|
92
|
+
if self._refresh_token and is_expired(self._refresh_token):
|
93
|
+
self._refresh_token = None
|
94
|
+
|
95
|
+
return self._refresh_token
|
@@ -1,12 +1,12 @@
|
|
1
1
|
import os
|
2
|
-
|
2
|
+
import inspect
|
3
|
+
from typing import Callable, Any
|
3
4
|
import httpx
|
4
5
|
|
5
6
|
from usso.core import is_expired
|
6
7
|
|
7
8
|
from .base_session import BaseUssoSession
|
8
9
|
|
9
|
-
|
10
10
|
class UssoSession(httpx.Client, BaseUssoSession):
|
11
11
|
|
12
12
|
def __init__(
|
@@ -19,8 +19,12 @@ class UssoSession(httpx.Client, BaseUssoSession):
|
|
19
19
|
usso_api_key: str | None = os.getenv("USSO_ADMIN_API_KEY"),
|
20
20
|
user_id: str | None = None,
|
21
21
|
client: "UssoSession" = None,
|
22
|
+
**kwargs,
|
22
23
|
):
|
23
|
-
|
24
|
+
|
25
|
+
httpx_kwargs = _filter_kwargs(kwargs, httpx.Client.__init__)
|
26
|
+
|
27
|
+
httpx.Client.__init__(self, **httpx_kwargs)
|
24
28
|
|
25
29
|
BaseUssoSession.__init__(
|
26
30
|
self,
|
@@ -1,6 +1,6 @@
|
|
1
|
-
Metadata-Version: 2.
|
1
|
+
Metadata-Version: 2.2
|
2
2
|
Name: usso
|
3
|
-
Version: 0.27.
|
3
|
+
Version: 0.27.19
|
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,86 +0,0 @@
|
|
1
|
-
import os
|
2
|
-
from urllib.parse import urlparse
|
3
|
-
|
4
|
-
from usso.core import is_expired
|
5
|
-
|
6
|
-
|
7
|
-
class BaseUssoSession:
|
8
|
-
|
9
|
-
def __init__(
|
10
|
-
self,
|
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
|
-
client: "BaseUssoSession" = None,
|
18
|
-
):
|
19
|
-
if client:
|
20
|
-
self.copy_attributes_from(client)
|
21
|
-
return
|
22
|
-
|
23
|
-
assert (
|
24
|
-
usso_base_url or usso_refresh_url
|
25
|
-
), "usso_base_url or usso_refresh_url is required"
|
26
|
-
assert (
|
27
|
-
refresh_token or api_key or usso_api_key
|
28
|
-
), "refresh_token or api_key or usso_api_key is required"
|
29
|
-
|
30
|
-
if not usso_base_url:
|
31
|
-
url_parts = urlparse(usso_refresh_url)
|
32
|
-
usso_base_url = f"{url_parts.scheme}://{url_parts.netloc}"
|
33
|
-
if usso_base_url.endswith("/"):
|
34
|
-
usso_base_url = usso_base_url[:-1]
|
35
|
-
|
36
|
-
self.usso_base_url = usso_base_url
|
37
|
-
self.usso_refresh_url = usso_refresh_url or f"{usso_base_url}/auth/refresh"
|
38
|
-
self._refresh_token = refresh_token
|
39
|
-
self.access_token = None
|
40
|
-
self.api_key = api_key
|
41
|
-
self.usso_api_key = usso_api_key
|
42
|
-
self.user_id = user_id
|
43
|
-
self.headers = getattr(self, "headers", {})
|
44
|
-
if self.api_key:
|
45
|
-
self.headers.update({"x-api-key": api_key})
|
46
|
-
|
47
|
-
def copy_attributes_from(self, client: "BaseUssoSession"):
|
48
|
-
self.usso_base_url = client.usso_base_url
|
49
|
-
self.usso_refresh_url = client.usso_refresh_url
|
50
|
-
self._refresh_token = client._refresh_token
|
51
|
-
self.access_token = client.access_token
|
52
|
-
self.api_key = client.api_key
|
53
|
-
self.usso_api_key = client.usso_api_key
|
54
|
-
self.user_id = client.user_id
|
55
|
-
self.headers = client.headers.copy()
|
56
|
-
|
57
|
-
@property
|
58
|
-
def refresh_token(self):
|
59
|
-
if self._refresh_token and is_expired(self._refresh_token):
|
60
|
-
self._refresh_token = None
|
61
|
-
|
62
|
-
return self._refresh_token
|
63
|
-
|
64
|
-
def request(self, method: str, url: str, **kwargs):
|
65
|
-
return self._request(method, url, **kwargs)
|
66
|
-
|
67
|
-
def get(self, url: str, **kwargs):
|
68
|
-
return self._request("GET", url, **kwargs)
|
69
|
-
|
70
|
-
def post(self, url: str, **kwargs):
|
71
|
-
return self._request("POST", url, **kwargs)
|
72
|
-
|
73
|
-
def put(self, url: str, **kwargs):
|
74
|
-
return self._request("PUT", url, **kwargs)
|
75
|
-
|
76
|
-
def patch(self, url: str, **kwargs):
|
77
|
-
return self._request("PATCH", url, **kwargs)
|
78
|
-
|
79
|
-
def delete(self, url: str, **kwargs):
|
80
|
-
return self._request("DELETE", url, **kwargs)
|
81
|
-
|
82
|
-
def head(self, url: str, **kwargs):
|
83
|
-
return self._request("HEAD", url, **kwargs)
|
84
|
-
|
85
|
-
def options(self, url: str, **kwargs):
|
86
|
-
return self._request("OPTIONS", 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
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|