From 5ae871da2f8f3a97cce45920e2790f8933590f62 Mon Sep 17 00:00:00 2001 From: Michal Domonkos Date: Tue, 7 May 2024 12:46:16 +0200 Subject: [PATCH 1/5] Fix up the pattern in "countme disabled" scenario Make sure *no* countme flag is present. --- dnf-behave-tests/dnf/countme.feature | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/dnf-behave-tests/dnf/countme.feature b/dnf-behave-tests/dnf/countme.feature index 19f1f54b7..48003fe4c 100644 --- a/dnf-behave-tests/dnf/countme.feature +++ b/dnf-behave-tests/dnf/countme.feature @@ -117,4 +117,4 @@ Feature: Better user counting When I execute dnf with args "makecache" 4 times Then no HTTP GET request should match: | path | - | */metalink.xml?countme=1 | + | */metalink.xml?countme=* | From b66c9d9c56cc33fdc57622324bfede3b52694832 Mon Sep 17 00:00:00 2001 From: Michal Domonkos Date: Tue, 7 May 2024 12:46:21 +0200 Subject: [PATCH 2/5] Add $releasever to metalink URL Also adapt the HTTP matching patterns in the countme feature file accordingly. This will be used in the next commit, no functional change right now. --- dnf-behave-tests/dnf/countme.feature | 32 ++++++++++++++-------------- dnf-behave-tests/dnf/steps/repo.py | 2 +- 2 files changed, 17 insertions(+), 17 deletions(-) diff --git a/dnf-behave-tests/dnf/countme.feature b/dnf-behave-tests/dnf/countme.feature index 48003fe4c..0939a4f3c 100644 --- a/dnf-behave-tests/dnf/countme.feature +++ b/dnf-behave-tests/dnf/countme.feature @@ -52,43 +52,43 @@ Feature: Better user counting # flag (see COUNTME_BUDGET=4 in libdnf/repo/Repo.cpp for details) When I execute dnf with args "makecache" 4 times Then exactly one HTTP GET request should match: - | path | - | */metalink.xml?countme=1 | + | path | + | */metalink.xml*&countme=1 | # Same week (should not be sent) When today is Friday, August 09, 2019 And I forget any HTTP requests captured so far And I execute dnf with args "makecache" 4 times Then no HTTP GET request should match: - | path | - | */metalink.xml?countme=* | + | path | + | */metalink.xml*&countme=* | # Next week (bucket 1) When today is Tuesday, August 13, 2019 And I forget any HTTP requests captured so far And I execute dnf with args "makecache" 4 times Then exactly one HTTP GET request should match: - | path | - | */metalink.xml?countme=1 | + | path | + | */metalink.xml*&countme=1 | # Next week (bucket 2) When today is Tuesday, August 21, 2019 And I forget any HTTP requests captured so far And I execute dnf with args "makecache" 4 times Then exactly one HTTP GET request should match: - | path | - | */metalink.xml?countme=2 | + | path | + | */metalink.xml*&countme=2 | # 1 month later (bucket 3) When today is Tuesday, September 16, 2019 And I forget any HTTP requests captured so far And I execute dnf with args "makecache" 4 times Then exactly one HTTP GET request should match: - | path | - | */metalink.xml?countme=3 | + | path | + | */metalink.xml*&countme=3 | # 6 months later (bucket 4) When today is Tuesday, March 15, 2020 And I forget any HTTP requests captured so far And I execute dnf with args "makecache" 4 times Then exactly one HTTP GET request should match: - | path | - | */metalink.xml?countme=4 | + | path | + | */metalink.xml*&countme=4 | Scenario: Countme flag is not sent repeatedly on retries Given I set config option "countme" to "1" @@ -105,8 +105,8 @@ Feature: Better user counting | path | | */metalink.xml* | And exactly one HTTP GET request should match: - | path | - | */metalink.xml?countme=1 | + | path | + | */metalink.xml*&countme=1 | Scenario: Countme feature is disabled Given I set config option "countme" to "0" @@ -116,5 +116,5 @@ Feature: Better user counting And I start capturing outbound HTTP requests When I execute dnf with args "makecache" 4 times Then no HTTP GET request should match: - | path | - | */metalink.xml?countme=* | + | path | + | */metalink.xml*&countme=* | diff --git a/dnf-behave-tests/dnf/steps/repo.py b/dnf-behave-tests/dnf/steps/repo.py index 5c130b4fc..274e95795 100644 --- a/dnf-behave-tests/dnf/steps/repo.py +++ b/dnf-behave-tests/dnf/steps/repo.py @@ -356,7 +356,7 @@ def step_set_up_metalink_for_repository(context, repo): generate_metalink(repo_info.path, url) repo_info.update_config({ "baseurl": "", - "metalink": url + "metalink.xml", + "metalink": url + "metalink.xml?releasever=$releasever", }) create_repo_conf(context, repo) From b09246100986ad63b5ca3081d03a42e7a1bccca7 Mon Sep 17 00:00:00 2001 From: Michal Domonkos Date: Tue, 7 May 2024 13:00:23 +0200 Subject: [PATCH 3/5] Improve readability of main countme scenario Shuffle some steps around, add blank lines and also emphasize that we're dealing with *calendar* weeks/months (i.e. aligned with the calendar, not with the first countme hit). No functional change. --- dnf-behave-tests/dnf/countme.feature | 22 ++++++++++++++-------- 1 file changed, 14 insertions(+), 8 deletions(-) diff --git a/dnf-behave-tests/dnf/countme.feature b/dnf-behave-tests/dnf/countme.feature index 0939a4f3c..81daa4ace 100644 --- a/dnf-behave-tests/dnf/countme.feature +++ b/dnf-behave-tests/dnf/countme.feature @@ -40,49 +40,55 @@ Feature: Better user counting | header | value | | User-Agent | Agent 007 | - Scenario: Countme flag is sent once per week + Scenario: Countme flag is sent once per calendar week Given I set config option "countme" to "1" - And today is Wednesday, August 07, 2019 And I copy repository "dnf-ci-fedora" for modification And I use repository "dnf-ci-fedora" as http And I set up metalink for repository "dnf-ci-fedora" And I start capturing outbound HTTP requests - # First week (bucket 1) + + # First calendar week (bucket 1) # Note: One in the first 4 requests is randomly chosen to include the # flag (see COUNTME_BUDGET=4 in libdnf/repo/Repo.cpp for details) + When today is Wednesday, August 07, 2019 When I execute dnf with args "makecache" 4 times Then exactly one HTTP GET request should match: | path | | */metalink.xml*&countme=1 | - # Same week (should not be sent) + + # Same calendar week (should not be sent) When today is Friday, August 09, 2019 And I forget any HTTP requests captured so far And I execute dnf with args "makecache" 4 times Then no HTTP GET request should match: | path | | */metalink.xml*&countme=* | - # Next week (bucket 1) + + # Next calendar week (bucket 1) When today is Tuesday, August 13, 2019 And I forget any HTTP requests captured so far And I execute dnf with args "makecache" 4 times Then exactly one HTTP GET request should match: | path | | */metalink.xml*&countme=1 | - # Next week (bucket 2) + + # Next calendar week (bucket 2) When today is Tuesday, August 21, 2019 And I forget any HTTP requests captured so far And I execute dnf with args "makecache" 4 times Then exactly one HTTP GET request should match: | path | | */metalink.xml*&countme=2 | - # 1 month later (bucket 3) + + # 1 calendar month later (bucket 3) When today is Tuesday, September 16, 2019 And I forget any HTTP requests captured so far And I execute dnf with args "makecache" 4 times Then exactly one HTTP GET request should match: | path | | */metalink.xml*&countme=3 | - # 6 months later (bucket 4) + + # 6 calendar months later (bucket 4) When today is Tuesday, March 15, 2020 And I forget any HTTP requests captured so far And I execute dnf with args "makecache" 4 times From 1bc3fb5d8a24ca41d5a06cba5187915e7070dbb6 Mon Sep 17 00:00:00 2001 From: Michal Domonkos Date: Thu, 9 May 2024 10:08:58 +0200 Subject: [PATCH 4/5] Convert $releasever "given" step to generic step We'll need this step in a "when" context in the next commit. No functional change. --- dnf-behave-tests/dnf/steps/repo.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/dnf-behave-tests/dnf/steps/repo.py b/dnf-behave-tests/dnf/steps/repo.py index 274e95795..b78b8dc69 100644 --- a/dnf-behave-tests/dnf/steps/repo.py +++ b/dnf-behave-tests/dnf/steps/repo.py @@ -137,7 +137,7 @@ def generate_metalink(destdir, url): f.write(data + '\n') -@behave.given("I set releasever to \"{releasever}\"") +@behave.step("I set releasever to \"{releasever}\"") def step_impl(context, releasever): context.dnf._set("releasever", releasever) for repo, info in context.dnf.repos.items(): From eb7f732eb6e3bbf9b893c28c5afd96356d97d1d4 Mon Sep 17 00:00:00 2001 From: Michal Domonkos Date: Thu, 2 May 2024 17:02:51 +0200 Subject: [PATCH 5/5] Add coverage for improved countme system age Adapt the countme feature to the libdnf fix for issue #1611, namely: - Turn the main scenario into a scenario outline to capture the various machine-id(5) configurations (see the table). - Add an upgrade scenario (from F39 to F40) that verifies that system age is now independent of $releasever on systems with a machine-id file. - Use NO_FAKE_STAT=1 in faketime invocations so that filesystem timestamps are *not* reported relative to the target time (this would break our custom machine-id timestamps we set here), see faketime(1) for details. - Add a new MachineId class to encapsulate the machine-id file, similar to OSRelease. - Mark the touched scenarios as destructive (due to them overriding the machine-id file). --- dnf-behave-tests/common/lib/cmd.py | 2 +- dnf-behave-tests/dnf/countme.feature | 82 +++++++++++++------ .../dnf/steps/fixtures/machineid.py | 63 ++++++++++++++ dnf-behave-tests/dnf/steps/repo.py | 17 ++++ 4 files changed, 136 insertions(+), 28 deletions(-) create mode 100644 dnf-behave-tests/dnf/steps/fixtures/machineid.py diff --git a/dnf-behave-tests/common/lib/cmd.py b/dnf-behave-tests/common/lib/cmd.py index 52c4d25bb..6b0a7b6bf 100644 --- a/dnf-behave-tests/common/lib/cmd.py +++ b/dnf-behave-tests/common/lib/cmd.py @@ -31,7 +31,7 @@ def run(cmd, shell=True, cwd=None): def run_in_context(context, cmd, can_fail=False, expected_exit_code=None, **run_args): if getattr(context, "faketime", None) is not None: - cmd = context.faketime + cmd + cmd = 'NO_FAKE_STAT=1 ' + context.faketime + cmd if getattr(context, "fake_kernel_release", None) is not None: cmd = context.fake_kernel_release + cmd diff --git a/dnf-behave-tests/dnf/countme.feature b/dnf-behave-tests/dnf/countme.feature index 81daa4ace..0a9d46841 100644 --- a/dnf-behave-tests/dnf/countme.feature +++ b/dnf-behave-tests/dnf/countme.feature @@ -40,64 +40,90 @@ Feature: Better user counting | header | value | | User-Agent | Agent 007 | - Scenario: Countme flag is sent once per calendar week - Given I set config option "countme" to "1" + @destructive + Scenario Outline: Countme flag is sent once per calendar week + Given the machine-id file is as of + And I set config option "countme" to "1" + And I set releasever to "39" And I copy repository "dnf-ci-fedora" for modification And I use repository "dnf-ci-fedora" as http And I set up metalink for repository "dnf-ci-fedora" And I start capturing outbound HTTP requests - # First calendar week (bucket 1) + # First calendar week # Note: One in the first 4 requests is randomly chosen to include the # flag (see COUNTME_BUDGET=4 in libdnf/repo/Repo.cpp for details) - When today is Wednesday, August 07, 2019 + When today is When I execute dnf with args "makecache" 4 times Then exactly one HTTP GET request should match: - | path | - | */metalink.xml*&countme=1 | + | path | + | */metalink.xml*&countme= | # Same calendar week (should not be sent) - When today is Friday, August 09, 2019 + When today is + 3 days And I forget any HTTP requests captured so far And I execute dnf with args "makecache" 4 times Then no HTTP GET request should match: - | path | - | */metalink.xml*&countme=* | + | path | + | */metalink.xml*&countme=* | - # Next calendar week (bucket 1) - When today is Tuesday, August 13, 2019 + # Next calendar week + When today is + 8 days And I forget any HTTP requests captured so far And I execute dnf with args "makecache" 4 times Then exactly one HTTP GET request should match: - | path | - | */metalink.xml*&countme=1 | + | path | + | */metalink.xml*&countme= | - # Next calendar week (bucket 2) - When today is Tuesday, August 21, 2019 + # Next calendar week + When today is + 15 days And I forget any HTTP requests captured so far And I execute dnf with args "makecache" 4 times Then exactly one HTTP GET request should match: - | path | - | */metalink.xml*&countme=2 | + | path | + | */metalink.xml*&countme= | - # 1 calendar month later (bucket 3) - When today is Tuesday, September 16, 2019 + # Next calendar month + When today is + 40 days And I forget any HTTP requests captured so far And I execute dnf with args "makecache" 4 times Then exactly one HTTP GET request should match: - | path | - | */metalink.xml*&countme=3 | + | path | + | */metalink.xml*&countme= | - # 6 calendar months later (bucket 4) - When today is Tuesday, March 15, 2020 + # 6 calendar months later + When today is + 182 days And I forget any HTTP requests captured so far And I execute dnf with args "makecache" 4 times Then exactly one HTTP GET request should match: - | path | - | */metalink.xml*&countme=4 | + | path | + | */metalink.xml*&countme= | + + # Even later, after a system upgrade + When today is + 365 days + And I set releasever to "40" + And I forget any HTTP requests captured so far + And I execute dnf with args "makecache" 4 times + Then exactly one HTTP GET request should match: + | path | + | */metalink.xml*&countme= | + Examples: + | machine-id | epoch | date | age#1 | age#2 | age#3 | age#4 | age#5 | age#6 | + # Absolute age counting (since "epoch") + | initialized | Aug 06, 2019 | Aug 07, 2019 | 1 | 1 | 2 | 3 | 4 | 4 | + | initialized | Aug 06, 2019 | Aug 20, 2019 | 2 | 2 | 2 | 3 | 4 | 4 | + | initialized | Aug 06, 2019 | Sep 12, 2019 | 3 | 3 | 3 | 3 | 4 | 4 | + | initialized | Aug 06, 2019 | Jun 18, 2020 | 4 | 4 | 4 | 4 | 4 | 4 | + # Relative age counting (since "date") + | uninitialized | Aug 06, 2019 | Jun 18, 2020 | 1 | 1 | 2 | 3 | 4 | 1 | + | empty | --- | Jun 18, 2020 | 1 | 1 | 2 | 3 | 4 | 1 | + | absent | --- | Jun 18, 2020 | 1 | 1 | 2 | 3 | 4 | 1 | + + @destructive Scenario: Countme flag is not sent repeatedly on retries - Given I set config option "countme" to "1" + Given the machine-id file is initialized as of today + And I set config option "countme" to "1" And I copy repository "dnf-ci-fedora" for modification And I use repository "dnf-ci-fedora" as http And I set up metalink for repository "dnf-ci-fedora" @@ -114,8 +140,10 @@ Feature: Better user counting | path | | */metalink.xml*&countme=1 | + @destructive Scenario: Countme feature is disabled - Given I set config option "countme" to "0" + Given the machine-id file is initialized as of today + And I set config option "countme" to "0" And I copy repository "dnf-ci-fedora" for modification And I use repository "dnf-ci-fedora" as http And I set up metalink for repository "dnf-ci-fedora" diff --git a/dnf-behave-tests/dnf/steps/fixtures/machineid.py b/dnf-behave-tests/dnf/steps/fixtures/machineid.py new file mode 100644 index 000000000..0d706c61e --- /dev/null +++ b/dnf-behave-tests/dnf/steps/fixtures/machineid.py @@ -0,0 +1,63 @@ +# -*- coding: utf-8 -*- + +from __future__ import absolute_import +from __future__ import print_function + +from behave import fixture +from datetime import datetime +import os + + +class MachineId(object): + """Represents the machine-id(5) file.""" + def __init__(self, path): + self._path = path + self._backup = path + '.bak' + if os.path.exists(path): + os.rename(path, self._backup) + + def _set_mtime(self, value): + """Set the given mtime on this file.""" + times = None + if value is not None and value != 'today': + ts = int(datetime.strptime(value, '%b %d, %Y').timestamp()) + times = (ts, ts) + os.utime(self._path, times=times) + + def initialize(self, mtime): + """Initialize the file and set the given mtime.""" + with open(self._path, 'w') as f: + f.write('dummy\n') + self._set_mtime(mtime) + + def uninitialize(self, mtime): + """Uninitialize the file and set the given mtime.""" + with open(self._path, 'w') as f: + f.write('uninitialized\n') + self._set_mtime(mtime) + + def empty(self): + """Empty the file.""" + open(self._path, 'w').close() + + def delete(self): + """Delete the file.""" + if os.path.exists(self._path): + os.remove(self._path) + + def __del__(self): + """Restore the backup.""" + if os.path.exists(self._backup): + os.rename(self._backup, self._path) + + +@fixture +def machineid_fixture(context): + try: + if not hasattr(context, "machineid"): + path = os.path.realpath('/etc/machine-id') + context.scenario.machineid = MachineId(path) + + yield context.scenario.machineid + finally: + del context.scenario.machineid diff --git a/dnf-behave-tests/dnf/steps/repo.py b/dnf-behave-tests/dnf/steps/repo.py index b78b8dc69..083cd9cb7 100644 --- a/dnf-behave-tests/dnf/steps/repo.py +++ b/dnf-behave-tests/dnf/steps/repo.py @@ -23,6 +23,7 @@ ) from fixtures import start_server_based_on_type, stop_server_type from fixtures.osrelease import osrelease_fixture +from fixtures.machineid import machineid_fixture def repo_config(repo, new={}): @@ -381,6 +382,22 @@ def given_no_osrelease(context): context.scenario.osrelease.delete() +@behave.step("the machine-id file is {what} as of {when}") +def step_machine_id_file(context, what, when): + behave.use_fixture(machineid_fixture, context) + machineid = context.scenario.machineid + if when.startswith('on '): + when = when[3:] + if what == 'initialized': + machineid.initialize(when) + elif what == 'uninitialized': + machineid.uninitialize(when) + elif what == 'empty': + machineid.empty() + elif what == 'absent': + machineid.delete() + + @behave.step("I invalidate solvfile version of \"{path}\"") def rewrite_solvfile_version(context, path): path = path.format(context=context)