Skip to content

Commit

Permalink
Gating: assume 'waiting' for two hours on missing results (#4219)
Browse files Browse the repository at this point in the history
This is part of addressing #4219. Currently we always count the
gating status as failed if requirements are unsatisfied. However,
it's usual for test results to not be present yet for a little
while after the update is created or edited. Ideally we would
have a mechanism for knowing for sure whether the tests in
question are pending, but we don't, so let's use a dumb rule
instead: if there are no failures (only missing results), and
it's less than two hours since the update was last modified, we'll
return 'waiting' as the status. After two hours, maybe the tests
actually didn't get scheduled or the policy is wrong, so we'll
return 'failed'.

This also simplifies the logic used to decide the status, and
reduces code duplication. We can determine the status using only
the number of satisfied and unsatisfied requirements: if there are
none of either it is ignored, if there are satisfied requirements
and no unsatisfied requirements it is passed, otherwise it is
failed or waiting.

As part of this we change the mock greenwave responses used in
a lot of other tests to use test-result-failed not -missing,
because they currently use -missing but expect the gating status
to be 'failed', which it no longer always will be. We also have
to make the mock greenwave responses used in tests that hit this
code a bit more realistic - they have to always include the
satisfied and unsatisfied requirement lists. Note I've checked
greenwave code and it does always include these in a successful
query response.

Signed-off-by: Adam Williamson <awilliam@redhat.com>
  • Loading branch information
AdamWill authored and AdamSaleh committed Sep 9, 2021
1 parent 923522c commit 37c9e74
Show file tree
Hide file tree
Showing 5 changed files with 363 additions and 70 deletions.
89 changes: 65 additions & 24 deletions bodhi/server/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@

from collections import defaultdict
from copy import copy
from datetime import datetime
from datetime import datetime, timedelta
from textwrap import wrap
import hashlib
import json
Expand Down Expand Up @@ -2211,42 +2211,83 @@ def get_test_gating_info(self):
}
return util.greenwave_api_post(self._greenwave_api_url, data)

def _get_test_gating_status(self):
@property
def _greenwave_requirements_generator(self):
"""
Query Greenwave about this update and return the information retrieved.
Query Greenwave about this update and return satisfied and unsatisfied requirements.
Returns:
TestGatingStatus:
- TestGatingStatus.ignored if no tests are required
- TestGatingStatus.failed if policies are not satisfied
- TestGatingStatus.passed if policies are satisfied, and there
are required tests
generator: An iterable of 2-tuples of lists of requirement dicts from each request
batch: first tuple item satisfied requirements, second item unsatisfied requirements.
Either list or both may be empty.
Raises:
BodhiException: When the ``greenwave_api_url`` is undefined in configuration.
RuntimeError: If Greenwave did not give us a 200 code.
RuntimeError: If Greenwave did not give us a 200 code, including if no applicable
policies were found.
"""
# If an unrestricted policy is applied and no tests are required
# on this update, let's set the test gating as ignored in Bodhi.
status = TestGatingStatus.ignored
for data in self.greenwave_request_batches(verbose=False):
response = util.greenwave_api_post(self._greenwave_api_url, data)
if not response['policies_satisfied']:
return TestGatingStatus.failed

if status != TestGatingStatus.ignored or response['summary'] != 'no tests are required':
status = TestGatingStatus.passed

return status
satisfied = response.get('satisfied_requirements', [])
unsatisfied = response.get('unsatisfied_requirements', [])
yield (satisfied, unsatisfied)

@property
def _unsatisfied_requirements(self):
unsatisfied_requirements = []
for data in self.greenwave_request_batches(verbose=False):
response = util.greenwave_api_post(self._greenwave_api_url, data)
unsatisfied_requirements.extend(response['unsatisfied_requirements'])
"""Query Greenwave about this update and return all unsatisfied requirements.
Returns:
list: A list of unsatisfied requirement dicts.
Raises:
BodhiException: When the ``greenwave_api_url`` is undefined in configuration.
RuntimeError: If Greenwave did not give us a 200 code, including if no applicable
policies were found.
"""
ret = []
for (_, unsatisfied) in self._greenwave_requirements_generator:
ret.extend(unsatisfied)
return ret

def _get_test_gating_status(self):
"""
Query Greenwave about this update and return the information retrieved.
return unsatisfied_requirements
Returns:
TestGatingStatus:
- TestGatingStatus.ignored if no tests are required
- TestGatingStatus.passed if there are required tests and policies are satisfied
- TestGatingStatus.waiting if there are required tests that have not yet completed,
no required test has failed, and update was last modified less than 2 hours ago
- TestGatingStatus.failed otherwise (failed required tests, missing required
test results and last modified more than 2 hours ago)
Raises:
BodhiException: When the ``greenwave_api_url`` is undefined in configuration.
RuntimeError: If Greenwave did not give us a 200 code, including if no applicable
policies were found.
"""
gotsat = False
gotunsat = False
recent = datetime.utcnow() - self.last_modified < timedelta(hours=2)
for (satisfied, unsatisfied) in self._greenwave_requirements_generator:
if satisfied:
gotsat = True
if unsatisfied:
gotunsat = True
if not recent or not all(req.get('type', '') == 'test-result-missing'
for req in unsatisfied):
return TestGatingStatus.failed

if not gotsat and not gotunsat:
return TestGatingStatus.ignored

if gotsat and not gotunsat:
return TestGatingStatus.passed

# here, we have unsatisfied requirements but we never bailed early
# in the for loop, so they are all 'missing' and update was recently modified
return TestGatingStatus.waiting

@property
def install_command(self) -> str:
Expand Down
34 changes: 32 additions & 2 deletions bodhi/tests/server/consumers/test_signed.py
Original file line number Diff line number Diff line change
Expand Up @@ -239,7 +239,22 @@ def test_consume_from_tag(self):
with mock.patch('bodhi.server.models.util.greenwave_api_post') as mock_greenwave:
greenwave_response = {
'policies_satisfied': True,
'summary': "all tests have passed"
'summary': "All required tests passed",
'applicable_policies': [
'kojibuild_bodhipush_no_requirements',
'kojibuild_bodhipush_remoterule',
'bodhiupdate_bodhipush_no_requirements',
'bodhiupdate_bodhipush_openqa'
],
'satisfied_requirements': [
{
'result_id': 39603316,
'subject_type': 'bodhi_update',
'testcase': 'update.install_default_update_netinst',
'type': 'test-result-passed'
},
],
'unsatisfied_requirements': []
}
mock_greenwave.return_value = greenwave_response
with fml_testing.mock_sends(update_schemas.UpdateReadyForTestingV1):
Expand Down Expand Up @@ -276,7 +291,22 @@ def test_consume_from_tag_composed_by_bodhi(self, add_tag):
with mock.patch('bodhi.server.models.util.greenwave_api_post') as mock_greenwave:
greenwave_response = {
'policies_satisfied': True,
'summary': "all tests have passed"
'summary': "All required tests passed",
'applicable_policies': [
'kojibuild_bodhipush_no_requirements',
'kojibuild_bodhipush_remoterule',
'bodhiupdate_bodhipush_no_requirements',
'bodhiupdate_bodhipush_openqa'
],
'satisfied_requirements': [
{
'result_id': 39603316,
'subject_type': 'bodhi_update',
'testcase': 'update.install_default_update_netinst',
'type': 'test-result-passed'
},
],
'unsatisfied_requirements': []
}
mock_greenwave.return_value = greenwave_response
with fml_testing.mock_sends(update_schemas.UpdateRequestTestingV1):
Expand Down
18 changes: 9 additions & 9 deletions bodhi/tests/server/services/test_updates.py
Original file line number Diff line number Diff line change
Expand Up @@ -5843,7 +5843,7 @@ def test_waive_test_results_1_unsatisfied_requirement(
},
'scenario': None,
'testcase': 'dist.rpmdeplint',
'type': 'test-result-missing'
'type': 'test-result-failed'
}
],
}
Expand Down Expand Up @@ -5912,7 +5912,7 @@ def test_waive_test_results_2_unsatisfied_requirements(
},
'scenario': None,
'testcase': 'dist.rpmdeplint',
'type': 'test-result-missing'
'type': 'test-result-failed'
},
{
'item': {
Expand All @@ -5921,7 +5921,7 @@ def test_waive_test_results_2_unsatisfied_requirements(
},
'scenario': None,
'testcase': 'atomic_ci_pipeline_results',
'type': 'test-result-missing'
'type': 'test-result-failed'
}
],
}
Expand Down Expand Up @@ -6006,7 +6006,7 @@ def test_waive_test_results_1_of_2_unsatisfied_requirements(
},
'scenario': None,
'testcase': 'dist.rpmdeplint',
'type': 'test-result-missing'
'type': 'test-result-failed'
},
{
'item': {
Expand All @@ -6015,7 +6015,7 @@ def test_waive_test_results_1_of_2_unsatisfied_requirements(
},
'scenario': None,
'testcase': 'atomic_ci_pipeline_results',
'type': 'test-result-missing'
'type': 'test-result-failed'
}
],
}
Expand Down Expand Up @@ -6088,7 +6088,7 @@ def test_waive_test_results_2_of_2_unsatisfied_requirements(
},
'scenario': None,
'testcase': 'dist.rpmdeplint',
'type': 'test-result-missing'
'type': 'test-result-failed'
},
{
'item': {
Expand All @@ -6097,7 +6097,7 @@ def test_waive_test_results_2_of_2_unsatisfied_requirements(
},
'scenario': None,
'testcase': 'atomic_ci_pipeline_results',
'type': 'test-result-missing'
'type': 'test-result-failed'
}
],
}
Expand Down Expand Up @@ -6186,7 +6186,7 @@ def test_waive_test_results_unfailing_tests(
},
'scenario': None,
'testcase': 'dist.rpmdeplint',
'type': 'test-result-missing'
'type': 'test-result-failed'
},
{
'item': {
Expand All @@ -6195,7 +6195,7 @@ def test_waive_test_results_unfailing_tests(
},
'scenario': None,
'testcase': 'atomic_ci_pipeline_results',
'type': 'test-result-missing'
'type': 'test-result-failed'
}
],
}
Expand Down
Loading

0 comments on commit 37c9e74

Please sign in to comment.