nlbone 0.6.10__tar.gz → 0.6.11__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.
- {nlbone-0.6.10 → nlbone-0.6.11}/PKG-INFO +2 -1
- {nlbone-0.6.10 → nlbone-0.6.11}/pyproject.toml +3 -2
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/auth/keycloak.py +72 -2
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/http_clients/uploadchi/uploadchi.py +14 -13
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/percolation/connection.py +1 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/ports/auth.py +1 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/api/dependencies/auth.py +13 -5
- {nlbone-0.6.10 → nlbone-0.6.11}/.gitignore +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/LICENSE +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/README.md +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/auth/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/auth/token_provider.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/cache/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/cache/async_redis.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/cache/memory.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/cache/pubsub_listener.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/cache/redis.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/db/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/db/postgres/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/db/postgres/audit.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/db/postgres/base.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/db/postgres/engine.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/db/postgres/query_builder.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/db/postgres/repository.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/db/postgres/schema.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/db/postgres/uow.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/db/redis/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/db/redis/client.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/http_clients/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/http_clients/pricing/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/http_clients/pricing/pricing_service.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/http_clients/uploadchi/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/http_clients/uploadchi/uploadchi_async.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/messaging/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/messaging/event_bus.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/messaging/redis.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/percolation/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/config/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/config/logging.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/config/settings.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/container.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/application/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/application/base_worker.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/application/events.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/application/services/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/application/services.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/application/use_case.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/domain/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/domain/base.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/domain/events.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/domain/models.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/ports/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/ports/cache.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/ports/event_bus.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/ports/files.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/ports/messaging.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/ports/repo.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/core/ports/uow.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/api/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/api/dependencies/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/api/dependencies/async_auth.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/api/dependencies/db.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/api/dependencies/uow.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/api/exception_handlers.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/api/exceptions.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/api/middleware/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/api/middleware/access_log.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/api/middleware/add_request_context.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/api/middleware/authentication.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/api/pagination/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/api/pagination/offset_base.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/api/routers.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/api/schemas.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/cli/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/cli/init_db.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/cli/main.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/jobs/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/interfaces/jobs/sync_tokens.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/types.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/utils/__init__.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/utils/cache.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/utils/cache_keys.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/utils/cache_registry.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/utils/context.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/utils/http.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/utils/redactor.py +0 -0
- {nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/utils/time.py +0 -0
|
@@ -1,12 +1,13 @@
|
|
|
1
1
|
Metadata-Version: 2.4
|
|
2
2
|
Name: nlbone
|
|
3
|
-
Version: 0.6.
|
|
3
|
+
Version: 0.6.11
|
|
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
|
|
7
7
|
License-File: LICENSE
|
|
8
8
|
Requires-Python: >=3.10
|
|
9
9
|
Requires-Dist: anyio>=4.0
|
|
10
|
+
Requires-Dist: cachetools>=6.2.0
|
|
10
11
|
Requires-Dist: dependency-injector>=4.48.1
|
|
11
12
|
Requires-Dist: elasticsearch==8.14.0
|
|
12
13
|
Requires-Dist: fastapi>=0.116
|
|
@@ -4,7 +4,7 @@ build-backend = "hatchling.build"
|
|
|
4
4
|
|
|
5
5
|
[project]
|
|
6
6
|
name = "nlbone"
|
|
7
|
-
version = "0.6.
|
|
7
|
+
version = "0.6.11"
|
|
8
8
|
description = "Backbone package for interfaces and infrastructure in Python projects"
|
|
9
9
|
readme = "README.md"
|
|
10
10
|
requires-python = ">=3.10"
|
|
@@ -27,7 +27,8 @@ dependencies = [
|
|
|
27
27
|
"redis~=6.4.0",
|
|
28
28
|
"python-dateutil~=2.9.0.post0",
|
|
29
29
|
"typer>=0.17.4",
|
|
30
|
-
"makefun>=1.16.0"
|
|
30
|
+
"makefun>=1.16.0",
|
|
31
|
+
"cachetools>=6.2.0",
|
|
31
32
|
]
|
|
32
33
|
|
|
33
34
|
[tool.ruff]
|
|
@@ -1,9 +1,31 @@
|
|
|
1
|
+
import functools
|
|
2
|
+
from datetime import datetime, timezone
|
|
3
|
+
from threading import RLock
|
|
4
|
+
|
|
5
|
+
from cachetools import LRUCache
|
|
1
6
|
from keycloak import KeycloakOpenID
|
|
2
7
|
from keycloak.exceptions import KeycloakAuthenticationError
|
|
3
8
|
|
|
4
|
-
from nlbone.config.settings import Settings, get_settings
|
|
9
|
+
from nlbone.config.settings import Settings, get_settings
|
|
5
10
|
from nlbone.core.ports.auth import AuthService
|
|
6
11
|
|
|
12
|
+
_permissions_cache: LRUCache = LRUCache(maxsize=2048)
|
|
13
|
+
_permissions_lock = RLock()
|
|
14
|
+
|
|
15
|
+
|
|
16
|
+
def _now_ts() -> int:
|
|
17
|
+
return int(datetime.now(timezone.utc).timestamp())
|
|
18
|
+
|
|
19
|
+
|
|
20
|
+
def _ttl_from_decoded(decoded: dict) -> int:
|
|
21
|
+
exp = int(decoded.get("exp", 0))
|
|
22
|
+
ttl = max(1, exp - _now_ts())
|
|
23
|
+
return ttl
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
def _cache_key(sub: str | None, exp: int | None) -> tuple[str | None, int | None]:
|
|
27
|
+
return sub, exp
|
|
28
|
+
|
|
7
29
|
|
|
8
30
|
class KeycloakAuthService(AuthService):
|
|
9
31
|
def __init__(self, settings: Settings | None = None):
|
|
@@ -14,7 +36,7 @@ class KeycloakAuthService(AuthService):
|
|
|
14
36
|
realm_name=s.KEYCLOAK_REALM_NAME,
|
|
15
37
|
client_secret_key=s.KEYCLOAK_CLIENT_SECRET.get_secret_value().strip(),
|
|
16
38
|
)
|
|
17
|
-
self.bypass =
|
|
39
|
+
self.bypass = s.ENV != 'prod'
|
|
18
40
|
|
|
19
41
|
def has_access(self, token, permissions):
|
|
20
42
|
if self.bypass:
|
|
@@ -29,6 +51,49 @@ class KeycloakAuthService(AuthService):
|
|
|
29
51
|
print(f"Token verification failed: {e}")
|
|
30
52
|
return False
|
|
31
53
|
|
|
54
|
+
def _fetch_permissions_from_keycloak(self, token: str) -> tuple[list[str], dict]:
|
|
55
|
+
permissions = self.keycloak_openid.uma_permissions(token)
|
|
56
|
+
decoded_token = self.keycloak_openid.decode_token(token)
|
|
57
|
+
result: list[str] = []
|
|
58
|
+
for p in permissions or []:
|
|
59
|
+
rsname = p.get("rsname")
|
|
60
|
+
for s in p.get("scopes", []) or []:
|
|
61
|
+
result.append(f"{rsname}#{s}")
|
|
62
|
+
return result, decoded_token
|
|
63
|
+
|
|
64
|
+
def get_permissions(self, token: str) -> list[str]:
|
|
65
|
+
try:
|
|
66
|
+
decoded = self.keycloak_openid.decode_token(token)
|
|
67
|
+
sub = decoded.get("sub")
|
|
68
|
+
exp = decoded.get("exp")
|
|
69
|
+
key = _cache_key(sub, exp)
|
|
70
|
+
|
|
71
|
+
now = _now_ts()
|
|
72
|
+
with _permissions_lock:
|
|
73
|
+
entry = _permissions_cache.get(key)
|
|
74
|
+
if entry is not None:
|
|
75
|
+
perms, exp_ts = entry
|
|
76
|
+
if exp_ts > now:
|
|
77
|
+
return perms
|
|
78
|
+
_permissions_cache.pop(key, None)
|
|
79
|
+
|
|
80
|
+
perms, decoded2 = self._fetch_permissions_from_keycloak(token)
|
|
81
|
+
decoded_final = decoded2 or decoded
|
|
82
|
+
sub_f = decoded_final.get("sub")
|
|
83
|
+
exp_f = int(decoded_final.get("exp") or 0)
|
|
84
|
+
key_f = _cache_key(sub_f, exp_f)
|
|
85
|
+
|
|
86
|
+
with _permissions_lock:
|
|
87
|
+
_permissions_cache[key_f] = (perms, exp_f)
|
|
88
|
+
|
|
89
|
+
return perms
|
|
90
|
+
|
|
91
|
+
except KeycloakAuthenticationError:
|
|
92
|
+
return []
|
|
93
|
+
except Exception as e:
|
|
94
|
+
print(f"Getting permissions failed: {e}")
|
|
95
|
+
return []
|
|
96
|
+
|
|
32
97
|
def verify_token(self, token: str) -> dict | None:
|
|
33
98
|
try:
|
|
34
99
|
result = self.keycloak_openid.introspect(token)
|
|
@@ -76,3 +141,8 @@ class KeycloakAuthService(AuthService):
|
|
|
76
141
|
if not self.is_client_token(token, allowed_clients):
|
|
77
142
|
return False
|
|
78
143
|
return self.has_access(token, permissions)
|
|
144
|
+
|
|
145
|
+
|
|
146
|
+
@functools.lru_cache(maxsize=1)
|
|
147
|
+
def get_auth_service() -> KeycloakAuthService:
|
|
148
|
+
return KeycloakAuthService()
|
|
@@ -35,11 +35,11 @@ def _filename_from_cd(cd: str | None, fallback: str) -> str:
|
|
|
35
35
|
|
|
36
36
|
class UploadchiClient(FileServicePort):
|
|
37
37
|
def __init__(
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
38
|
+
self,
|
|
39
|
+
token_provider: ClientTokenProvider | None = None,
|
|
40
|
+
base_url: Optional[str] = None,
|
|
41
|
+
timeout_seconds: Optional[float] = None,
|
|
42
|
+
client: httpx.Client | None = None,
|
|
43
43
|
) -> None:
|
|
44
44
|
s = get_settings()
|
|
45
45
|
self._base_url = normalize_https_base(base_url or str(s.UPLOADCHI_BASE_URL))
|
|
@@ -51,7 +51,7 @@ class UploadchiClient(FileServicePort):
|
|
|
51
51
|
self._client.close()
|
|
52
52
|
|
|
53
53
|
def upload_file(
|
|
54
|
-
|
|
54
|
+
self, file_bytes: bytes, filename: str, params: dict[str, Any] | None = None, token: str | None = None
|
|
55
55
|
) -> dict:
|
|
56
56
|
tok = _resolve_token(token)
|
|
57
57
|
files = {"file": (filename, file_bytes)}
|
|
@@ -84,12 +84,12 @@ class UploadchiClient(FileServicePort):
|
|
|
84
84
|
raise UploadchiError(r.status_code, r.text)
|
|
85
85
|
|
|
86
86
|
def list_files(
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
87
|
+
self,
|
|
88
|
+
limit: int = 10,
|
|
89
|
+
offset: int = 0,
|
|
90
|
+
filters: dict[str, Any] | None = None,
|
|
91
|
+
sort: list[tuple[str, str]] | None = None,
|
|
92
|
+
token: str | None = None,
|
|
93
93
|
) -> dict:
|
|
94
94
|
tok = _resolve_token(token)
|
|
95
95
|
q = build_list_query(limit, offset, filters, sort)
|
|
@@ -116,6 +116,7 @@ class UploadchiClient(FileServicePort):
|
|
|
116
116
|
|
|
117
117
|
def delete_file(self, file_id: str, token: str | None = None) -> None:
|
|
118
118
|
tok = _resolve_token(token)
|
|
119
|
-
r = self._client.delete(f"{self._base_url}/{file_id}",
|
|
119
|
+
r = self._client.delete(f"{self._base_url}/{file_id}",
|
|
120
|
+
headers=auth_headers(tok or self._token_provider.get_access_token()))
|
|
120
121
|
if r.status_code not in (204, 200):
|
|
121
122
|
raise UploadchiError(r.status_code, r.text)
|
|
@@ -8,3 +8,4 @@ class AuthService(Protocol):
|
|
|
8
8
|
def get_client_token(self) -> dict | None: ...
|
|
9
9
|
def is_client_token(self, token: str, allowed_clients: set[str] | None = None) -> bool: ...
|
|
10
10
|
def client_has_access(self, token: str, perms: list[str], allowed_clients: set[str] | None = None) -> bool: ...
|
|
11
|
+
def get_permissions(self, token: str) -> list[str]: ...
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
import functools
|
|
2
2
|
|
|
3
3
|
from nlbone.adapters.auth import KeycloakAuthService
|
|
4
|
+
from nlbone.adapters.auth.keycloak import get_auth_service
|
|
4
5
|
from nlbone.interfaces.api.exceptions import ForbiddenException, UnauthorizedException
|
|
5
6
|
from nlbone.utils.context import current_request
|
|
6
7
|
|
|
@@ -51,8 +52,10 @@ def has_access(*, permissions=None):
|
|
|
51
52
|
request = current_request()
|
|
52
53
|
if not current_user_id():
|
|
53
54
|
raise UnauthorizedException()
|
|
54
|
-
|
|
55
|
-
|
|
55
|
+
user_permissions = get_auth_service().get_permissions(request.state.token)
|
|
56
|
+
for p in permissions or []:
|
|
57
|
+
if p not in user_permissions:
|
|
58
|
+
raise ForbiddenException(f"Forbidden {permissions}")
|
|
56
59
|
|
|
57
60
|
return func(*args, **kwargs)
|
|
58
61
|
|
|
@@ -79,9 +82,14 @@ def client_or_user_has_access(*, permissions=None, client_permissions=None):
|
|
|
79
82
|
else:
|
|
80
83
|
if not current_user_id():
|
|
81
84
|
raise UnauthorizedException()
|
|
82
|
-
|
|
83
|
-
|
|
85
|
+
|
|
86
|
+
user_permissions = get_auth_service().get_permissions(request.state.token)
|
|
87
|
+
for p in permissions or []:
|
|
88
|
+
if p not in user_permissions:
|
|
89
|
+
raise ForbiddenException(f"Forbidden {permissions}")
|
|
84
90
|
|
|
85
91
|
return func(*args, **kwargs)
|
|
92
|
+
|
|
86
93
|
return wrapper
|
|
87
|
-
|
|
94
|
+
|
|
95
|
+
return decorator
|
|
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
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
{nlbone-0.6.10 → nlbone-0.6.11}/src/nlbone/adapters/http_clients/uploadchi/uploadchi_async.py
RENAMED
|
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
|
|
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
|
|
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
|