blob: 297aee10cff0b9282547a0bae8315a024124c7a7 [file] [log] [blame]
Frank Farzan37761d12011-12-01 14:29:08 -08001#!/usr/bin/python
2#
3# Copyright (c) 2011 The Chromium OS Authors. All rights reserved.
4# Use of this source code is governed by a BSD-style license that can be
5# found in the LICENSE file.
6
7"""Unit tests for devserver_util module."""
8
Chris Sosaea148d92012-03-06 16:22:04 -08009import mox
Frank Farzan37761d12011-12-01 14:29:08 -080010import os
11import shutil
Chris Sosaea148d92012-03-06 16:22:04 -080012import subprocess
Frank Farzan37761d12011-12-01 14:29:08 -080013import tempfile
14import unittest
15
16import devserver_util
17
18
19# Fake Dev Server Layout:
20TEST_LAYOUT = {
21 'test-board-1': ['R17-1413.0.0-a1-b1346', 'R17-18.0.0-a1-b1346'],
22 'test-board-2': ['R16-2241.0.0-a0-b2', 'R17-2.0.0-a1-b1346']
23}
24
25
Chris Sosaea148d92012-03-06 16:22:04 -080026class DevServerUtilTest(mox.MoxTestBase):
Frank Farzan37761d12011-12-01 14:29:08 -080027
28 def setUp(self):
Chris Sosaea148d92012-03-06 16:22:04 -080029 mox.MoxTestBase.setUp(self)
Frank Farzan37761d12011-12-01 14:29:08 -080030 self._static_dir = tempfile.mkdtemp()
31 self._outside_sandbox_dir = tempfile.mkdtemp()
32
33 for board, builds in TEST_LAYOUT.iteritems():
34 board_path = os.path.join(self._static_dir, board)
35 os.mkdir(board_path)
36 for build in builds:
37 build_path = os.path.join(board_path, build)
38 os.mkdir(build_path)
39 with open(os.path.join(build_path,
40 devserver_util.TEST_IMAGE), 'w') as f:
41 f.write('TEST_IMAGE')
42 with open(os.path.join(build_path,
43 devserver_util.STATEFUL_UPDATE), 'w') as f:
44 f.write('STATEFUL_UPDATE')
45 with open(os.path.join(build_path,
46 devserver_util.ROOT_UPDATE), 'w') as f:
47 f.write('ROOT_UPDATE')
48 # AU payloads.
49 au_dir = os.path.join(build_path, devserver_util.AU_BASE)
50 nton_dir = os.path.join(au_dir, build + devserver_util.NTON_DIR_SUFFIX)
51 os.makedirs(nton_dir)
52 with open(os.path.join(nton_dir,
53 devserver_util.ROOT_UPDATE), 'w') as f:
54 f.write('ROOT_UPDATE')
55 mton_dir = os.path.join(au_dir, build + devserver_util.MTON_DIR_SUFFIX)
56 os.makedirs(mton_dir)
57 with open(os.path.join(mton_dir,
58 devserver_util.ROOT_UPDATE), 'w') as f:
59 f.write('ROOT_UPDATE')
60
Chris Sosaea148d92012-03-06 16:22:04 -080061 self._good_mock_process = self.mox.CreateMock(subprocess.Popen)
62 self._good_mock_process.returncode = 0
63 self._bad_mock_process = self.mox.CreateMock(subprocess.Popen)
64 self._bad_mock_process.returncode = 1
65
Frank Farzan37761d12011-12-01 14:29:08 -080066 def tearDown(self):
67 shutil.rmtree(self._static_dir)
68 shutil.rmtree(self._outside_sandbox_dir)
69
70 def testParsePayloadList(self):
71 archive_url_prefix = ('gs://chromeos-image-archive/x86-mario-release/'
72 'R17-1413.0.0-a1-b1346/')
73 mton_url = (archive_url_prefix + 'chromeos_R17-1412.0.0-a1-b1345_'
74 'R17-1413.0.0-a1_x86-mario_delta_dev.bin')
75 nton_url = (archive_url_prefix + 'chromeos_R17-1413.0.0-a1_'
76 'R17-1413.0.0-a1_x86-mario_delta_dev.bin')
77 full_url = (archive_url_prefix + 'chromeos_R17-1413.0.0-a1_'
78 'x86-mario_full_dev.bin')
79 full_url_out, nton_url_out, mton_url_out = (
80 devserver_util.ParsePayloadList([full_url, nton_url, mton_url]))
81 self.assertEqual([full_url, nton_url, mton_url],
82 [full_url_out, nton_url_out, mton_url_out])
83
84 archive_url_prefix = ('gs://chromeos-image-archive/x86-alex_he-release/'
85 'R18-1420.0.0-a1-b541')
86 mton_url = (archive_url_prefix + 'chromeos_R18-1418.0.0-a1-b540_'
87 'R18-1420.0.0-a1_x86-alex_he_delta_dev.bin')
88 nton_url = (archive_url_prefix + 'chromeos_R18-1420.0.0-a1_'
89 'R18-1420.0.0-a1_x86-alex_he_delta_dev.bin')
90 full_url = (archive_url_prefix + 'chromeos_R18-1420.0.0-a1_'
91 'x86-alex_he_full_dev.bin')
92 full_url_out, nton_url_out, mton_url_out = (
93 devserver_util.ParsePayloadList([full_url, nton_url, mton_url]))
94 self.assertEqual([full_url, nton_url, mton_url],
95 [full_url_out, nton_url_out, mton_url_out])
96
Chris Sosaea148d92012-03-06 16:22:04 -080097 def _CallRunGS(self, output, str_should_contain, attempts=1):
98 """Helper that wraps a RunGS for tests."""
99 for attempt in range(attempts):
100 if attempt == devserver_util.GSUTIL_ATTEMPTS:
101 # We can't mock more than we can attempt.
102 return
103
104 # Return 1's for all but last attempt.
105 if attempt != attempts - 1:
106 mock_process = self._bad_mock_process
107 else:
108 mock_process = self._good_mock_process
109
110 subprocess.Popen(mox.StrContains(str_should_contain), shell=True,
111 stdout=subprocess.PIPE).AndReturn(mock_process)
112 mock_process.communicate().AndReturn((output, None))
113
Frank Farzan37761d12011-12-01 14:29:08 -0800114 def testDownloadBuildFromGS(self):
Chris Sosaea148d92012-03-06 16:22:04 -0800115 """Tests that we can run download build from gs with one error."""
116 build = 'R17-1413.0.0-a1-b1346'
117 archive_url_prefix = ('gs://chromeos-image-archive/x86-mario-release/' +
118 build)
119 mock_data = 'mock data\nmock_data\nmock_data'
120 payloads = ['p1', 'p2', 'p3']
121 self.mox.StubOutWithMock(subprocess, 'Popen', use_mock_anything=True)
122 self.mox.StubOutWithMock(devserver_util, 'ParsePayloadList')
123
124 # Make sure we our retry works.
125 self._CallRunGS(mock_data, archive_url_prefix, attempts=2)
126 devserver_util.ParsePayloadList(mock_data.splitlines()).AndReturn(
127 payloads)
128
129 for payload in payloads + [devserver_util.STATEFUL_UPDATE,
130 devserver_util.AUTOTEST_PACKAGE]:
131 self._CallRunGS(mock_data, payload,)
132
133 self.mox.ReplayAll()
134 devserver_util.DownloadBuildFromGS(self._static_dir, archive_url_prefix,
135 build)
136 self.mox.VerifyAll()
137
138 def testDownloadBuildFromGSButGSDown(self):
139 """Tests that we fail correctly if we can't reach GS."""
140 build = 'R17-1413.0.0-a1-b1346'
141 archive_url_prefix = ('gs://chromeos-image-archive/x86-mario-release/' +
142 build)
143 mock_data = 'mock data\nmock_data\nmock_data'
144 self.mox.StubOutWithMock(subprocess, 'Popen', use_mock_anything=True)
145
146 self._CallRunGS(mock_data, archive_url_prefix,
147 attempts=devserver_util.GSUTIL_ATTEMPTS + 1)
148
149 self.mox.ReplayAll()
150 self.assertRaises(
151 devserver_util.DevServerUtilError,
152 devserver_util.DownloadBuildFromGS,
153 self._static_dir, archive_url_prefix, build)
154 self.mox.VerifyAll()
Frank Farzan37761d12011-12-01 14:29:08 -0800155
156 def testInstallBuild(self):
157 self.fail('Not implemented.')
158
159 def testPrepareAutotestPkgs(self):
160 self.fail('Not implemented.')
161
162 def testSafeSandboxAccess(self):
163 # Path is in sandbox.
164 self.assertTrue(
165 devserver_util.SafeSandboxAccess(
166 self._static_dir, os.path.join(self._static_dir, 'some-board')))
167
168 # Path is sandbox.
169 self.assertFalse(
170 devserver_util.SafeSandboxAccess(self._static_dir, self._static_dir))
171
172 # Path is outside the sandbox.
173 self.assertFalse(
174 devserver_util.SafeSandboxAccess(self._static_dir,
175 self._outside_sandbox_dir))
176
177 # Path contains '..'.
178 self.assertFalse(
179 devserver_util.SafeSandboxAccess(
180 self._static_dir, os.path.join(self._static_dir, os.pardir)))
181
182 # Path contains symbolic link references.
183 os.chdir(self._static_dir)
184 os.symlink(os.pardir, 'parent')
185 self.assertFalse(
186 devserver_util.SafeSandboxAccess(
187 self._static_dir, os.path.join(self._static_dir, os.pardir)))
188
189 def testAcquireReleaseLocks(self):
190 # Successful lock and unlock.
191 lock_file = devserver_util.AcquireLock(self._static_dir, 'test-lock')
192 self.assertTrue(os.path.exists(lock_file))
193 devserver_util.ReleaseLock(self._static_dir, 'test-lock')
194 self.assertFalse(os.path.exists(lock_file))
195
196 # Attempt to lock an existing directory.
197 devserver_util.AcquireLock(self._static_dir, 'test-lock')
198 self.assertRaises(devserver_util.DevServerUtilError,
199 devserver_util.AcquireLock, self._static_dir, 'test-lock')
200
201 def testFindMatchingBoards(self):
202 for key in TEST_LAYOUT:
203 # Partial match with multiple boards.
204 self.assertEqual(
205 set(devserver_util.FindMatchingBoards(self._static_dir, key[:-5])),
206 set(TEST_LAYOUT.keys()))
207
208 # Absolute match.
209 self.assertEqual(
210 devserver_util.FindMatchingBoards(self._static_dir, key), [key])
211
212 # Invalid partial match.
213 self.assertEqual(
214 devserver_util.FindMatchingBoards(self._static_dir, 'asdfsadf'), [])
215
216 def testFindMatchingBuilds(self):
217 # Try a partial board and build match with single match.
218 self.assertEqual(
219 devserver_util.FindMatchingBuilds(self._static_dir, 'test-board',
220 'R17-1413'),
221 [('test-board-1', 'R17-1413.0.0-a1-b1346')])
222
223 # Try a partial board and build match with multiple match.
224 actual = set(devserver_util.FindMatchingBuilds(
225 self._static_dir, 'test-board', 'R17'))
226 expected = set([('test-board-1', 'R17-1413.0.0-a1-b1346'),
227 ('test-board-1', 'R17-18.0.0-a1-b1346'),
228 ('test-board-2', 'R17-2.0.0-a1-b1346')])
229 self.assertEqual(actual, expected)
230
231 def testGetLatestBuildVersion(self):
232 self.assertEqual(
233 devserver_util.GetLatestBuildVersion(self._static_dir, 'test-board-1'),
234 'R17-1413.0.0-a1-b1346')
235
236 def testFindBuild(self):
237 # Ensure no matching boards raises exception for latest.
238 self.assertRaises(
239 devserver_util.DevServerUtilError, devserver_util.FindBuild,
240 self._static_dir, 'aasdf', 'latest')
241
242 # Ensure no matching builds or boards raises an exception.
243 self.assertRaises(
244 devserver_util.DevServerUtilError, devserver_util.FindBuild,
245 self._static_dir, 'aasdf', 'asdgsadg')
246
247 # Ensure latest returns the proper build.
248 self.assertEqual(
249 devserver_util.FindBuild(self._static_dir, 'test-board-1', 'latest'),
250 ('test-board-1', 'R17-1413.0.0-a1-b1346'))
251
252 # Ensure specific board, build is returned.
253 self.assertEqual(
254 devserver_util.FindBuild(self._static_dir, 'test-board-1',
255 'R17-18.0.0-a1-b1346'),
256 ('test-board-1', 'R17-18.0.0-a1-b1346'))
257
258 # Ensure too many matches raises an exception.
259 self.assertRaises(
260 devserver_util.DevServerUtilError, devserver_util.FindBuild,
261 self._static_dir, 'test-board', 'R17')
262
263 def testCloneBuild(self):
264 test_prefix = 'abc'
265 test_tag = test_prefix + '/123'
266 abc_path = os.path.join(self._static_dir, devserver_util.DEV_BUILD_PREFIX,
267 test_tag)
268
269 os.mkdir(os.path.join(self._static_dir, test_prefix))
270
271 # Verify leaf path is created and proper values returned.
272 board, builds = TEST_LAYOUT.items()[0]
273 dev_build = devserver_util.CloneBuild(self._static_dir, board, builds[0],
274 test_tag)
275 self.assertEquals(dev_build, abc_path)
276 self.assertTrue(os.path.exists(abc_path))
277 self.assertTrue(os.path.isfile(os.path.join(
278 abc_path, devserver_util.TEST_IMAGE)))
279 self.assertTrue(os.path.isfile(os.path.join(
280 abc_path, devserver_util.ROOT_UPDATE)))
281 self.assertTrue(os.path.isfile(os.path.join(
282 abc_path, devserver_util.STATEFUL_UPDATE)))
283
284 # Verify force properly removes the old directory.
285 junk_path = os.path.join(dev_build, 'junk')
286 with open(junk_path, 'w') as f:
287 f.write('hello!')
288 remote_dir = devserver_util.CloneBuild(
289 self._static_dir, board, builds[0], test_tag, force=True)
290 self.assertEquals(remote_dir, abc_path)
291 self.assertTrue(os.path.exists(abc_path))
292 self.assertTrue(os.path.isfile(os.path.join(
293 abc_path, devserver_util.TEST_IMAGE)))
294 self.assertTrue(os.path.isfile(os.path.join(
295 abc_path, devserver_util.ROOT_UPDATE)))
296 self.assertTrue(os.path.isfile(os.path.join(
297 abc_path, devserver_util.STATEFUL_UPDATE)))
298 self.assertFalse(os.path.exists(junk_path))
299
300 def testGetControlFile(self):
301 control_file_dir = os.path.join(
Chris Sosaea148d92012-03-06 16:22:04 -0800302 self._static_dir, 'test-board-1', 'R17-1413.0.0-a1-b1346', 'autotest',
303 'server', 'site_tests', 'network_VPN')
Frank Farzan37761d12011-12-01 14:29:08 -0800304 os.makedirs(control_file_dir)
305 with open(os.path.join(control_file_dir, 'control'), 'w') as f:
306 f.write('hello!')
307
308 control_content = devserver_util.GetControlFile(
Chris Sosaea148d92012-03-06 16:22:04 -0800309 self._static_dir, 'test-board-1/R17-1413.0.0-a1-b1346',
Frank Farzan37761d12011-12-01 14:29:08 -0800310 os.path.join('server', 'site_tests', 'network_VPN', 'control'))
311 self.assertEqual(control_content, 'hello!')
312
313 def testListAutoupdateTargets(self):
314 for board, builds in TEST_LAYOUT.iteritems():
315 for build in builds:
316 au_targets = devserver_util.ListAutoupdateTargets(self._static_dir,
317 board, build)
318 self.assertEqual(set(au_targets),
319 set([build + devserver_util.NTON_DIR_SUFFIX,
320 build + devserver_util.MTON_DIR_SUFFIX]))
321
322
323if __name__ == '__main__':
324 unittest.main()