mirror of
https://github.com/ansible/awx.git
synced 2026-04-10 12:39:22 -02:30
more multicredential JobTemplate changes
* allow for filtering Jobs and JobTemplates by v1 `cloud_credential` and `network_credential` fields * properly validate uniqueness of `extra_credentials` types see: #5807
This commit is contained in:
@@ -265,6 +265,11 @@ class FieldLookupBackend(BaseFilterBackend):
|
|||||||
key = key[5:]
|
key = key[5:]
|
||||||
q_not = True
|
q_not = True
|
||||||
|
|
||||||
|
# Make legacy v1 Job/Template fields work for backwards compatability
|
||||||
|
# TODO: remove after API v1 deprecation period
|
||||||
|
if queryset.model._meta.object_name in ('JobTemplate', 'Job') and key in ('cloud_credential', 'network_credential'):
|
||||||
|
key = 'extra_credentials'
|
||||||
|
|
||||||
# Make legacy v1 Credential fields work for backwards compatability
|
# Make legacy v1 Credential fields work for backwards compatability
|
||||||
# TODO: remove after API v1 deprecation period
|
# TODO: remove after API v1 deprecation period
|
||||||
#
|
#
|
||||||
|
|||||||
@@ -2704,6 +2704,12 @@ class JobTemplateExtraCredentialsList(SubListCreateAttachDetachAPIView):
|
|||||||
new_in_api_v2 = True
|
new_in_api_v2 = True
|
||||||
|
|
||||||
def is_valid_relation(self, parent, sub, created=False):
|
def is_valid_relation(self, parent, sub, created=False):
|
||||||
|
current_extra_types = [
|
||||||
|
cred.credential_type.pk for cred in parent.extra_credentials.all()
|
||||||
|
]
|
||||||
|
if sub.credential_type.pk in current_extra_types:
|
||||||
|
return {'error': _('Cannot assign multiple %s credentials.' % sub.credential_type.name)}
|
||||||
|
|
||||||
if sub.credential_type.kind not in ('net', 'cloud'):
|
if sub.credential_type.kind not in ('net', 'cloud'):
|
||||||
return {'error': _('Extra credentials must be network or cloud.')}
|
return {'error': _('Extra credentials must be network or cloud.')}
|
||||||
return super(JobTemplateExtraCredentialsList, self).is_valid_relation(parent, sub, created)
|
return super(JobTemplateExtraCredentialsList, self).is_valid_relation(parent, sub, created)
|
||||||
@@ -3470,7 +3476,7 @@ class JobDetail(RetrieveUpdateDestroyAPIView):
|
|||||||
return super(JobDetail, self).destroy(request, *args, **kwargs)
|
return super(JobDetail, self).destroy(request, *args, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
class JobExtraCredentialsList(SubListCreateAttachDetachAPIView):
|
class JobExtraCredentialsList(SubListAPIView):
|
||||||
|
|
||||||
model = Credential
|
model = Credential
|
||||||
serializer_class = CredentialSerializer
|
serializer_class = CredentialSerializer
|
||||||
@@ -3479,11 +3485,6 @@ class JobExtraCredentialsList(SubListCreateAttachDetachAPIView):
|
|||||||
new_in_320 = True
|
new_in_320 = True
|
||||||
new_in_api_v2 = True
|
new_in_api_v2 = True
|
||||||
|
|
||||||
def is_valid_relation(self, parent, sub, created=False):
|
|
||||||
if sub.credential_type.kind not in ('net', 'cloud'):
|
|
||||||
return {'error': _('Extra credentials must be network or cloud.')}
|
|
||||||
return super(JobExtraCredentialsList, self).is_valid_relation(parent, sub, created)
|
|
||||||
|
|
||||||
|
|
||||||
class JobLabelList(SubListAPIView):
|
class JobLabelList(SubListAPIView):
|
||||||
|
|
||||||
|
|||||||
@@ -181,7 +181,7 @@ class JobOptions(BaseModel):
|
|||||||
|
|
||||||
@property
|
@property
|
||||||
def all_credentials(self):
|
def all_credentials(self):
|
||||||
credentials = self.extra_credentials.all()
|
credentials = list(self.extra_credentials.all())
|
||||||
if self.vault_credential:
|
if self.vault_credential:
|
||||||
credentials.insert(0, self.vault_credential)
|
credentials.insert(0, self.vault_credential)
|
||||||
if self.credential:
|
if self.credential:
|
||||||
|
|||||||
@@ -5,52 +5,7 @@ from awx.api.versioning import reverse
|
|||||||
|
|
||||||
# TODO: test this with RBAC and lower-priveleged users
|
# TODO: test this with RBAC and lower-priveleged users
|
||||||
@pytest.mark.django_db
|
@pytest.mark.django_db
|
||||||
def test_extra_credential_creation(get, post, organization_factory, job_template_factory, credentialtype_aws):
|
def test_extra_credentials(get, organization_factory, job_template_factory, credential):
|
||||||
objs = organization_factory("org", superusers=['admin'])
|
|
||||||
jt = job_template_factory("jt", organization=objs.organization,
|
|
||||||
inventory='test_inv', project='test_proj').job_template
|
|
||||||
job = jt.create_unified_job()
|
|
||||||
|
|
||||||
url = reverse('api:job_extra_credentials_list', kwargs={'version': 'v2', 'pk': job.pk})
|
|
||||||
response = post(url, {
|
|
||||||
'name': 'My Cred',
|
|
||||||
'credential_type': credentialtype_aws.pk,
|
|
||||||
'inputs': {
|
|
||||||
'username': 'bob',
|
|
||||||
'password': 'secret',
|
|
||||||
}
|
|
||||||
}, objs.superusers.admin)
|
|
||||||
assert response.status_code == 201
|
|
||||||
|
|
||||||
response = get(url, user=objs.superusers.admin)
|
|
||||||
assert response.data.get('count') == 1
|
|
||||||
|
|
||||||
|
|
||||||
# TODO: test this with RBAC and lower-priveleged users
|
|
||||||
@pytest.mark.django_db
|
|
||||||
def test_attach_extra_credential(get, post, organization_factory, job_template_factory, credential):
|
|
||||||
objs = organization_factory("org", superusers=['admin'])
|
|
||||||
jt = job_template_factory("jt", organization=objs.organization,
|
|
||||||
inventory='test_inv', project='test_proj').job_template
|
|
||||||
job = jt.create_unified_job()
|
|
||||||
|
|
||||||
url = reverse('api:job_extra_credentials_list', kwargs={'version': 'v2', 'pk': job.pk})
|
|
||||||
response = get(url, user=objs.superusers.admin)
|
|
||||||
assert response.data.get('count') == 0
|
|
||||||
|
|
||||||
response = post(url, {
|
|
||||||
'associate': True,
|
|
||||||
'id': credential.id,
|
|
||||||
}, objs.superusers.admin)
|
|
||||||
assert response.status_code == 204
|
|
||||||
|
|
||||||
response = get(url, user=objs.superusers.admin)
|
|
||||||
assert response.data.get('count') == 1
|
|
||||||
|
|
||||||
|
|
||||||
# TODO: test this with RBAC and lower-priveleged users
|
|
||||||
@pytest.mark.django_db
|
|
||||||
def test_detach_extra_credential(get, post, organization_factory, job_template_factory, credential):
|
|
||||||
objs = organization_factory("org", superusers=['admin'])
|
objs = organization_factory("org", superusers=['admin'])
|
||||||
jt = job_template_factory("jt", organization=objs.organization,
|
jt = job_template_factory("jt", organization=objs.organization,
|
||||||
inventory='test_inv', project='test_proj').job_template
|
inventory='test_inv', project='test_proj').job_template
|
||||||
@@ -61,28 +16,3 @@ def test_detach_extra_credential(get, post, organization_factory, job_template_f
|
|||||||
url = reverse('api:job_extra_credentials_list', kwargs={'version': 'v2', 'pk': job.pk})
|
url = reverse('api:job_extra_credentials_list', kwargs={'version': 'v2', 'pk': job.pk})
|
||||||
response = get(url, user=objs.superusers.admin)
|
response = get(url, user=objs.superusers.admin)
|
||||||
assert response.data.get('count') == 1
|
assert response.data.get('count') == 1
|
||||||
|
|
||||||
response = post(url, {
|
|
||||||
'disassociate': True,
|
|
||||||
'id': credential.id,
|
|
||||||
}, objs.superusers.admin)
|
|
||||||
assert response.status_code == 204
|
|
||||||
|
|
||||||
response = get(url, user=objs.superusers.admin)
|
|
||||||
assert response.data.get('count') == 0
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.django_db
|
|
||||||
def test_attach_extra_credential_wrong_kind_xfail(get, post, organization_factory, job_template_factory, machine_credential):
|
|
||||||
"""Extra credentials only allow net + cloud credentials"""
|
|
||||||
objs = organization_factory("org", superusers=['admin'])
|
|
||||||
jt = job_template_factory("jt", organization=objs.organization,
|
|
||||||
inventory='test_inv', project='test_proj').job_template
|
|
||||||
job = jt.create_unified_job()
|
|
||||||
|
|
||||||
url = reverse('api:job_extra_credentials_list', kwargs={'version': 'v2', 'pk': job.pk})
|
|
||||||
response = post(url, {
|
|
||||||
'associate': True,
|
|
||||||
'id': machine_credential.id,
|
|
||||||
}, objs.superusers.admin)
|
|
||||||
assert response.status_code == 400
|
|
||||||
|
|||||||
@@ -58,6 +58,42 @@ def test_extra_credential_creation(get, post, organization_factory, job_template
|
|||||||
assert response.data.get('count') == 1
|
assert response.data.get('count') == 1
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_extra_credential_unique_type_xfail(get, post, organization_factory, job_template_factory, credentialtype_aws):
|
||||||
|
objs = organization_factory("org", superusers=['admin'])
|
||||||
|
jt = job_template_factory("jt", organization=objs.organization,
|
||||||
|
inventory='test_inv', project='test_proj').job_template
|
||||||
|
|
||||||
|
url = reverse('api:job_template_extra_credentials_list', kwargs={'version': 'v2', 'pk': jt.pk})
|
||||||
|
response = post(url, {
|
||||||
|
'name': 'My Cred',
|
||||||
|
'credential_type': credentialtype_aws.pk,
|
||||||
|
'inputs': {
|
||||||
|
'username': 'bob',
|
||||||
|
'password': 'secret',
|
||||||
|
}
|
||||||
|
}, objs.superusers.admin)
|
||||||
|
assert response.status_code == 201
|
||||||
|
|
||||||
|
response = get(url, user=objs.superusers.admin)
|
||||||
|
assert response.data.get('count') == 1
|
||||||
|
|
||||||
|
# this request should fail because you can't assign the same type (aws)
|
||||||
|
# twice
|
||||||
|
response = post(url, {
|
||||||
|
'name': 'My Cred',
|
||||||
|
'credential_type': credentialtype_aws.pk,
|
||||||
|
'inputs': {
|
||||||
|
'username': 'joe',
|
||||||
|
'password': 'another-secret',
|
||||||
|
}
|
||||||
|
}, objs.superusers.admin)
|
||||||
|
assert response.status_code == 400
|
||||||
|
|
||||||
|
response = get(url, user=objs.superusers.admin)
|
||||||
|
assert response.data.get('count') == 1
|
||||||
|
|
||||||
|
|
||||||
# TODO: test this with RBAC and lower-priveleged users
|
# TODO: test this with RBAC and lower-priveleged users
|
||||||
@pytest.mark.django_db
|
@pytest.mark.django_db
|
||||||
def test_attach_extra_credential(get, post, organization_factory, job_template_factory, credential):
|
def test_attach_extra_credential(get, post, organization_factory, job_template_factory, credential):
|
||||||
@@ -130,7 +166,7 @@ def test_v1_extra_credentials_detail(get, organization_factory, job_template_fac
|
|||||||
|
|
||||||
|
|
||||||
@pytest.mark.django_db
|
@pytest.mark.django_db
|
||||||
def test_v1_set_extra_credentials(get, patch, organization_factory, job_template_factory, credential, net_credential):
|
def test_v1_set_extra_credentials_assignment(get, patch, organization_factory, job_template_factory, credential, net_credential):
|
||||||
objs = organization_factory("org", superusers=['admin'])
|
objs = organization_factory("org", superusers=['admin'])
|
||||||
jt = job_template_factory("jt", organization=objs.organization,
|
jt = job_template_factory("jt", organization=objs.organization,
|
||||||
inventory='test_inv', project='test_proj').job_template
|
inventory='test_inv', project='test_proj').job_template
|
||||||
@@ -163,6 +199,28 @@ def test_v1_set_extra_credentials(get, patch, organization_factory, job_template
|
|||||||
assert response.data.get('network_credential') is None
|
assert response.data.get('network_credential') is None
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_filter_by_v1(get, organization_factory, job_template_factory, credential, net_credential):
|
||||||
|
objs = organization_factory("org", superusers=['admin'])
|
||||||
|
jt = job_template_factory("jt", organization=objs.organization,
|
||||||
|
inventory='test_inv', project='test_proj').job_template
|
||||||
|
jt.extra_credentials.add(credential)
|
||||||
|
jt.extra_credentials.add(net_credential)
|
||||||
|
jt.save()
|
||||||
|
|
||||||
|
for query in (
|
||||||
|
('cloud_credential', str(credential.pk)),
|
||||||
|
('network_credential', str(net_credential.pk))
|
||||||
|
):
|
||||||
|
url = reverse('api:job_template_list', kwargs={'version': 'v1'})
|
||||||
|
response = get(
|
||||||
|
url,
|
||||||
|
user=objs.superusers.admin,
|
||||||
|
QUERY_STRING='='.join(query)
|
||||||
|
)
|
||||||
|
assert response.data.get('count') == 1
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.django_db
|
@pytest.mark.django_db
|
||||||
@pytest.mark.parametrize(
|
@pytest.mark.parametrize(
|
||||||
"grant_project, grant_credential, grant_inventory, expect", [
|
"grant_project, grant_credential, grant_inventory, expect", [
|
||||||
|
|||||||
Reference in New Issue
Block a user