diff --git a/awx/conf/fields.py b/awx/conf/fields.py index e68523271b..b9b3503bb1 100644 --- a/awx/conf/fields.py +++ b/awx/conf/fields.py @@ -9,6 +9,8 @@ from django.utils.translation import ugettext_lazy as _ # Django REST Framework from rest_framework.fields import * # noqa +import six + logger = logging.getLogger('awx.conf.fields') # Use DRF fields to convert/validate settings: @@ -84,3 +86,17 @@ class URLField(CharField): except: raise # If something fails here, just fall through and let the validators check it. super(URLField, self).run_validators(value) + + +class KeyValueField(DictField): + child = CharField() + default_error_messages = { + 'invalid_child': _('"{input}" is not a valid string.') + } + + def to_internal_value(self, data): + ret = super(KeyValueField, self).to_internal_value(data) + for value in data.values(): + if not isinstance(value, six.string_types + six.integer_types + (float,)): + self.fail('invalid_child', input=value) + return ret diff --git a/awx/main/conf.py b/awx/main/conf.py index 365336255c..6453ea1c3b 100644 --- a/awx/main/conf.py +++ b/awx/main/conf.py @@ -227,7 +227,7 @@ register( register( 'AWX_TASK_ENV', - field_class=fields.DictField, + field_class=fields.KeyValueField, default={}, label=_('Extra Environment Variables'), help_text=_('Additional environment variables set for playbook runs, inventory updates, project updates, and notification sending.'), diff --git a/awx/main/tests/functional/api/test_settings.py b/awx/main/tests/functional/api/test_settings.py index 0eb21711fc..5a10e2d0b4 100644 --- a/awx/main/tests/functional/api/test_settings.py +++ b/awx/main/tests/functional/api/test_settings.py @@ -13,6 +13,8 @@ from awx.api.versioning import reverse from awx.conf.models import Setting from awx.main.utils.handlers import BaseHTTPSHandler, LoggingConnectivityException +import six + TEST_GIF_LOGO = '' TEST_PNG_LOGO = '' TEST_JPEG_LOGO = '' @@ -75,6 +77,26 @@ def test_jobs_settings(get, put, patch, delete, admin): 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) + + if expected == 200: + resp = get(url, user=admin) + assert resp.data['AWX_TASK_ENV'] == dict((k, six.text_type(v)) for k, v in value.items()) + + @pytest.mark.django_db def test_ldap_settings(get, put, patch, delete, admin, enterprise_license): url = reverse('api:setting_singleton_detail', kwargs={'category_slug': 'ldap'})