blob: 0c156b778c4a3fe6f260fd103f565a873ce4cb82 [file] [log] [blame]
David James8c846492011-01-25 17:07:29 -08001#!/usr/bin/python
2# Copyright (c) 2010 The Chromium OS Authors. All rights reserved.
3# Use of this source code is governed by a BSD-style license that can be
4# found in the LICENSE file.
5
6import copy
7import mox
8import os
David James8c846492011-01-25 17:07:29 -08009import shutil
Chris Sosa471532a2011-02-01 15:10:06 -080010import sys
David James8c846492011-01-25 17:07:29 -080011import tempfile
12import unittest
13import urllib
Chris Sosa471532a2011-02-01 15:10:06 -080014
15import constants
16sys.path.append(constants.SOURCE_ROOT)
17import prebuilt
David James8c846492011-01-25 17:07:29 -080018from chromite.lib import cros_build_lib
19from chromite.lib.binpkg import PackageIndex
20
21PUBLIC_PACKAGES = [{'CPV': 'gtk+/public1', 'SHA1': '1'},
22 {'CPV': 'gtk+/public2', 'SHA1': '2',
David James3cbba5e2011-02-14 14:53:15 -080023 'PATH': 'gtk+/foo.tgz'}]
David James8c846492011-01-25 17:07:29 -080024PRIVATE_PACKAGES = [{'CPV': 'private', 'SHA1': '3'}]
25
26
27def SimplePackageIndex(header=True, packages=True):
28 pkgindex = PackageIndex()
29 if header:
30 pkgindex.header['URI'] = 'http://www.example.com'
31 if packages:
32 pkgindex.packages = copy.deepcopy(PUBLIC_PACKAGES + PRIVATE_PACKAGES)
33 return pkgindex
34
35
36class TestUpdateFile(unittest.TestCase):
37
38 def setUp(self):
39 self.contents_str = ['# comment that should be skipped',
40 'PKGDIR="/var/lib/portage/pkgs"',
41 'PORTAGE_BINHOST="http://no.thanks.com"',
42 'portage portage-20100310.tar.bz2',
43 'COMPILE_FLAGS="some_value=some_other"',
44 ]
45 temp_fd, self.version_file = tempfile.mkstemp()
46 os.write(temp_fd, '\n'.join(self.contents_str))
47 os.close(temp_fd)
48
49 def tearDown(self):
50 os.remove(self.version_file)
51
52 def _read_version_file(self, version_file=None):
53 """Read the contents of self.version_file and return as a list."""
54 if not version_file:
55 version_file = self.version_file
56
57 version_fh = open(version_file)
58 try:
59 return [line.strip() for line in version_fh.readlines()]
60 finally:
61 version_fh.close()
62
63 def _verify_key_pair(self, key, val):
64 file_contents = self._read_version_file()
65 # ensure key for verify is wrapped on quotes
66 if '"' not in val:
67 val = '"%s"' % val
68 for entry in file_contents:
69 if '=' not in entry:
70 continue
71 file_key, file_val = entry.split('=')
72 if file_key == key:
73 if val == file_val:
74 break
75 else:
76 self.fail('Could not find "%s=%s" in version file' % (key, val))
77
78 def testAddVariableThatDoesNotExist(self):
79 """Add in a new variable that was no present in the file."""
80 key = 'PORTAGE_BINHOST'
81 value = '1234567'
82 prebuilt.UpdateLocalFile(self.version_file, value)
83 print self.version_file
84 current_version_str = self._read_version_file()
85 self._verify_key_pair(key, value)
86 print self.version_file
87
88 def testUpdateVariable(self):
89 """Test updating a variable that already exists."""
90 key, val = self.contents_str[2].split('=')
91 new_val = 'test_update'
92 self._verify_key_pair(key, val)
93 prebuilt.UpdateLocalFile(self.version_file, new_val)
94 self._verify_key_pair(key, new_val)
95
96 def testUpdateNonExistentFile(self):
97 key = 'PORTAGE_BINHOST'
98 value = '1234567'
99 non_existent_file = tempfile.mktemp()
100 try:
101 prebuilt.UpdateLocalFile(non_existent_file, value)
102 file_contents = self._read_version_file(non_existent_file)
103 self.assertEqual(file_contents, ['%s=%s' % (key, value)])
104 finally:
105 if os.path.exists(non_existent_file):
106 os.remove(non_existent_file)
107
108
109class TestPrebuiltFilters(unittest.TestCase):
110
111 def setUp(self):
112 self.tmp_dir = tempfile.mkdtemp()
113 self.private_dir = os.path.join(self.tmp_dir,
114 prebuilt._PRIVATE_OVERLAY_DIR)
115 self.private_structure_base = 'chromeos-overlay/chromeos-base'
116 self.private_pkgs = ['test-package/salt-flavor-0.1.r3.ebuild',
117 'easy/alpha_beta-0.1.41.r3.ebuild',
Chris Sosa471532a2011-02-01 15:10:06 -0800118 'dev/j-t-r-0.1.r3.ebuild', ]
David James8c846492011-01-25 17:07:29 -0800119 self.expected_filters = set(['salt-flavor', 'alpha_beta', 'j-t-r'])
120
121 def tearDown(self):
122 if self.tmp_dir:
123 shutil.rmtree(self.tmp_dir)
124
125 def _CreateNestedDir(self, tmp_dir, dir_structure):
126 for entry in dir_structure:
127 full_path = os.path.join(os.path.join(tmp_dir, entry))
128 # ensure dirs are created
129 try:
130 os.makedirs(os.path.dirname(full_path))
131 if full_path.endswith('/'):
132 # we only want to create directories
133 return
134 except OSError, err:
135 if err.errno == errno.EEXIST:
136 # we don't care if the dir already exists
137 pass
138 else:
139 raise
140 # create dummy files
141 tmp = open(full_path, 'w')
142 tmp.close()
143
144 def _LoadPrivateMockFilters(self):
145 """Load mock filters as defined in the setUp function."""
146 dir_structure = [os.path.join(self.private_structure_base, entry)
147 for entry in self.private_pkgs]
148
149 self._CreateNestedDir(self.private_dir, dir_structure)
150 prebuilt.LoadPrivateFilters(self.tmp_dir)
151
152 def testFilterPattern(self):
153 """Check that particular packages are filtered properly."""
154 self._LoadPrivateMockFilters()
155 packages = ['/some/dir/area/j-t-r-0.1.r3.tbz',
156 '/var/pkgs/new/alpha_beta-0.2.3.4.tbz',
157 '/usr/local/cache/good-0.1.3.tbz',
158 '/usr-blah/b_d/salt-flavor-0.0.3.tbz']
159 expected_list = ['/usr/local/cache/good-0.1.3.tbz']
160 filtered_list = [file for file in packages if not
161 prebuilt.ShouldFilterPackage(file)]
162 self.assertEqual(expected_list, filtered_list)
163
164 def testLoadPrivateFilters(self):
165 self._LoadPrivateMockFilters()
166 prebuilt.LoadPrivateFilters(self.tmp_dir)
167 self.assertEqual(self.expected_filters, prebuilt._FILTER_PACKAGES)
168
169 def testEmptyFiltersErrors(self):
170 """Ensure LoadPrivateFilters errors if an empty list is generated."""
171 os.makedirs(os.path.join(self.tmp_dir, prebuilt._PRIVATE_OVERLAY_DIR))
172 self.assertRaises(prebuilt.FiltersEmpty, prebuilt.LoadPrivateFilters,
173 self.tmp_dir)
174
175
176class TestPrebuilt(unittest.TestCase):
177
178 def setUp(self):
179 self.mox = mox.Mox()
180
181 def tearDown(self):
182 self.mox.UnsetStubs()
183 self.mox.VerifyAll()
184
185 def testGenerateUploadDict(self):
186 base_local_path = '/b/cbuild/build/chroot/build/x86-dogfood/'
187 gs_bucket_path = 'gs://chromeos-prebuilt/host/version'
188 local_path = os.path.join(base_local_path, 'public1.tbz2')
189 self.mox.StubOutWithMock(prebuilt.os.path, 'exists')
190 prebuilt.os.path.exists(local_path).AndReturn(True)
191 self.mox.ReplayAll()
192 pkgs = [{ 'CPV': 'public1' }]
193 result = prebuilt.GenerateUploadDict(base_local_path, gs_bucket_path, pkgs)
194 expected = { local_path: gs_bucket_path + '/public1.tbz2' }
195 self.assertEqual(result, expected)
196
197 def testFailonUploadFail(self):
198 """Make sure we fail if one of the upload processes fail."""
199 files = {'test': '/uasd'}
200 self.assertEqual(prebuilt.RemoteUpload(files), set([('test', '/uasd')]))
201
202 def testDeterminePrebuiltConfHost(self):
203 """Test that the host prebuilt path comes back properly."""
204 expected_path = os.path.join(prebuilt._PREBUILT_MAKE_CONF['amd64'])
205 self.assertEqual(prebuilt.DeterminePrebuiltConfFile('fake_path', 'amd64'),
206 expected_path)
207
208 def testDeterminePrebuiltConf(self):
209 """Test the different known variants of boards for proper path discovery."""
210 fake_path = '/b/cbuild'
Chris Sosa471532a2011-02-01 15:10:06 -0800211 script_path = os.path.join(fake_path, 'src/platform/dev/host')
David James8c846492011-01-25 17:07:29 -0800212 public_overlay_path = os.path.join(fake_path, 'src/overlays')
213 private_overlay_path = os.path.join(fake_path,
214 prebuilt._PRIVATE_OVERLAY_DIR)
215 path_dict = {'private_overlay_path': private_overlay_path,
216 'public_overlay_path': public_overlay_path}
217 # format for targets
218 # board target key in dictionar
219 # Tuple containing cmd run, expected results as cmd obj, and expected output
220
221 # Mock output from cros_overlay_list
222 x86_out = ('%(private_overlay_path)s/chromeos-overlay\n'
223 '%(public_overlay_path)s/overlay-x86-generic\n' % path_dict)
224
225 x86_cmd = './cros_overlay_list --board x86-generic'
226 x86_expected_path = os.path.join(public_overlay_path, 'overlay-x86-generic',
227 'prebuilt.conf')
228 # Mock output from cros_overlay_list
229 tegra2_out = ('%(private_overlay_path)s/chromeos-overlay\n'
230 '%(public_overlay_path)s/overlay-tegra2\n'
231 '%(public_overlay_path)s/overlay-variant-tegra2-seaboard\n'
232 '%(private_overlay_path)s/overlay-tegra2-private\n'
233 '%(private_overlay_path)s/'
234 'overlay-variant-tegra2-seaboard-private\n' % path_dict)
235 tegra2_cmd = './cros_overlay_list --board tegra2 --variant seaboard'
236 tegra2_expected_path = os.path.join(
237 private_overlay_path, 'overlay-variant-tegra2-seaboard-private',
238 'prebuilt.conf')
239
240
241 targets = {'x86-generic': {'cmd': x86_cmd,
242 'output': x86_out,
243 'result': x86_expected_path},
244 'tegra2_seaboard': {'cmd': tegra2_cmd,
245 'output': tegra2_out,
246 'result': tegra2_expected_path}
247 }
248
249 self.mox.StubOutWithMock(prebuilt.cros_build_lib, 'RunCommand')
250 for target, expected_results in targets.iteritems():
251 # create command object for output
252 cmd_result_obj = cros_build_lib.CommandResult()
253 cmd_result_obj.output = expected_results['output']
254 prebuilt.cros_build_lib.RunCommand(
255 expected_results['cmd'].split(), redirect_stdout=True,
256 cwd=script_path).AndReturn(cmd_result_obj)
Chris Sosa471532a2011-02-01 15:10:06 -0800257
David James8c846492011-01-25 17:07:29 -0800258 self.mox.ReplayAll()
259 for target, expected_results in targets.iteritems():
260 self.assertEqual(
261 prebuilt.DeterminePrebuiltConfFile(fake_path, target),
262 expected_results['result'])
263
264 def testDeterminePrebuiltConfGarbage(self):
265 """Ensure an exception is raised on bad input."""
266 self.assertRaises(prebuilt.UnknownBoardFormat,
267 prebuilt.DeterminePrebuiltConfFile,
268 'fake_path', 'asdfasdf')
269
270
271class TestPackagesFileFiltering(unittest.TestCase):
272
273 def testFilterPkgIndex(self):
274 pkgindex = SimplePackageIndex()
275 pkgindex.RemoveFilteredPackages(lambda pkg: pkg in PRIVATE_PACKAGES)
276 self.assertEqual(pkgindex.packages, PUBLIC_PACKAGES)
277 self.assertEqual(pkgindex.modified, True)
278
279
280class TestPopulateDuplicateDB(unittest.TestCase):
281
282 def testEmptyIndex(self):
283 pkgindex = SimplePackageIndex(packages=False)
284 db = {}
285 pkgindex._PopulateDuplicateDB(db)
286 self.assertEqual(db, {})
287
288 def testNormalIndex(self):
289 pkgindex = SimplePackageIndex()
290 db = {}
291 pkgindex._PopulateDuplicateDB(db)
292 self.assertEqual(len(db), 3)
David James3cbba5e2011-02-14 14:53:15 -0800293 self.assertEqual(db['1'], 'http://www.example.com/gtk+/public1.tbz2')
294 self.assertEqual(db['2'], 'http://www.example.com/gtk+/foo.tgz')
David James8c846492011-01-25 17:07:29 -0800295 self.assertEqual(db['3'], 'http://www.example.com/private.tbz2')
296
297 def testMissingSHA1(self):
298 db = {}
299 pkgindex = SimplePackageIndex()
300 del pkgindex.packages[0]['SHA1']
301 pkgindex._PopulateDuplicateDB(db)
302 self.assertEqual(len(db), 2)
David James3cbba5e2011-02-14 14:53:15 -0800303 self.assertEqual(db['2'], 'http://www.example.com/gtk+/foo.tgz')
David James8c846492011-01-25 17:07:29 -0800304 self.assertEqual(db['3'], 'http://www.example.com/private.tbz2')
305
306 def testFailedPopulate(self):
307 db = {}
308 pkgindex = SimplePackageIndex(header=False)
309 self.assertRaises(KeyError, pkgindex._PopulateDuplicateDB, db)
310 pkgindex = SimplePackageIndex()
311 del pkgindex.packages[0]['CPV']
312 self.assertRaises(KeyError, pkgindex._PopulateDuplicateDB, db)
313
314
315class TestResolveDuplicateUploads(unittest.TestCase):
316
317 def testEmptyList(self):
318 pkgindex = SimplePackageIndex()
319 pristine = SimplePackageIndex()
320 uploads = pkgindex.ResolveDuplicateUploads([])
321 self.assertEqual(uploads, pristine.packages)
322 self.assertEqual(pkgindex.packages, pristine.packages)
323 self.assertEqual(pkgindex.modified, False)
324
325 def testEmptyIndex(self):
326 pkgindex = SimplePackageIndex()
327 pristine = SimplePackageIndex()
328 empty = SimplePackageIndex(packages=False)
329 uploads = pkgindex.ResolveDuplicateUploads([empty])
330 self.assertEqual(uploads, pristine.packages)
331 self.assertEqual(pkgindex.packages, pristine.packages)
332 self.assertEqual(pkgindex.modified, False)
333
334 def testDuplicates(self):
335 pkgindex = SimplePackageIndex()
336 dup_pkgindex = SimplePackageIndex()
337 expected_pkgindex = SimplePackageIndex()
338 for pkg in expected_pkgindex.packages:
David James3cbba5e2011-02-14 14:53:15 -0800339 pkg.setdefault('PATH', pkg['CPV'] + '.tbz2')
David James8c846492011-01-25 17:07:29 -0800340 uploads = pkgindex.ResolveDuplicateUploads([dup_pkgindex])
341 self.assertEqual(pkgindex.packages, expected_pkgindex.packages)
342
343 def testMissingSHA1(self):
344 db = {}
345 pkgindex = SimplePackageIndex()
346 dup_pkgindex = SimplePackageIndex()
347 expected_pkgindex = SimplePackageIndex()
348 del pkgindex.packages[0]['SHA1']
349 del expected_pkgindex.packages[0]['SHA1']
350 for pkg in expected_pkgindex.packages[1:]:
351 pkg.setdefault('PATH', pkg['CPV'] + '.tbz2')
352 uploads = pkgindex.ResolveDuplicateUploads([dup_pkgindex])
353 self.assertEqual(pkgindex.packages, expected_pkgindex.packages)
354
355
356class TestWritePackageIndex(unittest.TestCase):
357
358 def setUp(self):
359 self.mox = mox.Mox()
360
361 def tearDown(self):
362 self.mox.UnsetStubs()
363 self.mox.VerifyAll()
364
365 def testSimple(self):
366 pkgindex = SimplePackageIndex()
367 self.mox.StubOutWithMock(pkgindex, 'Write')
368 pkgindex.Write(mox.IgnoreArg())
369 self.mox.ReplayAll()
370 f = pkgindex.WriteToNamedTemporaryFile()
371 self.assertEqual(f.read(), '')
372
373
David James05bcb2b2011-02-09 09:25:47 -0800374class TestUploadPrebuilt(unittest.TestCase):
375
376 def setUp(self):
377 class MockTemporaryFile(object):
378 def __init__(self, name):
379 self.name = name
380 self.mox = mox.Mox()
381 self.pkgindex = SimplePackageIndex()
382 self.mox.StubOutWithMock(prebuilt, 'GrabLocalPackageIndex')
383 prebuilt.GrabLocalPackageIndex('/packages').AndReturn(self.pkgindex)
384 self.mox.StubOutWithMock(prebuilt, 'RemoteUpload')
385 self.mox.StubOutWithMock(self.pkgindex, 'ResolveDuplicateUploads')
386 self.pkgindex.ResolveDuplicateUploads([]).AndReturn(PRIVATE_PACKAGES)
387 self.mox.StubOutWithMock(self.pkgindex, 'WriteToNamedTemporaryFile')
388 fake_pkgs_file = MockTemporaryFile('fake')
389 self.pkgindex.WriteToNamedTemporaryFile().AndReturn(fake_pkgs_file)
390
391 def tearDown(self):
392 self.mox.UnsetStubs()
393 self.mox.VerifyAll()
394
395 def doRsyncUpload(self, suffix):
396 self.mox.StubOutWithMock(prebuilt, '_RetryRun')
397 remote_path = '/dir/%s' % suffix.rstrip('/')
398 full_remote_path = 'chromeos-prebuilt:%s' % remote_path
399 cmds = ['ssh chromeos-prebuilt mkdir -p %s' % remote_path,
400 'rsync -av --chmod=a+r fake %s/Packages' % full_remote_path,
401 'rsync -Rav private.tbz2 %s/' % full_remote_path]
402 for cmd in cmds:
403 prebuilt._RetryRun(cmd, shell=True, cwd='/packages').AndReturn(True)
404 self.mox.ReplayAll()
405 prebuilt.UploadPrebuilt('/packages', 'chromeos-prebuilt:/dir', suffix,
406 self.pkgindex.header['URI'], [])
407
408 def testSuccessfulGsUpload(self):
409 uploads = {'/packages/private.tbz2': 'gs://foo/private.tbz2'}
410 self.mox.StubOutWithMock(prebuilt, 'GenerateUploadDict')
411 prebuilt.GenerateUploadDict('/packages', 'gs://foo/suffix',
412 PRIVATE_PACKAGES).AndReturn(uploads)
413 uploads = uploads.copy()
414 uploads['fake'] = 'gs://foo/suffix/Packages'
415 prebuilt.RemoteUpload(uploads).AndReturn([None])
416 self.mox.ReplayAll()
417 prebuilt.UploadPrebuilt('/packages', 'gs://foo', 'suffix',
418 self.pkgindex.header['URI'], [])
419
420 def testSuccessfulRsyncUploadWithNoTrailingSlash(self):
421 self.doRsyncUpload('suffix')
422
423 def testSuccessfulRsyncUploadWithTrailingSlash(self):
424 self.doRsyncUpload('suffix/')
425
426
427class TestSyncPrebuilts(unittest.TestCase):
428
429 def setUp(self):
430 self.mox = mox.Mox()
431 self.mox.StubOutWithMock(prebuilt, '_GetCrossCompilerSubdirectories')
432 self.mox.StubOutWithMock(prebuilt, 'DeterminePrebuiltConfFile')
433 self.mox.StubOutWithMock(prebuilt, 'UploadPrebuilt')
434 self.mox.StubOutWithMock(prebuilt, 'RevGitFile')
435 self.mox.StubOutWithMock(prebuilt, 'UpdateBinhostConfFile')
436 self.build_path = '/trunk'
437 self.upload_location = 'gs://upload/'
438 self.version = '1'
439 self.binhost = 'http://prebuilt/'
440 self.key = 'PORTAGE_BINHOST'
441
442 def tearDown(self):
443 self.mox.UnsetStubs()
444 self.mox.VerifyAll()
445
446 def testSyncHostPrebuilts(self):
447 package_path = os.path.join(self.build_path,
448 prebuilt._HOST_PACKAGES_PATH)
449 url_suffix = prebuilt._REL_HOST_PATH % {'version': self.version,
450 'target': prebuilt._HOST_TARGET }
451 subdir = 'cross/x86'
452 prebuilt._GetCrossCompilerSubdirectories(self.build_path).AndReturn(
453 [subdir])
454 cross_url_suffix = '%s/%s' % (url_suffix.rstrip('/'), subdir)
455 prebuilt.UploadPrebuilt(os.path.join(package_path, subdir),
456 self.upload_location, cross_url_suffix, self.binhost, [])
457 prebuilt.UploadPrebuilt(package_path, self.upload_location,
458 url_suffix, self.binhost, [])
459 url_value = '%s/%s/' % (self.binhost.rstrip('/'), url_suffix.rstrip('/'))
460 prebuilt.RevGitFile(mox.IgnoreArg(), url_value, key=self.key)
461 prebuilt.UpdateBinhostConfFile(mox.IgnoreArg(), self.key, url_value)
462 self.mox.ReplayAll()
463 prebuilt._SyncHostPrebuilts(self.build_path, self.upload_location,
464 self.version, self.binhost, [], self.key, True, True)
465
466 def testSyncBoardPrebuilts(self):
467 board = 'x86-generic'
468 board_path = os.path.join(self.build_path,
469 prebuilt._BOARD_PATH % {'board': board})
470 package_path = os.path.join(board_path, 'packages')
471 url_suffix = prebuilt._REL_BOARD_PATH % {'version': self.version,
472 'board': board }
473 prebuilt.UploadPrebuilt(package_path, self.upload_location,
474 url_suffix, self.binhost, [])
475 url_value = '%s/%s/' % (self.binhost.rstrip('/'), url_suffix.rstrip('/'))
476 prebuilt.DeterminePrebuiltConfFile(self.build_path, board).AndReturn('foo')
477 prebuilt.RevGitFile('foo', url_value, key=self.key)
478 prebuilt.UpdateBinhostConfFile(mox.IgnoreArg(), self.key, url_value)
479 self.mox.ReplayAll()
480 prebuilt._SyncBoardPrebuilts(self.build_path, self.upload_location,
481 self.version, self.binhost, [], board, self.key, True, True)
482
483
David James8c846492011-01-25 17:07:29 -0800484if __name__ == '__main__':
485 unittest.main()