fastapi-factory-utilities 0.3.10__tar.gz → 0.4.0__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 (81) hide show
  1. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/PKG-INFO +1 -1
  2. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/pyproject.toml +1 -1
  3. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/app/config.py +2 -0
  4. fastapi_factory_utilities-0.4.0/src/fastapi_factory_utilities/core/services/hydra/__init__.py +13 -0
  5. fastapi_factory_utilities-0.4.0/src/fastapi_factory_utilities/core/services/hydra/exceptions.py +15 -0
  6. fastapi_factory_utilities-0.4.0/src/fastapi_factory_utilities/core/services/hydra/objects.py +26 -0
  7. fastapi_factory_utilities-0.4.0/src/fastapi_factory_utilities/core/services/hydra/services.py +122 -0
  8. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/LICENSE +0 -0
  9. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/README.md +0 -0
  10. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/__main__.py +0 -0
  11. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/__init__.py +0 -0
  12. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/api/__init__.py +0 -0
  13. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/api/tags.py +0 -0
  14. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/api/v1/sys/__init__.py +0 -0
  15. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/api/v1/sys/health.py +0 -0
  16. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/api/v1/sys/readiness.py +0 -0
  17. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/app/__init__.py +0 -0
  18. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/app/application.py +0 -0
  19. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/app/builder.py +0 -0
  20. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/app/enums.py +0 -0
  21. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/app/exceptions.py +0 -0
  22. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/app/fastapi_builder.py +0 -0
  23. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/app/plugin_manager/__init__.py +0 -0
  24. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/app/plugin_manager/exceptions.py +0 -0
  25. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/app/plugin_manager/plugin_manager.py +0 -0
  26. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/exceptions.py +0 -0
  27. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/plugins/__init__.py +0 -0
  28. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/plugins/example/__init__.py +0 -0
  29. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/plugins/httpx_plugin/__init__.py +0 -0
  30. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/plugins/odm_plugin/__init__.py +0 -0
  31. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/plugins/odm_plugin/builder.py +0 -0
  32. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/plugins/odm_plugin/configs.py +0 -0
  33. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/plugins/odm_plugin/depends.py +0 -0
  34. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/plugins/odm_plugin/documents.py +0 -0
  35. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/plugins/odm_plugin/exceptions.py +0 -0
  36. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/plugins/odm_plugin/repositories.py +0 -0
  37. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/plugins/opentelemetry_plugin/__init__.py +0 -0
  38. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/plugins/opentelemetry_plugin/builder.py +0 -0
  39. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/plugins/opentelemetry_plugin/configs.py +0 -0
  40. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/plugins/opentelemetry_plugin/exceptions.py +0 -0
  41. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/plugins/opentelemetry_plugin/helpers.py +0 -0
  42. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/protocols.py +0 -0
  43. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/security/jwt.py +0 -0
  44. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/security/kratos.py +0 -0
  45. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/services/kratos/__init__.py +0 -0
  46. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/services/kratos/enums.py +0 -0
  47. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/services/kratos/exceptions.py +0 -0
  48. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/services/kratos/objects.py +0 -0
  49. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/services/kratos/services.py +0 -0
  50. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/services/status/__init__.py +0 -0
  51. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/services/status/enums.py +0 -0
  52. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/services/status/exceptions.py +0 -0
  53. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/services/status/health_calculator_strategies.py +0 -0
  54. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/services/status/readiness_calculator_strategies.py +0 -0
  55. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/services/status/services.py +0 -0
  56. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/services/status/types.py +0 -0
  57. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/utils/configs.py +0 -0
  58. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/utils/importlib.py +0 -0
  59. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/utils/log.py +0 -0
  60. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/utils/status.py +0 -0
  61. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/utils/uvicorn.py +0 -0
  62. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/core/utils/yaml_reader.py +0 -0
  63. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/example/__init__.py +0 -0
  64. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/example/__main__.py +0 -0
  65. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/example/api/__init__.py +0 -0
  66. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/example/api/books/__init__.py +0 -0
  67. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/example/api/books/responses.py +0 -0
  68. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/example/api/books/routes.py +0 -0
  69. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/example/app.py +0 -0
  70. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/example/application.yaml +0 -0
  71. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/example/entities/books/__init__.py +0 -0
  72. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/example/entities/books/entities.py +0 -0
  73. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/example/entities/books/enums.py +0 -0
  74. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/example/entities/books/types.py +0 -0
  75. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/example/models/__init__.py +0 -0
  76. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/example/models/books/__init__.py +0 -0
  77. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/example/models/books/document.py +0 -0
  78. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/example/models/books/repository.py +0 -0
  79. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/example/services/books/__init__.py +0 -0
  80. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/example/services/books/services.py +0 -0
  81. {fastapi_factory_utilities-0.3.10 → fastapi_factory_utilities-0.4.0}/src/fastapi_factory_utilities/py.typed +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.3
2
2
  Name: fastapi_factory_utilities
3
- Version: 0.3.10
3
+ Version: 0.4.0
4
4
  Summary: Consolidate libraries and utilities to create microservices in Python with FastAPI, Beanie, Httpx, AioPika and OpenTelemetry.
5
5
  License: MIT
6
6
  Keywords: python,fastapi,beanie,httpx,opentelemetry,microservices
@@ -3,7 +3,7 @@ name = "fastapi_factory_utilities"
3
3
  homepage = "https://github.com/miragecentury/fastapi_factory_utilities"
4
4
  repository = "https://github.com/miragecentury/fastapi_factory_utilities"
5
5
  keywords = ["python", "fastapi", "beanie", "httpx", "opentelemetry", "microservices"]
6
- version = "0.3.10"
6
+ version = "0.4.0"
7
7
  description = "Consolidate libraries and utilities to create microservices in Python with FastAPI, Beanie, Httpx, AioPika and OpenTelemetry."
8
8
  authors = ["miragecentury <victorien.vanroye@gmail.com>"]
9
9
  maintainers = ["miragecentury <victorien.vanroye@gmail.com>"]
@@ -82,6 +82,8 @@ class DependencyConfig(BaseModel):
82
82
  """Dependency config."""
83
83
 
84
84
  kratos: HttpServiceDependencyConfig | None = Field(default=None, description="Kratos dependency config")
85
+ hydra_admin: HttpServiceDependencyConfig | None = Field(default=None, description="Hydra admin dependency config")
86
+ hydra_public: HttpServiceDependencyConfig | None = Field(default=None, description="Hydra public dependency config")
85
87
 
86
88
 
87
89
  class RootConfig(BaseModel):
@@ -0,0 +1,13 @@
1
+ """Hydra service module."""
2
+
3
+ from .exceptions import HydraOperationError, HydraTokenInvalidError
4
+ from .objects import HydraTokenIntrospectObject
5
+ from .services import HydraService, depends_hydra_service
6
+
7
+ __all__: list[str] = [
8
+ "HydraOperationError",
9
+ "HydraService",
10
+ "HydraTokenIntrospectObject",
11
+ "HydraTokenInvalidError",
12
+ "depends_hydra_service",
13
+ ]
@@ -0,0 +1,15 @@
1
+ """Python exceptions for the Hydra service."""
2
+
3
+ from fastapi_factory_utilities.core.exceptions import FastAPIFactoryUtilitiesError
4
+
5
+
6
+ class HydraError(FastAPIFactoryUtilitiesError):
7
+ """Base class for all exceptions raised by the Hydra service."""
8
+
9
+
10
+ class HydraOperationError(HydraError):
11
+ """Exception raised when a Hydra operation fails."""
12
+
13
+
14
+ class HydraTokenInvalidError(HydraError):
15
+ """Exception raised when a Hydra token is invalid."""
@@ -0,0 +1,26 @@
1
+ """Provides the objects for the Hydra service."""
2
+
3
+ from typing import ClassVar
4
+
5
+ from pydantic import BaseModel, ConfigDict
6
+
7
+
8
+ class HydraTokenIntrospectObject(BaseModel):
9
+ """Represents the object returned by the Hydra token introspection."""
10
+
11
+ model_config: ClassVar[ConfigDict] = ConfigDict(extra="ignore")
12
+
13
+ active: bool
14
+ aud: list[str]
15
+ client_id: str
16
+ exp: int
17
+ ext: dict[str, str] | None = None
18
+ iat: int
19
+ iss: str
20
+ nbf: int
21
+ obfuscated_subject: str | None = None
22
+ scope: str
23
+ sub: str
24
+ token_type: str
25
+ token_use: str
26
+ username: str | None = None
@@ -0,0 +1,122 @@
1
+ """Provides a service to interact with the Hydra service."""
2
+
3
+ from base64 import b64encode
4
+ from http import HTTPStatus
5
+ from typing import Annotated
6
+
7
+ import aiohttp
8
+ from fastapi import Depends
9
+ from pydantic import ValidationError
10
+
11
+ from fastapi_factory_utilities.core.app import (
12
+ DependencyConfig,
13
+ HttpServiceDependencyConfig,
14
+ depends_dependency_config,
15
+ )
16
+
17
+ from .exceptions import HydraOperationError, HydraTokenInvalidError
18
+ from .objects import HydraTokenIntrospectObject
19
+
20
+
21
+ class HydraService:
22
+ """Service to interact with the Hydra service."""
23
+
24
+ INTROSPECT_ENDPOINT: str = "/admin/oauth2/introspect"
25
+ CLIENT_CREDENTIALS_ENDPOINT: str = "/oauth2/token"
26
+
27
+ def __init__(
28
+ self,
29
+ hydra_admin_http_config: HttpServiceDependencyConfig,
30
+ hydra_public_http_config: HttpServiceDependencyConfig,
31
+ ) -> None:
32
+ """Instanciate the Hydra service.
33
+
34
+ Args:
35
+ hydra_admin_http_config (HttpServiceDependencyConfig): The Hydra admin HTTP configuration.
36
+ hydra_public_http_config (HttpServiceDependencyConfig): The Hydra public HTTP configuration.
37
+ """
38
+ self._hydra_admin_http_config: HttpServiceDependencyConfig = hydra_admin_http_config
39
+ self._hydra_public_http_config: HttpServiceDependencyConfig = hydra_public_http_config
40
+
41
+ async def introspect(self, token: str) -> HydraTokenIntrospectObject:
42
+ """Introspects a token using the Hydra service.
43
+
44
+ Args:
45
+ token (str): The token to introspect.
46
+
47
+ Raises:
48
+ HydraOperationError: If the introspection fails.
49
+ HydraTokenInvalidError: If the token is invalid.
50
+ """
51
+ async with aiohttp.ClientSession(
52
+ base_url=str(self._hydra_admin_http_config.url),
53
+ ) as session:
54
+ async with session.post(
55
+ url=self.INTROSPECT_ENDPOINT,
56
+ data={"token": token},
57
+ ) as response:
58
+ if response.status != HTTPStatus.OK:
59
+ raise HydraTokenInvalidError()
60
+
61
+ try:
62
+ instrospect: HydraTokenIntrospectObject = HydraTokenIntrospectObject(**await response.json())
63
+ except ValidationError as error:
64
+ raise HydraOperationError() from error
65
+
66
+ return instrospect
67
+
68
+ async def oauth2_client_credentials(self, client_id: str, client_secret: str, scope: str) -> str:
69
+ """Get the OAuth2 client credentials.
70
+
71
+ Args:
72
+ client_id (str): The client ID.
73
+ client_secret (str): The client secret.
74
+ scope (str): The scope.
75
+
76
+ Returns:
77
+ str: The access token.
78
+
79
+ Raises:
80
+ HydraOperationError: If the client credentials request fails.
81
+ """
82
+ # Create base64 encoded Basic Auth header
83
+ auth_string = f"{client_id}:{client_secret}"
84
+ auth_bytes = auth_string.encode("utf-8")
85
+ auth_b64 = b64encode(auth_bytes).decode("utf-8")
86
+
87
+ async with aiohttp.ClientSession(
88
+ base_url=str(self._hydra_public_http_config.url),
89
+ ) as session:
90
+ async with session.post(
91
+ url=self.CLIENT_CREDENTIALS_ENDPOINT,
92
+ headers={"Authorization": f"Basic {auth_b64}"},
93
+ data={"grant_type": "client_credentials", "scope": scope},
94
+ ) as response:
95
+ response_data = await response.json()
96
+ if response.status != HTTPStatus.OK:
97
+ raise HydraOperationError(f"Failed to get client credentials: {response_data}")
98
+
99
+ return response_data["access_token"]
100
+
101
+
102
+ def depends_hydra_service(
103
+ dependency_config: Annotated[DependencyConfig, Depends(depends_dependency_config)],
104
+ ) -> HydraService:
105
+ """Dependency injection for the Hydra service.
106
+
107
+ Args:
108
+ dependency_config (DependencyConfig): The dependency configuration.
109
+
110
+ Returns:
111
+ HydraService: The Hydra service instance.
112
+
113
+ Raises:
114
+ HydraOperationError: If the Hydra admin or public dependency is not configured.
115
+ """
116
+ if dependency_config.hydra_admin is None or dependency_config.hydra_public is None:
117
+ raise HydraOperationError(message="Hydra admin or public dependency not configured")
118
+
119
+ return HydraService(
120
+ hydra_admin_http_config=dependency_config.hydra_admin,
121
+ hydra_public_http_config=dependency_config.hydra_public,
122
+ )