Server IP : 85.214.239.14 / Your IP : 3.146.65.209 Web Server : Apache/2.4.62 (Debian) System : Linux h2886529.stratoserver.net 4.9.0 #1 SMP Tue Jan 9 19:45:01 MSK 2024 x86_64 User : www-data ( 33) PHP Version : 7.4.18 Disable Function : pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wifcontinued,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_get_handler,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,pcntl_async_signals,pcntl_unshare, MySQL : OFF | cURL : OFF | WGET : ON | Perl : ON | Python : ON | Sudo : ON | Pkexec : OFF Directory : /usr/lib/python3/dist-packages/ansible_collections/awx/awx/test/awx/ |
Upload File : |
from __future__ import absolute_import, division, print_function __metaclass__ = type import io import os import json import datetime import importlib from contextlib import redirect_stdout, suppress from unittest import mock import logging from requests.models import Response, PreparedRequest import pytest from ansible.module_utils.six import raise_from from awx.main.tests.functional.conftest import _request from awx.main.models import Organization, Project, Inventory, JobTemplate, Credential, CredentialType, ExecutionEnvironment, UnifiedJob from django.db import transaction try: import tower_cli # noqa HAS_TOWER_CLI = True except ImportError: HAS_TOWER_CLI = False try: # Because awxkit will be a directory at the root of this makefile and we are using python3, import awxkit will work even if its not installed. # However, awxkit will not contain api whih causes a stack failure down on line 170 when we try to mock it. # So here we are importing awxkit.api to prevent that. Then you only get an error on tests for awxkit functionality. import awxkit.api # noqa HAS_AWX_KIT = True except ImportError: HAS_AWX_KIT = False logger = logging.getLogger('awx.main.tests') def sanitize_dict(din): """Sanitize Django response data to purge it of internal types so it may be used to cast a requests response object """ if isinstance(din, (int, str, type(None), bool)): return din # native JSON types, no problem elif isinstance(din, datetime.datetime): return din.isoformat() elif isinstance(din, list): for i in range(len(din)): din[i] = sanitize_dict(din[i]) return din elif isinstance(din, dict): for k in din.copy().keys(): din[k] = sanitize_dict(din[k]) return din else: return str(din) # translation proxies often not string but stringlike @pytest.fixture(autouse=True) def collection_path_set(monkeypatch): """Monkey patch sys.path, insert the root of the collection folder so that content can be imported without being fully packaged """ base_folder = os.path.abspath(os.path.join(os.path.dirname(__file__), os.pardir, os.pardir)) monkeypatch.syspath_prepend(base_folder) @pytest.fixture def collection_import(): """These tests run assuming that the awx_collection folder is inserted into the PATH before-hand by collection_path_set. But all imports internally to the collection go through this fixture so that can be changed if needed. For instance, we could switch to fully-qualified import paths. """ def rf(path): return importlib.import_module(path) return rf @pytest.fixture def run_module(request, collection_import): def rf(module_name, module_params, request_user): def new_request(self, method, url, **kwargs): kwargs_copy = kwargs.copy() if 'data' in kwargs: if isinstance(kwargs['data'], dict): kwargs_copy['data'] = kwargs['data'] elif kwargs['data'] is None: pass elif isinstance(kwargs['data'], str): kwargs_copy['data'] = json.loads(kwargs['data']) else: raise RuntimeError('Expected data to be dict or str, got {0}, data: {1}'.format(type(kwargs['data']), kwargs['data'])) if 'params' in kwargs and method == 'GET': # query params for GET are handled a bit differently by # tower-cli and python requests as opposed to REST framework APIRequestFactory if not kwargs_copy.get('data'): kwargs_copy['data'] = {} if isinstance(kwargs['params'], dict): kwargs_copy['data'].update(kwargs['params']) elif isinstance(kwargs['params'], list): for k, v in kwargs['params']: kwargs_copy['data'][k] = v # make request with transaction.atomic(): rf = _request(method.lower()) django_response = rf(url, user=request_user, expect=None, **kwargs_copy) # requests library response object is different from the Django response, but they are the same concept # this converts the Django response object into a requests response object for consumption resp = Response() py_data = django_response.data sanitize_dict(py_data) resp._content = bytes(json.dumps(django_response.data), encoding='utf8') resp.status_code = django_response.status_code resp.headers = {'X-API-Product-Name': 'AWX', 'X-API-Product-Version': '0.0.1-devel'} if request.config.getoption('verbose') > 0: logger.info('%s %s by %s, code:%s', method, '/api/' + url.split('/api/')[1], request_user.username, resp.status_code) resp.request = PreparedRequest() resp.request.prepare(method=method, url=url) return resp def new_open(self, method, url, **kwargs): r = new_request(self, method, url, **kwargs) m = mock.MagicMock(read=mock.MagicMock(return_value=r._content), status=r.status_code, getheader=mock.MagicMock(side_effect=r.headers.get)) return m stdout_buffer = io.StringIO() # Requies specific PYTHONPATH, see docs # Note that a proper Ansiballz explosion of the modules will have an import path like: # ansible_collections.awx.awx.plugins.modules.{} # We should consider supporting that in the future resource_module = collection_import('plugins.modules.{0}'.format(module_name)) if not isinstance(module_params, dict): raise RuntimeError('Module params must be dict, got {0}'.format(type(module_params))) # Ansible params can be passed as an invocation argument or over stdin # this short circuits within the AnsibleModule interface def mock_load_params(self): self.params = module_params if getattr(resource_module, 'ControllerAWXKitModule', None): resource_class = resource_module.ControllerAWXKitModule elif getattr(resource_module, 'ControllerAPIModule', None): resource_class = resource_module.ControllerAPIModule elif getattr(resource_module, 'TowerLegacyModule', None): resource_class = resource_module.TowerLegacyModule else: raise RuntimeError("The module has neither a TowerLegacyModule, ControllerAWXKitModule or a ControllerAPIModule") with mock.patch.object(resource_class, '_load_params', new=mock_load_params): # Call the test utility (like a mock server) instead of issuing HTTP requests with mock.patch('ansible.module_utils.urls.Request.open', new=new_open): if HAS_TOWER_CLI: tower_cli_mgr = mock.patch('tower_cli.api.Session.request', new=new_request) elif HAS_AWX_KIT: tower_cli_mgr = mock.patch('awxkit.api.client.requests.Session.request', new=new_request) else: tower_cli_mgr = suppress() with tower_cli_mgr: try: # Ansible modules return data to the mothership over stdout with redirect_stdout(stdout_buffer): resource_module.main() except SystemExit: pass # A system exit indicates successful execution except Exception: # dump the stdout back to console for debugging print(stdout_buffer.getvalue()) raise module_stdout = stdout_buffer.getvalue().strip() try: result = json.loads(module_stdout) except Exception as e: raise_from(Exception('Module did not write valid JSON, error: {0}, stdout:\n{1}'.format(str(e), module_stdout)), e) # A module exception should never be a test expectation if 'exception' in result: if "ModuleNotFoundError: No module named 'tower_cli'" in result['exception']: pytest.skip('The tower-cli library is needed to run this test, module no longer supported.') raise Exception('Module encountered error:\n{0}'.format(result['exception'])) return result return rf @pytest.fixture def survey_spec(): return { "spec": [{"index": 0, "question_name": "my question?", "default": "mydef", "variable": "myvar", "type": "text", "required": False}], "description": "test", "name": "test", } @pytest.fixture def organization(): return Organization.objects.create(name='Default') @pytest.fixture def project(organization): return Project.objects.create( name="test-proj", description="test-proj-desc", organization=organization, playbook_files=['helloworld.yml'], local_path='_92__test_proj', scm_revision='1234567890123456789012345678901234567890', scm_url='localhost', scm_type='git', ) @pytest.fixture def inventory(organization): return Inventory.objects.create(name='test-inv', organization=organization) @pytest.fixture def job_template(project, inventory): return JobTemplate.objects.create(name='test-jt', project=project, inventory=inventory, playbook='helloworld.yml') @pytest.fixture def machine_credential(organization): ssh_type = CredentialType.defaults['ssh']() ssh_type.save() return Credential.objects.create(credential_type=ssh_type, name='machine-cred', inputs={'username': 'test_user', 'password': 'pas4word'}) @pytest.fixture def vault_credential(organization): ct = CredentialType.defaults['vault']() ct.save() return Credential.objects.create(credential_type=ct, name='vault-cred', inputs={'vault_id': 'foo', 'vault_password': 'pas4word'}) @pytest.fixture def kube_credential(): ct = CredentialType.defaults['kubernetes_bearer_token']() ct.save() return Credential.objects.create( credential_type=ct, name='kube-cred', inputs={'host': 'my.cluster', 'bearer_token': 'my-token', 'verify_ssl': False} ) @pytest.fixture def silence_deprecation(): """The deprecation warnings are stored in a global variable they will create cross-test interference. Use this to turn them off. """ with mock.patch('ansible.module_utils.basic.AnsibleModule.deprecate') as this_mock: yield this_mock @pytest.fixture(autouse=True) def silence_warning(): """Warnings use global variable, same as deprecations.""" with mock.patch('ansible.module_utils.basic.AnsibleModule.warn') as this_mock: yield this_mock @pytest.fixture def execution_environment(): return ExecutionEnvironment.objects.create(name="test-ee", description="test-ee", managed=False) @pytest.fixture(scope='session', autouse=True) def mock_has_unpartitioned_events(): # has_unpartitioned_events determines if there are any events still # left in the old, unpartitioned job events table. In order to work, # this method looks up when the partition migration occurred. When # Django's unit tests run, however, there will be no record of the migration. # We mock this out to circumvent the migration query. with mock.patch.object(UnifiedJob, 'has_unpartitioned_events', new=False) as _fixture: yield _fixture