Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

route53_key_signing_key - new module #2412

Merged
1 change: 1 addition & 0 deletions meta/runtime.yml
Original file line number Diff line number Diff line change
Expand Up @@ -140,6 +140,7 @@ action_groups:
- route53
- route53_health_check
- route53_info
- route53_key_signing_key
- route53_zone
- s3_bucket
- s3_bucket_info
Expand Down
367 changes: 367 additions & 0 deletions plugins/modules/route53_key_signing_key.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,367 @@
#!/usr/bin/python
# -*- coding: utf-8 -*-

# Copyright: Contributors to the Ansible project
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)

DOCUMENTATION = r"""
module: route53_key_signing_key
short_description: Manages a key-signing key (KSK)
version_added: 9.2.0
description:
- Creates a new key-signing key (KSK) associated with a hosted zone.
You can only have two KSKs per hosted zone.
- When O(state=absent), it deactivates and deletes a key-signing key (KSK).
- Activates a key-signing key (KSK) so that it can be used for signing by DNSSEC.
- Deactivates a key-signing key (KSK) so that it will not be used for signing by DNSSEC.
options:
state:
description:
- Whether or not the zone should exist.
default: present
choices: [ "present", "absent" ]
type: str
caller_reference:
description:
- A unique string that identifies the request.
- Required when O(state=present).
type: str
hosted_zone_id:
description:
- The unique string (ID) used to identify a hosted zone.
type: str
required: true
aliases: ["zone_id"]
key_management_service_arn:
description:
- The Amazon resource name (ARN) for a customer managed key in Key Management Service (KMS).
- Required when O(state=present).
type: str
aliases: ["kms_arn"]
name:
description:
- A string used to identify a key-signing key (KSK).
type: str
required: true
status:
description:
- A string specifying the initial status of the key-signing key (KSK).
- When O(state=presnent), you can set the value to V(ACTIVE) or V(INACTIVE).
type: str
default: "ACTIVE"
choices: ["ACTIVE", "INACTIVE"]
wait:
description:
- Wait until the changes have been replicated.
type: bool
default: false
wait_timeout:
description:
- How long to wait for the changes to be replicated, in seconds.
default: 300
type: int
extends_documentation_fragment:
- amazon.aws.common.modules
- amazon.aws.region.modules
- amazon.aws.boto3
author:
- Alina Buzachis (@alinabuzachis)
"""

EXAMPLES = r"""
- name: Create a Key Signing Key Request
amazon.aws.route53_ksk:
name: "{{ resource_prefix }}-ksk"
hosted_zone_id: "ZZZ1111112222"
key_management_service_arn: "arn:aws:kms:us-west-2:111122223333:key/1234abcd-12ab-34cd-56ef-1234567890ab"
caller_reference: "arn:aws:iam::123456789012:user/SomeUser"
status: "INACTIVE"
state: present

- name: Activate a Key Signing Key Request
amazon.aws.route53_ksk:
name: "{{ resource_prefix }}-ksk"
hosted_zone_id: "ZZZ1111112222"
key_management_service_arn: "arn:aws:kms:us-west-2:111122223333:key/1234abcd-12ab-34cd-56ef-1234567890ab"
caller_reference: "arn:aws:iam::123456789012:user/SomeUser"
status: "ACTIVE"
state: present

- name: Delete a Key Signing Key Request and deactivate it
amazon.aws.route53_ksk:
name: "{{ resource_prefix }}-ksk"
hosted_zone_id: "ZZZ1111112222"
state: absent
"""

RETURN = r"""
change_info:
description: A dictionary that describes change information about changes made to the hosted zone.
returned: when a Key Signing Key is created or it exists and is updated/deleted
type: dict
contains:
id:
description: Change ID.
type: str
status:
description: The current state of the request.
type: str
submitted_at:
description: The date and time that the change request was submitted in ISO 8601 format and Coordinated Universal Time (UTC).
type: str
comment:
description: A comment you can provide.
type: str
sample: {
"id": "/change/C090307813XORZJ5J3U4",
"status": "PENDING",
"submitted_at": "2024-12-04T15:15:36.743000+00:00"
}
location:
description: The unique URL representing the new key-signing key (KSK).
returned: when only a new Key Signing Key is created
type: str
sample: "https://route53.amazonaws.com/2013-04-01/keysigningkey/xxx/ansible-test-ksk"
key_signing_key:
description: The key-signing key (KSK) that the request creates.
returned: always
type: dict
contains:
name:
description: A string used to identify a key-signing key (KSK).
type: str
kms_arn:
description: The Amazon resource name (ARN) used to identify the customer managed key in Key Management Service (KMS).
type: str
flag:
description: An integer that specifies how the key is used.
type: int
signing_algorithm_mnemonic:
description: A string used to represent the signing algorithm.
type: str
signing_algorithm_type:
description: An integer used to represent the signing algorithm.
type: int
digest_algorithm_mnemonic:
description: A string used to represent the delegation signer digest algorithm.
type: str
digest_algorithm_type:
description: An integer used to represent the delegation signer digest algorithm.
type: int
key_tag:
description: An integer used to identify the DNSSEC record for the domain name.
type: int
digest_value:
description: A cryptographic digest of a DNSKEY resource record (RR).
type: str
public_key:
description: The public key, represented as a Base64 encoding.
type: str
ds_record:
description: A string that represents a delegation signer (DS) record.
type: str
dnskey_record:
description: A string that represents a DNSKEY record.
type: str
status:
description: A string that represents the current key-signing key (KSK) status.
type: str
status_message:
description: The status message provided for ACTION_NEEDED or INTERNAL_FAILURE statuses.
type: str
created_date:
description: The date when the key-signing key (KSK) was created.
type: str
last_modified_date:
description: The last time that the key-signing key (KSK) was changed.
type: str
sample: {
"created_date": "2024-12-04T15:15:36.715000+00:00",
"digest_algorithm_mnemonic": "SHA-256",
"digest_algorithm_type": 2,
"digest_value": "xxx",
"dnskey_record": "xxx",
"ds_record": "xxx",
"flag": 257,
"key_tag": 18948,
"kms_arn": "arn:aws:kms:us-east-1:xxx:key/xxx",
"last_modified_date": "2024-12-04T15:15:36.715000+00:00",
"name": "ansible-test-44230979--ksk",
"public_key": "xxxx",
"signing_algorithm_mnemonic": "ECDSAP256SHA256",
"signing_algorithm_type": 13,
"status": "INACTIVE"
}
"""


try:
import botocore
except ImportError:
pass # Handled by AnsibleAWSModule

from typing import Any
from typing import Dict
from typing import Tuple

from ansible.module_utils.common.dict_transformations import camel_dict_to_snake_dict

from ansible_collections.amazon.aws.plugins.module_utils.exceptions import AnsibleAWSError
from ansible_collections.amazon.aws.plugins.module_utils.modules import AnsibleAWSModule
from ansible_collections.amazon.aws.plugins.module_utils.waiters import get_waiter


def deactivate_key_signing_key(client, hosted_zone_id: str, name: str) -> Dict[str, Any]:
return client.deactivate_key_signing_key(HostedZoneId=hosted_zone_id, Name=name)


def activate_key_signing_key(client, hosted_zone_id: str, name: str) -> Dict[str, Any]:
return client.activate_key_signing_key(HostedZoneId=hosted_zone_id, Name=name)


def get_change(client, change_id: str) -> Dict[str, Any]:
return client.get_change(Id=change_id)


def get_dnssec(client, hosted_zone_id: str) -> Dict[str, Any]:
return client.get_dnssec(HostedZoneId=hosted_zone_id)


def find_ksk(client, module: AnsibleAWSModule) -> Dict[str, Any]:
hosted_zone_dnssec = get_dnssec(client, module.params.get("hosted_zone_id"))
if hosted_zone_dnssec["KeySigningKeys"] != []:
for ksk in hosted_zone_dnssec["KeySigningKeys"]:
if ksk["Name"] == module.params.get("name"):
return ksk
return {}


def wait(client, module: AnsibleAWSModule, change_id: str) -> None:
try:
waiter = get_waiter(client, "resource_record_sets_changed")
waiter.wait(
Id=change_id,
WaiterConfig=dict(
Delay=5,
MaxAttempts=module.params.get("wait_timeout") // 5,
),
)
except botocore.exceptions.WaiterError as e:
module.fail_json_aws(e, msg="Timeout waiting for changes to be applied")


def create_or_update(client, module: AnsibleAWSModule, ksk: Dict[str, Any]) -> Tuple[bool, Dict[str, Any]]:
changed: bool = False
zone_id = module.params.get("hosted_zone_id")
name = module.params.get("name")
status = module.params.get("status")
response = {}

if ksk:
if ksk["Status"] != status:
changed = True

if module.check_mode:
ksk["Status"] = status
module.exit_json(
changed=changed,
key_signing_key=camel_dict_to_snake_dict(ksk),
msg=f"Would have updated the Key Signing Key status to {status} if not in check_mode.",
)

if status == "ACTIVE":
response.update(activate_key_signing_key(client, zone_id, name))
elif status == "INACTIVE":
response.update(deactivate_key_signing_key(client, zone_id, name))
else:
changed = True
if module.check_mode:
module.exit_json(changed=changed, msg="Would have created the Key Signing Key if not in check_mode.")

response = client.create_key_signing_key(
CallerReference=module.params.get("caller_reference"),
KeyManagementServiceArn=module.params.get("key_management_service_arn"),
HostedZoneId=zone_id,
Name=name,
Status=status,
)

return changed, response


def delete(client, module: AnsibleAWSModule, ksk: Dict[str, Any]) -> Tuple[bool, Dict[str, Any]]:
changed: bool = False
zone_id = module.params.get("hosted_zone_id")
name = module.params.get("name")
response = {}

if ksk:
changed = True
if module.check_mode:
module.exit_json(
changed=changed,
key_signing_key={},
msg="Would have deleted the Key Signing Key if not in check_mode.",
)

result = deactivate_key_signing_key(client, zone_id, name)
change_id = result["ChangeInfo"]["Id"]
wait(client, module, change_id)

response.update(client.delete_key_signing_key(HostedZoneId=zone_id, Name=name))

return changed, response


def main() -> None:
argument_spec = dict(
caller_reference=dict(type="str"),
hosted_zone_id=dict(type="str", aliases=["zone_id"], required=True),
key_management_service_arn=dict(type="str", aliases=["kms_arn"], no_log=False),
name=dict(type="str", required=True),
status=dict(type="str", default="ACTIVE", choices=["ACTIVE", "INACTIVE"]),
state=dict(default="present", choices=["present", "absent"]),
wait=dict(type="bool", default=False),
wait_timeout=dict(type="int", default=300),
)

module = AnsibleAWSModule(
argument_spec=argument_spec,
supports_check_mode=True,
required_if=[["state", "present", ["caller_reference", "key_management_service_arn"]]],
)

try:
client = module.client("route53")
except (botocore.exceptions.ClientError, botocore.exceptions.BotoCoreError) as e:
module.fail_json_aws(e, msg="Failed to connect to AWS.")

changed = False
state = module.params.get("state")

try:
ksk = find_ksk(client, module)
if state == "present":
changed, result = create_or_update(client, module, ksk)
else:
changed, result = delete(client, module, ksk)

if module.params.get("wait") and result.get("ChangeInfo"):
change_id = result["ChangeInfo"]["Id"]
wait(client, module, change_id)
result.update(get_change(client, change_id))

# Get updated information about KSK
result["KeySigningKey"] = find_ksk(client, module)

if "ResponseMetadata" in result:
del result["ResponseMetadata"]

except AnsibleAWSError as e:
module.fail_json_aws_error(e)

module.exit_json(changed=changed, **camel_dict_to_snake_dict(result))


if __name__ == "__main__":
main()
1 change: 1 addition & 0 deletions tests/integration/targets/route53_key_signing_key/aliases
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
cloud/aws
Loading
Loading