mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-25 13:34:01 -07:00 
			
		
		
		
	Porting tests to pytest (#33387)
* Porting tests to pytest
* Achievement Get: No longer need mock/generator.py
  * Now done via pytest's parametrization
  * Port safe_eval to pytest
  * Port text tests to pytest
  * Port test_set_mode_if_different to pytest
* Change conftest AnsibleModule fixtures to be more flexible
  * Move the AnsibleModules fixtures to module_utils/conftest.py for sharing
  * Testing the argspec code requires:
    * injecting both the argspec and the arguments.
    * Patching the arguments into sys.stdin at a different level
* More porting to obsolete mock/procenv.py
  * Port run_command to pytest
  * Port known_hosts tests to pytest
  * Port safe_eval to pytest
  * Port test_distribution_version.py to pytest
  * Port test_log to pytest
  * Port test__log_invocation to pytest
  * Remove unneeded import of procenv in test_postgresql
* Port test_pip to pytest style
  * As part of this, create a pytest ansiblemodule fixture in
    modules/conftest.py.  This is slightly different than the
    approach taken in module_utils because here we need to override the
    AnsibleModule that the modules will inherit from instead of one that
    we're instantiating ourselves.
* Fixup usage of parametrization in test_deprecate_warn
* Check that the pip module failed in our test
	
	
This commit is contained in:
		
					parent
					
						
							
								ed376abe42
							
						
					
				
			
			
				commit
				
					
						cd36164239
					
				
			
		
					 16 changed files with 812 additions and 1117 deletions
				
			
		|  | @ -1,99 +1,70 @@ | |||
| # -*- coding: utf-8 -*- | ||||
| # (c) 2015-2016, Toshio Kuratomi <tkuratomi@ansible.com> | ||||
| # | ||||
| # This file is part of Ansible | ||||
| # | ||||
| # Ansible is free software: you can redistribute it and/or modify | ||||
| # it under the terms of the GNU General Public License as published by | ||||
| # the Free Software Foundation, either version 3 of the License, or | ||||
| # (at your option) any later version. | ||||
| # | ||||
| # Ansible is distributed in the hope that it will be useful, | ||||
| # but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
| # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
| # GNU General Public License for more details. | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License | ||||
| # along with Ansible.  If not, see <http://www.gnu.org/licenses/>. | ||||
| # (c) 2015-2017, Toshio Kuratomi <tkuratomi@ansible.com> | ||||
| # GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) | ||||
| 
 | ||||
| # Make coding more python3-ish | ||||
| from __future__ import (absolute_import, division) | ||||
| __metaclass__ = type | ||||
| 
 | ||||
| import sys | ||||
| import json | ||||
| 
 | ||||
| from ansible.compat.tests import unittest | ||||
| from units.mock.procenv import ModuleTestCase | ||||
| from units.mock.generator import add_method | ||||
| from itertools import chain | ||||
| import pytest | ||||
| 
 | ||||
| 
 | ||||
| # Strings that should be converted into a typed value | ||||
| VALID_STRINGS = ( | ||||
|     [("'a'", 'a')], | ||||
|     [("'1'", '1')], | ||||
|     [("1", 1)], | ||||
|     [("True", True)], | ||||
|     [("False", False)], | ||||
|     [("{}", {})], | ||||
|     ("'a'", 'a'), | ||||
|     ("'1'", '1'), | ||||
|     ("1", 1), | ||||
|     ("True", True), | ||||
|     ("False", False), | ||||
|     ("{}", {}), | ||||
| ) | ||||
| 
 | ||||
| # Passing things that aren't strings should just return the object | ||||
| NONSTRINGS = ( | ||||
|     [({'a': 1}, {'a': 1})], | ||||
|     ({'a': 1}, {'a': 1}), | ||||
| ) | ||||
| 
 | ||||
| # These strings are not basic types.  For security, these should not be | ||||
| # executed.  We return the same string and get an exception for some | ||||
| INVALID_STRINGS = ( | ||||
|     [("a=1", "a=1", SyntaxError)], | ||||
|     [("a.foo()", "a.foo()", None)], | ||||
|     [("import foo", "import foo", None)], | ||||
|     [("__import__('foo')", "__import__('foo')", ValueError)], | ||||
|     ("a=1", "a=1", SyntaxError), | ||||
|     ("a.foo()", "a.foo()", None), | ||||
|     ("import foo", "import foo", None), | ||||
|     ("__import__('foo')", "__import__('foo')", ValueError), | ||||
| ) | ||||
| 
 | ||||
| 
 | ||||
| def _check_simple_types(self, code, expected): | ||||
| @pytest.mark.parametrize('code, expected, stdin', | ||||
|                          ((c, e, {}) for c, e in chain(VALID_STRINGS, NONSTRINGS)), | ||||
|                          indirect=['stdin']) | ||||
| def test_simple_types(am, code, expected): | ||||
|     # test some basic usage for various types | ||||
|     self.assertEqual(self.am.safe_eval(code), expected) | ||||
|     assert am.safe_eval(code) == expected | ||||
| 
 | ||||
| 
 | ||||
| def _check_simple_types_with_exceptions(self, code, expected): | ||||
| @pytest.mark.parametrize('code, expected, stdin', | ||||
|                          ((c, e, {}) for c, e in chain(VALID_STRINGS, NONSTRINGS)), | ||||
|                          indirect=['stdin']) | ||||
| def test_simple_types_with_exceptions(am, code, expected): | ||||
|     # Test simple types with exceptions requested | ||||
|     self.assertEqual(self.am.safe_eval(code, include_exceptions=True), (expected, None)) | ||||
|     assert am.safe_eval(code, include_exceptions=True), (expected, None) | ||||
| 
 | ||||
| 
 | ||||
| def _check_invalid_strings(self, code, expected): | ||||
|     self.assertEqual(self.am.safe_eval(code), expected) | ||||
| @pytest.mark.parametrize('code, expected, stdin', | ||||
|                          ((c, e, {}) for c, e, dummy in INVALID_STRINGS), | ||||
|                          indirect=['stdin']) | ||||
| def test_invalid_strings(am, code, expected): | ||||
|     assert am.safe_eval(code) == expected | ||||
| 
 | ||||
| 
 | ||||
| def _check_invalid_strings_with_exceptions(self, code, expected, exception): | ||||
|     res = self.am.safe_eval("a=1", include_exceptions=True) | ||||
|     self.assertEqual(res[0], "a=1") | ||||
|     self.assertEqual(type(res[1]), SyntaxError) | ||||
| 
 | ||||
| 
 | ||||
| @add_method(_check_simple_types, *VALID_STRINGS) | ||||
| @add_method(_check_simple_types, *NONSTRINGS) | ||||
| @add_method(_check_simple_types_with_exceptions, *VALID_STRINGS) | ||||
| @add_method(_check_simple_types_with_exceptions, *NONSTRINGS) | ||||
| @add_method(_check_invalid_strings, *[[i[0][0:-1]] for i in INVALID_STRINGS]) | ||||
| @add_method(_check_invalid_strings_with_exceptions, *INVALID_STRINGS) | ||||
| class TestSafeEval(ModuleTestCase): | ||||
| 
 | ||||
|     def setUp(self): | ||||
|         super(TestSafeEval, self).setUp() | ||||
| 
 | ||||
|         from ansible.module_utils import basic | ||||
|         self.old_ansible_args = basic._ANSIBLE_ARGS | ||||
| 
 | ||||
|         basic._ANSIBLE_ARGS = None | ||||
|         self.am = basic.AnsibleModule( | ||||
|             argument_spec=dict(), | ||||
|         ) | ||||
| 
 | ||||
|     def tearDown(self): | ||||
|         super(TestSafeEval, self).tearDown() | ||||
| 
 | ||||
|         from ansible.module_utils import basic | ||||
|         basic._ANSIBLE_ARGS = self.old_ansible_args | ||||
| @pytest.mark.parametrize('code, expected, exception, stdin', | ||||
|                          ((c, e, ex, {}) for c, e, ex in INVALID_STRINGS), | ||||
|                          indirect=['stdin']) | ||||
| def test_invalid_strings_with_exceptions(am, code, expected, exception): | ||||
|     res = am.safe_eval(code, include_exceptions=True) | ||||
|     assert res[0] == expected | ||||
|     if exception is None: | ||||
|         assert res[1] == exception | ||||
|     else: | ||||
|         assert type(res[1]) == exception | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue