Skip to content
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
25 changes: 25 additions & 0 deletions tests/gold_tests/autest-site/ats_replay.test.ext
Original file line number Diff line number Diff line change
Expand Up @@ -119,6 +119,21 @@ def configure_ats(obj: 'TestRun', server: 'Process', ats_config: dict, dns: Opti
gold_file = diags_log['gold_file']
ts.Disk.diags_log.Content += gold_file

# error_log validation.
error_log = log_validation.get('error_log', {})
for contains_entry in error_log.get('contains', []):
expression = contains_entry['expression']
description = contains_entry.get('description', f'Verify error_log contains: {expression}')
ts.Disk.error_log.Content += Testers.ContainsExpression(expression, description)
for excludes_entry in error_log.get('excludes', []):
expression = excludes_entry['expression']
description = excludes_entry.get('description', f'Verify error_log excludes: {expression}')
ts.Disk.error_log.Content += Testers.ExcludesExpression(expression, description)
# Gold file validation for error_log.
if 'gold_file' in error_log:
gold_file = error_log['gold_file']
ts.Disk.error_log.Content += gold_file

# access_log validation.
access_log = log_validation.get('access_log', {})
if 'gold_file' in access_log:
Expand Down Expand Up @@ -182,6 +197,8 @@ def ATSReplayTest(obj, replay_file: str):
dns = tr.MakeDNServer(name, **process_config)
else:
dns = tr.MakeDNServer(name, default='127.0.0.1')
if 'records' in dns_config:
dns.addRecords(dns_config['records'])

# Proxy Verifier Server configuration.
if not 'server' in autest_config:
Expand Down Expand Up @@ -215,6 +232,7 @@ def ATSReplayTest(obj, replay_file: str):
ats_config = autest_config['ats']
enable_tls = ats_config.get('enable_tls', False)
metric_checks = ats_config.get('metric_checks', [])
log_validation = ats_config.get('log_validation', None)

ats_owner = obj if _requires_persistent_ats(ats_config) else tr
ts = configure_ats(ats_owner, server=server, ats_config=ats_config, dns=dns)
Expand Down Expand Up @@ -272,6 +290,13 @@ def ATSReplayTest(obj, replay_file: str):
f'^{re.escape(metric_name)}\\s+{expected_value}$', f'{metric_name} should be {expected_value}')
check_tr.StillRunningAfter = ts

# wait for error log
if log_validation != None and log_validation['error_log'] != None:
wait_for_log = obj.AddTestRun('Wait for logs of ' + ats_config.get('name', 'ts'))
wait_for_log.Processes.Default.Command = (
os.path.join(obj.Variables.AtsTestToolsDir, 'condwait') + ' 60 1 -f ' + os.path.join(ts.Variables.LOGDIR, 'error.log'))
wait_for_log.Processes.Default.ReturnCode = 0
Comment on lines +293 to +298
Copy link

Copilot AI Mar 2, 2026

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In ATSReplayTest, the guard if log_validation != None and log_validation['error_log'] != None: can raise a KeyError when autest.ats.log_validation exists but does not define error_log (e.g. many existing replay tests only use traffic_out / diags_log). Please switch to a safe lookup (e.g. error_log_cfg = (log_validation or {}).get('error_log')) and check that value instead.

Copilot uses AI. Check for mistakes.

return tr


Expand Down
31 changes: 31 additions & 0 deletions tests/gold_tests/dns/connect_attempts.test.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
'''
Verify Origin Server Connect Attempts Behavior
'''
# Licensed to the Apache Software Foundation (ASF) under one
# or more contributor license agreements. See the NOTICE file
# distributed with this work for additional information
# regarding copyright ownership. The ASF licenses this file
# to you under the Apache License, Version 2.0 (the
# "License"); you may not use this file except in compliance
# with the License. You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

Test.Summary = '''
Verify Origin Server Connect Attempts Behavior
'''

# No retry
Test.ATSReplayTest(replay_file="replay/connect_attempts_rr_no_retry.replay.yaml")

# max_retries
Test.ATSReplayTest(replay_file="replay/connect_attempts_rr_max_retries.replay.yaml")

# rr_retries
Test.ATSReplayTest(replay_file="replay/connect_attempts_rr_retries.replay.yaml")
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
`` CONNECT: attempt fail [CONNECTION_ERROR] to 0.0.0.1:`` for host='example.com' `` retry_attempts=0 url='http://backend.example.com:``/path/'
`` CONNECT: attempt fail [CONNECTION_ERROR] to 0.0.0.1:`` for host='example.com' `` retry_attempts=1 url='http://backend.example.com:``/path/'
`` CONNECT: attempt fail [CONNECTION_ERROR] to 0.0.0.1:`` for host='example.com' `` retry_attempts=2 url='http://backend.example.com:``/path/'
`` CONNECT : `` connecting to 0.0.0.1:`` for host='example.com' url='http://backend.example.com:``/path/' fail_count='1' marking down
`` CONNECT: attempt fail [CONNECTION_ERROR] to 0.0.0.2:`` for host='example.com' `` retry_attempts=0 url='http://backend.example.com:``/path/'
`` CONNECT: attempt fail [CONNECTION_ERROR] to 0.0.0.2:`` for host='example.com' `` retry_attempts=1 url='http://backend.example.com:``/path/'
`` CONNECT: attempt fail [CONNECTION_ERROR] to 0.0.0.2:`` for host='example.com' `` retry_attempts=2 url='http://backend.example.com:``/path/'
`` CONNECT : `` connecting to 0.0.0.2:`` for host='example.com' url='http://backend.example.com:``/path/' fail_count='1' marking down
`` DNS Error: no valid server http://backend.example.com:``/path/
``
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
`` CONNECT: attempt fail [CONNECTION_ERROR] to 0.0.0.1:`` for host='example.com' `` retry_attempts=0 url='http://backend.example.com:``/path/'
`` CONNECT : `` connecting to 0.0.0.1:`` for host='example.com' url='http://backend.example.com:``/path/' fail_count='1' marking down
`` CONNECT: attempt fail [CONNECTION_ERROR] to 0.0.0.2:`` for host='example.com' `` retry_attempts=0 url='http://backend.example.com:``/path/'
`` CONNECT : `` connecting to 0.0.0.2:`` for host='example.com' url='http://backend.example.com:``/path/' fail_count='1' marking down
`` DNS Error: no valid server http://backend.example.com:``/path/
``
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
`` CONNECT: attempt fail [CONNECTION_ERROR] to 0.0.0.1:`` for host='example.com' `` retry_attempts=0 url='http://backend.example.com:``/path/'
`` CONNECT: attempt fail [CONNECTION_ERROR] to 0.0.0.1:`` for host='example.com' `` retry_attempts=0 url='http://backend.example.com:``/path/'
`` CONNECT : `` connecting to 0.0.0.1:`` for host='example.com' url='http://backend.example.com:``/path/' fail_count='2' marking down
`` CONNECT: attempt fail [CONNECTION_ERROR] to 0.0.0.2:`` for host='example.com' `` retry_attempts=0 url='http://backend.example.com:``/path/'
`` CONNECT: attempt fail [CONNECTION_ERROR] to 0.0.0.2:`` for host='example.com' `` retry_attempts=0 url='http://backend.example.com:``/path/'
`` CONNECT : `` connecting to 0.0.0.2:`` for host='example.com' url='http://backend.example.com:``/path/' fail_count='2' marking down
`` DNS Error: no valid server http://backend.example.com:``/path/
Original file line number Diff line number Diff line change
@@ -0,0 +1,133 @@
# Licensed to the Apache Software Foundation (ASF) under one
# or more contributor license agreements. See the NOTICE file
# distributed with this work for additional information
# regarding copyright ownership. The ASF licenses this file
# to you under the Apache License, Version 2.0 (the
# "License"); you may not use this file except in compliance
# with the License. You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

#
# This replay file assumes that caching is enabled.
Copy link

Copilot AI Mar 2, 2026

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The header comment says caching is enabled, but this replay config explicitly sets process_config.enable_cache: false. Please update or remove the comment so it matches the test configuration.

Suggested change
# This replay file assumes that caching is enabled.
# This replay file runs with ATS caching disabled.

Copilot uses AI. Check for mistakes.
#

meta:
version: "1.0"

# Configuration section for autest integration
autest:
description: 'Verify connect attempts behavior - round robin'

dns:
name: 'dns-rr'
records: {"backend.example.com": ["0.0.0.1", "0.0.0.2"]}

server:
name: 'server-rr'

client:
name: 'client-rr'

ats:
name: 'ts-rr'
process_config:
enable_cache: false

records_config:
proxy.config.diags.debug.enabled: 1
proxy.config.diags.debug.tags: 'http|hostdb|dns'
proxy.config.http.connect_attempts_rr_retries: 0
proxy.config.http.connect_attempts_max_retries: 2
proxy.config.http.connect_attempts_max_retries_down_server: 0
proxy.config.http.connect_attempts_timeout: 1
proxy.config.http.down_server.cache_time: 5

remap_config:
- from: "http://example.com/"
to: "http://backend.example.com:{SERVER_HTTP_PORT}/"

log_validation:
error_log:
gold_file: "gold/connect_attempts_rr_max_retries_error_log.gold"

sessions:
- transactions:
# try 0.0.0.1
- client-request:
method: GET
url: /path/
version: '1.1'
headers:
fields:
- [Host, example.com]
- [uuid, 1]

# should not hit
server-response:
status: 200
reason: OK

proxy-response:
status: 502

# try 0.0.0.2
- client-request:
method: GET
url: /path/
version: '1.1'
headers:
fields:
- [Host, example.com]
- [uuid, 2]

# should not hit
server-response:
status: 200
reason: OK

proxy-response:
status: 502

# request expected hit down_server cache and immidiately get 500
Copy link

Copilot AI Mar 2, 2026

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Spelling: "immidiately" should be "immediately" in this comment (and consider rewording to a full sentence for clarity).

Suggested change
# request expected hit down_server cache and immidiately get 500
# The request is expected to hit the down_server cache and immediately receive a 500 response.

Copilot uses AI. Check for mistakes.
- client-request:
method: GET
url: /path/
version: '1.1'
headers:
fields:
- [Host, example.com]
- [uuid, 10]

# should not hit
server-response:
status: 200
reason: OK

proxy-response:
status: 500

# when down_server.cache_time is expired, try connect attempts
- client-request:
method: GET
url: /path/
version: '1.1'
headers:
fields:
- [Host, example.com]
- [uuid, 20]
delay: 10s

# should not hit
server-response:
status: 200
reason: OK

proxy-response:
status: 502
133 changes: 133 additions & 0 deletions tests/gold_tests/dns/replay/connect_attempts_rr_no_retry.replay.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,133 @@
# Licensed to the Apache Software Foundation (ASF) under one
# or more contributor license agreements. See the NOTICE file
# distributed with this work for additional information
# regarding copyright ownership. The ASF licenses this file
# to you under the Apache License, Version 2.0 (the
# "License"); you may not use this file except in compliance
# with the License. You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

#
# This replay file assumes that caching is enabled.
Copy link

Copilot AI Mar 2, 2026

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The header comment says caching is enabled, but this replay config explicitly sets process_config.enable_cache: false. Please update or remove the comment so it matches the test configuration.

Suggested change
# This replay file assumes that caching is enabled.
# This replay file runs with caching disabled.

Copilot uses AI. Check for mistakes.
#

meta:
version: "1.0"

# Configuration section for autest integration
autest:
description: 'Verify connect attempts behavior - no retry'

dns:
name: 'dns-no'
records: {"backend.example.com": ["0.0.0.1", "0.0.0.2"]}

server:
name: 'server-no'

client:
name: 'client-no'

ats:
name: 'ts-no'
process_config:
enable_cache: false

records_config:
proxy.config.diags.debug.enabled: 1
proxy.config.diags.debug.tags: 'http|hostdb|dns'
proxy.config.http.connect_attempts_rr_retries: 0
proxy.config.http.connect_attempts_max_retries: 0
proxy.config.http.connect_attempts_max_retries_down_server: 0
proxy.config.http.connect_attempts_timeout: 1
proxy.config.http.down_server.cache_time: 5

remap_config:
- from: "http://example.com/"
to: "http://backend.example.com:{SERVER_HTTP_PORT}/"

log_validation:
error_log:
gold_file: "gold/connect_attempts_rr_no_error_log.gold"

sessions:
- transactions:
# try 0.0.0.1
- client-request:
method: GET
url: /path/
version: '1.1'
headers:
fields:
- [Host, example.com]
- [uuid, 1]

# should not hit
server-response:
status: 200
reason: OK

proxy-response:
status: 502

# try 0.0.0.2
- client-request:
method: GET
url: /path/
version: '1.1'
headers:
fields:
- [Host, example.com]
- [uuid, 2]

# should not hit
server-response:
status: 200
reason: OK

proxy-response:
status: 502

# request expected hit down_server cache and immidiately get 500
Copy link

Copilot AI Mar 2, 2026

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Spelling: "immidiately" should be "immediately" in this comment (and consider rewording to a full sentence for clarity).

Suggested change
# request expected hit down_server cache and immidiately get 500
# This request is expected to hit the down_server cache and immediately receive a 500 response.

Copilot uses AI. Check for mistakes.
- client-request:
method: GET
url: /path/
version: '1.1'
headers:
fields:
- [Host, example.com]
- [uuid, 10]

# should not hit
server-response:
status: 200
reason: OK

proxy-response:
status: 500

# when down_server.cache_time is expired, try connect attempts
- client-request:
method: GET
url: /path/
version: '1.1'
headers:
fields:
- [Host, example.com]
- [uuid, 20]
delay: 10s

# should not hit
server-response:
status: 200
reason: OK

proxy-response:
status: 502
Loading