Skip to content

module/awsses.py

module.awsses

AWS SDK API Integration

Integration S3.client, SES.client

AWSS3

AWSS3

Parameters:

Name Type Description Default
aws_access_key_id str

aws_access_key_id.

required
aws_secret_access_key str

aws_secret_access_key.

required
bucket str

bucket name.

required
Source code in module/awsses.py
class AWSS3:
    ''' AWSS3

    Args:
        aws_access_key_id (str): aws_access_key_id.
        aws_secret_access_key (str): aws_secret_access_key.
        bucket (str): bucket name.

    '''
    __slots__ = ('client', 'bucket')

    def __init__(self, aws_access_key_id: str, aws_secret_access_key: str, bucket: str) -> None:
        self.client = boto3.client('s3',
                                   aws_access_key_id=aws_access_key_id,
                                   aws_secret_access_key=aws_secret_access_key,
                                   )
        self.bucket = bucket

    def get_object(self, key: str) -> Any:
        ''' Get object

        Args:
            key (str): Object key.

        Returns:
            Return the object. -> [S3.Client.get_object][]

        '''
        return self.client.get_object(Bucket=self.bucket, Key=key)

    def convert_to_attachment(self, key: str) -> MIMEBase:
        ''' Convert the object to match attachment

        Args:
            key (str): Object key.

        Returns:
            Get the object and turn into attachment format to use.

        '''
        s3object = self.get_object(key)
        attachment = MIMEBase(
            s3object['ContentType'].split('/')[0],
            f'''{s3object['ContentType'].split('/')[1]}; name="{Charset('utf-8').header_encode(basename(key))}"'''  # pylint: disable=line-too-long
        )
        attachment.add_header(
            'Content-Disposition',
            f'''attachment; filename="{Charset('utf-8').header_encode(basename(key))}"''')
        attachment.set_payload(s3object['Body'].read())
        encoders.encode_base64(attachment)

        return attachment

convert_to_attachment

convert_to_attachment(key: str) -> MIMEBase

Convert the object to match attachment

Parameters:

Name Type Description Default
key str

Object key.

required

Returns:

Type Description
MIMEBase

Get the object and turn into attachment format to use.

Source code in module/awsses.py
def convert_to_attachment(self, key: str) -> MIMEBase:
    ''' Convert the object to match attachment

    Args:
        key (str): Object key.

    Returns:
        Get the object and turn into attachment format to use.

    '''
    s3object = self.get_object(key)
    attachment = MIMEBase(
        s3object['ContentType'].split('/')[0],
        f'''{s3object['ContentType'].split('/')[1]}; name="{Charset('utf-8').header_encode(basename(key))}"'''  # pylint: disable=line-too-long
    )
    attachment.add_header(
        'Content-Disposition',
        f'''attachment; filename="{Charset('utf-8').header_encode(basename(key))}"''')
    attachment.set_payload(s3object['Body'].read())
    encoders.encode_base64(attachment)

    return attachment

get_object

get_object(key: str) -> Any

Get object

Parameters:

Name Type Description Default
key str

Object key.

required

Returns:

Type Description
Any

Return the object. -> S3.Client.get_object

Source code in module/awsses.py
def get_object(self, key: str) -> Any:
    ''' Get object

    Args:
        key (str): Object key.

    Returns:
        Return the object. -> [S3.Client.get_object][]

    '''
    return self.client.get_object(Bucket=self.bucket, Key=key)

AWSSES

AWSSES

Parameters:

Name Type Description Default
aws_access_key_id str

aws_access_key_id.

required
aws_secret_access_key str

aws_secret_access_key.

required
source dict

{'name': <str>, 'mail': <str>}, for mail FROM.

required
Source code in module/awsses.py
class AWSSES:
    ''' AWSSES

    Args:
        aws_access_key_id (str): aws_access_key_id.
        aws_secret_access_key (str): aws_secret_access_key.
        source (dict): `{'name': <str>, 'mail': <str>}`, for mail `FROM`.

    '''

    __slots__ = ('client', 'source')

    def __init__(self, aws_access_key_id: str, aws_secret_access_key: str,
                 source: dict[str, str]) -> None:
        self.client = boto3.client('ses',
                                   aws_access_key_id=aws_access_key_id,
                                   aws_secret_access_key=aws_secret_access_key,
                                   region_name='us-east-1')
        self.source = source

    @staticmethod
    def format_mail(name: str, mail: str) -> str:
        ''' Encode the `user`, `mail` to base64 for Email-Headers format.

            Args:
                name (str): name.
                mail (str): mail address.

            Returns:
                Return the excoded string.

        '''
        if name:
            return formataddr((name, mail))

        return mail

    def send_email(self, *args: Any, **kwargs: Any) -> Any:
        ''' Send mail

        ``*args``, ``**kwargs`` are the same with [SES.Client.send_email][]

        See Also:
            [SES.Client.send_email][]

        '''
        return self.client.send_email(*args, **kwargs)

    def raw_mail(self, **kwargs: Any) -> Any:
        ''' To make raw mail content

        Args:
            **kwargs:

                - to_addresses (list): List of user/mail.
                - subject (str): Mail subject.
                - body (str): Body content in `text/html`
                - text_body (str): Body content in `text/plain`
                - x_coscup (str): *Optional* For Email-Headers `X-Coscup`
                - cc_addresses (str): *Optional* List of user/mail for Cc.
                - attachment (MIMEBase): *Optional* Using with
                    [module.awsses.AWSS3.convert_to_attachment][].
                - list_unsubscribe (str): *Optional* An mail or link for user
                    to unsubscribe.

        Returns:
            [email.mime.multipart.MIMEMultipart][]

        Tips:
            This function return the mail object, the next step is using
            [module.awsses.AWSSES.send_raw_email][] to send.

        '''
        msg_all = MIMEMultipart('mixed')

        msg_all['From'] = self.format_mail(
            self.source['name'], self.source['mail'])

        to_list = []
        for to_user in kwargs['to_addresses']:
            to_list.append(self.format_mail(to_user['name'], to_user['mail']))
        msg_all['To'] = ','.join(to_list)

        cc_list = []
        if 'cc_addresses' in kwargs and kwargs['cc_addresses']:
            for cc_user in kwargs['cc_addresses']:
                cc_list.append(self.format_mail(
                    cc_user['name'], cc_user['mail']))

            if cc_list:
                msg_all['Cc'] = ','.join(cc_list)

        msg_all['Subject'] = kwargs['subject']

        if 'x_coscup' in kwargs and kwargs['x_coscup']:
            msg_all['X-Coscup'] = kwargs['x_coscup']

        if 'list_unsubscribe' in kwargs and kwargs['list_unsubscribe']:
            msg_all['List-Unsubscribe'] = kwargs['list_unsubscribe']

        body_mine = MIMEMultipart('alternative')

        if 'text_body' in kwargs and kwargs['text_body']:
            body_mine.attach(MIMEText(kwargs['text_body'], 'plain', 'utf-8'))

        body_mine.attach(MIMEText(kwargs['body'], 'html', 'utf-8'))

        msg_all.attach(body_mine)

        if 'attachment' in kwargs and kwargs['attachment']:
            for attach in kwargs['attachment']:
                msg_all.attach(attach)

        return msg_all

    def send_raw_email(self, **kwargs: Any) -> Any:
        ''' send raw email
        If the `data` has generated by [module.awsses.AWSSES.raw_mail][],
            directly put into `data`.
        If the `data` is empty, the attributes are the same
            with [module.awsses.AWSSES.raw_mail][].

        What is the difference between `data`, `data_str`. The `data` is the
        type of [email.mime.multipart.MIMEMultipart][], the `data_str` is the
        type of [email.message.Message.as_string][].

        Args:
            **kwargs: The same with [module.awsses.AWSSES.raw_mail][].

        Returns:
            dict

        See Also:
            [SES.Client.send_raw_email][]

        '''
        if 'data_str' in kwargs:
            return self.client.send_raw_email(
                RawMessage={'Data': kwargs['data_str']})

        data = kwargs.get('data')
        if not data:
            data = self.raw_mail(**kwargs)

        return self.client.send_raw_email(
            RawMessage={'Data': data.as_string()})

format_mail staticmethod

format_mail(name: str, mail: str) -> str

Encode the user, mail to base64 for Email-Headers format.

Parameters:

Name Type Description Default
name str

name.

required
mail str

mail address.

required

Returns:

Type Description
str

Return the excoded string.

Source code in module/awsses.py
@staticmethod
def format_mail(name: str, mail: str) -> str:
    ''' Encode the `user`, `mail` to base64 for Email-Headers format.

        Args:
            name (str): name.
            mail (str): mail address.

        Returns:
            Return the excoded string.

    '''
    if name:
        return formataddr((name, mail))

    return mail

raw_mail

raw_mail(**kwargs: Any) -> Any

To make raw mail content

Parameters:

Name Type Description Default
**kwargs Any
  • to_addresses (list): List of user/mail.
  • subject (str): Mail subject.
  • body (str): Body content in text/html
  • text_body (str): Body content in text/plain
  • x_coscup (str): Optional For Email-Headers X-Coscup
  • cc_addresses (str): Optional List of user/mail for Cc.
  • attachment (MIMEBase): Optional Using with module.awsses.AWSS3.convert_to_attachment.
  • list_unsubscribe (str): Optional An mail or link for user to unsubscribe.
{}

Returns:

Type Description
Any
Tips

This function return the mail object, the next step is using module.awsses.AWSSES.send_raw_email to send.

Source code in module/awsses.py
def raw_mail(self, **kwargs: Any) -> Any:
    ''' To make raw mail content

    Args:
        **kwargs:

            - to_addresses (list): List of user/mail.
            - subject (str): Mail subject.
            - body (str): Body content in `text/html`
            - text_body (str): Body content in `text/plain`
            - x_coscup (str): *Optional* For Email-Headers `X-Coscup`
            - cc_addresses (str): *Optional* List of user/mail for Cc.
            - attachment (MIMEBase): *Optional* Using with
                [module.awsses.AWSS3.convert_to_attachment][].
            - list_unsubscribe (str): *Optional* An mail or link for user
                to unsubscribe.

    Returns:
        [email.mime.multipart.MIMEMultipart][]

    Tips:
        This function return the mail object, the next step is using
        [module.awsses.AWSSES.send_raw_email][] to send.

    '''
    msg_all = MIMEMultipart('mixed')

    msg_all['From'] = self.format_mail(
        self.source['name'], self.source['mail'])

    to_list = []
    for to_user in kwargs['to_addresses']:
        to_list.append(self.format_mail(to_user['name'], to_user['mail']))
    msg_all['To'] = ','.join(to_list)

    cc_list = []
    if 'cc_addresses' in kwargs and kwargs['cc_addresses']:
        for cc_user in kwargs['cc_addresses']:
            cc_list.append(self.format_mail(
                cc_user['name'], cc_user['mail']))

        if cc_list:
            msg_all['Cc'] = ','.join(cc_list)

    msg_all['Subject'] = kwargs['subject']

    if 'x_coscup' in kwargs and kwargs['x_coscup']:
        msg_all['X-Coscup'] = kwargs['x_coscup']

    if 'list_unsubscribe' in kwargs and kwargs['list_unsubscribe']:
        msg_all['List-Unsubscribe'] = kwargs['list_unsubscribe']

    body_mine = MIMEMultipart('alternative')

    if 'text_body' in kwargs and kwargs['text_body']:
        body_mine.attach(MIMEText(kwargs['text_body'], 'plain', 'utf-8'))

    body_mine.attach(MIMEText(kwargs['body'], 'html', 'utf-8'))

    msg_all.attach(body_mine)

    if 'attachment' in kwargs and kwargs['attachment']:
        for attach in kwargs['attachment']:
            msg_all.attach(attach)

    return msg_all

send_email

send_email(*args: Any, **kwargs: Any) -> Any

Send mail

*args, **kwargs are the same with SES.Client.send_email

See Also

SES.Client.send_email

Source code in module/awsses.py
def send_email(self, *args: Any, **kwargs: Any) -> Any:
    ''' Send mail

    ``*args``, ``**kwargs`` are the same with [SES.Client.send_email][]

    See Also:
        [SES.Client.send_email][]

    '''
    return self.client.send_email(*args, **kwargs)

send_raw_email

send_raw_email(**kwargs: Any) -> Any

send raw email If the data has generated by module.awsses.AWSSES.raw_mail, directly put into data. If the data is empty, the attributes are the same with module.awsses.AWSSES.raw_mail.

What is the difference between data, data_str. The data is the type of email.mime.multipart.MIMEMultipart, the data_str is the type of email.message.Message.as_string.

Parameters:

Name Type Description Default
**kwargs Any {}

Returns:

Type Description
Any

dict

See Also

SES.Client.send_raw_email

Source code in module/awsses.py
def send_raw_email(self, **kwargs: Any) -> Any:
    ''' send raw email
    If the `data` has generated by [module.awsses.AWSSES.raw_mail][],
        directly put into `data`.
    If the `data` is empty, the attributes are the same
        with [module.awsses.AWSSES.raw_mail][].

    What is the difference between `data`, `data_str`. The `data` is the
    type of [email.mime.multipart.MIMEMultipart][], the `data_str` is the
    type of [email.message.Message.as_string][].

    Args:
        **kwargs: The same with [module.awsses.AWSSES.raw_mail][].

    Returns:
        dict

    See Also:
        [SES.Client.send_raw_email][]

    '''
    if 'data_str' in kwargs:
        return self.client.send_raw_email(
            RawMessage={'Data': kwargs['data_str']})

    data = kwargs.get('data')
    if not data:
        data = self.raw_mail(**kwargs)

    return self.client.send_raw_email(
        RawMessage={'Data': data.as_string()})