nlbone 0.7.29__py3-none-any.whl → 0.7.31__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.
@@ -1,7 +1,11 @@
1
1
  from pathlib import Path
2
2
 
3
+ from nlbone.config.settings import get_settings
4
+
3
5
  from .engine import I18nAdapter
4
- from .loaders import JSONFileLoader
6
+ from .loaders import JSONFileLoader, CompositeLoader
5
7
 
6
- translator = I18nAdapter(loader=JSONFileLoader(locales_path=Path(__file__).parent.joinpath("./locales").__str__()),
7
- default_locale="fa")
8
+ loader = CompositeLoader()
9
+ loader.add_loader(JSONFileLoader(locales_path=Path(__file__).parent.joinpath("./locales").__str__()))
10
+ JSONFileLoader(locales_path=lambda: get_settings().PROJECT_LOCALE_PATH)
11
+ translator = I18nAdapter(loader=loader, default_locale="fa-IR")
@@ -7,12 +7,19 @@ from .loaders import BaseLoader
7
7
 
8
8
 
9
9
  class I18nAdapter(TranslationPort):
10
- def __init__(self, loader: BaseLoader, default_locale: str = "fa"):
10
+ def __init__(self, loader: BaseLoader, default_locale: str = "fa-IR"):
11
11
  self.default_locale = default_locale
12
- self._translations: Dict[str, Dict[str, str]] = loader.load()
12
+ self.loader = loader
13
+ self._translations: Optional[Dict[str, Dict[str, str]]] = None
14
+
15
+ def _ensure_loaded(self):
16
+ if self._translations is None:
17
+ self._translations = self.loader.load()
13
18
 
14
19
  def translate(self, key: str, locale: Optional[str] = None, **kwargs) -> str:
15
- target_locale = locale or get_locale() or self.default_locale
20
+ self._ensure_loaded()
21
+
22
+ target_locale = locale or self.default_locale
16
23
 
17
24
  locale_data = self._translations.get(target_locale, {})
18
25
  text = locale_data.get(key)
@@ -2,7 +2,7 @@ import json
2
2
  import logging
3
3
  from abc import ABC, abstractmethod
4
4
  from pathlib import Path
5
- from typing import Any, Dict
5
+ from typing import Any, Dict, List, Callable, Union
6
6
 
7
7
  logger = logging.getLogger(__name__)
8
8
 
@@ -13,33 +13,38 @@ class BaseLoader(ABC):
13
13
  pass
14
14
 
15
15
 
16
+ from typing import Callable, Union
17
+
18
+
16
19
  class JSONFileLoader(BaseLoader):
17
- def __init__(self, locales_path: str):
18
- self.locales_path = locales_path
20
+ def __init__(self, locales_path: Union[str, Callable[[], str]]):
21
+ self.locales_path_provider = locales_path
19
22
 
20
23
  def load(self) -> Dict[str, Dict[str, str]]:
21
24
  translations: Dict[str, Dict[str, str]] = {}
22
25
 
23
- path_obj = Path(self.locales_path)
26
+ if callable(self.locales_path_provider):
27
+ path_str = self.locales_path_provider()
28
+ else:
29
+ path_str = self.locales_path_provider
30
+
31
+ if not path_str:
32
+ return translations
33
+
34
+ path_obj = Path(path_str)
24
35
 
25
36
  if not path_obj.exists():
26
- logger.warning(f"Locales directory not found at: {self.locales_path}")
27
37
  return translations
28
38
 
29
39
  for file_path in path_obj.glob("*.json"):
30
40
  try:
31
41
  lang_code = file_path.stem
32
-
33
42
  with open(file_path, "r", encoding="utf-8") as f:
34
43
  data = json.load(f)
35
44
 
36
45
  flat_data = self._flatten_dict(data)
37
46
  translations[lang_code] = flat_data
38
47
 
39
- logger.info(f"Loaded locale '{lang_code}' with {len(flat_data)} keys.")
40
-
41
- except json.JSONDecodeError as e:
42
- logger.error(f"Invalid JSON format in {file_path}: {e}")
43
48
  except Exception as e:
44
49
  logger.error(f"Error loading locale {file_path}: {e}")
45
50
 
@@ -49,10 +54,28 @@ class JSONFileLoader(BaseLoader):
49
54
  items = []
50
55
  for k, v in d.items():
51
56
  new_key = f"{parent_key}{sep}{k}" if parent_key else k
52
-
53
57
  if isinstance(v, dict):
54
58
  items.extend(self._flatten_dict(v, new_key, sep=sep).items())
55
59
  else:
56
60
  items.append((new_key, str(v)))
57
-
58
61
  return dict(items)
62
+
63
+
64
+ class CompositeLoader(BaseLoader):
65
+ def __init__(self):
66
+ self.loaders = []
67
+
68
+ def add_loader(self, loader: BaseLoader):
69
+ self.loaders.append(loader)
70
+
71
+ def load(self) -> Dict[str, Dict[str, str]]:
72
+ merged_translations: Dict[str, Dict[str, str]] = {}
73
+
74
+ for loader in self.loaders:
75
+ data = loader.load()
76
+ for lang, keys in data.items():
77
+ if lang not in merged_translations:
78
+ merged_translations[lang] = {}
79
+ merged_translations[lang].update(keys)
80
+
81
+ return merged_translations
@@ -1,11 +1,4 @@
1
1
  {
2
- "general": {
3
- "welcome": "خوش آمدید",
4
- "buttons": {
5
- "save": "ذخیره",
6
- "cancel": "لغو"
7
- }
8
- },
9
2
  "auth": {
10
3
  "otp_sent": "کد تایید برای {mobile} ارسال شد."
11
4
  },
nlbone/config/settings.py CHANGED
@@ -113,6 +113,8 @@ class Settings(BaseSettings):
113
113
  RABBITMQ_TICKETING_EXCHANGE: str = "crm_stage.ticket"
114
114
  RABBITMQ_TICKETING_ROUTING_KEY_CREATE_V1: str = "crm_stage.ticket.create.v1"
115
115
 
116
+ PROJECT_LOCALE_PATH: str = ""
117
+
116
118
  model_config = SettingsConfigDict(
117
119
  env_prefix="",
118
120
  env_file=None,
@@ -60,7 +60,8 @@ def install_exception_handlers(
60
60
  "http_error",
61
61
  extra={"status": exc.status_code, "detail": exc.detail, "path": request.url.path},
62
62
  )
63
- return _json_response(request, exc.status_code, detail=_(exc.detail))
63
+ locale = locale = getattr(request.state, "locale", None)
64
+ return _json_response(request, exc.status_code, detail=_(exc.detail, locale=locale))
64
65
 
65
66
  @app.exception_handler(FastAPIHTTPException)
66
67
  async def _handle_fastapi_http_exception(request: Request, exc: FastAPIHTTPException):
@@ -69,7 +70,8 @@ def install_exception_handlers(
69
70
  "fastapi_http_error",
70
71
  extra={"status": exc.status_code, "detail": exc.detail, "path": request.url.path},
71
72
  )
72
- return _json_response(request, exc.status_code, detail=exc.detail)
73
+ locale = locale = getattr(request.state, "locale", None)
74
+ return _json_response(request, exc.status_code, detail=_(exc.detail, locale=locale))
73
75
 
74
76
  @app.exception_handler(StarletteHTTPException)
75
77
  async def _handle_starlette_http_exception(request: Request, exc: StarletteHTTPException):
@@ -78,7 +80,8 @@ def install_exception_handlers(
78
80
  "starlette_http_error",
79
81
  extra={"status": exc.status_code, "detail": exc.detail, "path": request.url.path},
80
82
  )
81
- return _json_response(request, exc.status_code, detail=exc.detail)
83
+ locale = locale = getattr(request.state, "locale", None)
84
+ return _json_response(request, exc.status_code, detail=_(exc.detail, locale=locale))
82
85
 
83
86
  # 3) خطاهای اعتبارسنجی FastAPI (request body/query/path)
84
87
  @app.exception_handler(RequestValidationError)
@@ -107,4 +110,5 @@ def install_exception_handlers(
107
110
  if logger:
108
111
  logger.exception("unhandled_exception", extra={"trace_id": tid, "path": request.url.path})
109
112
  detail = str(exc) if expose_server_errors else "internal server error"
110
- return _json_response(request, 500, detail=detail, trace_id=tid)
113
+ locale = locale = getattr(request.state, "locale", None)
114
+ return _json_response(request, 500, detail=_(detail, locale=locale), trace_id=tid)
@@ -99,3 +99,10 @@ class GoneException(BaseHttpException):
99
99
  status_code=status.HTTP_410_GONE,
100
100
  detail=detail,
101
101
  )
102
+
103
+ class TooManyRequestsException(BaseHttpException):
104
+ def __init__(self, detail: str = "TooManyRequests"):
105
+ super().__init__(
106
+ status_code=status.HTTP_429_TOO_MANY_REQUESTS,
107
+ detail=detail,
108
+ )
@@ -46,6 +46,7 @@ class AddRequestContextMiddleware(BaseHTTPMiddleware):
46
46
  ip = request.client.host if request.client else None
47
47
  ua = request.headers.get("user-agent")
48
48
  locale = request.headers.get("Accept-Language", 'fa-IR')
49
+ request.state.locale = locale
49
50
 
50
51
  tokens = bind_context(request=request, request_id=req_id, user_id=user_id, ip=ip, user_agent=ua, locale=locale)
51
52
  try:
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: nlbone
3
- Version: 0.7.29
3
+ Version: 0.7.31
4
4
  Summary: Backbone package for interfaces and infrastructure in Python projects
5
5
  Author-email: Amir Hosein Kahkbazzadeh <a.khakbazzadeh@gmail.com>
6
6
  License: MIT
@@ -30,10 +30,10 @@ nlbone/adapters/http_clients/pricing/pricing_service.py,sha256=_15vyEwJD3S2DJG-y
30
30
  nlbone/adapters/http_clients/uploadchi/__init__.py,sha256=uBzEOuVtY22teWW2b36Pitkdk5yVdSqa6xbg22JfTNg,105
31
31
  nlbone/adapters/http_clients/uploadchi/uploadchi.py,sha256=erpjOees25FW0nuK1PkYS-oU0h8MeRV9Rhs1cf3gaEs,4881
32
32
  nlbone/adapters/http_clients/uploadchi/uploadchi_async.py,sha256=PQbVNeaYde5CmgT3vcnQoI1PGeSs9AxHlPFuB8biOmU,4717
33
- nlbone/adapters/i18n/__init__.py,sha256=19C_u8ZzWLQRfhG96FmEyk9x3pP9Iv8yEk59Or7UMU8,257
34
- nlbone/adapters/i18n/engine.py,sha256=f_nxxy-_T5CITnPp3vnxBl0hZZ8MSn0aA-PzajSx_dQ,908
35
- nlbone/adapters/i18n/loaders.py,sha256=Qz2B8YFaYx8u5OusO4Vo2zCwvnnVfTH7fetAOJZiK3c,1778
36
- nlbone/adapters/i18n/locales/fa-IR.json,sha256=SMShgJErN-tDLOSH3hAsjg08O8GPlDfDVyFGjIUQpGU,340
33
+ nlbone/adapters/i18n/__init__.py,sha256=fS97TR7HEc7fiDC2ufQKoFOxXDNkGA4njAFIB3EmhLk,426
34
+ nlbone/adapters/i18n/engine.py,sha256=55DbbVPPHG64o640BzWDsjaxRfu6GykZzVBqME03iDE,1078
35
+ nlbone/adapters/i18n/loaders.py,sha256=7td0Cmn0ehjDO2S2qeH31zwl8UyWI7quzedP9PnPhWk,2381
36
+ nlbone/adapters/i18n/locales/fa-IR.json,sha256=QL4XSuSIL82wYhBrvyOAoMM0rG9yjKuOue43_YItuSM,208
37
37
  nlbone/adapters/messaging/__init__.py,sha256=o6ZiMihm_MhRXfcEpcjHBB3JGQovQbg3pxe0qS6516c,41
38
38
  nlbone/adapters/messaging/event_bus.py,sha256=MSM70JPHoDK17efWlh20ATT1O0KW7xWnwZ5D-gI6U_w,3773
39
39
  nlbone/adapters/messaging/rabbitmq.py,sha256=Io3flSj8DMIHExJguh3hDgWT4LuDoPYb_xmQQ3k47Cs,1894
@@ -47,7 +47,7 @@ nlbone/adapters/ticketing/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJW
47
47
  nlbone/adapters/ticketing/client.py,sha256=J1-eT3qQDAJqrHcVpP1oqWNsRNnJ54dDdBeez-m9wyY,1291
48
48
  nlbone/config/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
49
49
  nlbone/config/logging.py,sha256=Ot6Ctf7EQZlW8YNB-uBdleqI6wixn5fH0Eo6QRgNkQk,4358
50
- nlbone/config/settings.py,sha256=2Kx7RZ7IwqTNTKB_0jLxy6G-qkeTWKGVqohLIIwA1DU,4750
50
+ nlbone/config/settings.py,sha256=vOB8RQ5iBdbRpvcbBF7Cdpr9IwdarrHPz-ncQ_yQA8g,4785
51
51
  nlbone/core/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
52
52
  nlbone/core/application/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
53
53
  nlbone/core/application/base_worker.py,sha256=5brIToSd-vi6tw0ukhHnUZGZhOLq1SQ-NRRy-kp6D24,1193
@@ -70,8 +70,8 @@ nlbone/core/ports/translation.py,sha256=pnqbxhdRCR7eprm8UI8ZKKx7VDUPntvBtlytrnTG
70
70
  nlbone/core/ports/uow.py,sha256=VhqSc-Ryt9m-rlNMiXTzD3dPGz6mM_JxND8D0UJGRu4,962
71
71
  nlbone/interfaces/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
72
72
  nlbone/interfaces/api/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
73
- nlbone/interfaces/api/exception_handlers.py,sha256=pk7ehq15fZ6If57BvDjo_Ot-R5hTEFMhBP5KDwp61AM,4102
74
- nlbone/interfaces/api/exceptions.py,sha256=Phv0nXaEHCgJVEAuQ2q8QHxpn_diXahKarupSHOATY0,2964
73
+ nlbone/interfaces/api/exception_handlers.py,sha256=jH2VAgx63auFSat1H3uQ0Z9JrJ6YYN57dFuqwKzT7EU,4431
74
+ nlbone/interfaces/api/exceptions.py,sha256=06S677YplJgODjVaa3fP00dPy9i-bNIlDWg1t5Q8EZc,3194
75
75
  nlbone/interfaces/api/routers.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
76
76
  nlbone/interfaces/api/schemas.py,sha256=NIEKeTdJtwwIkIxL7WURNZF8g34I4TlRAqs-x1Uq7YI,108
77
77
  nlbone/interfaces/api/additional_filed/__init__.py,sha256=BWemliLSQV9iq1vdUaF733q0FOSipSWBOQk9eYj732Q,318
@@ -88,7 +88,7 @@ nlbone/interfaces/api/dependencies/db.py,sha256=-UD39J_86UU7ZJs2ZncpdND0yhAG0Nee
88
88
  nlbone/interfaces/api/dependencies/uow.py,sha256=QfLEvLYLNWZJQN1k-0q0hBVtUld3D75P4j39q_RjcnE,1181
89
89
  nlbone/interfaces/api/middleware/__init__.py,sha256=zbX2vaEAfxRMIYwO2MVY_2O6bqG5H9o7HqGpX14U3Is,158
90
90
  nlbone/interfaces/api/middleware/access_log.py,sha256=vIkxxxfy2HcjqqKb8XCfGCcSrivAC8u6ie75FMq5x-U,1032
91
- nlbone/interfaces/api/middleware/add_request_context.py,sha256=rRxsE34XvsefN3HufZQyXqPFzCeWhdn3qel0-cS2QZM,1914
91
+ nlbone/interfaces/api/middleware/add_request_context.py,sha256=o8mdo-D6fODM9OyHunE5UodkVxsh4F__5tDv8ju8Sxg,1952
92
92
  nlbone/interfaces/api/middleware/authentication.py,sha256=Bt6sYu4KtXAyUQnSIp-Z2Z1yKNNtfRy9Y3rOZcYTFhw,3299
93
93
  nlbone/interfaces/api/pagination/__init__.py,sha256=pA1uC4rK6eqDI5IkLVxmgO2B6lExnOm8Pje2-hifJZw,431
94
94
  nlbone/interfaces/api/pagination/offset_base.py,sha256=60X8a9uDOSd3qG45M49dqNG_FUjSxEDrgEyb9JD9V-o,4113
@@ -114,8 +114,8 @@ nlbone/utils/http.py,sha256=0yeI34j5FfelqvX3PJnKknSXji1jl15VYbVIIvrSbXg,997
114
114
  nlbone/utils/normalize_mobile.py,sha256=sGH4tV9gX-6eVKozviNWJhm1DN1J28Nj-ERldCYkS_E,732
115
115
  nlbone/utils/redactor.py,sha256=-V4HrHmHwPi3Kez587Ek1uJlgK35qGSrwBOvcbw8Jas,1279
116
116
  nlbone/utils/time.py,sha256=DjjyQ9GLsfXoT6NK8RDW2rOlJg3e6sF04Jw6PBUrSvg,1268
117
- nlbone-0.7.29.dist-info/METADATA,sha256=BjeBTqBMp4Ob1lqx8s8ihRQ5URDcY3e-P8sc5hTyaxY,2295
118
- nlbone-0.7.29.dist-info/WHEEL,sha256=WLgqFyCfm_KASv4WHyYy0P3pM_m7J5L9k2skdKLirC8,87
119
- nlbone-0.7.29.dist-info/entry_points.txt,sha256=CpIL45t5nbhl1dGQPhfIIDfqqak3teK0SxPGBBr7YCk,59
120
- nlbone-0.7.29.dist-info/licenses/LICENSE,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
121
- nlbone-0.7.29.dist-info/RECORD,,
117
+ nlbone-0.7.31.dist-info/METADATA,sha256=18DKVQix7O0AX3SAPyjkekbvQpeNzlizIQ0KVPW3E8s,2295
118
+ nlbone-0.7.31.dist-info/WHEEL,sha256=WLgqFyCfm_KASv4WHyYy0P3pM_m7J5L9k2skdKLirC8,87
119
+ nlbone-0.7.31.dist-info/entry_points.txt,sha256=CpIL45t5nbhl1dGQPhfIIDfqqak3teK0SxPGBBr7YCk,59
120
+ nlbone-0.7.31.dist-info/licenses/LICENSE,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
121
+ nlbone-0.7.31.dist-info/RECORD,,