blob: 2b4d9c3d93851f9b98d650631792d1c9e703a808 [file] [log] [blame]
Alex Klein2966e302019-01-17 13:29:38 -07001# -*- coding: utf-8 -*-
2# Copyright 2019 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
6"""Image service tests."""
7
8from __future__ import print_function
9
Alex Klein56355682019-02-07 10:36:54 -070010import mock
Alex Klein2966e302019-01-17 13:29:38 -070011import os
12
Alex Klein8cb365a2019-05-15 16:24:53 -060013from chromite.api import controller
14from chromite.api.controller import image as image_controller
Alex Klein7107bdd2019-03-14 17:14:31 -060015from chromite.api.gen.chromite.api import image_pb2
David Burger13e06be2019-05-13 20:33:16 -060016from chromite.api.gen.chromiumos import common_pb2
Alex Klein56355682019-02-07 10:36:54 -070017from chromite.lib import constants
Alex Klein4f0eb432019-05-02 13:56:04 -060018from chromite.lib import cros_build_lib
Alex Klein2966e302019-01-17 13:29:38 -070019from chromite.lib import cros_test_lib
Michael Mortensenc83c9952019-08-05 12:15:12 -060020from chromite.lib import image_lib
Alex Klein2966e302019-01-17 13:29:38 -070021from chromite.lib import osutils
Alex Kleinb7cdbe62019-02-22 11:41:32 -070022from chromite.service import image as image_service
Alex Klein2966e302019-01-17 13:29:38 -070023
24
Alex Klein1bcd9882019-03-19 13:25:24 -060025class CreateTest(cros_test_lib.MockTempDirTestCase):
Alex Klein56355682019-02-07 10:36:54 -070026 """Create image tests."""
27
Alex Klein21b95022019-05-09 14:14:46 -060028 def _GetRequest(self, board=None, types=None, version=None, builder_path=None,
29 disable_rootfs_verification=False):
30 """Helper to build a request instance."""
31 return image_pb2.CreateImageRequest(
32 build_target={'name': board},
33 image_types=types,
34 disable_rootfs_verification=disable_rootfs_verification,
35 version=version,
36 builder_path=builder_path,
37 )
38
39 def _GetResponse(self):
40 """Helper to build an empty response instance."""
41 return image_pb2.CreateImageResult()
42
Alex Klein56355682019-02-07 10:36:54 -070043 def testArgumentValidation(self):
44 """Test the argument validation."""
45 input_proto = image_pb2.CreateImageRequest()
46 output_proto = image_pb2.CreateImageResult()
47
48 # No board should cause it to fail.
Alex Klein4f0eb432019-05-02 13:56:04 -060049 with self.assertRaises(cros_build_lib.DieSystemExit):
Alex Kleinb7cdbe62019-02-22 11:41:32 -070050 image_controller.Create(input_proto, output_proto)
Alex Klein56355682019-02-07 10:36:54 -070051
Alex Klein21b95022019-05-09 14:14:46 -060052 def testNoTypeSpecified(self):
53 """Test the image type default."""
54 request = self._GetRequest(board='board')
55 response = self._GetResponse()
56
Alex Klein1bcd9882019-03-19 13:25:24 -060057 # Failed result to avoid the success handling logic.
58 result = image_service.BuildResult(1, [])
59 build_patch = self.PatchObject(image_service, 'Build', return_value=result)
Alex Klein56355682019-02-07 10:36:54 -070060
Alex Klein21b95022019-05-09 14:14:46 -060061 image_controller.Create(request, response)
Alex Klein56355682019-02-07 10:36:54 -070062 build_patch.assert_called_with(images=[constants.IMAGE_TYPE_BASE],
Alex Klein21b95022019-05-09 14:14:46 -060063 board='board', config=mock.ANY)
Alex Klein56355682019-02-07 10:36:54 -070064
Alex Klein21b95022019-05-09 14:14:46 -060065 def testSingleTypeSpecified(self):
66 """Test it's properly using a specified type."""
67 request = self._GetRequest(board='board', types=[common_pb2.DEV])
68 response = self._GetResponse()
69
70 # Failed result to avoid the success handling logic.
71 result = image_service.BuildResult(1, [])
72 build_patch = self.PatchObject(image_service, 'Build', return_value=result)
73
74 image_controller.Create(request, response)
Alex Klein56355682019-02-07 10:36:54 -070075 build_patch.assert_called_with(images=[constants.IMAGE_TYPE_DEV],
Alex Klein21b95022019-05-09 14:14:46 -060076 board='board', config=mock.ANY)
Alex Klein56355682019-02-07 10:36:54 -070077
Alex Klein21b95022019-05-09 14:14:46 -060078 def testMultipleAndImpliedTypes(self):
79 """Test multiple types and implied type handling."""
80 # The TEST_VM type should force it to build the test image.
81 types = [common_pb2.BASE, common_pb2.TEST_VM]
82 expected_images = [constants.IMAGE_TYPE_BASE, constants.IMAGE_TYPE_TEST]
83
84 request = self._GetRequest(board='board', types=types)
85 response = self._GetResponse()
86
87 # Failed result to avoid the success handling logic.
88 result = image_service.BuildResult(1, [])
89 build_patch = self.PatchObject(image_service, 'Build', return_value=result)
90
91 image_controller.Create(request, response)
92 build_patch.assert_called_with(images=expected_images, board='board',
Alex Klein56355682019-02-07 10:36:54 -070093 config=mock.ANY)
94
Alex Klein1bcd9882019-03-19 13:25:24 -060095 def testFailedPackageHandling(self):
96 """Test failed packages are populated correctly."""
97 result = image_service.BuildResult(1, ['foo/bar', 'cat/pkg'])
98 expected_packages = [('foo', 'bar'), ('cat', 'pkg')]
99 self.PatchObject(image_service, 'Build', return_value=result)
100
101 input_proto = image_pb2.CreateImageRequest()
102 input_proto.build_target.name = 'board'
103 output_proto = image_pb2.CreateImageResult()
104
Alex Klein8cb365a2019-05-15 16:24:53 -0600105 rc = image_controller.Create(input_proto, output_proto)
106 self.assertEqual(controller.RETURN_CODE_UNSUCCESSFUL_RESPONSE_AVAILABLE, rc)
Alex Klein1bcd9882019-03-19 13:25:24 -0600107 for package in output_proto.failed_packages:
108 self.assertIn((package.category, package.package_name), expected_packages)
109
Alex Klein2557b4f2019-07-11 14:34:00 -0600110 def testNoPackagesFailureHandling(self):
111 """Test failed packages are populated correctly."""
112 result = image_service.BuildResult(1, [])
113 self.PatchObject(image_service, 'Build', return_value=result)
Alex Kleinb7cdbe62019-02-22 11:41:32 -0700114
Alex Klein2557b4f2019-07-11 14:34:00 -0600115 input_proto = image_pb2.CreateImageRequest()
116 input_proto.build_target.name = 'board'
117 output_proto = image_pb2.CreateImageResult()
118
119 rc = image_controller.Create(input_proto, output_proto)
120 self.assertTrue(rc)
121 self.assertNotEqual(controller.RETURN_CODE_UNSUCCESSFUL_RESPONSE_AVAILABLE,
122 rc)
123 self.assertFalse(output_proto.failed_packages)
124
125
Michael Mortensenc83c9952019-08-05 12:15:12 -0600126class ImageSignerTestTest(cros_test_lib.MockTempDirTestCase):
127 """Image signer test tests."""
128
129 def setUp(self):
130 self.image_path = os.path.join(self.tempdir, 'image.bin')
131 self.board = 'board'
132 self.result_directory = os.path.join(self.tempdir, 'results')
133
134 osutils.SafeMakedirs(self.result_directory)
135 osutils.Touch(self.image_path)
136
137 def testSignerTestArgumentValidation(self):
138 """Test function argument validation tests."""
139 self.PatchObject(image_lib, 'SecurityTest', return_value=True)
140 input_proto = image_pb2.TestImageRequest()
141 output_proto = image_pb2.TestImageResult()
142
143 # Nothing provided.
144 with self.assertRaises(cros_build_lib.DieSystemExit):
145 image_controller.Test(input_proto, output_proto)
146
147 # Just one argument.
148 input_proto.build_target.name = self.board
149 with self.assertRaises(cros_build_lib.DieSystemExit):
150 image_controller.Test(input_proto, output_proto)
151
152 # Two arguments provided.
153 input_proto.result.directory = self.result_directory
154 with self.assertRaises(cros_build_lib.DieSystemExit):
155 image_controller.Test(input_proto, output_proto)
156
157 # Invalid image path.
158 input_proto.image.path = '/invalid/image/path'
159 with self.assertRaises(cros_build_lib.DieSystemExit):
160 image_controller.Test(input_proto, output_proto)
161
162 # All valid arguments.
163 input_proto.image.path = self.image_path
164 image_controller.Test(input_proto, output_proto)
165
166 def testSignerTestOutputHandling(self):
167 """Test function output tests."""
168 input_proto = image_pb2.TestImageRequest()
169 input_proto.image.path = self.image_path
170 input_proto.build_target.name = self.board
171 input_proto.result.directory = self.result_directory
172 output_proto = image_pb2.TestImageResult()
173
174 self.PatchObject(image_lib, 'SecurityTest', return_value=True)
175 image_controller.SignerTest(input_proto, output_proto)
176 self.assertTrue(output_proto.success)
177
178 self.PatchObject(image_lib, 'SecurityTest', return_value=False)
179 image_controller.SignerTest(input_proto, output_proto)
180 self.assertFalse(output_proto.success)
181
182 def testSignerTestWithoutMocks(self):
183 """Test function with fake image to sign."""
184 input_proto = image_pb2.TestImageRequest()
185 input_proto.image.path = self.image_path
186 input_proto.build_target.name = self.board
187 input_proto.result.directory = self.result_directory
188 output_proto = image_pb2.TestImageResult()
189
190 image_controller.SignerTest(input_proto, output_proto)
191 self.assertTrue(output_proto.success)
192
193
Alex Klein2557b4f2019-07-11 14:34:00 -0600194class ImageTestTest(cros_test_lib.MockTempDirTestCase):
195 """Image test tests."""
Alex Klein2966e302019-01-17 13:29:38 -0700196
197 def setUp(self):
198 self.image_path = os.path.join(self.tempdir, 'image.bin')
199 self.board = 'board'
200 self.result_directory = os.path.join(self.tempdir, 'results')
201
202 osutils.SafeMakedirs(self.result_directory)
203 osutils.Touch(self.image_path)
204
205 def testTestArgumentValidation(self):
206 """Test function argument validation tests."""
Alex Kleinb7cdbe62019-02-22 11:41:32 -0700207 self.PatchObject(image_service, 'Test', return_value=True)
Alex Klein2966e302019-01-17 13:29:38 -0700208 input_proto = image_pb2.TestImageRequest()
209 output_proto = image_pb2.TestImageResult()
210
211 # Nothing provided.
Alex Klein4f0eb432019-05-02 13:56:04 -0600212 with self.assertRaises(cros_build_lib.DieSystemExit):
Alex Kleinb7cdbe62019-02-22 11:41:32 -0700213 image_controller.Test(input_proto, output_proto)
Alex Klein2966e302019-01-17 13:29:38 -0700214
215 # Just one argument.
216 input_proto.build_target.name = self.board
Alex Klein4f0eb432019-05-02 13:56:04 -0600217 with self.assertRaises(cros_build_lib.DieSystemExit):
Alex Kleinb7cdbe62019-02-22 11:41:32 -0700218 image_controller.Test(input_proto, output_proto)
Alex Klein2966e302019-01-17 13:29:38 -0700219
220 # Two arguments provided.
221 input_proto.result.directory = self.result_directory
Alex Klein4f0eb432019-05-02 13:56:04 -0600222 with self.assertRaises(cros_build_lib.DieSystemExit):
Alex Kleinb7cdbe62019-02-22 11:41:32 -0700223 image_controller.Test(input_proto, output_proto)
Alex Klein2966e302019-01-17 13:29:38 -0700224
225 # Invalid image path.
226 input_proto.image.path = '/invalid/image/path'
Alex Klein4f0eb432019-05-02 13:56:04 -0600227 with self.assertRaises(cros_build_lib.DieSystemExit):
Alex Kleinb7cdbe62019-02-22 11:41:32 -0700228 image_controller.Test(input_proto, output_proto)
Alex Klein2966e302019-01-17 13:29:38 -0700229
230 # All valid arguments.
231 input_proto.image.path = self.image_path
Alex Kleinb7cdbe62019-02-22 11:41:32 -0700232 image_controller.Test(input_proto, output_proto)
Alex Klein2966e302019-01-17 13:29:38 -0700233
234 def testTestOutputHandling(self):
235 """Test function output tests."""
236 input_proto = image_pb2.TestImageRequest()
237 input_proto.image.path = self.image_path
238 input_proto.build_target.name = self.board
239 input_proto.result.directory = self.result_directory
240 output_proto = image_pb2.TestImageResult()
241
Alex Kleinb7cdbe62019-02-22 11:41:32 -0700242 self.PatchObject(image_service, 'Test', return_value=True)
243 image_controller.Test(input_proto, output_proto)
Alex Klein2966e302019-01-17 13:29:38 -0700244 self.assertTrue(output_proto.success)
245
Alex Kleinb7cdbe62019-02-22 11:41:32 -0700246 self.PatchObject(image_service, 'Test', return_value=False)
247 image_controller.Test(input_proto, output_proto)
Alex Klein2966e302019-01-17 13:29:38 -0700248 self.assertFalse(output_proto.success)