xoscar 0.3.0__tar.gz → 0.3.1__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.

Potentially problematic release.


This version of xoscar might be problematic. Click here for more details.

Files changed (84) hide show
  1. {xoscar-0.3.0 → xoscar-0.3.1}/PKG-INFO +1 -1
  2. xoscar-0.3.1/pyproject.toml +47 -0
  3. {xoscar-0.3.0 → xoscar-0.3.1}/setup.cfg +2 -1
  4. {xoscar-0.3.0 → xoscar-0.3.1}/setup.py +4 -4
  5. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/__init__.py +0 -4
  6. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/context.py +6 -2
  7. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/core.py +2 -1
  8. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/pool.py +0 -3
  9. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/utils.py +38 -0
  10. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar.egg-info/PKG-INFO +1 -1
  11. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar.egg-info/SOURCES.txt +0 -1
  12. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar.egg-info/requires.txt +2 -1
  13. xoscar-0.3.0/pyproject.toml +0 -34
  14. xoscar-0.3.0/xoscar/entrypoints.py +0 -42
  15. {xoscar-0.3.0 → xoscar-0.3.1}/MANIFEST.in +0 -0
  16. {xoscar-0.3.0 → xoscar-0.3.1}/versioneer.py +0 -0
  17. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/_utils.pxd +0 -0
  18. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/_utils.pyx +0 -0
  19. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/_version.py +0 -0
  20. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/aio/__init__.py +0 -0
  21. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/aio/_threads.py +0 -0
  22. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/aio/base.py +0 -0
  23. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/aio/file.py +0 -0
  24. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/aio/lru.py +0 -0
  25. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/aio/parallelism.py +0 -0
  26. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/api.py +0 -0
  27. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backend.py +0 -0
  28. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/__init__.py +0 -0
  29. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/allocate_strategy.py +0 -0
  30. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/communication/__init__.py +0 -0
  31. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/communication/base.py +0 -0
  32. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/communication/core.py +0 -0
  33. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/communication/dummy.py +0 -0
  34. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/communication/errors.py +0 -0
  35. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/communication/socket.py +0 -0
  36. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/communication/ucx.py +0 -0
  37. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/communication/utils.py +0 -0
  38. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/config.py +0 -0
  39. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/indigen/__init__.py +0 -0
  40. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/indigen/backend.py +0 -0
  41. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/indigen/driver.py +0 -0
  42. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/indigen/pool.py +0 -0
  43. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/message.pyx +0 -0
  44. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/router.py +0 -0
  45. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/test/__init__.py +0 -0
  46. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/test/backend.py +0 -0
  47. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/backends/test/pool.py +0 -0
  48. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/batch.py +0 -0
  49. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/collective/__init__.py +0 -0
  50. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/collective/common.py +0 -0
  51. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/collective/core.py +0 -0
  52. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/collective/process_group.py +0 -0
  53. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/collective/utils.py +0 -0
  54. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/constants.py +0 -0
  55. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/context.pxd +0 -0
  56. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/context.pyx +0 -0
  57. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/core.pxd +0 -0
  58. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/core.pyx +0 -0
  59. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/debug.py +0 -0
  60. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/driver.py +0 -0
  61. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/errors.py +0 -0
  62. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/libcpp.pxd +0 -0
  63. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/metrics/__init__.py +0 -0
  64. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/metrics/api.py +0 -0
  65. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/metrics/backends/__init__.py +0 -0
  66. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/metrics/backends/console/__init__.py +0 -0
  67. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/metrics/backends/console/console_metric.py +0 -0
  68. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/metrics/backends/metric.py +0 -0
  69. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/metrics/backends/prometheus/__init__.py +0 -0
  70. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/metrics/backends/prometheus/prometheus_metric.py +0 -0
  71. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/nvutils.py +0 -0
  72. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/profiling.py +0 -0
  73. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/serialization/__init__.py +0 -0
  74. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/serialization/aio.py +0 -0
  75. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/serialization/core.pxd +0 -0
  76. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/serialization/core.pyx +0 -0
  77. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/serialization/cuda.py +0 -0
  78. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/serialization/exception.py +0 -0
  79. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/serialization/numpy.py +0 -0
  80. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/serialization/pyfury.py +0 -0
  81. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar/serialization/scipy.py +0 -0
  82. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar.egg-info/dependency_links.txt +0 -0
  83. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar.egg-info/not-zip-safe +0 -0
  84. {xoscar-0.3.0 → xoscar-0.3.1}/xoscar.egg-info/top_level.txt +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: xoscar
3
- Version: 0.3.0
3
+ Version: 0.3.1
4
4
  Summary: Python actor framework for heterogeneous computing.
5
5
  Home-page: http://github.com/xorbitsai/xoscar
6
6
  Author: Qin Xuye
@@ -0,0 +1,47 @@
1
+ [build-system]
2
+ requires = [
3
+ "setuptools<64",
4
+ "packaging",
5
+ "wheel",
6
+ "oldest-supported-numpy",
7
+ "scipy==1.4.1; python_version<'3.9' and platform_machine!='aarch64' and platform_machine!='arm64'",
8
+ "scipy==1.7.3; python_version<'3.9' and platform_machine=='arm64'",
9
+ "scipy==1.5.3; python_version<'3.9' and platform_machine=='aarch64'",
10
+ "scipy==1.5.4; python_version>='3.9' and python_version<'3.10' and platform_machine!='arm64'",
11
+ "scipy==1.7.2; python_version>='3.10' and python_version<'3.11' and platform_machine!='arm64'",
12
+ "scipy==1.7.3; python_version>='3.10' and python_version<'3.11' and platform_machine=='arm64'",
13
+ "scipy==1.9.2; python_version>='3.11' and python_version<'3.12'",
14
+ "scipy>=1.11.2; python_version>'3.11'",
15
+ "pandas==1.0.4; python_version<'3.9' and platform_machine!='aarch64' and platform_machine!='arm64'",
16
+ "pandas==1.1.3; python_version<'3.9' and platform_machine=='aarch64'",
17
+ "pandas==1.4.0; python_version<'3.9' and platform_machine=='arm64'",
18
+ "pandas==1.2.2; python_version>='3.9' and python_version<'3.10' and platform_machine!='arm64'",
19
+ "pandas==1.4.0; python_version>='3.9' and python_version<'3.10' and platform_machine=='arm64'",
20
+ "pandas==1.3.4; python_version>='3.10' and python_version<'3.11' and platform_machine!='arm64'",
21
+ "pandas==1.4.0; python_version>='3.10' and python_version<'3.11' and platform_machine=='arm64'",
22
+ "pandas==1.5.1; python_version>='3.11' and python_version<'3.12'",
23
+ "pandas>=2.1.1; python_version>'3.11'",
24
+ "numpy<2.0.0",
25
+ "cython>=0.29.33",
26
+ "requests>=2.4.0",
27
+ "cloudpickle>=2.2.1; python_version>='3.11'",
28
+ "cloudpickle==1.5.0; python_version<'3.11'",
29
+ ]
30
+ build-backend = "setuptools.build_meta"
31
+
32
+ [tool.black]
33
+ include = '\.pyi?$'
34
+ extend-exclude = '''
35
+ ^/xoscar/(_version.py)
36
+ '''
37
+
38
+ [tool.pytest.ini_options]
39
+ asyncio_mode = "auto"
40
+ markers = [
41
+ "cuda: mark a test as a cuda case.",
42
+ ]
43
+
44
+ [tool.cibuildwheel]
45
+ build = ["cp38-*", "cp39-*", "cp310-*", "cp311-*"]
46
+ skip = "pp* *musllinux* *i686 cp36* cp38-win32 cp39-win32 cp310-win32 cp311-win32"
47
+ manylinux-x86_64-image = "manylinux2014"
@@ -24,7 +24,7 @@ zip_safe = False
24
24
  include_package_data = True
25
25
  packages = find:
26
26
  install_requires =
27
- numpy>=1.14.0
27
+ numpy>=1.14.0,<2.0.0
28
28
  pandas>=1.0.0
29
29
  scipy>=1.0.0; sys_platform!="win32" or python_version>="3.10"
30
30
  scipy>=1.0.0,<=1.9.1; sys_platform=="win32" and python_version<"3.10"
@@ -33,6 +33,7 @@ install_requires =
33
33
  tblib>=1.7.0
34
34
  pickle5; python_version<"3.8"
35
35
  uvloop>=0.14.0; sys_platform!="win32"
36
+ packaging
36
37
 
37
38
  [options.packages.find]
38
39
  exclude =
@@ -25,7 +25,7 @@ from sysconfig import get_config_vars
25
25
 
26
26
  import numpy as np
27
27
  from Cython.Build import cythonize
28
- from pkg_resources import parse_version
28
+ from packaging.version import Version
29
29
  from setuptools import Extension, setup
30
30
  from setuptools.command.build_ext import build_ext
31
31
  from setuptools.extension import Library
@@ -63,9 +63,9 @@ if sys.platform == "darwin":
63
63
  )
64
64
  target_macos_version = "10.9"
65
65
 
66
- parsed_python_target = parse_version(python_target)
67
- parsed_current_system = parse_version(current_system)
68
- parsed_macos_version = parse_version(target_macos_version)
66
+ parsed_python_target = Version(python_target)
67
+ parsed_current_system = Version(current_system)
68
+ parsed_macos_version = Version(target_macos_version)
69
69
  if parsed_python_target <= parsed_macos_version <= parsed_current_system:
70
70
  os.environ["MACOSX_DEPLOYMENT_TARGET"] = target_macos_version
71
71
 
@@ -50,7 +50,6 @@ from ._utils import create_actor_ref
50
50
 
51
51
  # make sure methods are registered
52
52
  from .backends import indigen, test
53
- from .entrypoints import init_extension_entrypoints
54
53
  from . import _version
55
54
 
56
55
  del indigen, test
@@ -59,6 +58,3 @@ _T = TypeVar("_T")
59
58
  ActorRefType = Union[ActorRef, _T]
60
59
 
61
60
  __version__ = _version.get_versions()["version"]
62
-
63
- init_extension_entrypoints()
64
- del init_extension_entrypoints
@@ -26,7 +26,7 @@ from ..context import BaseActorContext
26
26
  from ..core import ActorRef, BufferRef, FileObjectRef, create_local_actor_ref
27
27
  from ..debug import debug_async_timeout, detect_cycle_send
28
28
  from ..errors import CannotCancelTask
29
- from ..utils import dataslots
29
+ from ..utils import dataslots, fix_all_zero_ip
30
30
  from .allocate_strategy import AddressSpecified, AllocateStrategy
31
31
  from .communication import Client, DummyClient, UCXClient
32
32
  from .core import ActorCaller
@@ -187,6 +187,7 @@ class IndigenActorContext(BaseActorContext):
187
187
 
188
188
  async def actor_ref(self, *args, **kwargs):
189
189
  actor_ref = create_actor_ref(*args, **kwargs)
190
+ connect_addr = actor_ref.address
190
191
  local_actor_ref = create_local_actor_ref(actor_ref.address, actor_ref.uid)
191
192
  if local_actor_ref is not None:
192
193
  return local_actor_ref
@@ -195,7 +196,10 @@ class IndigenActorContext(BaseActorContext):
195
196
  )
196
197
  future = await self._call(actor_ref.address, message, wait=False)
197
198
  result = await self._wait(future, actor_ref.address, message)
198
- return self._process_result_message(result)
199
+ res = self._process_result_message(result)
200
+ if res.address != connect_addr:
201
+ res.address = fix_all_zero_ip(res.address, connect_addr)
202
+ return res
199
203
 
200
204
  async def send(
201
205
  self,
@@ -85,7 +85,8 @@ class ActorCaller:
85
85
  f"Remote server {client.dest_address} closed"
86
86
  ) from None
87
87
  future = self._client_to_message_futures[client].pop(message.message_id)
88
- future.set_result(message)
88
+ if not future.done():
89
+ future.set_result(message)
89
90
  except DeserializeMessageFailed as e:
90
91
  message_id = e.message_id
91
92
  future = self._client_to_message_futures[client].pop(message_id)
@@ -33,7 +33,6 @@ from .._utils import TypeDispatcher, create_actor_ref, to_binary
33
33
  from ..api import Actor
34
34
  from ..core import ActorRef, BufferRef, FileObjectRef, register_local_pool
35
35
  from ..debug import debug_async_timeout, record_message_trace
36
- from ..entrypoints import init_extension_entrypoints
37
36
  from ..errors import (
38
37
  ActorAlreadyExist,
39
38
  ActorNotExist,
@@ -188,8 +187,6 @@ class AbstractActorPool(ABC):
188
187
  self._asyncio_task_timeout_detector_task = (
189
188
  register_asyncio_task_timeout_detector()
190
189
  )
191
- # load third party extensions.
192
- init_extension_entrypoints()
193
190
  # init metrics
194
191
  metric_configs = self._config.get_metric_configs()
195
192
  metric_backend = metric_configs.get("backend")
@@ -462,3 +462,41 @@ def is_windows():
462
462
 
463
463
  def is_linux():
464
464
  return sys.platform.startswith("linux")
465
+
466
+
467
+ def is_v4_zero_ip(ip_port_addr: str) -> bool:
468
+ return ip_port_addr.startswith("0.0.0.0:")
469
+
470
+
471
+ def is_v6_zero_ip(ip_port_addr: str) -> bool:
472
+ # tcp6 addr ":::123", ":: means all zero"
473
+ arr = ip_port_addr.split(":")
474
+ if len(arr) <= 2: # Not tcp6 or udp6
475
+ return False
476
+ for part in arr[0:-1]:
477
+ if part != "":
478
+ if int(part, 16) != 0:
479
+ return False
480
+ return True
481
+
482
+
483
+ def fix_all_zero_ip(remote_addr: str, connect_addr: str) -> str:
484
+ """
485
+ Use connect_addr to fix ActorRef.address return by remote server.
486
+ When remote server listen on "0.0.0.0:port" or ":::port", it will return ActorRef.address set to listening addr,
487
+ it cannot be use by client for the following interaction unless we fix it.
488
+ (client will treat 0.0.0.0 as 127.0.0.1)
489
+
490
+ NOTE: Server might return a different addr from a pool for load-balance purpose.
491
+ """
492
+ if remote_addr == connect_addr:
493
+ return remote_addr
494
+ if not is_v4_zero_ip(remote_addr) and not is_v6_zero_ip(remote_addr):
495
+ # Remote server returns on non-zero ip
496
+ return remote_addr
497
+ if is_v4_zero_ip(connect_addr) or is_v6_zero_ip(connect_addr):
498
+ # Client connect to local server
499
+ return remote_addr
500
+ remote_port = remote_addr.split(":")[-1]
501
+ connect_ip = ":".join(connect_addr.split(":")[0:-1]) # Remote the port
502
+ return f"{connect_ip}:{remote_port}"
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: xoscar
3
- Version: 0.3.0
3
+ Version: 0.3.1
4
4
  Summary: Python actor framework for heterogeneous computing.
5
5
  Home-page: http://github.com/xorbitsai/xoscar
6
6
  Author: Qin Xuye
@@ -17,7 +17,6 @@ xoscar/core.pxd
17
17
  xoscar/core.pyx
18
18
  xoscar/debug.py
19
19
  xoscar/driver.py
20
- xoscar/entrypoints.py
21
20
  xoscar/errors.py
22
21
  xoscar/libcpp.pxd
23
22
  xoscar/nvutils.py
@@ -1,8 +1,9 @@
1
- numpy>=1.14.0
1
+ numpy<2.0.0,>=1.14.0
2
2
  pandas>=1.0.0
3
3
  cloudpickle>=1.5.0
4
4
  psutil>=5.9.0
5
5
  tblib>=1.7.0
6
+ packaging
6
7
 
7
8
  [:python_version < "3.8"]
8
9
  pickle5
@@ -1,34 +0,0 @@
1
- [build-system]
2
- requires = [
3
- "setuptools<64",
4
- "wheel",
5
- "oldest-supported-numpy",
6
- "pandas>=1.0.4",
7
- "scipy==1.4.1; python_version<'3.9' and platform_machine!='aarch64'",
8
- "scipy==1.5.3; python_version<'3.9' and platform_machine=='aarch64'",
9
- "scipy==1.5.4; python_version>='3.9' and python_version<'3.10' and sys_platform=='win32'",
10
- "scipy>=1.4.1; python_version>='3.9' and python_version<'3.10' and sys_platform!='win32'",
11
- "scipy>=1.4.1; python_version>='3.10'",
12
- "cython>=0.29.33",
13
- "requests>=2.4.0",
14
- "cloudpickle>=2.2.1; python_version>='3.11'",
15
- "cloudpickle==1.5.0; python_version<'3.11'",
16
- ]
17
- build-backend = "setuptools.build_meta"
18
-
19
- [tool.black]
20
- include = '\.pyi?$'
21
- extend-exclude = '''
22
- ^/xoscar/(_version.py)
23
- '''
24
-
25
- [tool.pytest.ini_options]
26
- asyncio_mode = "auto"
27
- markers = [
28
- "cuda: mark a test as a cuda case.",
29
- ]
30
-
31
- [tool.cibuildwheel]
32
- build = ["cp38-*", "cp39-*", "cp310-*", "cp311-*"]
33
- skip = "pp* *musllinux* *i686 cp36* cp38-win32 cp39-win32 cp310-win32 cp311-win32"
34
- manylinux-x86_64-image = "manylinux2014"
@@ -1,42 +0,0 @@
1
- # Copyright 2022-2023 XProbe Inc.
2
- #
3
- # Licensed under the Apache License, Version 2.0 (the "License");
4
- # you may not use this file except in compliance with the License.
5
- # You may obtain a copy of the License at
6
- #
7
- # http://www.apache.org/licenses/LICENSE-2.0
8
- #
9
- # Unless required by applicable law or agreed to in writing, software
10
- # distributed under the License is distributed on an "AS IS" BASIS,
11
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
- # See the License for the specific language governing permissions and
13
- # limitations under the License.
14
-
15
- import functools
16
- import logging
17
- import warnings
18
-
19
- logger = logging.getLogger(__name__)
20
-
21
-
22
- # from https://github.com/numba/numba/blob/master/numba/core/entrypoints.py
23
- # Must put this here to avoid extensions re-triggering initialization
24
- @functools.lru_cache(maxsize=None)
25
- def init_extension_entrypoints():
26
- """Execute all `xoscar_extensions` entry points with the name `init`
27
- If extensions have already been initialized, this function does nothing.
28
- """
29
- from pkg_resources import iter_entry_points # type: ignore
30
-
31
- for entry_point in iter_entry_points("xoscar_extensions", "init"):
32
- logger.info("Loading extension: %s", entry_point)
33
- try:
34
- func = entry_point.load()
35
- func()
36
- except Exception as e:
37
- msg = "Xoscar extension module '{}' failed to load due to '{}({})'."
38
- warnings.warn(
39
- msg.format(entry_point.module_name, type(e).__name__, str(e)),
40
- stacklevel=2,
41
- )
42
- logger.info("Extension loading failed for: %s", entry_point)
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