alainivars/utils2devops

View on GitHub
utils2devops/aws/__init__.py

Summary

Maintainability
C
1 day
Test Coverage
class BaseFormat:
    begin_block = '{\n'
    end_block = '}\n\n'

    def format_item(self, name, item, indent: int = 1):
        return '\t' * indent + name + ' = "' + item + '"\n'

    def format_item_bool(self, name, item, indent: int = 1):
        return '\t' * indent + name + ' = ' + 'true\n' \
            if item else 'false\n'

    def format_item_cond(self, name, item, indent: int = 1):
        if item is not None:
            return '\t' * indent + name + ' = "' + item + '"\n'
        else:
            return ''

    def format_items(self, name, items, indent: int = 1):
        the_str = ''
        for item in items:
            the_str += '\t' * indent + name + ' ' + self.begin_block
            the_str += str(item)
            the_str += '\t' * indent + self.end_block
        return the_str


class BlockTypeBase(BaseFormat):
    """
    Described at: https://www.terraform.io/docs/configuration/index.html \n
    Generate the base block type with the format:

    <BLOCK TYPE> "<BLOCK LABEL>" "<BLOCK NAME>" {
      # Block body
      <IDENTIFIER> = <EXPRESSION> # Argument
    }
    """
    def __init__(self, btype, label, name=''):
        """
        Create new object, called automatically

        :param name:
        """
        self.type = btype
        self.label = label
        self.name = name

    def __str__(self):
        return self.type + ' "' + self.label + '" "' + self.name + '" ' + self.begin_block


class BlockTypeWithTagsBase(BlockTypeBase):
    """
    Described at: https://www.terraform.io/docs/configuration/index.html \n
    Generate the base block type with tags with the format:

    <BLOCK TYPE> "<BLOCK LABEL>" "<BLOCK NAME>" {
      # Block body
      <IDENTIFIER> = <EXPRESSION> # Argument
    }
    """
    def __init__(self, btype, label, name=''):
        """
        Create new object, called automatically

        :param name:
        """
        super(BlockTypeWithTagsBase, self).__init__(
            btype='resource', label=label, name=name)
        self.tags = []

    def add_str_tags(self):
        the_str = '\ttags {\n'
        tags = self.tags
        if type(tags) == list:
            for tag in self.tags:
                if 'Key' in tag:
                    the_str += '\t\t"' + tag['Key'] + '" = "' + tag['Value'] + '"\n'
                else:
                    raise NotImplementedError()
                    # TODO: the_str += '\t\t"' + tag[0] + '" = "' + tag[1] + '"\n'
        else:
            for tag in self.tags.items():
                if 'Key' in tag:
                    raise NotImplementedError()
                    # TODO: the_str += '\t\t"' + tag['Key'] + '" = "' + tag['Value'] + '"\n'
                else:
                    the_str += '\t\t"' + tag[0] + '" = "' + tag[1] + '"\n'
        the_str += '\t}\n'
        return the_str


"""
Virtual Private Cloud
"""


class Vpc(BlockTypeWithTagsBase):
    """
    Virtual Private Cloud
    """
    def __init__(self, name=''):
        """
        Create new object, called automatically

        :param name:
        """
        super(Vpc, self).__init__(
            btype='resource', label='aws_vpc', name=name)
        self.cidr_block = ''
        self.enable_dns_hostnames = 'true'
        self.enable_dns_support = 'true'
        self.instance_tenancy = 'default'

    def __str__(self):
        the_str = super(Vpc, self).__str__()
        the_str += self.format_item('tcidr_block', self.cidr_block)
        the_str += self.format_item_bool('enable_dns_hostnames', self.enable_dns_hostnames)
        the_str += self.format_item_bool('enable_dns_support', self.enable_dns_support)
        the_str += self.format_item('instance_tenancy', self.instance_tenancy)
        the_str += self.add_str_tags()
        the_str += self.end_block
        return the_str


class Subnet(BlockTypeWithTagsBase):
    """
    Subnet
    """
    def __init__(self, name=''):
        """
        Create new object, called automatically

        :param name:
        """
        super(Subnet, self).__init__(
            btype='resource', label='aws_subnet', name=name)
        self.vpc_id = ''
        self.cidr_block = ''
        self.availability_zone = ''
        self.map_public_ip_on_launch = True

    def __str__(self):
        if any('Name' in d['Key'] for d in self.tags):
            for tag in self.tags:
                if tag['Key'] == 'Name':
                    self.name = self.name + '-' + tag['Value']
                    break
        the_str = super(Subnet, self).__str__()
        the_str += self.format_item('vpc_id', self.vpc_id)
        the_str += self.format_item('cidr_block', self.cidr_block)
        the_str += self.format_item('availability_zone', self.availability_zone)
        the_str += self.format_item_bool('map_public_ip_on_launch', self.map_public_ip_on_launch)
        the_str += self.add_str_tags()
        the_str += self.end_block
        return the_str


class InternetGateway(BlockTypeWithTagsBase):
    """
    Internet Gateway
    """
    def __init__(self, name=''):
        """
        Create new object, called automatically

        :param name:
        """
        super(InternetGateway, self).__init__(
            btype='resource', label='aws_internet_gateway', name=name)
        self.vpc_id = ''

    def __str__(self):
        the_str = super(InternetGateway, self).__str__()
        the_str += self.format_item('vpc_id', self.vpc_id)
        the_str += self.add_str_tags()
        the_str += self.end_block
        return the_str


class Route(BlockTypeWithTagsBase):
    """
    Route
        The following arguments are supported:
            route_table_id - (Required) The ID of the routing table.

        One of the following destination arguments must be supplied:
            destination_cidr_block - (Optional) The destination CIDR block.
            destination_ipv6_cidr_block - (Optional) The destination IPv6 CIDR block.

        One of the following target arguments must be supplied:
            egress_only_gateway_id - (Optional) Identifier of a VPC Egress Only Internet Gateway.
            gateway_id - (Optional) Identifier of a VPC internet gateway or a virtual private gateway.
            instance_id - (Optional) Identifier of an EC2 instance.
            nat_gateway_id - (Optional) Identifier of a VPC NAT gateway.
            network_interface_id - (Optional) Identifier of an EC2 network interface.
            transit_gateway_id - (Optional) Identifier of an EC2 Transit Gateway.
            vpc_peering_connection_id - (Optional) Identifier of a VPC peering connection.

        Note that the default route, mapping the VPC's CIDR block to "local", is created implicitly and cannot be specified.
    """
    def __init__(self, name='', inserted=True):
        """
        Create new object, called automatically

        :param name:
        """
        super(Route, self).__init__(
            btype='resource', label='aws_route', name=name)
        self.inserted = inserted
        self.route_table_id = ''
        self.destination_cidr_block = None
        self.destination_ipv6_cidr_block = None

        self.egress_only_gateway_id = None
        self.gateway_id = None
        self.instance_id = None
        self.nat_gateway_id = None
        self.network_interface_id = None
        self.transit_gateway_id = None
        self.vpc_peering_connection_id = None
        # self.route = {
        #     'cidr_block': '',
        #     'gateway_id': '',
        # }

    def __str__(self):
        the_str = ''
        if not self.inserted:
            the_str += super(Route, self).__str__()
        the_str += '\n\troute {\n'
        for attr in self.__dict__.items():
            if attr[0] not in ('type', 'label', 'name', 'tags', 'inserted') and attr[1]:
                if self.inserted and attr[0] != 'vpc_id':
                    the_str += self.format_item(attr[0], attr[1], 2)
        the_str += '\t}\n'
        if not self.inserted:
            the_str += self.end_block
        return the_str


class RouteTable(BlockTypeWithTagsBase):
    """
    Route Table
    """
    def __init__(self, name=''):
        """
        Create new object, called automatically

        :param name:
        """
        super(RouteTable, self).__init__(
            btype='resource', label='aws_route_table', name=name)
        self.route_table_id = ''
        self.routes = []

    def __str__(self):
        the_str = super(RouteTable, self).__str__()
        the_str += self.format_item('vpc_id', self.vpc_id)
        for route in self.routes:
            the_str += str(route)
        if self.tags:
            the_str += self.add_str_tags()
        the_str += self.end_block
        return the_str


class RouteTableAssociation(BlockTypeWithTagsBase):
    """
    Route Table Association
    """
    def __init__(self, name=''):
        """
        Create new object, called automatically

        :param name:
        """
        super(RouteTableAssociation, self).__init__(
            btype='resource', label='aws_route_table_association', name=name)
        self.subnet_id = ''
        self.route_table_id = ''

    def __str__(self):
        the_str = super(RouteTableAssociation, self).__str__()
        the_str += self.format_item('subnet_id', self.subnet_id)
        the_str += self.format_item('route_table_id', self.route_table_id)
        the_str += self.end_block
        return the_str


"""
Security
"""


class Gress(BaseFormat):
    """
    Gress part
    """
    def __init__(self, _=''):
        """
        Create new object, called automatically

        :param _:
        """
        self.rule_no = 100
        self.action = 'deny'
        self.cidr_block = "0.0.0.0/0"
        self.ipv6_cidr_block = ""
        self.protocol = "-1"
        self.from_port = 0
        self.to_port = 0
        self.icmp_code = 0
        self.icmp_type = 0

    def __str__(self):
        the_str = '\t\trule_no = "' + str(self.rule_no) + '"\n'
        the_str += '\t\taction = "' + self.action + '"\n'
        the_str += '\t\tcidr_block = "' + self.cidr_block + '"\n'
        the_str += '\t\tipv6_cidr_block = "' + self.ipv6_cidr_block + '"\n'
        the_str += '\t\tprotocol = "' + self.protocol + '"\n'
        the_str += '\t\tfrom_port = ' + str(self.from_port) + '\n'
        the_str += '\t\tto_port = ' + str(self.to_port) + '\n'
        the_str += '\t\ticmp_code = ' + str(self.icmp_code) + '\n'
        the_str += '\t\ticmp_type = ' + str(self.icmp_type) + '\n'
        return the_str


class NetworkAcl(BlockTypeWithTagsBase):
    """
    Network Acl
    """
    def __init__(self, name=''):
        """
        Create new object, called automatically

        :param name:
        """
        super(NetworkAcl, self).__init__(
            btype='resource', label='aws_network_acl', name=name)
        self.vpc_id = ''
        self.subnet_ids = []
        self.ingress = []
        self.egress = []

    def __str__(self):
        if any('Name' in d['Key'] for d in self.tags):
            for tag in self.tags:
                if tag['Key'] == 'Name' and tag['Value'] not in self.name:
                    self.name = self.name + '-' + tag['Value']
                    break
        the_str = super(NetworkAcl, self).__str__()
        the_str += self.format_item('vpc_id', self.vpc_id)
        the_str += '\tsubnet_ids = ['
        for sid in self.subnet_ids:
            the_str += '"' + sid + '",'
        the_str += ']\n\n'
        the_str += self.format_items('ingress', self.ingress)
        the_str += self.format_items('egress', self.egress)
        the_str += self.add_str_tags()
        the_str += self.end_block
        return the_str


class SecurityGroup(BlockTypeBase):
    """
    Security Group
    """
    def __init__(self, name=''):
        """
        Create new object, called automatically

        :param name:
        """
        super(SecurityGroup, self).__init__(
            btype='resource', label='aws_security_group', name=name)
        self.name = ''
        self.description = ''
        self.vpc_id = ''
        self.ingress = []
        self.egress = []

    def __str__(self):
        the_str = super(SecurityGroup, self).__str__()
        the_str += self.format_item('name', self.name)
        the_str += self.format_item('description', self.description)
        the_str += self.format_item('vpc_id', self.vpc_id)
        the_str += self.format_items('ingress', self.ingress)
        the_str += self.format_items('egress', self.egress)
        the_str += '}\n'
        return the_str


"""
Lambda
"""


class Lambda(BlockTypeWithTagsBase):
    """
    Lambda
    https://www.terraform.io/docs/providers/aws/r/lambda_function.html
    """
    def __init__(self, name=''):
        """
        Create new object, called automatically

        :param name:
        """
        super(Lambda, self).__init__(
            btype='resource', label='aws_lambda_function', name=name)
        self.ofilename = None
        self.os3_bucket = None
        self.os3_key = None
        self.os3_object_version = None
        self.function_name = ''
        self.odead_letter_config = None
        self.handler = ''
        self.role = ''
        self.odescription = None
        self.olayers = None
        self.omemory_size = None
        self.runtime = ''
        self.otimeout = None
        self.oreserved_concurrent_executions = None
        self.opublish = None
        self.ovpc_config = None
        self.oenvironment = None
        self.okms_key_arn = None
        self.osource_code_hash = None
        self.otarget_arn = None
        self.osubnet_ids = None
        self.osecurity_group_ids = None

    def __str__(self):
        the_str = super(Lambda, self).__str__()
        the_str += self.format_item_cond('filename', self.ofilename)
        the_str += self.format_item_cond('s3_bucket', self.os3_bucket)
        the_str += self.format_item_cond('s3_key', self.os3_key)
        the_str += self.format_item_cond('s3_object_version', self.os3_object_version)
        the_str += self.format_item('function_name', self.function_name)
        the_str += self.format_item_cond('dead_letter_config', self.odead_letter_config)
        the_str += self.format_item('handler', self.handler)
        the_str += self.format_item('role', self.role)
        the_str += self.format_item_cond('description', self.odescription)
        if self.olayers:
            the_str += '\tlayers = {' + self.olayers + '"\n'
            for layer in self.olayers:
                the_str += '\t\t"' + layer['Key'] + '" = "' + layer['Value'] + '"\n'
            the_str += '\t}\n'
        the_str += self.format_item_cond('memory_size', str(self.omemory_size))
        the_str += self.format_item('runtime', self.runtime)
        the_str += self.format_item_cond('timeout', str(self.otimeout))
        the_str += self.format_item_cond('reserved_concurrent_executions',
                                         self.oreserved_concurrent_executions)
        the_str += self.format_item_cond('publish', self.opublish)
        the_str += self.format_item_cond('vpc_config', self.ovpc_config)
        if self.oenvironment:
            the_str += '\tenvironment {\n'
            the_str += '\n\tvariables = {\n'
            for k, v in self.oenvironment.items():
                the_str += '\t\t"' + k + '" = "' + v + '"\n'
            the_str += '\t}\n'
        the_str += self.format_item_cond('kms_key_arn', self.okms_key_arn)
        the_str += self.format_item_cond('source_code_hash', self.osource_code_hash)
        the_str += self.add_str_tags()
        the_str += self.format_item_cond('target_arn', self.otarget_arn)
        the_str += self.format_item_cond('subnet_ids', self.osubnet_ids)
        the_str += self.format_item_cond('security_group_ids', self.osecurity_group_ids)
        the_str += '}\n'
        return the_str


"""
S3
"""


class S3Bucket(BlockTypeBase):
    """
    S3 Bucket
    https://www.terraform.io/docs/providers/aws/r/s3_bucket.html
    """
    def __init__(self, name=''):
        """
        Create new object, called automatically

        :param name:
        """
        super(S3Bucket, self).__init__(
            btype='resource', label='aws_s3_bucket', name=name)
        self.bucket = None
        self.acl = None
        self.versioning = None

    def __str__(self):
        the_str = super(S3Bucket, self).__str__()
        the_str += self.format_item('bucket', self.bucket)
        the_str += self.format_item_cond('acl', self.acl)
        if self.versioning:
            the_str += '\tversioning {\n\t\tenabled = ' + self.versioning + '\n\t}\n'
        the_str += self.end_block
        # todo
        return the_str


"""
ApiGatewayV2
"""


class ApiGatewayV2(BlockTypeBase):
    """
    Api Gateway V2
    todo
    """
    def __init__(self, name=''):
        """
        Create new object, called automatically

        :param name:
        """
        super(ApiGatewayV2, self).__init__(
            btype='resource', label='aws_api_gateway_v2', name=name)
        # todo

    def __str__(self):
        the_str = super(ApiGatewayV2, self).__str__()
        the_str += self.end_block
        # todo
        return the_str