| # |
| # Copyright 2008 Google Inc. All Rights Reserved. |
| |
| """Test for cli.""" |
| |
| import os |
| import sys |
| import unittest |
| |
| import common |
| from autotest_lib.cli import atest, rpc |
| from autotest_lib.frontend.afe import rpc_client_lib |
| from autotest_lib.frontend.afe.json_rpc import proxy |
| from autotest_lib.client.common_lib.test_utils import mock |
| from autotest_lib.client.common_lib import autotemp |
| |
| CLI_USING_PDB = False |
| CLI_UT_DEBUG = False |
| |
| |
| class ExitException(Exception): |
| """Junk that should be removed.""" |
| pass |
| |
| def create_file(content): |
| """Create a temporary file for testing. |
| |
| @param content: string contents for file. |
| |
| @return: Instance of autotemp.tempfile with specified contents. |
| """ |
| file_temp = autotemp.tempfile(unique_id='cli_mock', text=True) |
| os.write(file_temp.fd, content) |
| return file_temp |
| |
| |
| class cli_unittest(unittest.TestCase): |
| """General mocks and setup / teardown for testing the atest cli. |
| """ |
| def setUp(self): |
| """Setup mocks for rpc calls and system exit. |
| """ |
| super(cli_unittest, self).setUp() |
| self.god = mock.mock_god(debug=CLI_UT_DEBUG, ut=self) |
| self.god.stub_class_method(rpc.afe_comm, 'run') |
| self.god.stub_function(sys, 'exit') |
| |
| def stub_authorization_headers(*args, **kwargs): |
| """No auth headers required for testing.""" |
| return {} |
| self.god.stub_with(rpc_client_lib, 'authorization_headers', |
| stub_authorization_headers) |
| |
| |
| def tearDown(self): |
| """Remove mocks. |
| """ |
| # Unstub first because super may need exit |
| self.god.unstub_all() |
| super(cli_unittest, self).tearDown() |
| |
| |
| def assertEqualNoOrder(self, x, y, message=None): |
| """Assert x and y contain the same elements. |
| |
| @param x: list like object for comparison. |
| @param y: second list like object for comparison |
| @param message: Message for AssertionError if x and y contain different |
| elements. |
| |
| @raises: AssertionError |
| """ |
| self.assertEqual(set(x), set(y), message) |
| |
| |
| def assertWords(self, string, to_find=[], not_in=[]): |
| """Assert the string contains all of the set of words to_find and none |
| of the set not_in. |
| |
| @param string: String to search. |
| @param to_find: List of strings that must be in string. |
| @param not_in: List of strings that must NOT be in string. |
| |
| @raises: AssertionError |
| """ |
| for word in to_find: |
| self.assert_(string.find(word) >= 0, |
| "Could not find '%s' in: %s" % (word, string)) |
| for word in not_in: |
| self.assert_(string.find(word) < 0, |
| "Found (and shouldn't have) '%s' in: %s" % (word, |
| string)) |
| |
| |
| def _check_output(self, out='', out_words_ok=[], out_words_no=[], |
| err='', err_words_ok=[], err_words_no=[]): |
| if out_words_ok or out_words_no: |
| self.assertWords(out, out_words_ok, out_words_no) |
| else: |
| self.assertEqual('', out) |
| |
| if err_words_ok or err_words_no: |
| self.assertWords(err, err_words_ok, err_words_no) |
| else: |
| self.assertEqual('', err) |
| |
| |
| def assertOutput(self, obj, results, |
| out_words_ok=[], out_words_no=[], |
| err_words_ok=[], err_words_no=[]): |
| """Assert that obj's output writes the expected strings to std(out/err). |
| |
| An empty list for out_words_ok or err_words_ok means that the stdout |
| or stderr (respectively) must be empty. |
| |
| @param obj: Command object (such as atest_add_or_remove). |
| @param results: Results of command for obj.output to format. |
| @param out_words_ok: List of strings that must be in stdout. |
| @param out_words_no: List of strings that must NOT be in stdout. |
| @param err_words_ok: List of strings that must be in stderr. |
| @param err_words_no: List of strings that must NOT be in stderr. |
| |
| @raises: AssertionError |
| """ |
| self.god.mock_io() |
| obj.output(results) |
| obj.show_all_failures() |
| (out, err) = self.god.unmock_io() |
| self._check_output(out, out_words_ok, out_words_no, |
| err, err_words_ok, err_words_no) |
| |
| |
| def mock_rpcs(self, rpcs): |
| """Expect and mock the results of a list of RPCs. |
| |
| @param rpcs: A list of tuples, each representing one RPC: |
| (op, args(dict), success, expected) |
| """ |
| for (op, dargs, success, expected) in rpcs: |
| comm = rpc.afe_comm.run |
| if success: |
| comm.expect_call(op, **dargs).and_return(expected) |
| else: |
| (comm.expect_call(op, **dargs). |
| and_raises(proxy.JSONRPCException(expected))) |
| |
| |
| def run_cmd(self, argv, rpcs=[], exit_code=None, |
| out_words_ok=[], out_words_no=[], |
| err_words_ok=[], err_words_no=[]): |
| """Run an atest command with arguments. |
| |
| An empty list for out_words_ok or err_words_ok means that the stdout |
| or stderr (respectively) must be empty. |
| |
| @param argv: List of command and arguments as strings. |
| @param rpcs: List of rpcs to expect the command to perform. |
| @param exit_code: Expected exit code of the command (if not 0). |
| @param out_words_ok: List of strings to expect in stdout. |
| @param out_words_no: List of strings that must not be in stdout. |
| @param err_words_ok: List of strings to expect in stderr. |
| @param err_words_no: List of strings that must not be in stderr. |
| |
| @raises: AssertionError or CheckPlaybackError. |
| |
| @returns: stdout, stderr |
| """ |
| sys.argv = argv |
| |
| self.mock_rpcs(rpcs) |
| |
| if not (CLI_USING_PDB and CLI_UT_DEBUG): |
| self.god.mock_io() |
| if exit_code is not None: |
| sys.exit.expect_call(exit_code).and_raises(ExitException) |
| self.assertRaises(ExitException, atest.main) |
| else: |
| atest.main() |
| (out, err) = self.god.unmock_io() |
| self.god.check_playback() |
| self._check_output(out, out_words_ok, out_words_no, |
| err, err_words_ok, err_words_no) |
| return (out, err) |
| |