| #!/usr/bin/env vpython3 |
| # Copyright 2014 The LUCI Authors. All rights reserved. |
| # Use of this source code is governed under the Apache License, Version 2.0 |
| # that can be found in the LICENSE file. |
| |
| import json |
| import os |
| import subprocess |
| |
| import test_env |
| |
| |
| class ErrorsTest(test_env.RecipeEngineUnitTest): |
| def _test_cmd(self, deps, cmd, asserts=None, retcode=0): |
| if cmd[0] == 'run': |
| path = self.tempfile() |
| cmd = [cmd[0]] + ['--output-result-json', path] + cmd[1:] |
| |
| try: |
| output, returncode = deps.main_repo.recipes_py(*cmd) |
| |
| if asserts: |
| asserts(output) |
| self.assertEqual( |
| returncode, retcode, |
| '%d != %d.\noutput:\n%s' % (returncode, retcode, output)) |
| |
| if cmd[0] == 'run': |
| if not os.path.exists(path): |
| return |
| |
| with open(path) as tf: |
| raw = tf.read() |
| data = None |
| if raw: |
| data = json.loads(raw) |
| return data |
| finally: |
| if cmd[0] == 'run': |
| if os.path.exists(path): |
| os.unlink(path) |
| |
| def test_missing_dependency(self): |
| deps = self.FakeRecipeDeps() |
| with deps.main_repo.write_recipe('foo') as recipe: |
| recipe.DEPS = ['aint_no_thang'] |
| |
| def _assert_nomodule(output): |
| self.assertRegex(output, |
| r"No module named 'aint_no_thang' in repo 'main'.") |
| |
| self._test_cmd( |
| deps, ['run', 'foo'], retcode=1, asserts=_assert_nomodule) |
| |
| def test_missing_module_dependency(self): |
| deps = self.FakeRecipeDeps() |
| with deps.main_repo.write_recipe('foo') as recipe: |
| recipe.DEPS = ['le_module'] |
| |
| with deps.main_repo.write_module('le_module') as mod: |
| mod.DEPS.append('love') |
| |
| def _assert_nomodule(output): |
| self.assertRegex(output, r"No module named 'love' in repo 'main'") |
| |
| self._test_cmd( |
| deps, ['run', 'foo'], retcode=1, asserts=_assert_nomodule) |
| |
| def test_no_such_recipe(self): |
| deps = self.FakeRecipeDeps() |
| result = self._test_cmd( |
| deps, ['run', 'nooope'], retcode=1) |
| self.assertNotIn('failure', result['failure']) |
| |
| def test_syntax_error(self): |
| deps = self.FakeRecipeDeps() |
| with deps.main_repo.write_file('recipes/foo.py') as buf: |
| buf.write(''' |
| DEPS = [ (sic) |
| ''') |
| |
| def assert_syntaxerror(output): |
| self.assertRegex(output, r'RecipeSyntaxError') |
| |
| self._test_cmd(deps, ['test', 'run', '--filter', 'foo'], |
| asserts=assert_syntaxerror, retcode=1) |
| self._test_cmd(deps, ['test', 'train', '--filter', 'foo'], |
| asserts=assert_syntaxerror, retcode=1) |
| self._test_cmd(deps, ['run', 'foo'], |
| asserts=assert_syntaxerror, retcode=1) |
| |
| def test_missing_path(self): |
| deps = self.FakeRecipeDeps() |
| with deps.main_repo.write_recipe('missing_path') as recipe: |
| recipe.DEPS.append('recipe_engine/path') |
| recipe.RunSteps.write(''' |
| api.step('do it, joe', ['echo', 'JOE'], |
| cwd=api.path['bippityboppityboo']) |
| ''') |
| recipe.GenTests.write(''' |
| yield api.test('basic') |
| ''') |
| |
| def _assert_keyerror(output): |
| self.assertRegex(output, |
| "KeyError.{1,3}Unknown path: bippityboppityboo.{1,3}") |
| |
| self._test_cmd(deps, ['test', 'train', '--filter', 'missing_path'], |
| asserts=_assert_keyerror, retcode=1) |
| self._test_cmd(deps, ['test', 'run', '--filter', 'missing_path'], |
| asserts=_assert_keyerror, retcode=1) |
| self._test_cmd(deps, ['run', 'missing_path'], |
| asserts=_assert_keyerror, retcode=1) |
| |
| def test_engine_failure(self): |
| deps = self.FakeRecipeDeps() |
| with deps.main_repo.write_recipe('print_step_error') as recipe: |
| recipe.imports = [ |
| 'from recipe_engine.internal import engine' |
| ] |
| recipe.RunSteps.write(''' |
| def bad_print_step(execution_log, step): |
| raise Exception("bad to the bone") |
| |
| engine._print_step = bad_print_step |
| try: |
| api.step('Be good', ['echo', 'Sunshine, lollipops, and rainbows']) |
| finally: |
| api.step.active_result.presentation.status = 'WARNING' |
| ''') |
| |
| self._test_cmd(deps, ['run', 'print_step_error'], |
| asserts=lambda output: self.assertIn( |
| '@@@STEP_LOG_LINE@$debug@Exception: bad to the bone@@@', |
| output), |
| retcode=1) |
| |
| def test_missing_method(self): |
| deps = self.FakeRecipeDeps() |
| with deps.main_repo.write_file('recipes/no_gen_tests.py') as buf: |
| buf.write(''' |
| def RunSteps(api): |
| pass |
| ''') |
| self._test_cmd( |
| deps, ['run', 'no_gen_tests'], |
| asserts=lambda output: self.assertRegex(output, |
| r'(?s)misspelled GenTests'), |
| retcode=1) |
| |
| with deps.main_repo.write_file('recipes/no_run_steps.py') as buf: |
| buf.write(''' |
| def GenTests(api): |
| pass |
| ''') |
| self._test_cmd( |
| deps, ['run', 'no_run_steps'], |
| asserts=lambda output: self.assertRegex(output, |
| r'(?s)misspelled RunSteps'), |
| retcode=1) |
| |
| |
| def test_unconsumed_assertion(self): |
| # There was a regression where unconsumed exceptions would not be detected |
| # if the exception was AssertionError. |
| deps = self.FakeRecipeDeps() |
| with deps.main_repo.write_recipe('unconsumed_assertion') as recipe: |
| recipe.DEPS = [] |
| recipe.GenTests.write(''' |
| yield api.test('basic') + api.expect_exception('AssertionError') |
| ''') |
| |
| self._test_cmd(deps, [ |
| 'test', 'train', '--filter', 'unconsumed_assertion'], |
| asserts=lambda output: self.assertIn( |
| 'FAIL (recipe crashed in an unexpected way)', output), |
| retcode=1) |
| |
| def test_run_recipe_help(self): |
| deps = self.FakeRecipeDeps() |
| with deps.main_repo.write_recipe('do_nothing') as recipe: |
| recipe.DEPS = [] |
| |
| def _assert_output(output): |
| self.assertRegex(output, r'from the root of a \'main\' checkout') |
| self.assertRegex(output, r'\./recipes\.py run .* do_nothing') |
| |
| self._test_cmd(deps, ['run', 'do_nothing'], |
| asserts=_assert_output) |
| |
| def test_bad_config_import(self): |
| deps = self.FakeRecipeDeps() |
| with deps.main_repo.write_module('mod') as mod: |
| mod.config.write(''' |
| import BAD_IMPORT |
| ''') |
| |
| with deps.main_repo.write_recipe('recipe') as recipe: |
| recipe.DEPS.append('mod') |
| |
| self._test_cmd( |
| deps, ['test', 'train'], |
| asserts=lambda output: self.assertRegex( |
| output, r"No module named 'BAD_IMPORT'"), |
| retcode=1) |
| |
| def test_bad_test_api_import(self): |
| deps = self.FakeRecipeDeps() |
| with deps.main_repo.write_module('mod') as mod: |
| mod.test_api.write(''' |
| import BAD_IMPORT |
| ''') |
| |
| with deps.main_repo.write_recipe('recipe') as recipe: |
| recipe.DEPS.append('mod') |
| |
| self._test_cmd( |
| deps, ['test', 'train'], |
| asserts=lambda output: self.assertRegex( |
| output, r"No module named 'BAD_IMPORT'"), |
| retcode=1) |
| |
| |
| if __name__ == '__main__': |
| test_env.main() |