blob: d3ee1cf81e8613c513575551f538f57d6670c189 [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',
23 'PATH': 'gtk%2B/foo.tgz'}]
24PRIVATE_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)
293 self.assertEqual(db['1'], 'http://www.example.com/gtk%2B/public1.tbz2')
294 self.assertEqual(db['2'], 'http://www.example.com/gtk%2B/foo.tgz')
295 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)
303 self.assertEqual(db['2'], 'http://www.example.com/gtk%2B/foo.tgz')
304 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:
339 pkg.setdefault('PATH', urllib.quote(pkg['CPV'] + '.tbz2'))
340 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
374if __name__ == '__main__':
375 unittest.main()