mirror of
https://github.com/ansible-collections/community.general.git
synced 2024-09-14 20:13:21 +02:00
cf4806d29d
remove cloudtrail.py from legacy-files as it passes pep8 tests now update return values to use snake case per coding guidelines update boto3 exception handling to use method outlined in coding guidelines update parameter spec and return value code per PR code review
615 lines
23 KiB
Python
615 lines
23 KiB
Python
#!/usr/bin/python
|
|
# This file is part of Ansible
|
|
#
|
|
# Ansible is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# Ansible is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
ANSIBLE_METADATA = {'metadata_version': '1.0',
|
|
'status': ['preview'],
|
|
'supported_by': 'community'}
|
|
|
|
|
|
DOCUMENTATION = '''
|
|
---
|
|
module: cloudtrail
|
|
short_description: manage CloudTrail create, delete, update
|
|
description:
|
|
- Creates, deletes, or updates CloudTrail configuration. Ensures logging is also enabled.
|
|
version_added: "2.0"
|
|
author:
|
|
- "Ansible Core Team"
|
|
- "Ted Timmons"
|
|
- "Daniel Shepherd (@shepdelacreme)"
|
|
requirements:
|
|
- boto3
|
|
- botocore
|
|
options:
|
|
state:
|
|
description:
|
|
- Add or remove CloudTrail configuration.
|
|
- The following states have been preserved for backwards compatibility. C(state=enabled) and C(state=disabled).
|
|
- enabled=present and disabled=absent.
|
|
required: true
|
|
choices: ['present', 'absent', 'enabled', 'disabled']
|
|
name:
|
|
description:
|
|
- Name for the CloudTrail.
|
|
- Names are unique per-region unless the CloudTrail is a mulit-region trail, in which case it is unique per-account.
|
|
required: true
|
|
enable_logging:
|
|
description:
|
|
- Start or stop the CloudTrail logging. If stopped the trail will be paused and will not record events or deliver log files.
|
|
default: true
|
|
version_added: "2.4"
|
|
s3_bucket_name:
|
|
description:
|
|
- An existing S3 bucket where CloudTrail will deliver log files.
|
|
- This bucket should exist and have the proper policy.
|
|
- See U(http://docs.aws.amazon.com/awscloudtrail/latest/userguide/aggregating_logs_regions_bucket_policy.html)
|
|
- Required when C(state=present)
|
|
version_added: "2.4"
|
|
s3_key_prefix:
|
|
description:
|
|
- S3 Key prefix for delivered log files. A trailing slash is not necessary and will be removed.
|
|
is_multi_region_trail:
|
|
description:
|
|
- Specify whether the trail belongs only to one region or exists in all regions.
|
|
default: false
|
|
version_added: "2.4"
|
|
enable_log_file_validation:
|
|
description:
|
|
- Specifies whether log file integrity validation is enabled.
|
|
- CloudTrail will create a hash for every log file delivered and produce a signed digest file that can be used to ensure log files have not been tampered.
|
|
default: false
|
|
version_added: "2.4"
|
|
include_global_events:
|
|
description:
|
|
- Record API calls from global services such as IAM and STS.
|
|
default: true
|
|
sns_topic_name:
|
|
description:
|
|
- SNS Topic name to send notifications to when a log file is delivered
|
|
version_added: "2.4"
|
|
cloudwatch_logs_role_arn:
|
|
description:
|
|
- Specifies a full ARN for an IAM role that assigns the proper permissions for CloudTrail to create and write to the log group listed below.
|
|
- See U(https://docs.aws.amazon.com/awscloudtrail/latest/userguide/send-cloudtrail-events-to-cloudwatch-logs.html)
|
|
- "Example arn:aws:iam::123456789012:role/CloudTrail_CloudWatchLogs_Role"
|
|
- Required when C(cloudwatch_logs_log_group_arn)
|
|
version_added: "2.4"
|
|
cloudwatch_logs_log_group_arn:
|
|
description:
|
|
- A full ARN specifying a valid CloudWatch log group to which CloudTrail logs will be delivered. The log group should already exist.
|
|
- See U(https://docs.aws.amazon.com/awscloudtrail/latest/userguide/send-cloudtrail-events-to-cloudwatch-logs.html)
|
|
- "Example arn:aws:logs:us-east-1:123456789012:log-group:CloudTrail/DefaultLogGroup:*"
|
|
- Required when C(cloudwatch_logs_role_arn)
|
|
version_added: "2.4"
|
|
kms_key_id:
|
|
description:
|
|
- Specifies the KMS key ID to use to encrypt the logs delivered by CloudTrail. This also has the effect of enabling log file encryption.
|
|
- The value can be an alias name prefixed by "alias/", a fully specified ARN to an alias, a fully specified ARN to a key, or a globally unique identifier.
|
|
- Examples
|
|
- alias/MyAliasName
|
|
- "arn:aws:kms:us-east-1:123456789012:alias/MyAliasName"
|
|
- "arn:aws:kms:us-east-1:123456789012:key/12345678-1234-1234-1234-123456789012"
|
|
- 12345678-1234-1234-1234-123456789012
|
|
- See U(https://docs.aws.amazon.com/awscloudtrail/latest/userguide/encrypting-cloudtrail-log-files-with-aws-kms.html)
|
|
version_added: "2.4"
|
|
tags:
|
|
description:
|
|
- A hash/dictionary of tags to be applied to the CloudTrail resource.
|
|
- Remove completely or specify an empty dictionary to remove all tags.
|
|
default: {}
|
|
version_added: "2.4"
|
|
|
|
extends_documentation_fragment:
|
|
- aws
|
|
- ec2
|
|
'''
|
|
|
|
EXAMPLES = '''
|
|
- name: create single region cloudtrail
|
|
cloudtrail:
|
|
state: present
|
|
name: default
|
|
s3_bucket_name: mylogbucket
|
|
s3_key_prefix: cloudtrail
|
|
region: us-east-1
|
|
|
|
- name: create multi-region trail with validation and tags
|
|
cloudtrail:
|
|
state: present
|
|
name: default
|
|
s3_bucket_name: mylogbucket
|
|
region: us-east-1
|
|
is_multi_region_trail: true
|
|
enable_log_file_validation: true
|
|
tags:
|
|
environment: dev
|
|
Name: default
|
|
|
|
- name: pause logging the trail we just created
|
|
cloudtrail:
|
|
state: present
|
|
name: default
|
|
enable_logging: false
|
|
s3_bucket_name: mylogbucket
|
|
region: us-east-1
|
|
is_multi_region_trail: true
|
|
enable_log_file_validation: true
|
|
tags:
|
|
environment: dev
|
|
Name: default
|
|
|
|
- name: delete a trail
|
|
cloudtrail:
|
|
state: absent
|
|
name: default
|
|
'''
|
|
|
|
RETURN = '''
|
|
exists:
|
|
description: whether the resource exists
|
|
returned: always
|
|
type: bool
|
|
sample: true
|
|
trail:
|
|
description: CloudTrail resource details
|
|
returned: always
|
|
type: complex
|
|
sample: hash/dictionary of values
|
|
contains:
|
|
trail_arn:
|
|
description: Full ARN of the CloudTrail resource
|
|
returned: success
|
|
type: string
|
|
sample: arn:aws:cloudtrail:us-east-1:123456789012:trail/default
|
|
name:
|
|
description: Name of the CloudTrail resource
|
|
returned: success
|
|
type: string
|
|
sample: default
|
|
is_logging:
|
|
description: Whether logging is turned on or paused for the Trail
|
|
returned: success
|
|
type: bool
|
|
sample: True
|
|
s3_bucket_name:
|
|
description: S3 bucket name where log files are delivered
|
|
returned: success
|
|
type: string
|
|
sample: myBucket
|
|
s3_key_prefix:
|
|
description: Key prefix in bucket where log files are delivered (if any)
|
|
returned: success when present
|
|
type: string
|
|
sample: myKeyPrefix
|
|
log_file_validation_enabled:
|
|
description: Whether log file validation is enabled on the trail
|
|
returned: success
|
|
type: bool
|
|
sample: true
|
|
include_global_service_events:
|
|
description: Whether global services (IAM, STS) are logged with this trail
|
|
returned: success
|
|
type: bool
|
|
sample: true
|
|
is_multi_region_trail:
|
|
description: Whether the trail applies to all regions or just one
|
|
returned: success
|
|
type: bool
|
|
sample: true
|
|
has_custom_event_selectors:
|
|
description: Whether any custom event selectors are used for this trail.
|
|
returned: success
|
|
type: bool
|
|
sample: False
|
|
home_region:
|
|
description: The home region where the trail was originally created and must be edited.
|
|
returned: success
|
|
type: string
|
|
sample: us-east-1
|
|
sns_topic_name:
|
|
description: The SNS topic name where log delivery notifications are sent.
|
|
returned: success when present
|
|
type: string
|
|
sample: myTopic
|
|
sns_topic_arn:
|
|
description: Full ARN of the SNS topic where log delivery notifications are sent.
|
|
returned: success when present
|
|
type: string
|
|
sample: arn:aws:sns:us-east-1:123456789012:topic/myTopic
|
|
cloud_watch_logs_log_group_arn:
|
|
description: Full ARN of the CloudWatch Logs log group where events are delivered.
|
|
returned: success when present
|
|
type: string
|
|
sample: arn:aws:logs:us-east-1:123456789012:log-group:CloudTrail/DefaultLogGroup:*
|
|
cloud_watch_logs_role_arn:
|
|
description: Full ARN of the IAM role that CloudTrail assumes to deliver events.
|
|
returned: success when present
|
|
type: string
|
|
sample: arn:aws:iam::123456789012:role/CloudTrail_CloudWatchLogs_Role
|
|
kms_key_id:
|
|
description: Full ARN of the KMS Key used to encrypt log files.
|
|
returned: success when present
|
|
type: string
|
|
sample: arn:aws:kms::123456789012:key/12345678-1234-1234-1234-123456789012
|
|
tags:
|
|
description: hash/dictionary of tags applied to this resource
|
|
returned: success
|
|
type: dict
|
|
sample: {'environment': 'dev', 'Name': 'default'}
|
|
'''
|
|
|
|
import traceback
|
|
from ansible.module_utils.basic import AnsibleModule
|
|
from ansible.module_utils.ec2 import boto3_conn, ec2_argument_spec
|
|
from ansible.module_utils.ec2 import get_aws_connection_info, HAS_BOTO3
|
|
from ansible.module_utils.ec2 import ansible_dict_to_boto3_tag_list
|
|
from ansible.module_utils.ec2 import boto3_tag_list_to_ansible_dict
|
|
from ansible.module_utils.ec2 import camel_dict_to_snake_dict
|
|
from botocore.exceptions import ClientError
|
|
|
|
|
|
def create_trail(module, client, ct_params):
|
|
"""
|
|
Creates a CloudTrail
|
|
|
|
module : AnisbleModule object
|
|
client : boto3 client connection object
|
|
ct_params : The parameters for the Trail to create
|
|
"""
|
|
resp = {}
|
|
try:
|
|
resp = client.create_trail(**ct_params)
|
|
except ClientError as err:
|
|
module.fail_json(msg=err.message, exception=traceback.format_exc(), **camel_dict_to_snake_dict(err.response))
|
|
|
|
return resp
|
|
|
|
|
|
def tag_trail(module, client, tags, trail_arn, curr_tags=None, dry_run=False):
|
|
"""
|
|
Creates, updates, removes tags on a CloudTrail resource
|
|
|
|
module : AnisbleModule object
|
|
client : boto3 client connection object
|
|
tags : Dict of tags converted from ansible_dict to boto3 list of dicts
|
|
trail_arn : The ARN of the CloudTrail to operate on
|
|
curr_tags : Dict of the current tags on resource, if any
|
|
dry_run : true/false to determine if changes will be made if needed
|
|
"""
|
|
adds = []
|
|
removes = []
|
|
updates = []
|
|
changed = False
|
|
|
|
if curr_tags is None:
|
|
# No current tags so just convert all to a tag list
|
|
adds = ansible_dict_to_boto3_tag_list(tags)
|
|
else:
|
|
curr_keys = set(curr_tags.keys())
|
|
new_keys = set(tags.keys())
|
|
add_keys = new_keys - curr_keys
|
|
remove_keys = curr_keys - new_keys
|
|
update_keys = dict()
|
|
for k in curr_keys.intersection(new_keys):
|
|
if curr_tags[k] != tags[k]:
|
|
update_keys.update({k: tags[k]})
|
|
|
|
adds = get_tag_list(add_keys, tags)
|
|
removes = get_tag_list(remove_keys, curr_tags)
|
|
updates = get_tag_list(update_keys, tags)
|
|
|
|
if removes or updates:
|
|
changed = True
|
|
if not dry_run:
|
|
try:
|
|
client.remove_tags(ResourceId=trail_arn, TagsList=removes + updates)
|
|
except ClientError as err:
|
|
module.fail_json(msg=err.message, exception=traceback.format_exc(), **camel_dict_to_snake_dict(err.response))
|
|
|
|
if updates or adds:
|
|
changed = True
|
|
if not dry_run:
|
|
try:
|
|
client.add_tags(ResourceId=trail_arn, TagsList=updates + adds)
|
|
except ClientError as err:
|
|
module.fail_json(msg=err.message, exception=traceback.format_exc(), **camel_dict_to_snake_dict(err.response))
|
|
|
|
return changed
|
|
|
|
|
|
def get_tag_list(keys, tags):
|
|
"""
|
|
Returns a list of dicts with tags to act on
|
|
keys : set of keys to get the values for
|
|
tags : the dict of tags to turn into a list
|
|
"""
|
|
tag_list = []
|
|
for k in keys:
|
|
tag_list.append({'Key': k, 'Value': tags[k]})
|
|
|
|
return tag_list
|
|
|
|
|
|
def set_logging(module, client, name, action):
|
|
"""
|
|
Starts or stops logging based on given state
|
|
|
|
module : AnsibleModule object
|
|
client : boto3 client connection object
|
|
name : The name or ARN of the CloudTrail to operate on
|
|
action : start or stop
|
|
"""
|
|
if action == 'start':
|
|
try:
|
|
client.start_logging(Name=name)
|
|
return client.get_trail_status(Name=name)
|
|
except ClientError as err:
|
|
module.fail_json(msg=err.message, exception=traceback.format_exc(), **camel_dict_to_snake_dict(err.response))
|
|
elif action == 'stop':
|
|
try:
|
|
client.stop_logging(Name=name)
|
|
return client.get_trail_status(Name=name)
|
|
except ClientError as err:
|
|
module.fail_json(msg=err.message, exception=traceback.format_exc(), **camel_dict_to_snake_dict(err.response))
|
|
else:
|
|
module.fail_json(msg="Unsupported logging action")
|
|
|
|
|
|
def get_trail_facts(module, client, name):
|
|
"""
|
|
Describes existing trail in an account
|
|
|
|
module : AnsibleModule object
|
|
client : boto3 client connection object
|
|
name : Name of the trail
|
|
"""
|
|
# get Trail info
|
|
try:
|
|
trail_resp = client.describe_trails(trailNameList=[name])
|
|
except ClientError as err:
|
|
module.fail_json(msg=err.message, exception=traceback.format_exc(), **camel_dict_to_snake_dict(err.response))
|
|
|
|
# Now check to see if our trail exists and get status and tags
|
|
if len(trail_resp['trailList']):
|
|
trail = trail_resp['trailList'][0]
|
|
try:
|
|
status_resp = client.get_trail_status(Name=trail['Name'])
|
|
tags_list = client.list_tags(ResourceIdList=[trail['TrailARN']])
|
|
except ClientError as err:
|
|
module.fail_json(msg=err.message, exception=traceback.format_exc(), **camel_dict_to_snake_dict(err.response))
|
|
|
|
trail['IsLogging'] = status_resp['IsLogging']
|
|
trail['tags'] = boto3_tag_list_to_ansible_dict(tags_list['ResourceTagList'][0]['TagsList'])
|
|
# Check for non-existent values and populate with None
|
|
optional_vals = set(['S3KeyPrefix', 'SnsTopicName', 'SnsTopicARN', 'CloudWatchLogsLogGroupArn', 'CloudWatchLogsRoleArn', 'KmsKeyId'])
|
|
for v in optional_vals - set(trail.keys()):
|
|
trail[v] = None
|
|
return trail
|
|
|
|
else:
|
|
# trail doesn't exist return None
|
|
return None
|
|
|
|
|
|
def delete_trail(module, client, trail_arn):
|
|
"""
|
|
Delete a CloudTrail
|
|
|
|
module : AnisbleModule object
|
|
client : boto3 client connection object
|
|
trail_arn : Full CloudTrail ARN
|
|
"""
|
|
try:
|
|
client.delete_trail(Name=trail_arn)
|
|
except ClientError as err:
|
|
module.fail_json(msg=err.message, exception=traceback.format_exc(), **camel_dict_to_snake_dict(err.response))
|
|
|
|
|
|
def update_trail(module, client, ct_params):
|
|
"""
|
|
Delete a CloudTrail
|
|
|
|
module : AnisbleModule object
|
|
client : boto3 client connection object
|
|
ct_params : The parameters for the Trail to update
|
|
"""
|
|
try:
|
|
client.update_trail(**ct_params)
|
|
except ClientError as err:
|
|
module.fail_json(msg=err.message, exception=traceback.format_exc(), **camel_dict_to_snake_dict(err.response))
|
|
|
|
|
|
def main():
|
|
argument_spec = ec2_argument_spec()
|
|
argument_spec.update(dict(
|
|
state=dict(default='present', choices=['present', 'absent', 'enabled', 'disabled']),
|
|
name=dict(default='default'),
|
|
enable_logging=dict(default=True, type='bool'),
|
|
s3_bucket_name=dict(),
|
|
s3_key_prefix=dict(),
|
|
sns_topic_name=dict(),
|
|
is_multi_region_trail=dict(default=False, type='bool'),
|
|
enable_log_file_validation=dict(default=False, type='bool'),
|
|
include_global_events=dict(default=True, type='bool'),
|
|
cloudwatch_logs_role_arn=dict(),
|
|
cloudwatch_logs_log_group_arn=dict(),
|
|
kms_key_id=dict(),
|
|
tags=dict(default={}, type='dict'),
|
|
))
|
|
|
|
required_if = [('state', 'present', ['s3_bucket_name']), ('state', 'enabled', ['s3_bucket_name'])]
|
|
required_together = [('cloudwatch_logs_role_arn', 'cloudwatch_logs_log_group_arn')]
|
|
|
|
module = AnsibleModule(argument_spec=argument_spec, supports_check_mode=True, required_together=required_together, required_if=required_if)
|
|
|
|
if not HAS_BOTO3:
|
|
module.fail_json(msg='boto3 is required for this module')
|
|
|
|
# collect parameters
|
|
if module.params['state'] in ('present', 'enabled'):
|
|
state = 'present'
|
|
elif module.params['state'] in ('absent', 'disabled'):
|
|
state = 'absent'
|
|
tags = module.params['tags']
|
|
enable_logging = module.params['enable_logging']
|
|
ct_params = dict(
|
|
Name=module.params['name'],
|
|
S3BucketName=module.params['s3_bucket_name'],
|
|
IncludeGlobalServiceEvents=module.params['include_global_events'],
|
|
IsMultiRegionTrail=module.params['is_multi_region_trail'],
|
|
EnableLogFileValidation=module.params['enable_log_file_validation'],
|
|
S3KeyPrefix='',
|
|
SnsTopicName='',
|
|
CloudWatchLogsRoleArn='',
|
|
CloudWatchLogsLogGroupArn='',
|
|
KmsKeyId=''
|
|
)
|
|
|
|
if module.params['s3_key_prefix']:
|
|
ct_params['S3KeyPrefix'] = module.params['s3_key_prefix'].rstrip('/')
|
|
|
|
if module.params['sns_topic_name']:
|
|
ct_params['SnsTopicName'] = module.params['sns_topic_name']
|
|
|
|
if module.params['cloudwatch_logs_role_arn']:
|
|
ct_params['CloudWatchLogsRoleArn'] = module.params['cloudwatch_logs_role_arn']
|
|
|
|
if module.params['cloudwatch_logs_log_group_arn']:
|
|
ct_params['CloudWatchLogsLogGroupArn'] = module.params['cloudwatch_logs_log_group_arn']
|
|
|
|
if module.params['kms_key_id']:
|
|
ct_params['KmsKeyId'] = module.params['kms_key_id']
|
|
|
|
try:
|
|
region, ec2_url, aws_connect_params = get_aws_connection_info(module, boto3=True)
|
|
client = boto3_conn(module, conn_type='client', resource='cloudtrail', region=region, endpoint=ec2_url, **aws_connect_params)
|
|
except ClientError as err:
|
|
module.fail_json(msg=err.message, exception=traceback.format_exc(), **camel_dict_to_snake_dict(err.response))
|
|
|
|
results = dict(
|
|
changed=False,
|
|
exists=False
|
|
)
|
|
|
|
# Get existing trail facts
|
|
trail = get_trail_facts(module, client, ct_params['Name'])
|
|
|
|
# If the trail exists set the result exists variable
|
|
if trail is not None:
|
|
results['exists'] = True
|
|
|
|
if state == 'absent' and results['exists']:
|
|
# If Trail exists go ahead and delete
|
|
results['changed'] = True
|
|
results['exists'] = False
|
|
results['trail'] = dict()
|
|
if not module.check_mode:
|
|
delete_trail(module, client, trail['TrailARN'])
|
|
|
|
elif state == 'present' and results['exists']:
|
|
# If Trail exists see if we need to update it
|
|
do_update = False
|
|
for key in ct_params:
|
|
tkey = str(key)
|
|
# boto3 has inconsistent parameter naming so we handle it here
|
|
if key == 'EnableLogFileValidation':
|
|
tkey = 'LogFileValidationEnabled'
|
|
# We need to make an empty string equal None
|
|
if ct_params.get(key) == '':
|
|
val = None
|
|
else:
|
|
val = ct_params.get(key)
|
|
if val != trail.get(tkey):
|
|
do_update = True
|
|
results['changed'] = True
|
|
# If we are in check mode copy the changed values to the trail facts in result output to show what would change.
|
|
if module.check_mode:
|
|
trail.update({tkey: ct_params.get(key)})
|
|
|
|
if not module.check_mode and do_update:
|
|
update_trail(module, client, ct_params)
|
|
trail = get_trail_facts(module, client, ct_params['Name'])
|
|
|
|
# Check if we need to start/stop logging
|
|
if enable_logging and not trail['IsLogging']:
|
|
results['changed'] = True
|
|
trail['IsLogging'] = True
|
|
if not module.check_mode:
|
|
set_logging(module, client, name=ct_params['Name'], action='start')
|
|
if not enable_logging and trail['IsLogging']:
|
|
results['changed'] = True
|
|
trail['IsLogging'] = False
|
|
if not module.check_mode:
|
|
set_logging(module, client, name=ct_params['Name'], action='stop')
|
|
|
|
# Check if we need to update tags on resource
|
|
tag_dry_run = False
|
|
if module.check_mode:
|
|
tag_dry_run = True
|
|
tags_changed = tag_trail(module, client, tags=tags, trail_arn=trail['TrailARN'], curr_tags=trail['tags'], dry_run=tag_dry_run)
|
|
if tags_changed:
|
|
results['changed'] = True
|
|
trail['tags'] = tags
|
|
# Populate trail facts in output
|
|
results['trail'] = camel_dict_to_snake_dict(trail)
|
|
|
|
elif state == 'present' and not results['exists']:
|
|
# Trail doesn't exist just go create it
|
|
results['changed'] = True
|
|
if not module.check_mode:
|
|
# If we aren't in check_mode then actually create it
|
|
created_trail = create_trail(module, client, ct_params)
|
|
# Apply tags
|
|
tag_trail(module, client, tags=tags, trail_arn=created_trail['TrailARN'])
|
|
# Get the trail status
|
|
try:
|
|
status_resp = client.get_trail_status(Name=created_trail['Name'])
|
|
except ClientError as err:
|
|
module.fail_json(msg=err.message, exception=traceback.format_exc(), **camel_dict_to_snake_dict(err.response))
|
|
# Set the logging state for the trail to desired value
|
|
if enable_logging and not status_resp['IsLogging']:
|
|
set_logging(module, client, name=ct_params['Name'], action='start')
|
|
if not enable_logging and status_resp['IsLogging']:
|
|
set_logging(module, client, name=ct_params['Name'], action='stop')
|
|
# Get facts for newly created Trail
|
|
trail = get_trail_facts(module, client, ct_params['Name'])
|
|
|
|
# If we are in check mode create a fake return structure for the newly minted trail
|
|
if module.check_mode:
|
|
acct_id = '123456789012'
|
|
try:
|
|
sts_client = boto3_conn(module, conn_type='client', resource='sts', region=region, endpoint=ec2_url, **aws_connect_params)
|
|
acct_id = sts_client.get_caller_identity()['Account']
|
|
except ClientError:
|
|
pass
|
|
trail = dict()
|
|
trail.update(ct_params)
|
|
trail['LogFileValidationEnabled'] = ct_params['EnableLogFileValidation']
|
|
trail.pop('EnableLogFileValidation')
|
|
fake_arn = 'arn:aws:cloudtrail:' + region + ':' + acct_id + ':trail/' + ct_params['Name']
|
|
trail['HasCustomEventSelectors'] = False
|
|
trail['HomeRegion'] = region
|
|
trail['TrailARN'] = fake_arn
|
|
trail['IsLogging'] = enable_logging
|
|
trail['tags'] = tags
|
|
# Populate trail facts in output
|
|
results['trail'] = camel_dict_to_snake_dict(trail)
|
|
|
|
module.exit_json(**results)
|
|
|
|
|
|
if __name__ == '__main__':
|
|
main()
|