Add tutorial code for generating emails

main
sgoudham 4 years ago
parent 5b5406f915
commit 30ed9b47e3
No known key found for this signature in database
GPG Key ID: EF51A29A50FB754C

@ -0,0 +1 @@
# Cuckoo

@ -0,0 +1,151 @@
# This file should help you with the commands we'll be working through in this module.
# I'll also include documentation and notes where I think they might be useful.
# Commands will be unindented and uncommented (no # symbol at the beginning)
# Part 1 - S3 Uploads
# When you open up your demos file you'll see a templates folder.
# Along with licence information are three modified html email templates.
# Each template is ready to use with Jinja2 but we'll need to move it to an S3 bucket.
# Keep in mind that bucket names are *globally* unique. Meaning you can't use mine!
# Replace gpc-email-templates with your bucket name.
# Also use the region you've been using throughout the course.
# The \ character is to escape line breaks and make the code more readable
# Here is the command to create a bucket with the AWS CLI:
aws s3 mb s3://gpc-email-templates
# Confirm the bucket was created
aws s3 ls
# Upload the templates to your s3 bucket
# Make sure you're in the /templates folder in your terminal first
aws s3 cp ./templates s3://gpc-email-templates --recursive
# Confirm the files were moved to s3
aws s3 ls gpc-email-templates
# Notes:
# Keep in mind that Lambda cron() scheduling is done on UTC.
# To get a cron expression in your timezone you may need to change the values
# 7am EST - cron(0 12 ? * MON-FRI *)
# Noon EST - cron(0 17 ? * MON-FRI *)
# 5pm EST - cron(0 22 ? * MON-FRI *)
# Keep in mind time zone differences like Daylight Savings!
# To add CloudWatch Events rules:
aws events put-rule \
--name come_to_work \
--schedule-expression 'cron(0 12 ? * MON-FRI *)'
aws events put-rule \
--name daily_tasks \
--schedule-expression 'cron(0 17 ? * MON-FRI *)'
aws events put-rule \
--name pickup \
--schedule-expression 'cron(0 22 ? * MON-FRI *)'
# With the events created make sure to modify your code skeleton
# Come back for these commands when you need to test locally
# Testing the function from the command line:
# Make sure you're in the directory with the cuckoo.py file
# Open a Python shell
python3
# In the python shell I'll put commands after >>>
# import the function
>>> import cuckoo
# Call the handler function and give it a sample event
# This event can either be the event ARN text or something else
# that contains text that contains some of the text that the
# function is looking for.
# The text should contain:
# 'come_to_work' or 'daily_tasks' or 'pickup'
# We'll also put them into the following format to match
# the way they will be passed to the function by the AWS event
>>> cuckoo.handler({'resources':['some text containing the word pickup']},'context')
>>> cuckoo.handler({'resources':['arn:aws:events:us-east-1:444510800003:rule/come_to_work']}, 'context')
>>> cuckoo.handler({'resources':['a great daily_tasks test']},'context')
# If successful you won't see any output so check your email for the results!
# Setup and Deployment
#
# To take a look at creating our function package locally you can open
# the setup.sh file in a text editor.
#
# Deploying your function with the AWS CLI
# First make sure you've created your zip function package
# and you're in the same directory as that package.zip file
# Then list the roles on your account.
# We'll need to look for the role we created for this function earlier
aws iam list-roles
# We'll see something like this in the output:
# arn:aws:iam::444510800003:role/gpc_cuckoo_role
# Copy that role ARN because you'll need to replace the value
# after --role in the below command:
aws lambda create-function \
--function-name gpc_cuckoo \
--runtime python3.7 \
--role arn:aws:iam::444510800003:role/gpc_cuckoo_role \
--handler cuckoo.handler \
--zip-file fileb://./package.zip
# Configure events to trigger the function
#
#
# List events
aws events list-rules
# We'll have the function trust events from each
# of these Cloudwatch Events
aws lambda add-permission \
--function-name gpc_cuckoo \
--statement-id 1 \
--action 'lambda:InvokeFunction' \
--principal events.amazonaws.com \
--source-arn arn:aws:events:us-east-1:444510800003:rule/pickup
aws lambda add-permission \
--function-name gpc_cuckoo \
--statement-id 2 \
--action 'lambda:InvokeFunction' \
--principal events.amazonaws.com \
--source-arn arn:aws:events:us-east-1:444510800003:rule/daily_tasks
aws lambda add-permission \
--function-name gpc_cuckoo \
--statement-id 3 \
--action 'lambda:InvokeFunction' \
--principal events.amazonaws.com \
--source-arn arn:aws:events:us-east-1:444510800003:rule/come_to_work
# This will add the lambda function to each of the rules to be triggered
aws events put-targets \
--rule daily_tasks \
--targets '{"Id" : "1", "Arn": "arn:aws:lambda:us-east-1:444510800003:function:gpc_cuckoo"}'
aws events put-targets \
--rule come_to_work \
--targets '{"Id" : "1", "Arn": "arn:aws:lambda:us-east-1:444510800003:function:gpc_cuckoo"}'
aws events put-targets \
--rule pickup \
--targets '{"Id" : "1", "Arn": "arn:aws:lambda:us-east-1:444510800003:function:gpc_cuckoo"}'

@ -0,0 +1,159 @@
import datetime
import boto3
from jinja2 import Template
# Start of some things you need to change
#
#
# Recipient emails or domains in the AWS Email Sandbox must be verified
# You'll want to change this to the email you verify in SES
FROM_ADDRESS = 'globomanticspetcare@gmail.com'
REPLY_TO_ADDRESS = 'globomanticspetcare@gmail.com'
CLIENTS = [
{
# You'll need to verify this email
'email': 'f_mcorey@yahoo.com',
'first_name': 'Fernando',
'last_name': 'Medina Corey',
'pet_name': 'Riley'
},
]
EMPLOYEES = [
{
# You'll need to verify this email
'email': 'springfield.homer@yahoo.com',
'first_name': 'Homer',
'last_name': 'Simpson'
},
]
# Change to the bucket you create on your AWS account
TEMPLATE_S3_BUCKET = 'gpc-email-templates'
#
#
# End of things you need to change
def get_template_from_s3(key):
"""Loads and returns html template from Amazon S3"""
s3 = boto3.client('s3')
s3_file = s3.get_object(
Bucket=TEMPLATE_S3_BUCKET,
Key=key
)
try:
template = Template(s3_file['Body'].read().decode('utf-8'))
except Exception as e:
print('Failed to load template')
raise e
return template
def render_come_to_work_template(employee_first_name):
template = get_template_from_s3('come_to_work.html')
html_email = template.render(first_name=employee_first_name)
plaintext_email = 'Hello {0}, \nPlease remember to be into work by 8am'.format(employee_first_name)
return html_email, plaintext_email
def render_daily_tasks_template():
template = get_template_from_s3('daily_tasks.html')
tasks = {
'Monday': '- Clean the dog areas\n',
'Tuesday': '- Clean the cat areas\n',
'Wednesday': '- Feed the aligator\n',
'Thursday': '- Clean the dog areas\n',
'Friday': '- Clean the cat areas\n',
'Saturday': '- Relax! Play with the puppies! It\'s the weekend!',
'Sunday': '- Relax! Play with the puppies! It\'s the weekend!'
}
# Gets an integer value from 0 to 6 for today (Monday - Sunday)
# Keep in mind this will run in GMT and you will need to adjust runtimes accordingly
days = ['Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday', 'Sunday']
today = days[datetime.date.today().weekday()]
html_email = template.render(
day_of_week=today,
daily_tasks=tasks[today]
)
plaintext_email = (
"Remember to do all of these today:\n"
"- Feed the dogs\n"
"- Feed the rabbits\n"
"- Feed the cats\n"
"- Feed the turtles\n"
"- Walk the dogs\n"
"- Empty cat litterboxes\n"
"{0}".format(tasks[today])
)
return html_email, plaintext_email
def render_pickup_template(client_first_name, client_pet_name):
template = get_template_from_s3('pickup.html')
html_email = template.render(
first_name=client_first_name,
pet_name=client_pet_name
)
plaintext_email = (
'Hello {0}, \nPlease remember to '
'pickup {1} by 7pm!'.format(
client_first_name,
client_pet_name
)
)
return html_email, plaintext_email
def send_email(html_email, plaintext_email, subject, recipients):
try:
ses = boto3.client('ses')
response = ses.send_email(
Source=FROM_ADDRESS,
Destination={
'ToAddresses': [recipients],
'CcAddresses': [],
'BccAddresses': []
},
Message={
'Subject': {
'Data': subject,
},
'Body': {
'Text': {
'Data': plaintext_email
},
'Html': {
'Data': html_email
}
}
},
ReplyToAddresses=[
REPLY_TO_ADDRESS,
]
)
except Exception as e:
print('Failed to send message via SES')
print(e)
raise e
def handler(event, context):
event_trigger = event['resources'][0]
print('event triggered by ' + event_trigger)
if 'come_to_work' in event_trigger:
for employee in EMPLOYEES:
html_email, plaintext_email = render_come_to_work_template(employee['first_name'])
send_email(html_email, plaintext_email, 'Work Schedule Reminder', employee['email'])
elif 'daily_tasks' in event_trigger:
for employee in EMPLOYEES:
html_email, plaintext_email = render_daily_tasks_template()
send_email(html_email, plaintext_email, 'Daily Tasks Reminder', employee['email'])
elif 'pickup' in event_trigger:
for client in CLIENTS:
html_email, plaintext_email = render_pickup_template(client['first_name'], client['pet_name'])
send_email(html_email, plaintext_email, 'Pickup Reminder', client['email'])
else:
return 'No template for this trigger!'

@ -0,0 +1,2 @@
Jinja2==2.11.1
MarkupSafe==1.1.1

@ -0,0 +1,35 @@
#!/bin/bash
# Create and initialize a Python Virtual Environment
echo "Creating virtual environment - .venv"
python3 -m venv .venv
echo "sourcing virtual environment - .venv"
source .venv/bin/activate
# Create a directory to put things in
echo "Creating 'setup' directory"
mkdir setup
# Move the relevant files into setup directory
echo "Moving function file(s) to setup dir"
cp cuckoo.py setup/
cd ./setup
# Install requirements
echo "pip installing requirements from requirements file in target directory"
pip install -r ../requirements.txt -t .
# Prepares the deployment package
echo "Zipping package"
zip -r ../package.zip ./*
# Remove the setup directory used
echo "Removing setup directory and virtual environment"
cd ..
rm -r ./setup
deactivate
rm -r .venv
# changing dirs back to dir from before
echo "Opening folder containg function package - 'package.zip'"
open .

@ -0,0 +1,37 @@
rem Create and initialize a Python Virtual Environment
echo "Creating the virtual environment - .venv"
python3 -m venv .venv
echo "starting the virtual environment - .venv"
call .venv\Scripts\activate.bat
rem Create a directory to put things in
echo "Creating 'setup' directory"
mkdir setup
rem Move the relevant files into setup directory
echo "Moving function file(s) to setup dir"
xcopy cuckoo.py setup\ /Q /R /Y
cd .\setup
rem Install requirements
echo "pip installing requirements from requirements file in target directory"
pip install -r ..\requirements.txt -t .
rem Prepares the deployment package
echo "Setting up your 7zip PATH - This assumes the installation location of 7zip is in C:\Program Files\7-Zip\"
set PATH=%PATH%;C:\Program Files\7-Zip\
echo "Zipping package"
7z a -r ..\package.zip .\*
rem Remove the setup directory used
echo "Removing setup directory and virtual environment"
cd ..
rd /Q /S .\setup
call .venv\Scripts\deactivate.bat
rd /Q /S .\.venv
rem changing dirs back to dir from before
echo "Opening folder containg function package - 'package.zip'"
explorer .
Loading…
Cancel
Save