# -*- coding: utf-8 -*- # Copyright (c) 2016 Ansible, Inc. # All Rights Reserved. # Python import pytest from django.conf import settings # AWX from awx.api.versioning import reverse from awx.conf.models import Setting from awx.conf.registry import settings_registry TEST_GIF_LOGO = '' # NOQA TEST_PNG_LOGO = '' # NOQA TEST_JPEG_LOGO = '' # NOQA @pytest.mark.django_db def test_url_base_defaults_to_request(options, admin): # If TOWER_URL_BASE is not set, default to the Tower request hostname resp = options(reverse('api:setting_singleton_detail', kwargs={'category_slug': 'system'}), user=admin, expect=200) assert resp.data['actions']['PUT']['TOWER_URL_BASE']['default'] == 'http://testserver' @pytest.mark.django_db def test_jobs_settings(get, put, patch, delete, admin): url = reverse('api:setting_singleton_detail', kwargs={'category_slug': 'jobs'}) get(url, user=admin, expect=200) delete(url, user=admin, expect=204) response = get(url, user=admin, expect=200) data = dict(response.data.items()) put(url, user=admin, data=data, expect=200) patch(url, user=admin, data={'AWX_PROOT_HIDE_PATHS': ['/home']}, expect=200) response = get(url, user=admin, expect=200) assert response.data['AWX_PROOT_HIDE_PATHS'] == ['/home'] data.pop('AWX_PROOT_HIDE_PATHS') data.pop('AWX_PROOT_SHOW_PATHS') data.pop('AWX_ANSIBLE_CALLBACK_PLUGINS') put(url, user=admin, data=data, expect=200) response = get(url, user=admin, expect=200) assert response.data['AWX_PROOT_HIDE_PATHS'] == [] assert response.data['AWX_PROOT_SHOW_PATHS'] == [] assert response.data['AWX_ANSIBLE_CALLBACK_PLUGINS'] == [] @pytest.mark.django_db @pytest.mark.parametrize('value, expected', [ [True, 400], ['invalid', 400], [['also', 'invalid'], 400], [{}, 200], [{'X_FOO': 'VALID'}, 200], [{'X_TOTAL': 100}, 200], [{'X_FOO': ['ALSO', 'INVALID']}, 400], [{'X_FOO': {'ALSO': 'INVALID'}}, 400], ]) def test_awx_task_env_validity(get, patch, admin, value, expected): url = reverse('api:setting_singleton_detail', kwargs={'category_slug': 'jobs'}) patch(url, user=admin, data={'AWX_TASK_ENV': value}, expect=expected) resp = get(url, user=admin) if expected == 200: assert resp.data['AWX_TASK_ENV'] == dict((k, str(v)) for k, v in value.items()) else: assert resp.data['AWX_TASK_ENV'] == dict() @pytest.mark.django_db def test_ldap_settings(get, put, patch, delete, admin): url = reverse('api:setting_singleton_detail', kwargs={'category_slug': 'ldap'}) get(url, user=admin, expect=200) # The PUT below will fail at the moment because AUTH_LDAP_GROUP_TYPE # defaults to None but cannot be set to None. # put(url, user=admin, data=response.data, expect=200) delete(url, user=admin, expect=204) patch(url, user=admin, data={'AUTH_LDAP_SERVER_URI': ''}, expect=200) patch(url, user=admin, data={'AUTH_LDAP_SERVER_URI': 'ldap.example.com'}, expect=400) patch(url, user=admin, data={'AUTH_LDAP_SERVER_URI': 'ldap://ldap.example.com'}, expect=200) patch(url, user=admin, data={'AUTH_LDAP_SERVER_URI': 'ldaps://ldap.example.com'}, expect=200) patch(url, user=admin, data={'AUTH_LDAP_SERVER_URI': 'ldap://ldap.example.com:389'}, expect=200) patch(url, user=admin, data={'AUTH_LDAP_SERVER_URI': 'ldaps://ldap.example.com:636'}, expect=200) patch(url, user=admin, data={'AUTH_LDAP_SERVER_URI': 'ldap://ldap.example.com ldap://ldap2.example.com'}, expect=200) patch(url, user=admin, data={'AUTH_LDAP_SERVER_URI': 'ldap://ldap.example.com,ldap://ldap2.example.com'}, expect=200) patch(url, user=admin, data={'AUTH_LDAP_SERVER_URI': 'ldap://ldap.example.com, ldap://ldap2.example.com'}, expect=200) patch(url, user=admin, data={'AUTH_LDAP_BIND_DN': 'cn=Manager,dc=example,dc=com'}, expect=200) patch(url, user=admin, data={'AUTH_LDAP_BIND_DN': u'cn=暴力膜,dc=大新闻,dc=真的粉丝'}, expect=200) @pytest.mark.django_db @pytest.mark.parametrize('value', [ None, '', 'INVALID', 1, [1], ['INVALID'], ]) def test_ldap_user_flags_by_group_invalid_dn(get, patch, admin, value): url = reverse('api:setting_singleton_detail', kwargs={'category_slug': 'ldap'}) patch(url, user=admin, data={'AUTH_LDAP_USER_FLAGS_BY_GROUP': {'is_superuser': value}}, expect=400) @pytest.mark.django_db def test_ldap_user_flags_by_group_string(get, patch, admin): expected = 'CN=Admins,OU=Groups,DC=example,DC=com' url = reverse('api:setting_singleton_detail', kwargs={'category_slug': 'ldap'}) patch(url, user=admin, data={'AUTH_LDAP_USER_FLAGS_BY_GROUP': {'is_superuser': expected}}, expect=200) resp = get(url, user=admin) assert resp.data['AUTH_LDAP_USER_FLAGS_BY_GROUP']['is_superuser'] == [expected] @pytest.mark.django_db def test_ldap_user_flags_by_group_list(get, patch, admin): expected = [ 'CN=Admins,OU=Groups,DC=example,DC=com', 'CN=Superadmins,OU=Groups,DC=example,DC=com' ] url = reverse('api:setting_singleton_detail', kwargs={'category_slug': 'ldap'}) patch(url, user=admin, data={'AUTH_LDAP_USER_FLAGS_BY_GROUP': {'is_superuser': expected}}, expect=200) resp = get(url, user=admin) assert resp.data['AUTH_LDAP_USER_FLAGS_BY_GROUP']['is_superuser'] == expected @pytest.mark.parametrize('setting', [ 'AUTH_LDAP_USER_DN_TEMPLATE', 'AUTH_LDAP_REQUIRE_GROUP', 'AUTH_LDAP_DENY_GROUP', ]) @pytest.mark.django_db def test_empty_ldap_dn(get, put, patch, delete, admin, setting): url = reverse('api:setting_singleton_detail', kwargs={'category_slug': 'ldap'}) patch(url, user=admin, data={setting: ''}, expect=200) resp = get(url, user=admin, expect=200) assert resp.data[setting] is None patch(url, user=admin, data={setting: None}, expect=200) resp = get(url, user=admin, expect=200) assert resp.data[setting] is None @pytest.mark.django_db def test_radius_settings(get, put, patch, delete, admin, settings): url = reverse('api:setting_singleton_detail', kwargs={'category_slug': 'radius'}) response = get(url, user=admin, expect=200) put(url, user=admin, data=response.data, expect=200) # Set secret via the API. patch(url, user=admin, data={'RADIUS_SECRET': 'mysecret'}, expect=200) response = get(url, user=admin, expect=200) assert response.data['RADIUS_SECRET'] == '$encrypted$' assert Setting.objects.filter(key='RADIUS_SECRET').first().value.startswith('$encrypted$') assert settings.RADIUS_SECRET == 'mysecret' # Set secret via settings wrapper. settings_wrapper = settings._awx_conf_settings settings_wrapper.RADIUS_SECRET = 'mysecret2' response = get(url, user=admin, expect=200) assert response.data['RADIUS_SECRET'] == '$encrypted$' assert Setting.objects.filter(key='RADIUS_SECRET').first().value.startswith('$encrypted$') assert settings.RADIUS_SECRET == 'mysecret2' # If we send back $encrypted$, the setting is not updated. patch(url, user=admin, data={'RADIUS_SECRET': '$encrypted$'}, expect=200) response = get(url, user=admin, expect=200) assert response.data['RADIUS_SECRET'] == '$encrypted$' assert Setting.objects.filter(key='RADIUS_SECRET').first().value.startswith('$encrypted$') assert settings.RADIUS_SECRET == 'mysecret2' # If we send an empty string, the setting is also set to an empty string. patch(url, user=admin, data={'RADIUS_SECRET': ''}, expect=200) response = get(url, user=admin, expect=200) assert response.data['RADIUS_SECRET'] == '' assert Setting.objects.filter(key='RADIUS_SECRET').first().value == '' assert settings.RADIUS_SECRET == '' @pytest.mark.django_db def test_tacacsplus_settings(get, put, patch, admin): url = reverse('api:setting_singleton_detail', kwargs={'category_slug': 'tacacsplus'}) response = get(url, user=admin, expect=200) put(url, user=admin, data=response.data, expect=200) patch(url, user=admin, data={'TACACSPLUS_SECRET': 'mysecret'}, expect=200) patch(url, user=admin, data={'TACACSPLUS_SECRET': ''}, expect=200) patch(url, user=admin, data={'TACACSPLUS_HOST': 'localhost'}, expect=400) patch(url, user=admin, data={'TACACSPLUS_SECRET': 'mysecret'}, expect=200) patch(url, user=admin, data={'TACACSPLUS_HOST': 'localhost'}, expect=200) patch(url, user=admin, data={'TACACSPLUS_HOST': '', 'TACACSPLUS_SECRET': ''}, expect=200) patch(url, user=admin, data={'TACACSPLUS_HOST': 'localhost', 'TACACSPLUS_SECRET': ''}, expect=400) patch(url, user=admin, data={'TACACSPLUS_HOST': 'localhost', 'TACACSPLUS_SECRET': 'mysecret'}, expect=200) @pytest.mark.django_db def test_ui_settings(get, put, patch, delete, admin): url = reverse('api:setting_singleton_detail', kwargs={'category_slug': 'ui'}) response = get(url, user=admin, expect=200) assert not response.data['CUSTOM_LOGO'] assert not response.data['CUSTOM_LOGIN_INFO'] put(url, user=admin, data=response.data, expect=200) patch(url, user=admin, data={'CUSTOM_LOGO': 'data:text/plain;base64,'}, expect=400) patch(url, user=admin, data={'CUSTOM_LOGO': ''}, expect=400) patch(url, user=admin, data={'CUSTOM_LOGO': TEST_GIF_LOGO}, expect=200) response = get(url, user=admin, expect=200) assert response.data['CUSTOM_LOGO'] == TEST_GIF_LOGO patch(url, user=admin, data={'CUSTOM_LOGO': TEST_PNG_LOGO}, expect=200) response = get(url, user=admin, expect=200) assert response.data['CUSTOM_LOGO'] == TEST_PNG_LOGO patch(url, user=admin, data={'CUSTOM_LOGO': TEST_JPEG_LOGO}, expect=200) response = get(url, user=admin, expect=200) assert response.data['CUSTOM_LOGO'] == TEST_JPEG_LOGO patch(url, user=admin, data={'CUSTOM_LOGO': ''}, expect=200) response = get(url, user=admin, expect=200) assert not response.data['CUSTOM_LOGO'] patch(url, user=admin, data={'CUSTOM_LOGIN_INFO': 'Customize Me!'}, expect=200) response = get(url, user=admin, expect=200) assert response.data['CUSTOM_LOGIN_INFO'] patch(url, user=admin, data={'CUSTOM_LOGIN_INFO': ''}, expect=200) response = get(url, user=admin, expect=200) assert not response.data['CUSTOM_LOGIN_INFO'] delete(url, user=admin, expect=204) response = get(url, user=admin, expect=200) assert not response.data['CUSTOM_LOGO'] assert not response.data['CUSTOM_LOGIN_INFO'] @pytest.mark.django_db def test_logging_aggregator_connection_test_requires_superuser(post, alice): url = reverse('api:setting_logging_test') post(url, {}, user=alice, expect=403) @pytest.mark.django_db def test_logging_aggregator_connection_test_not_enabled(post, admin): url = reverse('api:setting_logging_test') resp = post(url, {}, user=admin, expect=409) assert 'Logging not enabled' in resp.data.get('error') def _mock_logging_defaults(): # Pre-populate settings obj with defaults class MockSettings: pass mock_settings_obj = MockSettings() mock_settings_json = dict() for key in settings_registry.get_registered_settings(category_slug='logging'): value = settings_registry.get_setting_field(key).get_default() setattr(mock_settings_obj, key, value) mock_settings_json[key] = value setattr(mock_settings_obj, 'MAX_EVENT_RES_DATA', 700000) return mock_settings_obj, mock_settings_json @pytest.mark.parametrize('key, value, error', [ ['LOG_AGGREGATOR_TYPE', 'logstash', 'Cannot enable log aggregator without providing host.'], ['LOG_AGGREGATOR_HOST', 'https://logstash', 'Cannot enable log aggregator without providing type.'] ]) @pytest.mark.django_db def test_logging_aggregator_missing_settings(put, post, admin, key, value, error): _, mock_settings = _mock_logging_defaults() mock_settings['LOG_AGGREGATOR_ENABLED'] = True mock_settings[key] = value url = reverse('api:setting_singleton_detail', kwargs={'category_slug': 'logging'}) response = put(url, data=mock_settings, user=admin, expect=400) assert error in str(response.data) @pytest.mark.parametrize('type, host, port, username, password', [ ['logstash', 'localhost', 8080, 'logger', 'mcstash'], ['loggly', 'http://logs-01.loggly.com/inputs/1fd38090-hash-h4a$h-8d80-t0k3n71/tag/http/', None, None, None], ['splunk', 'https://yoursplunk:8088/services/collector/event', None, None, None], ['other', '97.221.40.41', 9000, 'logger', 'mcstash'], ['sumologic', 'https://endpoint5.collection.us2.sumologic.com/receiver/v1/http/Zagnw_f9XGr_zZgd-_EPM0hb8_rUU7_RU8Q==', None, None, None] ]) @pytest.mark.django_db def test_logging_aggregator_valid_settings(put, post, admin, type, host, port, username, password): _, mock_settings = _mock_logging_defaults() # type = 'splunk' # host = 'https://yoursplunk:8088/services/collector/event' mock_settings['LOG_AGGREGATOR_ENABLED'] = True mock_settings['LOG_AGGREGATOR_TYPE'] = type mock_settings['LOG_AGGREGATOR_HOST'] = host if port: mock_settings['LOG_AGGREGATOR_PORT'] = port if username: mock_settings['LOG_AGGREGATOR_USERNAME'] = username if password: mock_settings['LOG_AGGREGATOR_PASSWORD'] = password url = reverse('api:setting_singleton_detail', kwargs={'category_slug': 'logging'}) response = put(url, data=mock_settings, user=admin, expect=200) assert type in response.data.get('LOG_AGGREGATOR_TYPE') assert host in response.data.get('LOG_AGGREGATOR_HOST') if port: assert port == response.data.get('LOG_AGGREGATOR_PORT') if username: assert username in response.data.get('LOG_AGGREGATOR_USERNAME') if password: # Note: password should be encrypted assert '$encrypted$' in response.data.get('LOG_AGGREGATOR_PASSWORD') @pytest.mark.django_db def test_logging_aggregator_connection_test_valid(put, post, admin): _, mock_settings = _mock_logging_defaults() type = 'other' host = 'https://localhost' mock_settings['LOG_AGGREGATOR_ENABLED'] = True mock_settings['LOG_AGGREGATOR_TYPE'] = type mock_settings['LOG_AGGREGATOR_HOST'] = host # POST to save these mock settings url = reverse('api:setting_singleton_detail', kwargs={'category_slug': 'logging'}) put(url, data=mock_settings, user=admin, expect=200) # "Test" the logger url = reverse('api:setting_logging_test') post(url, {}, user=admin, expect=202) @pytest.mark.django_db @pytest.mark.parametrize('setting_name', [ 'AWX_ISOLATED_CHECK_INTERVAL', 'AWX_ISOLATED_LAUNCH_TIMEOUT', 'AWX_ISOLATED_CONNECTION_TIMEOUT', ]) def test_isolated_job_setting_validation(get, patch, admin, setting_name): url = reverse('api:setting_singleton_detail', kwargs={'category_slug': 'jobs'}) patch(url, user=admin, data={ setting_name: -1 }, expect=400) data = get(url, user=admin).data assert data[setting_name] != -1 @pytest.mark.django_db @pytest.mark.parametrize('key, expected', [ ['AWX_ISOLATED_PRIVATE_KEY', '$encrypted$'], ['AWX_ISOLATED_PUBLIC_KEY', 'secret'], ]) def test_isolated_keys_readonly(get, patch, delete, admin, key, expected): Setting.objects.create( key=key, value='secret' ).save() assert getattr(settings, key) == 'secret' url = reverse('api:setting_singleton_detail', kwargs={'category_slug': 'jobs'}) resp = get(url, user=admin) assert resp.data[key] == expected patch(url, user=admin, data={ key: 'new-secret' }) assert getattr(settings, key) == 'secret' delete(url, user=admin) assert getattr(settings, key) == 'secret' @pytest.mark.django_db def test_isolated_key_flag_readonly(get, patch, delete, admin): settings.AWX_ISOLATED_KEY_GENERATION = True url = reverse('api:setting_singleton_detail', kwargs={'category_slug': 'jobs'}) resp = get(url, user=admin) assert resp.data['AWX_ISOLATED_KEY_GENERATION'] is True patch(url, user=admin, data={ 'AWX_ISOLATED_KEY_GENERATION': False }) assert settings.AWX_ISOLATED_KEY_GENERATION is True delete(url, user=admin) assert settings.AWX_ISOLATED_KEY_GENERATION is True @pytest.mark.django_db @pytest.mark.parametrize('headers', [True, False]) def test_saml_x509cert_validation(patch, get, admin, headers): cert = "MIIEogIBAAKCAQEA1T4za6qBbHxFpN5f9eFvA74MFjrsjcp1uvzOaE23AYKMDEJghJ6dqQ7GwHLNIeIeumqDFmODauIzrgSDJTT5+NG30Rr+rRi0zDkrkBAj/AtA+SaVhbzqB6ZSd7LaMly9XAc+82OKlNpuWS9hPmFaSShzDTXRu5RRyvm4NDCAOGDu5hyVR2pV/ffKDNfNkChnqzvRRW9laQcVmliZhlTGn7nPZ+JbjpwEy0nwW+4zoAiEvwnT52N4xTqIcYOnXtGiaf13dh7FkUfYmS0tzF3+h8QRKwtIm4y+sq84R/kr79/0t5aRUpJynNrECajzmArpL4IjXKTPIyUpTKirJgGnCwIDAQABAoIBAC6bbbm2hpsjfkVOpUKkhxMWUqX5MwK6oYjBAIwjkEAwPFPhnh7eXC87H42oidVCCt1LsmMOVQbjcdAzBEb5kTkk/Twi3k8O+1U3maHfJT5NZ2INYNjeNXh+jb/Dw5UGWAzpOIUR2JQ4Oa4cgPCVbppW0O6uOKz6+fWXJv+hKiUoBCC0TiY52iseHJdUOaKNxYRD2IyIzCAxFSd5tZRaARIYDsugXp3E/TdbsVWA7bmjIBOXq+SquTrlB8x7j3B7+Pi09nAJ2U/uV4PHE+/2Fl009ywfmqancvnhwnz+GQ5jjP+gTfghJfbO+Z6M346rS0Vw+osrPgfyudNHlCswHOECgYEA/Cfq25gDP07wo6+wYWbx6LIzj/SSZy/Ux9P8zghQfoZiPoaq7BQBPAzwLNt7JWST8U11LZA8/wo6ch+HSTMk+m5ieVuru2cHxTDqeNlh94eCrNwPJ5ayA5U6LxAuSCTAzp+rv6KQUx1JcKSEHuh+nRYTKvUDE6iA6YtPLO96lLUCgYEA2H5rOPX2M4w1Q9zjol77lplbPRdczXNd0PIzhy8Z2ID65qvmr1nxBG4f2H96ykW8CKLXNvSXreNZ1BhOXc/3Hv+3mm46iitB33gDX4mlV4Jyo/w5IWhUKRyoW6qXquFFsScxRzTrx/9M+aZeRRLdsBk27HavFEg6jrbQ0SleZL8CgYAaM6Op8d/UgkVrHOR9Go9kmK/W85kK8+NuaE7Ksf57R0eKK8AzC9kc/lMuthfTyOG+n0ff1i8gaVWtai1Ko+/hvfqplacAsDIUgYK70AroB8LCZ5ODj5sr2CPVpB7LDFakod7c6O2KVW6+L7oy5AHUHOkc+5y4PDg5DGrLxo68SQKBgAlGoWF3aG0c/MtDk51JZI43U+lyLs++ua5SMlMAeaMFI7rucpvgxqrh7Qthqukvw7a7A22fXUBeFWM5B2KNnpD9c+hyAKAa6l+gzMQzKZpuRGsyS2BbEAAS8kO7M3Rm4o2MmFfstI2FKs8nibJ79HOvIONQ0n+T+K5Utu2/UAQRAoGAFB4fiIyQ0nYzCf18Z4Wvi/qeIOW+UoBonIN3y1h4wruBywINHxFMHx4aVImJ6R09hoJ9D3Mxli3xF/8JIjfTG5fBSGrGnuofl14d/XtRDXbT2uhVXrIkeLL/ojODwwEx0VhxIRUEjPTvEl6AFSRRcBp3KKzQ/cu7ENDY6GTlOUI=" # noqa if headers: cert = '-----BEGIN CERTIFICATE-----\n' + cert + '\n-----END CERTIFICATE-----' url = reverse('api:setting_singleton_detail', kwargs={'category_slug': 'saml'}) resp = patch(url, user=admin, data={ 'SOCIAL_AUTH_SAML_ENABLED_IDPS': { "okta": { "attr_last_name": "LastName", "attr_username": "login", "entity_id": "http://www.okta.com/abc123", "attr_user_permanent_id": "login", "url": "https://example.okta.com/app/abc123/xyz123/sso/saml", "attr_email": "Email", "x509cert": cert, "attr_first_name": "FirstName" } } }) assert resp.status_code == 200