2023-11-30 15:55:51 +00:00
|
|
|
from unittest import mock
|
|
|
|
|
2022-05-27 22:11:09 +00:00
|
|
|
import boto3
|
2022-08-24 10:48:37 +00:00
|
|
|
import pytest
|
2022-12-17 21:35:07 +00:00
|
|
|
from cryptography.hazmat.primitives.asymmetric import rsa
|
2022-05-27 22:11:09 +00:00
|
|
|
|
|
|
|
from moto import mock_kms
|
2022-08-13 09:49:43 +00:00
|
|
|
from moto.core import DEFAULT_ACCOUNT_ID as ACCOUNT_ID
|
2022-05-27 22:11:09 +00:00
|
|
|
|
|
|
|
grantee_principal = (
|
|
|
|
f"arn:aws:iam::{ACCOUNT_ID}:role/service-role/tf-acc-test-7071877926602081451"
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@mock_kms
|
|
|
|
def test_create_grant():
|
|
|
|
client = boto3.client("kms", region_name="us-east-1")
|
2022-12-17 21:35:07 +00:00
|
|
|
key_id = create_key(client)
|
2022-05-27 22:11:09 +00:00
|
|
|
|
|
|
|
resp = client.create_grant(
|
|
|
|
KeyId=key_id,
|
|
|
|
GranteePrincipal=grantee_principal,
|
|
|
|
Operations=["DECRYPT"],
|
|
|
|
Name="testgrant",
|
|
|
|
)
|
2023-08-02 09:41:44 +00:00
|
|
|
assert "GrantId" in resp
|
|
|
|
assert "GrantToken" in resp
|
2022-05-27 22:11:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_kms
|
|
|
|
def test_list_grants():
|
|
|
|
client = boto3.client("kms", region_name="us-east-1")
|
2022-12-17 21:35:07 +00:00
|
|
|
key_id = create_key(client)
|
2022-05-27 22:11:09 +00:00
|
|
|
|
2023-08-02 09:41:44 +00:00
|
|
|
assert client.list_grants(KeyId=key_id)["Grants"] == []
|
2022-05-27 22:11:09 +00:00
|
|
|
|
|
|
|
grant_id1 = client.create_grant(
|
|
|
|
KeyId=key_id,
|
|
|
|
GranteePrincipal=grantee_principal,
|
|
|
|
Operations=["DECRYPT"],
|
|
|
|
Name="testgrant",
|
|
|
|
)["GrantId"]
|
|
|
|
|
|
|
|
grant_id2 = client.create_grant(
|
|
|
|
KeyId=key_id,
|
|
|
|
GranteePrincipal=grantee_principal,
|
|
|
|
Operations=["DECRYPT", "ENCRYPT"],
|
|
|
|
Constraints={"EncryptionContextSubset": {"baz": "kaz", "foo": "bar"}},
|
|
|
|
)["GrantId"]
|
|
|
|
|
|
|
|
# List all
|
|
|
|
grants = client.list_grants(KeyId=key_id)["Grants"]
|
2023-08-02 09:41:44 +00:00
|
|
|
assert len(grants) == 2
|
2022-05-27 22:11:09 +00:00
|
|
|
grant_1 = [grant for grant in grants if grant["GrantId"] == grant_id1][0]
|
|
|
|
grant_2 = [grant for grant in grants if grant["GrantId"] == grant_id2][0]
|
|
|
|
|
2023-08-02 09:41:44 +00:00
|
|
|
assert grant_1["KeyId"] == key_id
|
|
|
|
assert grant_1["GrantId"] == grant_id1
|
|
|
|
assert grant_1["Name"] == "testgrant"
|
|
|
|
assert grant_1["GranteePrincipal"] == grantee_principal
|
|
|
|
assert grant_1["Operations"] == ["DECRYPT"]
|
|
|
|
|
|
|
|
assert grant_2["KeyId"] == key_id
|
|
|
|
assert grant_2["GrantId"] == grant_id2
|
|
|
|
assert "Name" not in grant_2
|
|
|
|
assert grant_2["GranteePrincipal"] == grantee_principal
|
|
|
|
assert grant_2["Operations"] == ["DECRYPT", "ENCRYPT"]
|
|
|
|
assert grant_2["Constraints"] == {
|
|
|
|
"EncryptionContextSubset": {"baz": "kaz", "foo": "bar"}
|
|
|
|
}
|
2022-05-27 22:11:09 +00:00
|
|
|
|
|
|
|
# List by grant_id
|
|
|
|
grants = client.list_grants(KeyId=key_id, GrantId=grant_id2)["Grants"]
|
2023-08-02 09:41:44 +00:00
|
|
|
assert len(grants) == 1
|
|
|
|
assert grants[0]["GrantId"] == grant_id2
|
2022-05-27 22:11:09 +00:00
|
|
|
|
|
|
|
# List by unknown grant_id
|
|
|
|
grants = client.list_grants(KeyId=key_id, GrantId="unknown")["Grants"]
|
2023-08-02 09:41:44 +00:00
|
|
|
assert len(grants) == 0
|
2022-05-27 22:11:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_kms
|
|
|
|
def test_list_retirable_grants():
|
|
|
|
client = boto3.client("kms", region_name="us-east-1")
|
2022-12-17 21:35:07 +00:00
|
|
|
key_id1 = create_key(client)
|
|
|
|
key_id2 = create_key(client)
|
2022-05-27 22:11:09 +00:00
|
|
|
|
|
|
|
client.create_grant(
|
|
|
|
KeyId=key_id1,
|
|
|
|
GranteePrincipal=grantee_principal,
|
|
|
|
Operations=["DECRYPT"],
|
|
|
|
)
|
|
|
|
|
|
|
|
client.create_grant(
|
|
|
|
KeyId=key_id1,
|
|
|
|
GranteePrincipal=grantee_principal,
|
|
|
|
RetiringPrincipal="sth else",
|
|
|
|
Operations=["DECRYPT"],
|
|
|
|
)
|
|
|
|
|
|
|
|
client.create_grant(
|
|
|
|
KeyId=key_id2,
|
|
|
|
GranteePrincipal=grantee_principal,
|
|
|
|
Operations=["DECRYPT"],
|
|
|
|
)
|
|
|
|
|
|
|
|
grant2_key2 = client.create_grant(
|
|
|
|
KeyId=key_id2,
|
|
|
|
GranteePrincipal=grantee_principal,
|
|
|
|
RetiringPrincipal="principal",
|
|
|
|
Operations=["DECRYPT"],
|
|
|
|
)["GrantId"]
|
|
|
|
|
|
|
|
# List only the grants from the retiring principal
|
|
|
|
grants = client.list_retirable_grants(RetiringPrincipal="principal")["Grants"]
|
2023-08-02 09:41:44 +00:00
|
|
|
assert len(grants) == 1
|
|
|
|
assert grants[0]["KeyId"] == key_id2
|
|
|
|
assert grants[0]["GrantId"] == grant2_key2
|
2022-05-27 22:11:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_kms
|
|
|
|
def test_revoke_grant():
|
|
|
|
|
|
|
|
client = boto3.client("kms", region_name="us-east-1")
|
2022-12-17 21:35:07 +00:00
|
|
|
key_id = create_key(client)
|
2022-05-27 22:11:09 +00:00
|
|
|
|
2023-08-02 09:41:44 +00:00
|
|
|
assert client.list_grants(KeyId=key_id)["Grants"] == []
|
2022-05-27 22:11:09 +00:00
|
|
|
|
|
|
|
grant_id = client.create_grant(
|
|
|
|
KeyId=key_id,
|
|
|
|
GranteePrincipal=grantee_principal,
|
|
|
|
Operations=["DECRYPT"],
|
|
|
|
Name="testgrant",
|
|
|
|
)["GrantId"]
|
|
|
|
|
|
|
|
client.revoke_grant(KeyId=key_id, GrantId=grant_id)
|
|
|
|
|
2023-08-02 09:41:44 +00:00
|
|
|
assert len(client.list_grants(KeyId=key_id)["Grants"]) == 0
|
2022-05-27 22:11:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_kms
|
2022-08-24 10:48:37 +00:00
|
|
|
def test_revoke_grant_raises_when_grant_does_not_exist():
|
2022-05-27 22:11:09 +00:00
|
|
|
client = boto3.client("kms", region_name="us-east-1")
|
2022-12-17 21:35:07 +00:00
|
|
|
key_id = create_key(client)
|
2022-08-24 10:48:37 +00:00
|
|
|
not_existent_grant_id = "aabbccdd"
|
2022-05-27 22:11:09 +00:00
|
|
|
|
2022-08-24 10:48:37 +00:00
|
|
|
with pytest.raises(client.exceptions.NotFoundException) as ex:
|
|
|
|
client.revoke_grant(KeyId=key_id, GrantId=not_existent_grant_id)
|
2022-05-27 22:11:09 +00:00
|
|
|
|
2023-08-02 09:41:44 +00:00
|
|
|
assert ex.value.response["Error"]["Code"] == "NotFoundException"
|
|
|
|
assert (
|
|
|
|
ex.value.response["Error"]["Message"]
|
|
|
|
== f"Grant ID {not_existent_grant_id} not found"
|
2022-08-24 10:48:37 +00:00
|
|
|
)
|
2022-05-27 22:11:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_kms
|
|
|
|
def test_retire_grant_by_token():
|
|
|
|
|
|
|
|
client = boto3.client("kms", region_name="us-east-1")
|
2022-12-17 21:35:07 +00:00
|
|
|
key_id = create_key(client)
|
2022-05-27 22:11:09 +00:00
|
|
|
|
|
|
|
for idx in range(0, 3):
|
|
|
|
grant_token = client.create_grant(
|
|
|
|
KeyId=key_id,
|
|
|
|
GranteePrincipal=grantee_principal,
|
|
|
|
Operations=["DECRYPT"],
|
|
|
|
Name=f"testgrant{idx}",
|
|
|
|
)["GrantToken"]
|
|
|
|
|
|
|
|
client.retire_grant(GrantToken=grant_token)
|
|
|
|
|
2023-08-02 09:41:44 +00:00
|
|
|
assert len(client.list_grants(KeyId=key_id)["Grants"]) == 2
|
2022-05-27 22:11:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_kms
|
|
|
|
def test_retire_grant_by_grant_id():
|
|
|
|
|
|
|
|
client = boto3.client("kms", region_name="us-east-1")
|
2022-12-17 21:35:07 +00:00
|
|
|
key_id = create_key(client)
|
2022-05-27 22:11:09 +00:00
|
|
|
|
|
|
|
for idx in range(0, 3):
|
|
|
|
grant_id = client.create_grant(
|
|
|
|
KeyId=key_id,
|
|
|
|
GranteePrincipal=grantee_principal,
|
|
|
|
Operations=["DECRYPT"],
|
|
|
|
Name=f"testgrant{idx}",
|
|
|
|
)["GrantId"]
|
|
|
|
|
|
|
|
client.retire_grant(KeyId=key_id, GrantId=grant_id)
|
|
|
|
|
2023-08-02 09:41:44 +00:00
|
|
|
assert len(client.list_grants(KeyId=key_id)["Grants"]) == 2
|
2022-12-17 21:35:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
def create_key(client):
|
|
|
|
with mock.patch.object(rsa, "generate_private_key", return_value=""):
|
|
|
|
return client.create_key(Policy="my policy")["KeyMetadata"]["KeyId"]
|