Skip to content

Commit aab9660

Browse files
tvaron3simorenohkushagraThaparjeet1995scbedd
authored andcommitted
Replace Endpoints with Regional Endpoints (Azure#39390)
* add new policy, add logic to use policy * added small test file I was using * initial regional endpoint work * groundwork * re-add AzureError logic, refactor, fix tests * Update _retry_utility.py * Updated location_cache with new design * Fixed key error with most_preferred_location * Update test_cosmos_http_logging_policy.py * Update _retry_utility.py * Added logic to refresh cache on previous endpoint usage * Added business logic update the regional endpoint based on success or failures * implementation * Update _retry_utility_async.py * fix some tests * changelog, versions, fixes * fixes * fix some tests * remove fake logic, count fix * fix some tests * Update _service_request_retry_policy.py * Update _retry_utility_async.py * retry utilities fixing * Update _retry_utility.py * additional enhancements * Update setup.py * Update _retry_utility_async.py * add tests, remove previous retry logic for ServiceRequestExceptions * clean up with finally * tests * retry utilities * disable tests * add logging to policies * GetDatabaseAccount Fix * Update _base.py * retry utilities fixes * Update _retry_utility.py * retry utulities part 34 * Update _service_request_retry_policy.py * remove extra logs * policy updates * Update _service_response_retry_policy.py * Update _service_response_retry_policy.py * policies updates and update operation types * trying out fixes * Update sdk/cosmos/azure-cosmos/CHANGELOG.md Co-authored-by: Abhijeet Mohanty <[email protected]> * Update sdk/cosmos/azure-cosmos/CHANGELOG.md Co-authored-by: Abhijeet Mohanty <[email protected]> * Skipped proxy test for debugging * annotation fix * Fixed some tests cases * test fixes * Update test_service_retry_policies_async.py * Fixed some mocking behavior * fixed pylint issues * Added aiohttp minimum dependency * Updated changelog and setup.py * Updated changelog * Add changelog and fix tests. * Fix tests * bootstrapping with global endpoint as previous for writes * Add headers and cleanup * cleanup and retry all service request headers * Don't retry on a none previous * Updated the business logic with current and previous, fixed database account refresh and some retry policies * fix client id * Reacting to comments * Added print statements and fixed some retry logic * Revert getDatabase in mark endpoint * Fixed some pylint and changelog issues * Fixed version * fix bug with type check, update tests * Update test_service_retry_policies_async.py * sync tests updates * Reacting to comments and fixing service request retry policy * Code review comments and pylint issues * Fixed tests and pylint * more sync mock tests - missing async copies * Fixed min aiohttp requirements * Update _retry_utility_async.py * Change to check operation type in operations * push initial GEM mock test * Update test_service_retry_policies.py * Fixed extra retries * sync tests * Update test_service_retry_policies_async.py * Fixed extra retries and relevant tests * Only delay retry by one second * async tests - need to split up inheritance ones since endpoint unavailable stops extra retries * Change retry strategy * add sub-class errors tests * change old tests, refactoring, fix mocking bleed * Fix a test * clear last routed location pythonic * Removed aiohttp dependency * catch import errors * Skipped global endpoint manager test for debugging * Fixed tests * Removed skips * fix live tests and print statements for debugging * cleanup of few tests * updated globaldb mock * Moved some of the high offer throughput tests to live tests * Fixed global endpoint retry async test * Tried fixing global endpoint retry async test * no swaps on success test * fix import * Tried fixing global endpoint retry async test * Added separate split live tests * Added live platform matrix * some test fixes * Fixed live test pipeline * Moved test resource id to cosmosLong * Updated live tests * Running live tests with proper flag * testing logging experiments * fix tests * honor testmark argument through a safe environment variable, versus accessing the value directly * more test fixes * remove accidental log files * Fixed issues with swapping and retry policies * Fixed issues with swapping and retry policies * Marking endpoint as down fix * more test fixes * Remove print statements * Fixed some minor issues with emulator tests * split change feed tests * Fixed emulator tests * updated changelog * Fixed emulator tests again * Fixed emulator tests and event loop * vector/fts query tests * Fix session token live tests * hybrid search query fixes * Fixed live test name * fallback to regional * fix ci tests * Update conftest.py * Database accounts call will timeout in 5 seconds * Change timeouts and update docs * call updates to endpoint policy and location cache * Health check for endpoitns * database account retry policy * Fix parameter error * Retry on cosmos error fix * Retry on service request error fix * None checks for request in retry utilities * lowercase constructed regional endpoint * fix global endpoint as unhealthy * fix parsing test * Added logic for swapping on health check failed * Fixed log statement * fix pylint, docs, and remove print statements * fix pylint * fix some tests * Prepared for release --------- Co-authored-by: Simon Moreno <[email protected]> Co-authored-by: Kushagra Thapar <[email protected]> Co-authored-by: Abhijeet Mohanty <[email protected]> Co-authored-by: Scott Beddall <[email protected]>
1 parent 13b1549 commit aab9660

File tree

77 files changed

+2365
-1187
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

77 files changed

+2365
-1187
lines changed

eng/pipelines/templates/steps/build-test.yml

+17-2
Original file line numberDiff line numberDiff line change
@@ -83,11 +83,18 @@ steps:
8383
}
8484
Write-Host (Get-Command python).Source
8585
86+
if ($env:TESTMARKARGUMENT) {
87+
$markArg = $env:TESTMARKARGUMENT
88+
}
89+
else {
90+
$markArg = "${{ parameters.TestMarkArgument }}"
91+
}
92+
8693
python scripts/devops_tasks/dispatch_tox.py
8794
"$(TargetingString)"
8895
${{ parameters.AdditionalTestArgs }}
8996
${{ parameters.CoverageArg }}
90-
--mark_arg="${{ parameters.TestMarkArgument }}"
97+
--mark_arg="$markArg"
9198
--service="${{ parameters.ServiceDirectory }}"
9299
--toxenv="${{ parameters.ToxTestEnv }}"
93100
--injected-packages="${{ parameters.InjectedPackages }}"
@@ -104,10 +111,18 @@ steps:
104111
. $(VENV_LOCATION)/bin/activate.ps1
105112
}
106113
Write-Host (Get-Command python).Source
114+
115+
if ($env:TESTMARKARGUMENT) {
116+
$markArg = $env:TESTMARKARGUMENT
117+
}
118+
else {
119+
$markArg = "${{ parameters.TestMarkArgument }}"
120+
}
121+
107122
python scripts/devops_tasks/dispatch_tox.py "$(TargetingString)" `
108123
${{ parameters.AdditionalTestArgs }} `
109124
${{ parameters.CoverageArg }} `
110-
--mark_arg="${{ parameters.TestMarkArgument }}" `
125+
--mark_arg="$markArg" `
111126
--service="${{ parameters.ServiceDirectory }}" `
112127
--toxenv="${{ parameters.ToxTestEnv }}" `
113128
--injected-packages="${{ parameters.InjectedPackages }}" `

sdk/cosmos/azure-cosmos/CHANGELOG.md

+17
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,22 @@
11
## Release History
22

3+
### 4.9.1b3 (2025-02-04)
4+
5+
#### Features Added
6+
* Improved retry logic by retrying alternative endpoint for writes within a region before performing a cross region retry. See [PR 39390](https://github.com/Azure/azure-sdk-for-python/pull/39390)
7+
* Added endpoint health check logic during database account calls. See [PR 39390](https://github.com/Azure/azure-sdk-for-python/pull/39390)
8+
9+
#### Bugs Fixed
10+
* Fixed unnecessary retries on the wrong region for timout retry policy. See [PR 39390](https://github.com/Azure/azure-sdk-for-python/pull/39390)
11+
* All client connection errors from aiohttp will be retried. See [PR 39390](https://github.com/Azure/azure-sdk-for-python/pull/39390)
12+
13+
#### Other Changes
14+
* Changed defaults for retry delays. See [PR 39390](https://github.com/Azure/azure-sdk-for-python/pull/39390)
15+
* Changed default connection timeout to be 5 seconds. See [PR 39390](https://github.com/Azure/azure-sdk-for-python/pull/39390)
16+
* Changed default read timeout to be 65 seconds. See [PR 39390](https://github.com/Azure/azure-sdk-for-python/pull/39390)
17+
* On database account calls send a client id header for load balancing. See [PR 39390](https://github.com/Azure/azure-sdk-for-python/pull/39390)
18+
* Removed aiohttp dependency. See [PR 39390](https://github.com/Azure/azure-sdk-for-python/pull/39390)
19+
320
### 4.9.1b2 (2025-01-24)
421

522
#### Features Added

sdk/cosmos/azure-cosmos/azure/cosmos/_base.py

+5
Original file line numberDiff line numberDiff line change
@@ -119,6 +119,7 @@ def GetHeaders( # pylint: disable=too-many-statements,too-many-branches
119119
operation_type: str,
120120
options: Mapping[str, Any],
121121
partition_key_range_id: Optional[str] = None,
122+
client_id: Optional[str] = None,
122123
) -> Dict[str, Any]:
123124
"""Gets HTTP request headers.
124125
@@ -131,6 +132,7 @@ def GetHeaders( # pylint: disable=too-many-statements,too-many-branches
131132
:param str operation_type:
132133
:param dict options:
133134
:param str partition_key_range_id:
135+
:param str client_id:
134136
:return: The HTTP request headers.
135137
:rtype: dict
136138
"""
@@ -280,6 +282,9 @@ def GetHeaders( # pylint: disable=too-many-statements,too-many-branches
280282
if partition_key_range_id is not None:
281283
headers[http_constants.HttpHeaders.PartitionKeyRangeID] = partition_key_range_id
282284

285+
if client_id is not None:
286+
headers[http_constants.HttpHeaders.ClientId] = client_id
287+
283288
if options.get("enableScriptLogging"):
284289
headers[http_constants.HttpHeaders.EnableScriptLogging] = options["enableScriptLogging"]
285290

sdk/cosmos/azure-cosmos/azure/cosmos/_cosmos_client_connection.py

+24-2
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@
2525
"""
2626
import os
2727
import urllib.parse
28+
import uuid
2829
from typing import Callable, Dict, Any, Iterable, List, Mapping, Optional, Sequence, Tuple, Union, cast, Type
2930
from typing_extensions import TypedDict
3031
from urllib3.util.retry import Retry
@@ -109,7 +110,7 @@ class _QueryCompatibilityMode:
109110
_DefaultStringHashPrecision = 3
110111
_DefaultStringRangePrecision = -1
111112

112-
def __init__(
113+
def __init__( # pylint: disable=too-many-statements
113114
self,
114115
url_connection: str,
115116
auth: CredentialDict,
@@ -131,6 +132,7 @@ def __init__(
131132
The default consistency policy for client operations.
132133
133134
"""
135+
self.client_id = str(uuid.uuid4())
134136
self.url_connection = url_connection
135137
self.master_key: Optional[str] = None
136138
self.resource_tokens: Optional[Mapping[str, Any]] = None
@@ -2555,7 +2557,7 @@ def GetDatabaseAccount(
25552557
url_connection = self.url_connection
25562558

25572559
headers = base.GetHeaders(self, self.default_headers, "get", "", "", "",
2558-
documents._OperationType.Read,{})
2560+
documents._OperationType.Read,{}, client_id=self.client_id)
25592561
request_params = RequestObject("databaseaccount", documents._OperationType.Read, url_connection)
25602562
result, last_response_headers = self.__Get("", request_params, headers, **kwargs)
25612563
self.last_response_headers = last_response_headers
@@ -2589,6 +2591,26 @@ def GetDatabaseAccount(
25892591
response_hook(last_response_headers, result)
25902592
return database_account
25912593

2594+
def _GetDatabaseAccountCheck(
2595+
self,
2596+
url_connection: Optional[str] = None,
2597+
**kwargs: Any
2598+
):
2599+
"""Gets database account info.
2600+
2601+
:param str url_connection: the endpoint used to get the database account
2602+
:return: The Database Account.
2603+
:rtype: documents.DatabaseAccount
2604+
"""
2605+
if url_connection is None:
2606+
url_connection = self.url_connection
2607+
2608+
headers = base.GetHeaders(self, self.default_headers, "get", "", "", "",
2609+
documents._OperationType.Read,{}, client_id=self.client_id)
2610+
request_params = RequestObject("databaseaccount", documents._OperationType.Read, url_connection)
2611+
self.__Get("", request_params, headers, **kwargs)
2612+
2613+
25922614
def Create(
25932615
self,
25942616
body: Dict[str, Any],
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
# The MIT License (MIT)
2+
# Copyright (c) 2014 Microsoft Corporation
3+
4+
# Permission is hereby granted, free of charge, to any person obtaining a copy
5+
# of this software and associated documentation files (the "Software"), to deal
6+
# in the Software without restriction, including without limitation the rights
7+
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
8+
# copies of the Software, and to permit persons to whom the Software is
9+
# furnished to do so, subject to the following conditions:
10+
11+
# The above copyright notice and this permission notice shall be included in all
12+
# copies or substantial portions of the Software.
13+
14+
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15+
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16+
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
17+
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
18+
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
19+
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
20+
# SOFTWARE.
21+
22+
"""Internal class for database account retry policy implementation in the
23+
Azure Cosmos database service.
24+
"""
25+
26+
class DatabaseAccountRetryPolicy(object):
27+
"""The database account retry policy which should only retry once regardless of errors.
28+
"""
29+
30+
def __init__(self, connection_policy):
31+
self.retry_count = 0
32+
self.retry_after_in_milliseconds = 0
33+
self.max_retry_attempt_count = 1
34+
self.connection_policy = connection_policy
35+
36+
def ShouldRetry(self, exception): # pylint: disable=unused-argument
37+
"""Returns true if the request should retry based on the passed-in exception.
38+
39+
:param exceptions.CosmosHttpResponseError exception:
40+
:returns: a boolean stating whether the request should be retried
41+
:rtype: bool
42+
"""
43+
44+
if self.retry_count >= self.max_retry_attempt_count:
45+
return False
46+
47+
self.retry_count += 1
48+
49+
return True

sdk/cosmos/azure-cosmos/azure/cosmos/_default_retry_policy.py

+3-2
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
Cosmos database service.
66
"""
77
from . import http_constants
8+
from .documents import _OperationType
89

910
# pylint: disable=protected-access
1011

@@ -36,12 +37,12 @@ def __init__(self, *args):
3637
self.current_retry_attempt_count = 0
3738
self.retry_after_in_milliseconds = 1000
3839
self.args = args
40+
self.request = args[0] if args else None
3941

4042
def needsRetry(self, error_code):
4143
if error_code in DefaultRetryPolicy.CONNECTION_ERROR_CODES:
4244
if self.args:
43-
if (self.args[3].method == "GET") or (http_constants.HttpHeaders.IsQuery in self.args[3].headers) \
44-
or (http_constants.HttpHeaders.IsQueryPlanRequest in self.args[3].headers):
45+
if _OperationType.IsReadOnlyOperation(self.request.operation_type):
4546
return True
4647
return False
4748
return True

sdk/cosmos/azure-cosmos/azure/cosmos/_endpoint_discovery_retry_policy.py

+7-10
Original file line numberDiff line numberDiff line change
@@ -50,14 +50,7 @@ def __init__(self, connection_policy, global_endpoint_manager, *args):
5050
self.retry_after_in_milliseconds = EndpointDiscoveryRetryPolicy.Retry_after_in_milliseconds
5151
self.connection_policy = connection_policy
5252
self.request = args[0] if args else None
53-
# clear previous location-based routing directive
54-
if self.request:
55-
self.request.clear_route_to_location()
5653

57-
# Resolve the endpoint for the request and pin the resolution to the resolved endpoint
58-
# This enables marking the endpoint unavailability on endpoint failover/unreachability
59-
self.location_endpoint = self.global_endpoint_manager.resolve_service_endpoint(self.request)
60-
self.request.route_to_location(self.location_endpoint)
6154

6255
def ShouldRetry(self, exception): # pylint: disable=unused-argument
6356
"""Returns true if the request should retry based on the passed-in exception.
@@ -77,12 +70,16 @@ def ShouldRetry(self, exception): # pylint: disable=unused-argument
7770

7871
self.failover_retry_count += 1
7972

80-
if self.location_endpoint:
73+
if self.request.location_endpoint_to_route:
8174
if _OperationType.IsReadOnlyOperation(self.request.operation_type):
8275
# Mark current read endpoint as unavailable
83-
self.global_endpoint_manager.mark_endpoint_unavailable_for_read(self.location_endpoint)
76+
self.global_endpoint_manager.mark_endpoint_unavailable_for_read(
77+
self.request.location_endpoint_to_route,
78+
True)
8479
else:
85-
self.global_endpoint_manager.mark_endpoint_unavailable_for_write(self.location_endpoint)
80+
self.global_endpoint_manager.mark_endpoint_unavailable_for_write(
81+
self.request.location_endpoint_to_route,
82+
True)
8683

8784
# set the refresh_needed flag to ensure that endpoint list is
8885
# refreshed with new writable and readable locations

sdk/cosmos/azure-cosmos/azure/cosmos/_global_endpoint_manager.py

+41-38
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,6 @@
2424
"""
2525

2626
import threading
27-
from urllib.parse import urlparse
2827

2928
from azure.core.exceptions import AzureError
3029

@@ -64,25 +63,28 @@ def get_refresh_time_interval_in_ms_stub(self):
6463
return constants._Constants.DefaultUnavailableLocationExpirationTime
6564

6665
def get_write_endpoint(self):
67-
return self.location_cache.get_write_endpoint()
66+
return self.location_cache.get_write_regional_endpoint()
6867

6968
def get_read_endpoint(self):
70-
return self.location_cache.get_read_endpoint()
69+
return self.location_cache.get_read_regional_endpoint()
70+
71+
def swap_regional_endpoint_values(self, request):
72+
return self.location_cache.swap_regional_endpoint_values(request)
7173

7274
def resolve_service_endpoint(self, request):
7375
return self.location_cache.resolve_service_endpoint(request)
7476

75-
def mark_endpoint_unavailable_for_read(self, endpoint):
76-
self.location_cache.mark_endpoint_unavailable_for_read(endpoint)
77+
def mark_endpoint_unavailable_for_read(self, endpoint, refresh_cache):
78+
self.location_cache.mark_endpoint_unavailable_for_read(endpoint, refresh_cache)
7779

78-
def mark_endpoint_unavailable_for_write(self, endpoint):
79-
self.location_cache.mark_endpoint_unavailable_for_write(endpoint)
80+
def mark_endpoint_unavailable_for_write(self, endpoint, refresh_cache):
81+
self.location_cache.mark_endpoint_unavailable_for_write(endpoint, refresh_cache)
8082

81-
def get_ordered_write_endpoints(self):
82-
return self.location_cache.get_ordered_write_endpoints()
83+
def get_ordered_write_locations(self):
84+
return self.location_cache.get_ordered_write_locations()
8385

84-
def get_ordered_read_endpoints(self):
85-
return self.location_cache.get_ordered_read_endpoints()
86+
def get_ordered_read_locations(self):
87+
return self.location_cache.get_ordered_read_locations()
8688

8789
def can_use_multiple_write_locations(self, request):
8890
return self.location_cache.can_use_multiple_write_locations_for_request(request)
@@ -91,6 +93,9 @@ def force_refresh(self, database_account):
9193
self.refresh_needed = True
9294
self.refresh_endpoint_list(database_account)
9395

96+
def update_location_cache(self):
97+
self.location_cache.update_location_cache()
98+
9499
def refresh_endpoint_list(self, database_account, **kwargs):
95100
if self.location_cache.current_time_millis() - self.last_refresh_time > self.refresh_time_interval_in_ms:
96101
self.refresh_needed = True
@@ -115,6 +120,8 @@ def _refresh_endpoint_list_private(self, database_account=None, **kwargs):
115120
self.last_refresh_time = self.location_cache.current_time_millis()
116121
database_account = self._GetDatabaseAccount(**kwargs)
117122
self.location_cache.perform_on_database_account_read(database_account)
123+
# this will perform getDatabaseAccount calls to check endpoint health
124+
self._endpoints_health_check(**kwargs)
118125

119126
def _GetDatabaseAccount(self, **kwargs):
120127
"""Gets the database account.
@@ -137,7 +144,7 @@ def _GetDatabaseAccount(self, **kwargs):
137144
# to get that info from any endpoints
138145
except (exceptions.CosmosHttpResponseError, AzureError):
139146
for location_name in self.PreferredLocations:
140-
locational_endpoint = _GlobalEndpointManager.GetLocationalEndpoint(self.DefaultEndpoint, location_name)
147+
locational_endpoint = LocationCache.GetLocationalEndpoint(self.DefaultEndpoint, location_name)
141148
try:
142149
database_account = self._GetDatabaseAccountStub(locational_endpoint, **kwargs)
143150
self._database_account_cache = database_account
@@ -146,6 +153,28 @@ def _GetDatabaseAccount(self, **kwargs):
146153
pass
147154
raise
148155

156+
def _endpoints_health_check(self, **kwargs):
157+
"""Gets the database account for each endpoint.
158+
159+
Validating if the endpoint is healthy else marking it as unavailable.
160+
"""
161+
all_endpoints = [self.location_cache.read_regional_endpoints[0]]
162+
all_endpoints.extend(self.location_cache.write_regional_endpoints)
163+
count = 0
164+
for endpoint in all_endpoints:
165+
count += 1
166+
if count > 3:
167+
break
168+
try:
169+
self.Client._GetDatabaseAccountCheck(endpoint.get_current(), **kwargs)
170+
except (exceptions.CosmosHttpResponseError, AzureError):
171+
if endpoint in self.location_cache.read_regional_endpoints:
172+
self.mark_endpoint_unavailable_for_read(endpoint.get_current(), False)
173+
if endpoint in self.location_cache.write_regional_endpoints:
174+
self.mark_endpoint_unavailable_for_write(endpoint.get_current(), False)
175+
endpoint.swap()
176+
self.location_cache.update_location_cache()
177+
149178
def _GetDatabaseAccountStub(self, endpoint, **kwargs):
150179
"""Stub for getting database account from the client.
151180
This can be used for mocking purposes as well.
@@ -155,29 +184,3 @@ def _GetDatabaseAccountStub(self, endpoint, **kwargs):
155184
:rtype: ~azure.cosmos.DatabaseAccount
156185
"""
157186
return self.Client.GetDatabaseAccount(endpoint, **kwargs)
158-
159-
@staticmethod
160-
def GetLocationalEndpoint(default_endpoint, location_name):
161-
# For default_endpoint like 'https://contoso.documents.azure.com:443/' parse it to
162-
# generate URL format. This default_endpoint should be global endpoint(and cannot
163-
# be a locational endpoint) and we agreed to document that
164-
endpoint_url = urlparse(default_endpoint)
165-
166-
# hostname attribute in endpoint_url will return 'contoso.documents.azure.com'
167-
if endpoint_url.hostname is not None:
168-
hostname_parts = str(endpoint_url.hostname).lower().split(".")
169-
if hostname_parts is not None:
170-
# global_database_account_name will return 'contoso'
171-
global_database_account_name = hostname_parts[0]
172-
173-
# Prepare the locational_database_account_name as contoso-EastUS for location_name 'East US'
174-
locational_database_account_name = global_database_account_name + "-" + location_name.replace(" ", "")
175-
176-
# Replace 'contoso' with 'contoso-EastUS' and return locational_endpoint
177-
# as https://contoso-EastUS.documents.azure.com:443/
178-
locational_endpoint = default_endpoint.lower().replace(
179-
global_database_account_name, locational_database_account_name, 1
180-
)
181-
return locational_endpoint
182-
183-
return None

0 commit comments

Comments
 (0)