481 lines
16 KiB
Python
481 lines
16 KiB
Python
import json
|
|
|
|
import boto3
|
|
import pytest
|
|
from botocore.exceptions import ClientError
|
|
|
|
from moto import mock_ssoadmin
|
|
from moto.iam.aws_managed_policies import aws_managed_policies_data
|
|
|
|
# See our Development Tips on writing tests for hints on how to write good tests:
|
|
# http://docs.getmoto.org/en/latest/docs/contributing/development_tips/tests.html
|
|
|
|
DUMMY_PERMISSIONSET_ID = (
|
|
"arn:aws:sso:::permissionSet/ins-eeeeffffgggghhhh/ps-hhhhkkkkppppoooo"
|
|
)
|
|
DUMMY_INSTANCE_ARN = "arn:aws:sso:::instance/ins-aaaabbbbccccdddd"
|
|
|
|
|
|
@pytest.fixture(name="managed_policies")
|
|
def get_managed_policies():
|
|
return json.loads(aws_managed_policies_data)
|
|
|
|
|
|
def create_permissionset(client) -> str:
|
|
"""Helper function to create a dummy permission set and returns the arn."""
|
|
|
|
response = client.create_permission_set(
|
|
Name="test-permission-set",
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
Description="test permission set",
|
|
)
|
|
|
|
return response["PermissionSet"]["PermissionSetArn"]
|
|
|
|
|
|
@mock_ssoadmin
|
|
def test_put_inline_policy_to_permission_set():
|
|
"""
|
|
Tests putting and getting an inline policy to a permission set.
|
|
"""
|
|
client = boto3.client("sso-admin", region_name="us-east-1")
|
|
|
|
permission_set_arn = create_permissionset(client)
|
|
dummy_policy = {
|
|
"Version": "2012-10-17",
|
|
"Statement": [
|
|
{
|
|
"Effect": "Allow",
|
|
"Action": "s3:GetObject",
|
|
"Resource": "arn:aws:s3:::your-bucket-name/*",
|
|
}
|
|
],
|
|
}
|
|
|
|
# Happy path
|
|
response = client.put_inline_policy_to_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
InlinePolicy=json.dumps(dummy_policy),
|
|
)
|
|
|
|
response = client.get_inline_policy_for_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
)
|
|
|
|
assert response["InlinePolicy"] == json.dumps(dummy_policy)
|
|
|
|
# Invalid permission set arn
|
|
not_create_ps_arn = (
|
|
"arn:aws:sso:::permissionSet/ins-eeeeffffgggghhhh/ps-hhhhkkkkppppoxyz"
|
|
)
|
|
with pytest.raises(ClientError) as e:
|
|
client.put_inline_policy_to_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=not_create_ps_arn,
|
|
InlinePolicy=json.dumps(dummy_policy),
|
|
)
|
|
err = e.value.response["Error"]
|
|
assert err["Code"] == "ResourceNotFoundException"
|
|
assert err["Message"] == "Could not find PermissionSet with id ps-hhhhkkkkppppoxyz"
|
|
|
|
|
|
@mock_ssoadmin
|
|
def test_get_inline_policy_to_permission_set_no_policy():
|
|
client = boto3.client("sso-admin", region_name="us-east-1")
|
|
|
|
permission_set_arn = create_permissionset(client)
|
|
|
|
response = client.get_inline_policy_for_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
)
|
|
|
|
assert response["InlinePolicy"] == ""
|
|
|
|
|
|
@mock_ssoadmin
|
|
def test_delete_inline_policy_to_permissionset():
|
|
client = boto3.client("sso-admin", region_name="us-east-1")
|
|
|
|
permission_set_arn = create_permissionset(client)
|
|
|
|
dummy_policy = {
|
|
"Version": "2012-10-17",
|
|
"Statement": [
|
|
{
|
|
"Effect": "Allow",
|
|
"Action": "s3:GetObject",
|
|
"Resource": "arn:aws:s3:::your-bucket-name/*",
|
|
}
|
|
],
|
|
}
|
|
|
|
client.put_inline_policy_to_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
InlinePolicy=json.dumps(dummy_policy),
|
|
)
|
|
|
|
response = client.get_inline_policy_for_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
)
|
|
|
|
assert response["InlinePolicy"] == json.dumps(dummy_policy)
|
|
|
|
client.delete_inline_policy_from_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
)
|
|
|
|
response = client.get_inline_policy_for_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
)
|
|
|
|
assert response["InlinePolicy"] == ""
|
|
|
|
|
|
@mock_ssoadmin
|
|
def test_attach_managed_policy_to_permission_set():
|
|
client = boto3.client("sso-admin", region_name="us-east-1")
|
|
|
|
permission_set_arn = create_permissionset(client)
|
|
permissionset_id = permission_set_arn.split("/")[-1]
|
|
managed_policy_arn = "arn:aws:iam::aws:policy/AdministratorAccess"
|
|
|
|
client.attach_managed_policy_to_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
ManagedPolicyArn=managed_policy_arn,
|
|
)
|
|
|
|
response = client.list_managed_policies_in_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
)
|
|
|
|
assert response["AttachedManagedPolicies"][0]["Name"] == "AdministratorAccess"
|
|
assert (
|
|
response["AttachedManagedPolicies"][0]["Arn"]
|
|
== "arn:aws:iam::aws:policy/AdministratorAccess"
|
|
)
|
|
|
|
# test for managed policy that is already attached
|
|
with pytest.raises(ClientError) as e:
|
|
client.attach_managed_policy_to_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
ManagedPolicyArn=managed_policy_arn,
|
|
)
|
|
err = e.value.response["Error"]
|
|
assert err["Code"] == "ConflictException"
|
|
assert (
|
|
err["Message"]
|
|
== f"Permission set with id {permissionset_id} already has a typed link attachment to a manged policy with {managed_policy_arn}"
|
|
)
|
|
|
|
# test for managed policy that does not exist
|
|
not_exist_managed_policy_arn = "arn:aws:iam::aws:policy/DoesNotExist"
|
|
with pytest.raises(ClientError) as e:
|
|
client.attach_managed_policy_to_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
ManagedPolicyArn=not_exist_managed_policy_arn,
|
|
)
|
|
err = e.value.response["Error"]
|
|
assert err["Code"] == "ResourceNotFoundException"
|
|
assert (
|
|
err["Message"]
|
|
== "Policy does not exist with ARN: arn:aws:iam::aws:policy/DoesNotExist"
|
|
)
|
|
|
|
|
|
@mock_ssoadmin
|
|
def test_list_managed_policies_quota_limit(managed_policies):
|
|
"""
|
|
Tests exceeding the managed policy quota limit.
|
|
"""
|
|
managed_policies_to_attach = []
|
|
policy_count = 0
|
|
for policy_name in managed_policies:
|
|
path = managed_policies[policy_name]["Path"]
|
|
# only attach policies with path "/"
|
|
if path != "/":
|
|
continue
|
|
managed_policies_to_attach.append(policy_name)
|
|
policy_count += 1
|
|
if policy_count >= 21: # 20 is the quota limit
|
|
break
|
|
|
|
client = boto3.client("sso-admin", region_name="us-east-1")
|
|
permission_set_arn = create_permissionset(client)
|
|
permission_set_id = permission_set_arn.split("/")[-1]
|
|
|
|
arn_string = "arn:aws:iam::aws:policy/"
|
|
with pytest.raises(ClientError) as e:
|
|
# the 21st policy should exceed the quota limit
|
|
for managed_policy in managed_policies_to_attach:
|
|
client.attach_managed_policy_to_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
ManagedPolicyArn=arn_string + managed_policy,
|
|
)
|
|
err = e.value.response["Error"]
|
|
assert err["Code"] == "ServiceQuotaExceededException"
|
|
assert (
|
|
err["Message"]
|
|
== f"You have exceeded AWS SSO limits. Cannot create ManagedPolicy more than 20 for id {permission_set_id}. Please refer to https://docs.aws.amazon.com/singlesignon/latest/userguide/limits.html"
|
|
)
|
|
|
|
|
|
@mock_ssoadmin
|
|
def test_list_managed_policies_in_permission_set(managed_policies):
|
|
"""
|
|
Tests functionality of listing aws managed policies attached to a permission set.
|
|
This also tests the pagination functionality.
|
|
"""
|
|
client = boto3.client("sso-admin", region_name="us-east-1")
|
|
|
|
arn_string = "arn:aws:iam::aws:policy/"
|
|
|
|
# create a dummy permission set
|
|
permission_set_arn = create_permissionset(client)
|
|
|
|
managed_policies_names = list(managed_policies.keys())
|
|
|
|
# attach 3 good managed policies
|
|
for idx in range(3):
|
|
managed_policy_name = managed_policies_names[idx]
|
|
|
|
client.attach_managed_policy_to_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
ManagedPolicyArn=arn_string + managed_policy_name,
|
|
)
|
|
|
|
response = client.list_managed_policies_in_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
MaxResults=2,
|
|
)
|
|
|
|
managed_policies = []
|
|
|
|
assert len(response["AttachedManagedPolicies"]) == 2
|
|
managed_policies.extend(response["AttachedManagedPolicies"])
|
|
next_token = response["NextToken"]
|
|
|
|
response = client.list_managed_policies_in_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
MaxResults=2,
|
|
NextToken=next_token,
|
|
)
|
|
|
|
assert len(response["AttachedManagedPolicies"]) == 1
|
|
managed_policies.extend(response["AttachedManagedPolicies"])
|
|
|
|
# ensure the 3 unique managed policies were returned
|
|
actual_managed_policy_names = [
|
|
managed_policy["Name"] for managed_policy in managed_policies
|
|
]
|
|
expected_managed_policy_names = managed_policies_names[:3]
|
|
assert all(
|
|
name in actual_managed_policy_names for name in expected_managed_policy_names
|
|
)
|
|
|
|
|
|
@mock_ssoadmin
|
|
def test_detach_managed_policy_from_permission_set():
|
|
client = boto3.client("sso-admin", region_name="us-east-1")
|
|
permission_set_arn = create_permissionset(client)
|
|
managed_policy_arn = "arn:aws:iam::aws:policy/AdministratorAccess"
|
|
|
|
# test for managed policy that is not attached
|
|
with pytest.raises(ClientError) as e:
|
|
client.detach_managed_policy_from_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
ManagedPolicyArn=managed_policy_arn,
|
|
)
|
|
err = e.value.response["Error"]
|
|
assert err["Code"] == "ResourceNotFoundException"
|
|
assert (
|
|
err["Message"] == f"Could not find ManagedPolicy with arn {managed_policy_arn}"
|
|
)
|
|
|
|
# attach managed policy
|
|
client.attach_managed_policy_to_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
ManagedPolicyArn=managed_policy_arn,
|
|
)
|
|
|
|
# detach managed policy
|
|
client.detach_managed_policy_from_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
ManagedPolicyArn=managed_policy_arn,
|
|
)
|
|
|
|
# ensure managed policy is detached
|
|
response = client.list_managed_policies_in_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
)
|
|
|
|
assert len(response["AttachedManagedPolicies"]) == 0
|
|
|
|
|
|
@mock_ssoadmin
|
|
def test_attach_customer_managed_policy_reference_to_permission_set():
|
|
client = boto3.client("sso-admin", region_name="us-east-1")
|
|
permission_set_arn = create_permissionset(client)
|
|
|
|
policy_name = "test-policy"
|
|
policy_path = "/test-path/"
|
|
|
|
client.attach_customer_managed_policy_reference_to_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
CustomerManagedPolicyReference={
|
|
"Name": policy_name,
|
|
"Path": policy_path,
|
|
},
|
|
)
|
|
|
|
response = client.list_customer_managed_policy_references_in_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
)
|
|
|
|
assert len(response["CustomerManagedPolicyReferences"]) == 1
|
|
assert response["CustomerManagedPolicyReferences"][0]["Name"] == policy_name
|
|
assert response["CustomerManagedPolicyReferences"][0]["Path"] == policy_path
|
|
|
|
# test for customer managed policy that is already attached
|
|
with pytest.raises(ClientError) as e:
|
|
client.attach_customer_managed_policy_reference_to_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
CustomerManagedPolicyReference={
|
|
"Name": policy_name,
|
|
"Path": policy_path,
|
|
},
|
|
)
|
|
err = e.value.response["Error"]
|
|
assert err["Code"] == "ConflictException"
|
|
assert (
|
|
err["Message"]
|
|
== f"Given customer managed policy with name: {policy_name} and path {policy_path} already attached"
|
|
)
|
|
|
|
|
|
@mock_ssoadmin
|
|
def test_list_customer_managed_policy_references_in_permission_set():
|
|
"""
|
|
Tests listing customer managed policies including pagination.
|
|
"""
|
|
client = boto3.client("sso-admin", region_name="us-east-1")
|
|
permission_set_arn = create_permissionset(client)
|
|
|
|
policy_name = "test-policy-"
|
|
|
|
# attach 3 customer managed policies
|
|
for idx in range(3):
|
|
client.attach_customer_managed_policy_reference_to_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
CustomerManagedPolicyReference={"Name": f"{policy_name}{idx}"},
|
|
)
|
|
|
|
response = client.list_customer_managed_policy_references_in_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
MaxResults=2,
|
|
)
|
|
|
|
customer_managed_policy_names = []
|
|
|
|
assert len(response["CustomerManagedPolicyReferences"]) == 2
|
|
next_token = response["NextToken"]
|
|
for name in response["CustomerManagedPolicyReferences"]:
|
|
customer_managed_policy_names.append(name["Name"])
|
|
|
|
response = client.list_customer_managed_policy_references_in_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
MaxResults=2,
|
|
NextToken=next_token,
|
|
)
|
|
for name in response["CustomerManagedPolicyReferences"]:
|
|
customer_managed_policy_names.append(name["Name"])
|
|
|
|
assert len(response["CustomerManagedPolicyReferences"]) == 1
|
|
|
|
# ensure the 3 unique customer managed policies were returned
|
|
assert len(set(customer_managed_policy_names)) == 3
|
|
|
|
|
|
@mock_ssoadmin
|
|
def test_detach_customer_managed_policy_reference_from_permission_set():
|
|
client = boto3.client("sso-admin", region_name="us-east-1")
|
|
permission_set_arn = create_permissionset(client)
|
|
|
|
# trying to detach a policy that doesn't exist yet
|
|
with pytest.raises(ClientError) as e:
|
|
client.detach_customer_managed_policy_reference_from_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
CustomerManagedPolicyReference={
|
|
"Name": "test-policy",
|
|
},
|
|
)
|
|
err = e.value.response["Error"]
|
|
assert err["Code"] == "ResourceNotFoundException"
|
|
assert (
|
|
err["Message"]
|
|
== "Given managed policy with name: test-policy and path / does not exist on PermissionSet"
|
|
)
|
|
|
|
# attach a policy
|
|
client.attach_customer_managed_policy_reference_to_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
CustomerManagedPolicyReference={
|
|
"Name": "test-policy",
|
|
"Path": "/some-path/",
|
|
},
|
|
)
|
|
|
|
# try to detach the policy but default path (should fail)
|
|
with pytest.raises(ClientError) as e:
|
|
client.detach_customer_managed_policy_reference_from_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
CustomerManagedPolicyReference={
|
|
"Name": "test-policy",
|
|
},
|
|
)
|
|
|
|
# detach the policy
|
|
client.detach_customer_managed_policy_reference_from_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
CustomerManagedPolicyReference={
|
|
"Name": "test-policy",
|
|
"Path": "/some-path/",
|
|
},
|
|
)
|
|
|
|
# ensure policy is detached
|
|
response = client.list_customer_managed_policy_references_in_permission_set(
|
|
InstanceArn=DUMMY_INSTANCE_ARN,
|
|
PermissionSetArn=permission_set_arn,
|
|
)
|
|
|
|
assert len(response["CustomerManagedPolicyReferences"]) == 0
|