Autopkgtest: Collect results for requested tests before submitting

When we need to blow away and rebuild results.cache we want to avoid
re-triggering all tests. Thus collect already existing results for requested
tests before submitting new requests.

This is rather hackish now, as fetch_one_result() now has to deal with both 
self.requested_tests and self.pending_tests. The code should be refactored to
eliminate one of these maps.
bzr-import-20160707
Martin Pitt 9 years ago
parent 19cd69cb47
commit 9b70fe361d

@ -418,38 +418,39 @@ class AutoPackageTest(object):
# remove matching test requests, remember triggers # remove matching test requests, remember triggers
satisfied_triggers = set() satisfied_triggers = set()
for pending_ver, pending_archinfo in self.pending_tests.get(src, {}).copy().items(): for request_map in [self.requested_tests, self.pending_tests]:
# don't consider newer requested versions for pending_ver, pending_archinfo in request_map.get(src, {}).copy().items():
if apt_pkg.version_compare(pending_ver, ver) > 0: # don't consider newer requested versions
continue if apt_pkg.version_compare(pending_ver, ver) > 0:
continue
if result_triggers: if result_triggers:
# explicitly recording/retrieving test triggers is the # explicitly recording/retrieving test triggers is the
# preferred (and robust) way of matching results to pending # preferred (and robust) way of matching results to pending
# requests # requests
for result_trigger in result_triggers: for result_trigger in result_triggers:
try:
request_map[src][pending_ver][arch].remove(result_trigger)
self.log_verbose('-> matches pending request %s/%s/%s for trigger %s' %
(src, pending_ver, arch, str(result_trigger)))
satisfied_triggers.add(result_trigger)
except (KeyError, ValueError):
self.log_verbose('-> does not match any pending request for %s/%s/%s' %
(src, pending_ver, arch))
else:
# ... but we still need to support results without
# testinfo.json and recorded triggers until we stop caring about
# existing wily and trusty results; match the latest result to all
# triggers for src that have at least the requested version
try: try:
self.pending_tests[src][pending_ver][arch].remove(result_trigger) t = pending_archinfo[arch]
self.log_verbose('-> matches pending request %s/%s/%s for trigger %s' % self.log_verbose('-> matches pending request %s/%s for triggers %s' %
(src, pending_ver, arch, str(result_trigger))) (src, pending_ver, str(t)))
satisfied_triggers.add(result_trigger) satisfied_triggers.update(t)
except (KeyError, ValueError): del request_map[src][pending_ver][arch]
self.log_verbose('-> does not match any pending request for %s/%s/%s' % except KeyError:
(src, pending_ver, arch)) self.log_verbose('-> does not match any pending request for %s/%s' %
else: (src, pending_ver))
# ... but we still need to support results without
# testinfo.json and recorded triggers until we stop caring about
# existing wily and trusty results; match the latest result to all
# triggers for src that have at least the requested version
try:
t = pending_archinfo[arch]
self.log_verbose('-> matches pending request %s/%s for triggers %s' %
(src, pending_ver, str(t)))
satisfied_triggers.update(t)
del self.pending_tests[src][pending_ver][arch]
except KeyError:
self.log_verbose('-> does not match any pending request for %s/%s' %
(src, pending_ver))
# FIXME: this is a hack that mostly applies to re-running tests # FIXME: this is a hack that mostly applies to re-running tests
# manually without giving a trigger. Tests which don't get # manually without giving a trigger. Tests which don't get
@ -581,10 +582,24 @@ class AutoPackageTest(object):
# mark them as pending now # mark them as pending now
self.update_pending_tests() self.update_pending_tests()
def collect_requested(self):
'''Update results from swift for all requested packages
This is normally redundant with collect(), but avoids actually
sending test requests if results are already available. This mostly
happens when you have to blow away results.cache and let it rebuild
from scratch.
'''
for pkg, verinfo in copy.deepcopy(self.requested_tests).items():
for archinfo in verinfo.values():
for arch in archinfo:
self.fetch_swift_results(self.britney.options.adt_swift_url, pkg, arch)
def collect(self, packages): def collect(self, packages):
# update results from swift for all packages that we are waiting '''Update results from swift for all pending packages
# for, and remove pending tests that we have results for on all
# arches Remove pending tests for which we have results.
'''
for pkg, verinfo in copy.deepcopy(self.pending_tests).items(): for pkg, verinfo in copy.deepcopy(self.pending_tests).items():
for archinfo in verinfo.values(): for archinfo in verinfo.values():
for arch in archinfo: for arch in archinfo:

@ -1902,6 +1902,7 @@ class Britney(object):
autopkgtest_packages.append((e.name, e.ver[1])) autopkgtest_packages.append((e.name, e.ver[1]))
autopkgtest.request(autopkgtest_packages, autopkgtest_excludes) autopkgtest.request(autopkgtest_packages, autopkgtest_excludes)
if not self.options.dry_run: if not self.options.dry_run:
autopkgtest.collect_requested()
autopkgtest.submit() autopkgtest.submit()
autopkgtest.collect(autopkgtest_packages) autopkgtest.collect(autopkgtest_packages)
cloud_url = "http://autopkgtest.ubuntu.com/packages/%(h)s/%(s)s/%(r)s/%(a)s" cloud_url = "http://autopkgtest.ubuntu.com/packages/%(h)s/%(s)s/%(r)s/%(a)s"

@ -365,7 +365,11 @@ lightgreen 1 i386 green 2
{'green': [('old-version', '1'), ('new-version', '2')]} {'green': [('old-version', '1'), ('new-version', '2')]}
)[0] )[0]
# we already had all results before the run, so this should not trigger
# any new requests
self.assertEqual(self.amqp_requests, set())
self.assertEqual(self.pending_requests, '') self.assertEqual(self.pending_requests, '')
# not expecting any failures to retrieve from swift # not expecting any failures to retrieve from swift
self.assertNotIn('Failure', out, out) self.assertNotIn('Failure', out, out)
@ -542,7 +546,7 @@ lightgreen 1 i386 green 2
} }
) )
self.assertEqual(len(self.amqp_requests), 6) self.assertEqual(self.amqp_requests, set())
self.assertEqual(self.pending_requests, '') self.assertEqual(self.pending_requests, '')
# next run should not trigger any new requests # next run should not trigger any new requests
@ -551,14 +555,22 @@ lightgreen 1 i386 green 2
self.assertEqual(self.pending_requests, '') self.assertEqual(self.pending_requests, '')
# now lightgreen 2 gets built, should trigger a new test run # now lightgreen 2 gets built, should trigger a new test run
self.data.remove_all(True)
self.do_test(
[('libgreen1', {'Version': '1.1', 'Source': 'green', 'Depends': 'libc6'}, 'autopkgtest'),
('lightgreen', {'Version': '2'}, 'autopkgtest')],
{})
self.assertEqual(self.amqp_requests,
set(['debci-series-amd64:lightgreen {"triggers": ["lightgreen/2"]}',
'debci-series-i386:lightgreen {"triggers": ["lightgreen/2"]}']))
# next run collects the results
self.swift.set_results({'autopkgtest-series': { self.swift.set_results({'autopkgtest-series': {
'series/i386/l/lightgreen/20150101_100200@': (0, 'lightgreen 2'), 'series/i386/l/lightgreen/20150101_100200@': (0, 'lightgreen 2'),
'series/amd64/l/lightgreen/20150101_102000@': (0, 'lightgreen 2'), 'series/amd64/l/lightgreen/20150101_102000@': (0, 'lightgreen 2'),
}}) }})
self.data.remove_all(True)
self.do_test( self.do_test(
[('libgreen1', {'Version': '1.1', 'Source': 'green', 'Depends': 'libc6'}, 'autopkgtest'), [],
('lightgreen', {'Version': '2'}, 'autopkgtest')],
{'green': (True, {'green 1.1': {'amd64': 'PASS', 'i386': 'PASS'}, {'green': (True, {'green 1.1': {'amd64': 'PASS', 'i386': 'PASS'},
# FIXME: expecting a lightgreen test here # FIXME: expecting a lightgreen test here
# 'lightgreen 2': {'amd64': 'PASS', 'i386': 'PASS'}, # 'lightgreen 2': {'amd64': 'PASS', 'i386': 'PASS'},
@ -570,9 +582,7 @@ lightgreen 1 i386 green 2
'lightgreen': [('old-version', '1'), ('new-version', '2')], 'lightgreen': [('old-version', '1'), ('new-version', '2')],
} }
) )
self.assertEqual(self.amqp_requests, self.assertEqual(self.amqp_requests, set())
set(['debci-series-amd64:lightgreen {"triggers": ["lightgreen/2"]}',
'debci-series-i386:lightgreen {"triggers": ["lightgreen/2"]}']))
self.assertEqual(self.pending_requests, '') self.assertEqual(self.pending_requests, '')
def test_rdepends_unbuilt_unstable_only(self): def test_rdepends_unbuilt_unstable_only(self):
@ -646,7 +656,7 @@ lightgreen 1 i386 green 2
('excuses', 'lightgreen has no up-to-date binaries on any arch')] ('excuses', 'lightgreen has no up-to-date binaries on any arch')]
} }
) )
self.assertEqual(len(self.amqp_requests), 6) self.assertEqual(self.amqp_requests, set())
self.assertEqual(self.pending_requests, '') self.assertEqual(self.pending_requests, '')
# lightgreen 2 stays unbuilt in britney, but we get a test result for it # lightgreen 2 stays unbuilt in britney, but we get a test result for it
@ -896,7 +906,7 @@ newgreen 2 i386 newgreen 2
}, },
{'newgreen': [('old-version', '-'), ('new-version', '2')]}) {'newgreen': [('old-version', '-'), ('new-version', '2')]})
self.assertEqual(len(self.amqp_requests), 6) self.assertEqual(self.amqp_requests, set())
self.assertEqual(self.pending_requests, '') self.assertEqual(self.pending_requests, '')
def test_result_from_older_version(self): def test_result_from_older_version(self):
@ -965,7 +975,7 @@ newgreen 2 i386 newgreen 2
}), }),
}) })
self.assertEqual(len(self.amqp_requests), 6) self.assertEqual(self.amqp_requests, set())
self.assertEqual(self.pending_requests, '') self.assertEqual(self.pending_requests, '')
self.data.remove_all(True) self.data.remove_all(True)
@ -1129,6 +1139,7 @@ lightgreen 1 i386 green 3
}), }),
}) })
self.assertEqual(self.pending_requests, '') self.assertEqual(self.pending_requests, '')
self.assertEqual(self.amqp_requests, set())
# remove new lightgreen by resetting archive indexes, and re-adding # remove new lightgreen by resetting archive indexes, and re-adding
# green # green
@ -1161,9 +1172,7 @@ lightgreen 1 i386 green 3
# should not trigger new requests # should not trigger new requests
self.assertEqual(self.pending_requests, '') self.assertEqual(self.pending_requests, '')
self.assertEqual(self.amqp_requests, self.assertEqual(self.amqp_requests, set())
set(['debci-series-amd64:lightgreen {"triggers": ["green/2"]}',
'debci-series-i386:lightgreen {"triggers": ["green/2"]}']))
# but the next run should not trigger anything new # but the next run should not trigger anything new
self.do_test( self.do_test(

Loading…
Cancel
Save