# Copyright 2018 The Chromium OS Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
"""Unit tests for the `stable_version` module and CLI."""
import mock
import unittest
import common
from autotest_lib.server import frontend
from autotest_lib.site_utils.stable_images import stable_version
class ParseArgsTestCase(unittest.TestCase):
"""Unit tests for `_parse_args()`."""
def test_default_options(self):
"""Test for an empty command line."""
arguments = stable_version._parse_args(['command'])
self.assertFalse(arguments.dry_run)
self.assertIsNone(arguments.type)
self.assertIsNone(arguments.web)
self.assertFalse(arguments.delete)
self.assertIsNone(arguments.key)
self.assertIsNone(arguments.version)
def test_web_option(self):
"""Test for the `--web` option."""
for option in ['-w', '--web']:
argv = ['command', option, 'server']
arguments = stable_version._parse_args(argv)
self.assertEqual(arguments.web, argv[2])
def test_dry_run_option(self):
"""Test for the `--dry-run` option."""
for option in ['-n', '--dry-run']:
argv = ['command', option]
arguments = stable_version._parse_args(argv)
self.assertTrue(arguments.dry_run)
def test_image_type_option(self):
"""Test for the `--type` option."""
for image_type in stable_version._ALL_IMAGE_TYPES:
for option in ['-t', '--type']:
argv = ['command', option, image_type]
arguments = stable_version._parse_args(argv)
def test_delete_option(self):
"""Test for the `--delete` option."""
for option in ['-d', '--delete']:
argv = ['command', option]
arguments = stable_version._parse_args(argv)
self.assertTrue(arguments.delete)
def test_key_argument(self):
"""Test for the BOARD_OR_MODEL argument."""
argv = ['command', 'key']
arguments = stable_version._parse_args(argv)
self.assertEqual(arguments.key, argv[1])
def test_version_argument(self):
"""Test for the VERSION argument."""
argv = ['command', 'key', 'version']
arguments = stable_version._parse_args(argv)
self.assertEqual(arguments.key, argv[1])
self.assertEqual(arguments.version, argv[2])
class ProcessCommandTestCase(unittest.TestCase):
"""Unit tests for `_dispatch_command()`."""
def _dispatch_command_success(self, afe, argv, name_called):
arguments = stable_version._parse_args(argv)
patches = mock.patch.multiple(stable_version,
list_all_mappings=mock.DEFAULT,
list_mapping_by_key=mock.DEFAULT,
set_mapping=mock.DEFAULT,
delete_mapping=mock.DEFAULT)
with patches as mocks:
stable_version._dispatch_command(afe, arguments)
for not_called in set(mocks) - set([name_called]):
mocks[not_called].assert_not_called()
return mocks[name_called]
def _assert_command_error(self, argv):
afe = object()
arguments = stable_version._parse_args(argv)
with self.assertRaises(stable_version._CommandError):
stable_version._dispatch_command(afe, arguments)
def test_list_all(self):
"""Test that `list_all_mappings` is called when required."""
argv = ['command']
afe = object()
called_mock = self._dispatch_command_success(
afe, argv, 'list_all_mappings')
called_mock.assert_called_once_with(afe, None)
def test_list_all_with_type(self):
"""Test that `list_all_mappings` is called with a supplied type."""
argv = ['command', '-t', frontend.AFE.CROS_IMAGE_TYPE]
afe = object()
called_mock = self._dispatch_command_success(
afe, argv, 'list_all_mappings')
called_mock.assert_called_once_with(afe, argv[2])
def test_list_mapping_by_key(self):
"""Test that `list_mapping_by_key` is called when required."""
argv = ['command', 'board']
afe = object()
called_mock = self._dispatch_command_success(
afe, argv, 'list_mapping_by_key')
called_mock.assert_called_once_with(afe, None, argv[1])
def test_list_mapping_by_key_with_type(self):
"""Test that `list_mapping_by_key` is called with a supplied type."""
argv = ['command', '-t', frontend.AFE.CROS_IMAGE_TYPE, 'board']
afe = object()
called_mock = self._dispatch_command_success(
afe, argv, 'list_mapping_by_key')
called_mock.assert_called_once_with(afe, argv[2], argv[3])
def test_set_mapping(self):
"""Test that `set_mapping` is called when required."""
argv = ['command', '-t', frontend.AFE.CROS_IMAGE_TYPE, 'board', 'V0.0']
afe = object()
called_mock = self._dispatch_command_success(
afe, argv, 'set_mapping')
called_mock.assert_called_once_with(afe, argv[2], argv[3],
argv[4], False)
def test_set_mapping_firmware(self):
"""Test error when `set_mapping` is called for firmware."""
argv = ['command', '-t', frontend.AFE.FIRMWARE_IMAGE_TYPE,
'board', 'V0.0']
self._assert_command_error(argv)
def test_set_mapping_no_type(self):
"""Test error when `set_mapping` is called without a type."""
argv = ['command', 'board', 'V0.0']
self._assert_command_error(argv)
def test_delete_mapping(self):
"""Test that `delete_mapping` is called when required."""
argv = ['command', '-t', frontend.AFE.CROS_IMAGE_TYPE, '-d', 'board']
afe = object()
called_mock = self._dispatch_command_success(
afe, argv, 'delete_mapping')
called_mock.assert_called_once_with(afe, argv[2], argv[4], False)
def test_delete_mapping_no_key(self):
"""Test error when `delete_mapping` is called without a key."""
argv = ['command', '-t', frontend.AFE.CROS_IMAGE_TYPE, '-d']
self._assert_command_error(argv)
def test_delete_mapping_no_type(self):
"""Test error when `delete_mapping` is called without a type."""
argv = ['command', '-d', 'board']
self._assert_command_error(argv)
def test_delete_mapping_with_version(self):
"""Test error when `delete_mapping` is called with extra arguments."""
argv = ['command', '-t', frontend.AFE.CROS_IMAGE_TYPE, '-d',
'board', 'V0.0']
self._assert_command_error(argv)
if __name__ == '__main__':
unittest.main()