blob: 256534fa828525a6bfb5154f51177ef91a630684 [file] [log] [blame]
Mike Frysingere58c0e22017-10-04 15:43:30 -04001# -*- coding: utf-8 -*-
David Jamesbb20ac82012-07-18 10:59:16 -07002# Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
David James8c846492011-01-25 17:07:29 -08003# Use of this source code is governed by a BSD-style license that can be
4# found in the LICENSE file.
5
Mike Frysingerc6824f62014-02-03 11:09:44 -05006"""Unittests for upload_prebuilts.py."""
7
Mike Frysinger383367e2014-09-16 15:06:17 -04008from __future__ import print_function
9
David James8c846492011-01-25 17:07:29 -080010import copy
David James8fa34ea2011-04-15 13:00:20 -070011import multiprocessing
Mike Frysinger40ffb532021-02-12 07:36:08 -050012import os
Mike Frysinger72b7cf92020-04-19 06:00:51 -040013import sys
David James8c846492011-01-25 17:07:29 -080014import tempfile
Chris Sosa471532a2011-02-01 15:10:06 -080015
Mike Frysinger40ffb532021-02-12 07:36:08 -050016from chromite.lib import binpkg
Brian Harringc92788f2012-09-21 18:07:15 -070017from chromite.lib import cros_test_lib
David James2c7ccb42012-11-04 15:34:28 -080018from chromite.lib import gs
Zdenek Behanc0e18762012-09-22 04:06:17 +020019from chromite.lib import osutils
Mike Frysinger36870f92015-04-12 02:59:55 -040020from chromite.lib import parallel_unittest
Prathmesh Prabhu421eef22014-10-16 17:13:19 -070021from chromite.lib import portage_util
Mike Frysinger40ffb532021-02-12 07:36:08 -050022from chromite.scripts import upload_prebuilts as prebuilt
23from chromite.third_party import mock
David James8c846492011-01-25 17:07:29 -080024
Mike Frysinger68182472014-11-05 22:38:39 -050025
Mike Frysinger27e21b72018-07-12 14:20:21 -040026# pylint: disable=protected-access
27
28
Mike Frysinger72b7cf92020-04-19 06:00:51 -040029assert sys.version_info >= (3, 6), 'This module requires Python 3.6+'
30
31
David James615e5b52011-06-03 11:10:15 -070032PUBLIC_PACKAGES = [{'CPV': 'gtk+/public1', 'SHA1': '1', 'MTIME': '1'},
David James8c846492011-01-25 17:07:29 -080033 {'CPV': 'gtk+/public2', 'SHA1': '2',
David James615e5b52011-06-03 11:10:15 -070034 'PATH': 'gtk+/foo.tgz', 'MTIME': '2'}]
35PRIVATE_PACKAGES = [{'CPV': 'private', 'SHA1': '3', 'MTIME': '3'}]
David James8c846492011-01-25 17:07:29 -080036
37
38def SimplePackageIndex(header=True, packages=True):
Chris Sosa58669192011-06-30 12:45:03 -070039 pkgindex = binpkg.PackageIndex()
40 if header:
David James2c7ccb42012-11-04 15:34:28 -080041 pkgindex.header['URI'] = 'gs://example'
Chris Sosa58669192011-06-30 12:45:03 -070042 if packages:
43 pkgindex.packages = copy.deepcopy(PUBLIC_PACKAGES + PRIVATE_PACKAGES)
44 return pkgindex
David James8c846492011-01-25 17:07:29 -080045
46
Brian Harringc92788f2012-09-21 18:07:15 -070047class TestUpdateFile(cros_test_lib.TempDirTestCase):
Mike Frysingerc6824f62014-02-03 11:09:44 -050048 """Tests for the UpdateLocalFile function."""
David James8c846492011-01-25 17:07:29 -080049
50 def setUp(self):
Mike Frysingerd6e2df02014-11-26 02:55:04 -050051 self.contents_str = [
52 '# comment that should be skipped',
53 'PKGDIR="/var/lib/portage/pkgs"',
54 'PORTAGE_BINHOST="http://no.thanks.com"',
55 'portage portage-20100310.tar.bz2',
56 'COMPILE_FLAGS="some_value=some_other"',
57 ]
Zdenek Behanc0e18762012-09-22 04:06:17 +020058 self.version_file = os.path.join(self.tempdir, 'version')
59 osutils.WriteFile(self.version_file, '\n'.join(self.contents_str))
David James8c846492011-01-25 17:07:29 -080060
David James8c846492011-01-25 17:07:29 -080061 def _read_version_file(self, version_file=None):
62 """Read the contents of self.version_file and return as a list."""
63 if not version_file:
64 version_file = self.version_file
65
66 version_fh = open(version_file)
67 try:
68 return [line.strip() for line in version_fh.readlines()]
69 finally:
70 version_fh.close()
71
72 def _verify_key_pair(self, key, val):
73 file_contents = self._read_version_file()
74 # ensure key for verify is wrapped on quotes
75 if '"' not in val:
76 val = '"%s"' % val
77 for entry in file_contents:
78 if '=' not in entry:
79 continue
80 file_key, file_val = entry.split('=')
81 if file_key == key:
82 if val == file_val:
83 break
84 else:
85 self.fail('Could not find "%s=%s" in version file' % (key, val))
86
87 def testAddVariableThatDoesNotExist(self):
88 """Add in a new variable that was no present in the file."""
89 key = 'PORTAGE_BINHOST'
90 value = '1234567'
91 prebuilt.UpdateLocalFile(self.version_file, value)
Mike Frysinger383367e2014-09-16 15:06:17 -040092 print(self.version_file)
Chris Sosa58669192011-06-30 12:45:03 -070093 self._read_version_file()
David James8c846492011-01-25 17:07:29 -080094 self._verify_key_pair(key, value)
Mike Frysinger383367e2014-09-16 15:06:17 -040095 print(self.version_file)
David James8c846492011-01-25 17:07:29 -080096
97 def testUpdateVariable(self):
98 """Test updating a variable that already exists."""
99 key, val = self.contents_str[2].split('=')
100 new_val = 'test_update'
101 self._verify_key_pair(key, val)
102 prebuilt.UpdateLocalFile(self.version_file, new_val)
103 self._verify_key_pair(key, new_val)
104
105 def testUpdateNonExistentFile(self):
106 key = 'PORTAGE_BINHOST'
107 value = '1234567'
108 non_existent_file = tempfile.mktemp()
109 try:
110 prebuilt.UpdateLocalFile(non_existent_file, value)
111 file_contents = self._read_version_file(non_existent_file)
Chris Sosa739c8752012-05-16 19:30:35 -0700112 self.assertEqual(file_contents, ['%s="%s"' % (key, value)])
David James8c846492011-01-25 17:07:29 -0800113 finally:
114 if os.path.exists(non_existent_file):
115 os.remove(non_existent_file)
116
117
Mike Frysinger68182472014-11-05 22:38:39 -0500118class TestPrebuilt(cros_test_lib.MockTestCase):
Mike Frysingerc6824f62014-02-03 11:09:44 -0500119 """Tests for Prebuilt logic."""
David James8c846492011-01-25 17:07:29 -0800120
Bertrand SIMONNET811bcde2014-11-20 15:21:25 -0800121 def setUp(self):
Mike Nicholsd0acc7f2021-05-21 17:18:24 +0000122 self._base_local_path = '/b/cbuild/build/chroot/build/x86-dogfood/'
Bertrand SIMONNET811bcde2014-11-20 15:21:25 -0800123 self._gs_bucket_path = 'gs://chromeos-prebuilt/host/version'
124 self._local_path = os.path.join(self._base_local_path, 'public1.tbz2')
125
David James8c846492011-01-25 17:07:29 -0800126 def testGenerateUploadDict(self):
Mike Frysinger68182472014-11-05 22:38:39 -0500127 self.PatchObject(prebuilt.os.path, 'exists', return_true=True)
Mike Frysingerd6e2df02014-11-26 02:55:04 -0500128 pkgs = [{'CPV': 'public1'}]
Bertrand SIMONNET811bcde2014-11-20 15:21:25 -0800129 result = prebuilt.GenerateUploadDict(self._base_local_path,
130 self._gs_bucket_path, pkgs)
131 expected = {self._local_path: self._gs_bucket_path + '/public1.tbz2', }
132 self.assertEqual(result, expected)
133
134 def testGenerateUploadDictWithDebug(self):
135 self.PatchObject(prebuilt.os.path, 'exists', return_true=True)
136 pkgs = [{'CPV': 'public1', 'DEBUG_SYMBOLS': 'yes'}]
137 result = prebuilt.GenerateUploadDict(self._base_local_path,
138 self._gs_bucket_path, pkgs)
139 expected = {self._local_path: self._gs_bucket_path + '/public1.tbz2',
140 self._local_path.replace('.tbz2', '.debug.tbz2'):
141 self._gs_bucket_path + '/public1.debug.tbz2'}
David James8c846492011-01-25 17:07:29 -0800142 self.assertEqual(result, expected)
143
David James8c846492011-01-25 17:07:29 -0800144 def testDeterminePrebuiltConfHost(self):
145 """Test that the host prebuilt path comes back properly."""
146 expected_path = os.path.join(prebuilt._PREBUILT_MAKE_CONF['amd64'])
147 self.assertEqual(prebuilt.DeterminePrebuiltConfFile('fake_path', 'amd64'),
148 expected_path)
149
David James8c846492011-01-25 17:07:29 -0800150
David James2c7ccb42012-11-04 15:34:28 -0800151class TestPkgIndex(cros_test_lib.TestCase):
Mike Frysingerc6824f62014-02-03 11:09:44 -0500152 """Helper for tests that update the Packages index file."""
David James2c7ccb42012-11-04 15:34:28 -0800153
154 def setUp(self):
155 self.db = {}
156 self.pkgindex = SimplePackageIndex()
157 self.empty = SimplePackageIndex(packages=False)
158
159 def assertURIs(self, uris):
160 """Verify that the duplicate DB has the specified URLs."""
161 expected = [v.uri for _, v in sorted(self.db.items())]
162 self.assertEqual(expected, uris)
163
164
165class TestPackagesFileFiltering(TestPkgIndex):
Mike Frysingerc6824f62014-02-03 11:09:44 -0500166 """Tests for Packages filtering behavior."""
David James8c846492011-01-25 17:07:29 -0800167
168 def testFilterPkgIndex(self):
David James2c7ccb42012-11-04 15:34:28 -0800169 """Test filtering out of private packages."""
170 self.pkgindex.RemoveFilteredPackages(lambda pkg: pkg in PRIVATE_PACKAGES)
171 self.assertEqual(self.pkgindex.packages, PUBLIC_PACKAGES)
172 self.assertEqual(self.pkgindex.modified, True)
David James8c846492011-01-25 17:07:29 -0800173
174
David James2c7ccb42012-11-04 15:34:28 -0800175class TestPopulateDuplicateDB(TestPkgIndex):
Mike Frysingerc6824f62014-02-03 11:09:44 -0500176 """Tests for the _PopulateDuplicateDB function."""
David James8c846492011-01-25 17:07:29 -0800177
178 def testEmptyIndex(self):
David James2c7ccb42012-11-04 15:34:28 -0800179 """Test population of the duplicate DB with an empty index."""
180 self.empty._PopulateDuplicateDB(self.db, 0)
181 self.assertEqual(self.db, {})
David James8c846492011-01-25 17:07:29 -0800182
183 def testNormalIndex(self):
David James2c7ccb42012-11-04 15:34:28 -0800184 """Test population of the duplicate DB with a full index."""
185 self.pkgindex._PopulateDuplicateDB(self.db, 0)
186 self.assertURIs(['gs://example/gtk+/public1.tbz2',
187 'gs://example/gtk+/foo.tgz',
188 'gs://example/private.tbz2'])
David James8c846492011-01-25 17:07:29 -0800189
190 def testMissingSHA1(self):
David James2c7ccb42012-11-04 15:34:28 -0800191 """Test population of the duplicate DB with a missing SHA1."""
192 del self.pkgindex.packages[0]['SHA1']
193 self.pkgindex._PopulateDuplicateDB(self.db, 0)
194 self.assertURIs(['gs://example/gtk+/foo.tgz',
195 'gs://example/private.tbz2'])
David James8c846492011-01-25 17:07:29 -0800196
197 def testFailedPopulate(self):
David James2c7ccb42012-11-04 15:34:28 -0800198 """Test failure conditions for the populate method."""
199 headerless = SimplePackageIndex(header=False)
200 self.assertRaises(KeyError, headerless._PopulateDuplicateDB, self.db, 0)
201 del self.pkgindex.packages[0]['CPV']
202 self.assertRaises(KeyError, self.pkgindex._PopulateDuplicateDB, self.db, 0)
David James8c846492011-01-25 17:07:29 -0800203
204
Mike Frysinger68182472014-11-05 22:38:39 -0500205class TestResolveDuplicateUploads(cros_test_lib.MockTestCase, TestPkgIndex):
Mike Frysingerc6824f62014-02-03 11:09:44 -0500206 """Tests for the ResolveDuplicateUploads function."""
David James8c846492011-01-25 17:07:29 -0800207
David James615e5b52011-06-03 11:10:15 -0700208 def setUp(self):
Mike Frysinger68182472014-11-05 22:38:39 -0500209 self.PatchObject(binpkg.time, 'time', return_value=binpkg.TWO_WEEKS)
David James2c7ccb42012-11-04 15:34:28 -0800210 self.db = {}
211 self.dup = SimplePackageIndex()
212 self.expected_pkgindex = SimplePackageIndex()
213
214 def assertNoDuplicates(self, candidates):
215 """Verify no duplicates are found with the specified candidates."""
216 uploads = self.pkgindex.ResolveDuplicateUploads(candidates)
217 self.assertEqual(uploads, self.pkgindex.packages)
218 self.assertEqual(len(self.pkgindex.packages),
219 len(self.expected_pkgindex.packages))
220 for pkg1, pkg2 in zip(self.pkgindex.packages,
221 self.expected_pkgindex.packages):
222 self.assertNotEqual(pkg1['MTIME'], pkg2['MTIME'])
223 del pkg1['MTIME']
224 del pkg2['MTIME']
225 self.assertEqual(self.pkgindex.modified, False)
226 self.assertEqual(self.pkgindex.packages, self.expected_pkgindex.packages)
227
228 def assertAllDuplicates(self, candidates):
229 """Verify every package is a duplicate in the specified list."""
230 for pkg in self.expected_pkgindex.packages:
231 pkg.setdefault('PATH', pkg['CPV'] + '.tbz2')
232 self.pkgindex.ResolveDuplicateUploads(candidates)
233 self.assertEqual(self.pkgindex.packages, self.expected_pkgindex.packages)
David James615e5b52011-06-03 11:10:15 -0700234
David James8c846492011-01-25 17:07:29 -0800235 def testEmptyList(self):
David James2c7ccb42012-11-04 15:34:28 -0800236 """If no candidates are supplied, no duplicates should be found."""
237 self.assertNoDuplicates([])
David James8c846492011-01-25 17:07:29 -0800238
239 def testEmptyIndex(self):
David James2c7ccb42012-11-04 15:34:28 -0800240 """If no packages are supplied, no duplicates should be found."""
241 self.assertNoDuplicates([self.empty])
David James8c846492011-01-25 17:07:29 -0800242
David James2c7ccb42012-11-04 15:34:28 -0800243 def testDifferentURI(self):
244 """If the URI differs, no duplicates should be found."""
245 self.dup.header['URI'] = 'gs://example2'
246 self.assertNoDuplicates([self.dup])
247
248 def testUpdateModificationTime(self):
249 """When duplicates are found, we should use the latest mtime."""
250 for pkg in self.expected_pkgindex.packages:
251 pkg['MTIME'] = '10'
252 for pkg in self.dup.packages:
253 pkg['MTIME'] = '4'
254 self.assertAllDuplicates([self.expected_pkgindex, self.dup])
255
256 def testCanonicalUrl(self):
257 """If the URL is in a different format, we should still find duplicates."""
258 self.dup.header['URI'] = gs.PUBLIC_BASE_HTTPS_URL + 'example'
259 self.assertAllDuplicates([self.dup])
David James8c846492011-01-25 17:07:29 -0800260
261 def testMissingSHA1(self):
David James2c7ccb42012-11-04 15:34:28 -0800262 """We should not find duplicates if there is no SHA1."""
263 del self.pkgindex.packages[0]['SHA1']
264 del self.expected_pkgindex.packages[0]['SHA1']
265 for pkg in self.expected_pkgindex.packages[1:]:
David James8c846492011-01-25 17:07:29 -0800266 pkg.setdefault('PATH', pkg['CPV'] + '.tbz2')
David James2c7ccb42012-11-04 15:34:28 -0800267 self.pkgindex.ResolveDuplicateUploads([self.dup])
268 self.assertNotEqual(self.pkgindex.packages[0]['MTIME'],
269 self.expected_pkgindex.packages[0]['MTIME'])
270 del self.pkgindex.packages[0]['MTIME']
271 del self.expected_pkgindex.packages[0]['MTIME']
272 self.assertEqual(self.pkgindex.packages, self.expected_pkgindex.packages)
David James8c846492011-01-25 17:07:29 -0800273
Bertrand SIMONNET811bcde2014-11-20 15:21:25 -0800274 def testSymbolsAvailable(self):
275 """If symbols are available remotely, re-use them and set DEBUG_SYMBOLS."""
276 self.dup.packages[0]['DEBUG_SYMBOLS'] = 'yes'
277
278 uploads = self.pkgindex.ResolveDuplicateUploads([self.dup])
279 self.assertEqual(uploads, [])
280 self.assertEqual(self.pkgindex.packages[0].get('DEBUG_SYMBOLS'), 'yes')
281
282 def testSymbolsAvailableLocallyOnly(self):
283 """If the symbols are only available locally, reupload them."""
284 self.pkgindex.packages[0]['DEBUG_SYMBOLS'] = 'yes'
285
286 uploads = self.pkgindex.ResolveDuplicateUploads([self.dup])
287 self.assertEqual(uploads, [self.pkgindex.packages[0]])
288
David James8c846492011-01-25 17:07:29 -0800289
Mike Frysinger68182472014-11-05 22:38:39 -0500290class TestWritePackageIndex(cros_test_lib.MockTestCase, TestPkgIndex):
Mike Frysingerc6824f62014-02-03 11:09:44 -0500291 """Tests for the WriteToNamedTemporaryFile function."""
David James8c846492011-01-25 17:07:29 -0800292
293 def testSimple(self):
David James2c7ccb42012-11-04 15:34:28 -0800294 """Test simple call of WriteToNamedTemporaryFile()"""
Mike Frysinger68182472014-11-05 22:38:39 -0500295 self.PatchObject(self.pkgindex, 'Write')
David James2c7ccb42012-11-04 15:34:28 -0800296 f = self.pkgindex.WriteToNamedTemporaryFile()
Mike Frysinger652057c2019-12-09 23:59:07 -0500297 self.assertEqual(f.read(), '')
David James8c846492011-01-25 17:07:29 -0800298
299
Mike Frysinger36870f92015-04-12 02:59:55 -0400300class TestUploadPrebuilt(cros_test_lib.MockTempDirTestCase):
Mike Frysingerc6824f62014-02-03 11:09:44 -0500301 """Tests for the _UploadPrebuilt function."""
David James05bcb2b2011-02-09 09:25:47 -0800302
303 def setUp(self):
304 class MockTemporaryFile(object):
Mike Frysingerc6824f62014-02-03 11:09:44 -0500305 """Mock out the temporary file logic."""
David James05bcb2b2011-02-09 09:25:47 -0800306 def __init__(self, name):
307 self.name = name
David James05bcb2b2011-02-09 09:25:47 -0800308 self.pkgindex = SimplePackageIndex()
Mike Frysinger36870f92015-04-12 02:59:55 -0400309 self.PatchObject(binpkg, 'GrabLocalPackageIndex',
310 return_value=self.pkgindex)
311 self.PatchObject(self.pkgindex, 'ResolveDuplicateUploads',
312 return_value=PRIVATE_PACKAGES)
313 self.PatchObject(self.pkgindex, 'WriteToNamedTemporaryFile',
314 return_value=MockTemporaryFile('fake'))
315 self.remote_up_mock = self.PatchObject(prebuilt, 'RemoteUpload')
316 self.gs_up_mock = self.PatchObject(prebuilt, '_GsUpload')
David James05bcb2b2011-02-09 09:25:47 -0800317
David James05bcb2b2011-02-09 09:25:47 -0800318 def testSuccessfulGsUpload(self):
Alex Deymo541ea6f2014-12-23 01:18:14 -0800319 uploads = {
320 os.path.join(self.tempdir, 'private.tbz2'): 'gs://foo/private.tbz2'}
Mike Frysinger9bb6cd82020-08-20 03:02:23 -0400321 dev_extras = os.path.join(self.tempdir, 'dev-only-extras.tar.xz')
322 osutils.Touch(dev_extras)
Mike Frysinger36870f92015-04-12 02:59:55 -0400323 self.PatchObject(prebuilt, 'GenerateUploadDict',
324 return_value=uploads)
David James05bcb2b2011-02-09 09:25:47 -0800325 uploads = uploads.copy()
326 uploads['fake'] = 'gs://foo/suffix/Packages'
Mike Frysinger9bb6cd82020-08-20 03:02:23 -0400327 uploads[dev_extras] = 'gs://foo/suffix/dev-only-extras.tar.xz'
David Jamesfd0b0852011-02-23 11:15:36 -0800328 acl = 'public-read'
David Jamesc0f158a2011-02-22 16:07:29 -0800329 uri = self.pkgindex.header['URI']
David James8ece7ee2011-06-29 16:02:30 -0700330 uploader = prebuilt.PrebuiltUploader('gs://foo', acl, uri, [], '/', [],
Mike Frysinger8092a632014-05-24 13:25:46 -0400331 False, 'foo', False, 'x86-foo', [], '')
Alex Deymo541ea6f2014-12-23 01:18:14 -0800332 uploader._UploadPrebuilt(self.tempdir, 'suffix')
Mike Frysinger36870f92015-04-12 02:59:55 -0400333 self.remote_up_mock.assert_called_once_with(mock.ANY, acl, uploads)
334 self.assertTrue(self.gs_up_mock.called)
David James05bcb2b2011-02-09 09:25:47 -0800335
David James05bcb2b2011-02-09 09:25:47 -0800336
Mike Frysinger36870f92015-04-12 02:59:55 -0400337class TestSyncPrebuilts(cros_test_lib.MockTestCase):
Mike Frysingerc6824f62014-02-03 11:09:44 -0500338 """Tests for the SyncHostPrebuilts function."""
David James05bcb2b2011-02-09 09:25:47 -0800339
340 def setUp(self):
Mike Frysinger36870f92015-04-12 02:59:55 -0400341 self.rev_mock = self.PatchObject(prebuilt, 'RevGitFile', return_value=None)
342 self.update_binhost_mock = self.PatchObject(
343 prebuilt, 'UpdateBinhostConfFile', return_value=None)
David James05bcb2b2011-02-09 09:25:47 -0800344 self.build_path = '/trunk'
345 self.upload_location = 'gs://upload/'
346 self.version = '1'
347 self.binhost = 'http://prebuilt/'
348 self.key = 'PORTAGE_BINHOST'
Mike Frysinger36870f92015-04-12 02:59:55 -0400349 self.upload_mock = self.PatchObject(prebuilt.PrebuiltUploader,
350 '_UploadPrebuilt', return_value=True)
David James05bcb2b2011-02-09 09:25:47 -0800351
David James05bcb2b2011-02-09 09:25:47 -0800352 def testSyncHostPrebuilts(self):
David Jamese2488642011-11-14 16:15:20 -0800353 board = 'x86-foo'
David James4058b0d2011-12-08 21:24:50 -0800354 target = prebuilt.BuildTarget(board, 'aura')
355 slave_targets = [prebuilt.BuildTarget('x86-bar', 'aura')]
David James05bcb2b2011-02-09 09:25:47 -0800356 package_path = os.path.join(self.build_path,
357 prebuilt._HOST_PACKAGES_PATH)
Mike Frysingerd6e2df02014-11-26 02:55:04 -0500358 url_suffix = prebuilt._REL_HOST_PATH % {
359 'version': self.version,
360 'host_arch': prebuilt._HOST_ARCH,
361 'target': target,
362 }
David James8fa34ea2011-04-15 13:00:20 -0700363 packages_url_suffix = '%s/packages' % url_suffix.rstrip('/')
David Jamesf0e6fd72011-04-15 15:58:07 -0700364 url_value = '%s/%s/' % (self.binhost.rstrip('/'),
365 packages_url_suffix.rstrip('/'))
David Jamese2488642011-11-14 16:15:20 -0800366 urls = [url_value.replace('foo', 'bar'), url_value]
David James20b2b6f2011-11-18 15:11:58 -0800367 binhost = ' '.join(urls)
David James615e5b52011-06-03 11:10:15 -0700368 uploader = prebuilt.PrebuiltUploader(
369 self.upload_location, 'public-read', self.binhost, [],
Mike Frysinger8092a632014-05-24 13:25:46 -0400370 self.build_path, [], False, 'foo', False, target, slave_targets,
371 self.version)
372 uploader.SyncHostPrebuilts(self.key, True, True)
Mike Frysinger36870f92015-04-12 02:59:55 -0400373 self.upload_mock.assert_called_once_with(package_path, packages_url_suffix)
374 self.rev_mock.assert_called_once_with(
375 mock.ANY, {self.key: binhost}, dryrun=False)
376 self.update_binhost_mock.assert_called_once_with(
377 mock.ANY, self.key, binhost)
David James05bcb2b2011-02-09 09:25:47 -0800378
379 def testSyncBoardPrebuilts(self):
David Jamese2488642011-11-14 16:15:20 -0800380 board = 'x86-foo'
David James4058b0d2011-12-08 21:24:50 -0800381 target = prebuilt.BuildTarget(board, 'aura')
382 slave_targets = [prebuilt.BuildTarget('x86-bar', 'aura')]
Mike Frysinger8092a632014-05-24 13:25:46 -0400383 board_path = os.path.join(
384 self.build_path, prebuilt._BOARD_PATH % {'board': board})
David James05bcb2b2011-02-09 09:25:47 -0800385 package_path = os.path.join(board_path, 'packages')
Mike Frysingerd6e2df02014-11-26 02:55:04 -0500386 url_suffix = prebuilt._REL_BOARD_PATH % {
387 'version': self.version,
388 'target': target,
389 }
David James8fa34ea2011-04-15 13:00:20 -0700390 packages_url_suffix = '%s/packages' % url_suffix.rstrip('/')
David Jamesf0e6fd72011-04-15 15:58:07 -0700391 url_value = '%s/%s/' % (self.binhost.rstrip('/'),
392 packages_url_suffix.rstrip('/'))
David Jamese2488642011-11-14 16:15:20 -0800393 bar_binhost = url_value.replace('foo', 'bar')
Mike Frysinger36870f92015-04-12 02:59:55 -0400394 determine_mock = self.PatchObject(prebuilt, 'DeterminePrebuiltConfFile',
395 side_effect=('bar', 'foo'))
396 self.PatchObject(prebuilt.PrebuiltUploader, '_UploadSdkTarball')
397 with parallel_unittest.ParallelMock():
398 multiprocessing.Process.exitcode = 0
399 uploader = prebuilt.PrebuiltUploader(
400 self.upload_location, 'public-read', self.binhost, [],
401 self.build_path, [], False, 'foo', False, target, slave_targets,
402 self.version)
Gilad Arnoldad333182015-05-27 15:50:41 -0700403 uploader.SyncBoardPrebuilts(self.key, True, True, True, None, None, None,
404 None, None)
Mike Frysinger36870f92015-04-12 02:59:55 -0400405 determine_mock.assert_has_calls([
406 mock.call(self.build_path, slave_targets[0]),
407 mock.call(self.build_path, target),
408 ])
409 self.upload_mock.assert_called_once_with(package_path, packages_url_suffix)
410 self.rev_mock.assert_has_calls([
411 mock.call('bar', {self.key: bar_binhost}, dryrun=False),
412 mock.call('foo', {self.key: url_value}, dryrun=False),
413 ])
414 self.update_binhost_mock.assert_has_calls([
415 mock.call(mock.ANY, self.key, bar_binhost),
416 mock.call(mock.ANY, self.key, url_value),
417 ])
David James05bcb2b2011-02-09 09:25:47 -0800418
419
Mike Frysinger36870f92015-04-12 02:59:55 -0400420class TestMain(cros_test_lib.MockTestCase):
Mike Frysingerc6824f62014-02-03 11:09:44 -0500421 """Tests for the main() function."""
David Jamesc0f158a2011-02-22 16:07:29 -0800422
423 def testMain(self):
424 """Test that the main function works."""
Mike Frysinger36870f92015-04-12 02:59:55 -0400425 options = mock.MagicMock()
David Jamesc0f158a2011-02-22 16:07:29 -0800426 old_binhost = 'http://prebuilt/1'
427 options.previous_binhost_url = [old_binhost]
David Jamese2488642011-11-14 16:15:20 -0800428 options.board = 'x86-foo'
David James4058b0d2011-12-08 21:24:50 -0800429 options.profile = None
Chris Sosa6a5dceb2012-05-14 13:48:56 -0700430 target = prebuilt.BuildTarget(options.board, options.profile)
David Jamesc0f158a2011-02-22 16:07:29 -0800431 options.build_path = '/trunk'
Mike Frysinger86509232014-05-24 13:18:37 -0400432 options.dryrun = False
David Jamesfd0b0852011-02-23 11:15:36 -0800433 options.private = True
David James615e5b52011-06-03 11:10:15 -0700434 options.packages = []
David Jamesc0f158a2011-02-22 16:07:29 -0800435 options.sync_host = True
436 options.git_sync = True
David James8fa34ea2011-04-15 13:00:20 -0700437 options.upload_board_tarball = True
Zdenek Behan62a57792012-08-31 15:09:08 +0200438 options.prepackaged_tarball = None
Gilad Arnold2b79e2d2015-06-02 11:26:07 -0700439 options.toolchains_overlay_tarballs = []
440 options.toolchains_overlay_upload_path = ''
Mike Frysinger9e979b92012-11-29 02:55:09 -0500441 options.toolchain_tarballs = []
442 options.toolchain_upload_path = ''
David Jamesc0f158a2011-02-22 16:07:29 -0800443 options.upload = 'gs://upload/'
David Jamesfd0b0852011-02-23 11:15:36 -0800444 options.binhost_base_url = options.upload
David Jamesc0f158a2011-02-22 16:07:29 -0800445 options.prepend_version = True
David James8ece7ee2011-06-29 16:02:30 -0700446 options.set_version = None
447 options.skip_upload = False
David Jamesc0f158a2011-02-22 16:07:29 -0800448 options.filters = True
449 options.key = 'PORTAGE_BINHOST'
David Jamesb26b9312014-12-15 11:26:46 -0800450 options.binhost_conf_dir = None
David Jamesc0f158a2011-02-22 16:07:29 -0800451 options.sync_binhost_conf = True
David James4058b0d2011-12-08 21:24:50 -0800452 options.slave_targets = [prebuilt.BuildTarget('x86-bar', 'aura')]
Mike Frysinger36870f92015-04-12 02:59:55 -0400453 self.PatchObject(prebuilt, 'ParseOptions',
454 return_value=tuple([options, target]))
455 self.PatchObject(binpkg, 'GrabRemotePackageIndex', return_value=True)
456 init_mock = self.PatchObject(prebuilt.PrebuiltUploader, '__init__',
457 return_value=None)
458 expected_gs_acl_path = os.path.join('/fake_path',
Prathmesh Prabhu5f14da02014-10-17 15:13:56 -0700459 prebuilt._GOOGLESTORAGE_GSUTIL_FILE)
Mike Frysinger36870f92015-04-12 02:59:55 -0400460 self.PatchObject(portage_util, 'FindOverlayFile',
461 return_value=expected_gs_acl_path)
462 host_mock = self.PatchObject(
463 prebuilt.PrebuiltUploader, 'SyncHostPrebuilts', return_value=None)
464 board_mock = self.PatchObject(
465 prebuilt.PrebuiltUploader, 'SyncBoardPrebuilts', return_value=None)
466
467 prebuilt.main([])
468
469 init_mock.assert_called_once_with(options.upload, expected_gs_acl_path,
470 options.upload, mock.ANY,
471 options.build_path, options.packages,
472 False, None, False,
473 target, options.slave_targets,
474 mock.ANY)
475 board_mock.assert_called_once_with(
Mike Frysinger8092a632014-05-24 13:25:46 -0400476 options.key, options.git_sync,
Gilad Arnoldad333182015-05-27 15:50:41 -0700477 options.sync_binhost_conf, options.upload_board_tarball, None,
478 [], '', [], '')
Mike Frysinger36870f92015-04-12 02:59:55 -0400479 host_mock.assert_called_once_with(
480 options.key, options.git_sync, options.sync_binhost_conf)
David Jamesc0f158a2011-02-22 16:07:29 -0800481
Mike Frysinger9e979b92012-11-29 02:55:09 -0500482
Mike Frysinger68182472014-11-05 22:38:39 -0500483class TestSdk(cros_test_lib.MockTestCase):
Mike Frysinger9e979b92012-11-29 02:55:09 -0500484 """Test logic related to uploading SDK binaries"""
485
486 def setUp(self):
Mike Frysinger68182472014-11-05 22:38:39 -0500487 self.PatchObject(prebuilt, '_GsUpload',
488 side_effect=Exception('should not get called'))
489 self.PatchObject(prebuilt, 'UpdateBinhostConfFile',
490 side_effect=Exception('should not get called'))
491 self.upload_mock = self.PatchObject(prebuilt.PrebuiltUploader, '_Upload')
Mike Frysinger9e979b92012-11-29 02:55:09 -0500492
493 self.acl = 'magic-acl'
494
495 # All these args pretty much get ignored. Whee.
496 self.uploader = prebuilt.PrebuiltUploader(
497 'gs://foo', self.acl, 'prebuilt', [], '/', [],
Mike Frysinger8092a632014-05-24 13:25:46 -0400498 False, 'foo', False, 'x86-foo', [], 'chroot-1234')
Mike Frysinger9e979b92012-11-29 02:55:09 -0500499
Gilad Arnold2b79e2d2015-06-02 11:26:07 -0700500 def testSdkUpload(self, to_tarballs=(), to_upload_path=None,
Gilad Arnoldad333182015-05-27 15:50:41 -0700501 tc_tarballs=(), tc_upload_path=None):
Mike Frysinger9e979b92012-11-29 02:55:09 -0500502 """Make sure we can upload just an SDK tarball"""
503 tar = 'sdk.tar.xz'
504 ver = '1234'
505 vtar = 'cros-sdk-%s.tar.xz' % ver
506
Mike Frysinger68182472014-11-05 22:38:39 -0500507 calls = [
508 mock.call('%s.Manifest' % tar,
509 'gs://chromiumos-sdk/%s.Manifest' % vtar),
510 mock.call(tar, 'gs://chromiumos-sdk/%s' % vtar),
511 ]
Gilad Arnold2b79e2d2015-06-02 11:26:07 -0700512 for to in to_tarballs:
513 to = to.split(':')
Gilad Arnoldad333182015-05-27 15:50:41 -0700514 calls.append(mock.call(
Gilad Arnold2b79e2d2015-06-02 11:26:07 -0700515 to[1],
516 ('gs://chromiumos-sdk/' + to_upload_path) % {'toolchains': to[0]}))
Mike Frysinger68182472014-11-05 22:38:39 -0500517 for tc in tc_tarballs:
518 tc = tc.split(':')
519 calls.append(mock.call(
520 tc[1], ('gs://chromiumos-sdk/' + tc_upload_path) % {'target': tc[0]}))
521 calls.append(mock.call(
522 mock.ANY, 'gs://chromiumos-sdk/cros-sdk-latest.conf'))
Mike Frysinger9e979b92012-11-29 02:55:09 -0500523
Mike Frysinger8092a632014-05-24 13:25:46 -0400524 self.uploader._UploadSdkTarball('amd64-host', '',
Gilad Arnold2b79e2d2015-06-02 11:26:07 -0700525 tar, to_tarballs, to_upload_path,
Gilad Arnoldad333182015-05-27 15:50:41 -0700526 tc_tarballs, tc_upload_path)
Mike Frysinger68182472014-11-05 22:38:39 -0500527 self.upload_mock.assert_has_calls(calls)
Mike Frysinger9e979b92012-11-29 02:55:09 -0500528
Gilad Arnoldad333182015-05-27 15:50:41 -0700529 def testBoardOverlayTarballUpload(self):
530 """Make sure processing of board-specific overlay tarballs works."""
Gilad Arnold2b79e2d2015-06-02 11:26:07 -0700531 to_tarballs = (
532 ('i686-pc-linux-gnu:'
533 '/some/path/built-sdk-overlay-toolchains-i686-pc-linux-gnu.tar.xz'),
534 ('armv7a-cros-linux-gnueabi-arm-none-eabi:'
535 '/some/path/built-sdk-overlay-toolchains-armv7a-cros-linux-gnueabi-'
536 'arm-none-eabi'),
Gilad Arnoldad333182015-05-27 15:50:41 -0700537 )
Gilad Arnold2b79e2d2015-06-02 11:26:07 -0700538 to_upload_path = (
539 '1994/04/cros-sdk-overlay-toolchains-%(toolchains)s-1994.04.02.tar.xz')
540 self.testSdkUpload(to_tarballs=to_tarballs, to_upload_path=to_upload_path)
Gilad Arnoldad333182015-05-27 15:50:41 -0700541
542 def testToolchainTarballUpload(self):
543 """Make sure processing of toolchain tarballs works."""
Mike Frysinger9e979b92012-11-29 02:55:09 -0500544 tc_tarballs = (
545 'i686:/some/i686.tar.xz',
546 'arm-none:/some/arm.tar.xz',
547 )
548 tc_upload_path = '1994/04/%(target)s-1994.04.02.tar.xz'
Gilad Arnoldad333182015-05-27 15:50:41 -0700549 self.testSdkUpload(tc_tarballs=tc_tarballs, tc_upload_path=tc_upload_path)