update event

This commit is contained in:
2025-09-20 14:37:36 -03:00
parent 0501dc8f16
commit 6d352ab3a3
14 changed files with 200 additions and 378 deletions

View File

@@ -0,0 +1,102 @@
from typing import TYPE_CHECKING, TypedDict
from aws_lambda_powertools import Logger
from layercake.dateutils import now
from layercake.dynamodb import DynamoDBPersistenceLayer
from layercake.email_ import Message
from layercake.strutils import first_word, truncate_str
from . import (
reminder_access_period_before_30_days,
reminder_cert_expiration_before_30_days,
reminder_no_access_after_3_days,
reminder_no_activity_after_7_days,
)
if TYPE_CHECKING:
from mypy_boto3_sesv2 import SESV2Client
else:
SESV2Client = object
Event = TypedDict('Event', {'id': str, 'sk': str})
TEMPLATES = {
'reminder_access_period_before_30_days': {
'subject': reminder_access_period_before_30_days.SUBJECT,
'message': reminder_access_period_before_30_days.MESSAGE,
},
'reminder_cert_expiration_before_30_days': {
'subject': reminder_cert_expiration_before_30_days.SUBJECT,
'message': reminder_cert_expiration_before_30_days.MESSAGE,
},
'reminder_no_access_after_3_days': {
'subject': reminder_no_access_after_3_days.SUBJECT,
'message': reminder_no_access_after_3_days.MESSAGE,
},
'reminder_no_activity_after_7_days': {
'subject': reminder_no_activity_after_7_days.SUBJECT,
'message': reminder_no_activity_after_7_days.MESSAGE,
},
}
logger = Logger(__name__)
def send_email(
to: tuple[str, str],
subject: str,
message: str,
context: dict = {},
*,
sender: tuple[str, str],
event: Event,
sesv2_client: SESV2Client,
dynamodb_persistence_layer: DynamoDBPersistenceLayer,
) -> bool:
now_ = now()
name, _ = to
event_name = event['sk']
emailmsg = Message(
from_=sender,
to=to,
subject=subject.format(
course=truncate_str(context['course']),
),
)
emailmsg.add_alternative(
message.format(
first_name=first_word(name),
course=context['course'],
)
)
try:
sesv2_client.send_email(
Content={
'Raw': {
'Data': emailmsg.as_bytes(),
},
}
)
dynamodb_persistence_layer.put_item(
item={
'id': event['id'],
'sk': f'{event_name}#EXECUTED',
'created_at': now_,
}
)
logger.info('Email sent')
except Exception as exc:
logger.exception(exc)
dynamodb_persistence_layer.put_item(
item={
'id': event['id'],
'sk': f'{event_name}#FAILED',
'created_at': now_,
}
)
return False
else:
return True

View File

@@ -1,76 +0,0 @@
from typing import TYPE_CHECKING, TypedDict
from aws_lambda_powertools import Logger
from layercake.dateutils import now
from layercake.dynamodb import DynamoDBPersistenceLayer
from layercake.email_ import Message
from layercake.strutils import first_word, truncate_str
logger = Logger(__name__)
if TYPE_CHECKING:
from mypy_boto3_sesv2 import SESV2Client
else:
SESV2Client = object
Event = TypedDict('Event', {'id': str, 'sk': str})
def send_email(
to: tuple[str, str],
subject: str,
message: str,
context: dict = {},
*,
sender: tuple[str, str],
event: Event,
sesv2_client: SESV2Client,
dynamodb_persistence_layer: DynamoDBPersistenceLayer,
) -> bool:
now_ = now()
name, _ = to
event_name = event['sk']
emailmsg = Message(
from_=sender,
to=to,
subject=subject.format(
course=truncate_str(context['course']),
),
)
emailmsg.add_alternative(
message.format(
first_name=first_word(name),
course=context['course'],
)
)
try:
sesv2_client.send_email(
Content={
'Raw': {
'Data': emailmsg.as_bytes(),
},
}
)
dynamodb_persistence_layer.put_item(
item={
'id': event['id'],
'sk': f'{event_name}#EXECUTED',
'created_at': now_,
}
)
logger.info('Email sent')
except Exception as exc:
logger.exception(exc)
dynamodb_persistence_layer.put_item(
item={
'id': event['id'],
'sk': f'{event_name}#FAILED',
'created_at': now_,
}
)
return False
else:
return True

View File

@@ -1,22 +1,4 @@
from aws_lambda_powertools import Logger
from aws_lambda_powertools.utilities.data_classes import (
EventBridgeEvent,
event_source,
)
from aws_lambda_powertools.utilities.typing import LambdaContext
from layercake.dynamodb import DynamoDBPersistenceLayer, KeyPair
from boto3clients import dynamodb_client, sesv2_client
from config import (
EMAIL_SENDER,
ENROLLMENT_TABLE,
)
from .email_ import send_email
logger = Logger(__name__)
enrollment_layer = DynamoDBPersistenceLayer(ENROLLMENT_TABLE, dynamodb_client)
# 30 days before the course access period ends
SUBJECT = 'Seu acesso ao curso {course} termina em 30 dias'
MESSAGE = """
@@ -27,34 +9,3 @@ Conclua dentro desse prazo para garantir sua certificação.<br/><br/>
<a href="https://saladeaula.digital">👉 Acesse agora seu curso</a>
"""
@event_source(data_class=EventBridgeEvent)
@logger.inject_lambda_context
def lambda_handler(event: EventBridgeEvent, context: LambdaContext) -> bool:
"""30 days before the course access period ends."""
old_image = event.detail['old_image']
# Post-migration: Remove the following lines
if 'email' not in old_image:
# If email is missing, use enrollment email
cur_image = enrollment_layer.get_item(KeyPair(old_image['id'], '0'))
old_image['name'] = cur_image['user']['name']
old_image['email'] = cur_image['user']['email']
old_image['course'] = cur_image['course']['name']
return send_email(
to=(old_image['name'], old_image['email']),
subject=SUBJECT,
message=MESSAGE,
context={
'course': old_image['course'],
},
sender=EMAIL_SENDER,
sesv2_client=sesv2_client,
event={
'id': old_image['id'],
'sk': 'SCHEDULE#REMINDER_ACCESS_PERIOD_BEFORE_30_DAYS',
},
dynamodb_persistence_layer=enrollment_layer,
)

View File

@@ -1,22 +1,7 @@
from aws_lambda_powertools import Logger
from aws_lambda_powertools.utilities.data_classes import (
EventBridgeEvent,
event_source,
)
from aws_lambda_powertools.utilities.typing import LambdaContext
from layercake.dynamodb import DynamoDBPersistenceLayer, KeyPair
from boto3clients import dynamodb_client, sesv2_client
from config import (
EMAIL_SENDER,
ENROLLMENT_TABLE,
)
from .email_ import send_email
logger = Logger(__name__)
enrollment_layer = DynamoDBPersistenceLayer(ENROLLMENT_TABLE, dynamodb_client)
"""
If a certificate exists, remind the user 30 days before
the certificate expires
"""
SUBJECT = 'Seu certificado {course} vai expirar em breve'
MESSAGE = """
@@ -28,35 +13,3 @@ antes da expiração.<br/><br/>
<a href="https://saladeaula.digital">👉 Acesse o curso e renove sua certificação</a>
"""
@event_source(data_class=EventBridgeEvent)
@logger.inject_lambda_context
def lambda_handler(event: EventBridgeEvent, context: LambdaContext) -> bool:
"""If a certificate exists, remind the user 30 days before
the certificate expires."""
old_image = event.detail['old_image']
# Post-migration: Remove the following lines
if 'email' not in old_image:
# If email is missing, use enrollment email
cur_image = enrollment_layer.get_item(KeyPair(old_image['id'], '0'))
old_image['name'] = cur_image['user']['name']
old_image['email'] = cur_image['user']['email']
old_image['course'] = cur_image['course']['name']
return send_email(
to=(old_image['name'], old_image['email']),
subject=SUBJECT,
message=MESSAGE,
context={
'course': old_image['course'],
},
sender=EMAIL_SENDER,
sesv2_client=sesv2_client,
event={
'id': old_image['id'],
'sk': 'SCHEDULE#REMINDER_CERT_EXPIRATION_BEFORE_30_DAYS',
},
dynamodb_persistence_layer=enrollment_layer,
)

View File

@@ -1,22 +1,7 @@
from aws_lambda_powertools import Logger
from aws_lambda_powertools.utilities.data_classes import (
EventBridgeEvent,
event_source,
)
from aws_lambda_powertools.utilities.typing import LambdaContext
from layercake.dynamodb import DynamoDBPersistenceLayer, KeyPair
from boto3clients import dynamodb_client, sesv2_client
from config import (
EMAIL_SENDER,
ENROLLMENT_TABLE,
)
from .email_ import send_email
logger = Logger(__name__)
enrollment_layer = DynamoDBPersistenceLayer(ENROLLMENT_TABLE, dynamodb_client)
"""
If the user does not access the course within 3 days
after enrollment creation
"""
SUBJECT = 'Seu curso {course} está esperando por você na EDUSEG®'
MESSAGE = """
@@ -28,35 +13,3 @@ ao máximo seu curso!<br/><br/>
<a href="https://saladeaula.digital">👉 Acesse seu curso agora</a>
"""
@event_source(data_class=EventBridgeEvent)
@logger.inject_lambda_context
def lambda_handler(event: EventBridgeEvent, context: LambdaContext) -> bool:
"""If the user does not access the course within 3 days after
enrollment creation."""
old_image = event.detail['old_image']
# Post-migration: Remove the following lines
if 'email' not in old_image:
# If email is missing, use enrollment email
cur_image = enrollment_layer.get_item(KeyPair(old_image['id'], '0'))
old_image['name'] = cur_image['user']['name']
old_image['email'] = cur_image['user']['email']
old_image['course'] = cur_image['course']['name']
return send_email(
to=(old_image['name'], old_image['email']),
subject=SUBJECT,
message=MESSAGE,
context={
'course': old_image['course'],
},
sender=EMAIL_SENDER,
sesv2_client=sesv2_client,
event={
'id': old_image['id'],
'sk': 'SCHEDULE#REMINDER_NO_ACCESS_AFTER_3_DAYS',
},
dynamodb_persistence_layer=enrollment_layer,
)

View File

@@ -1,22 +1,4 @@
from aws_lambda_powertools import Logger
from aws_lambda_powertools.utilities.data_classes import (
EventBridgeEvent,
event_source,
)
from aws_lambda_powertools.utilities.typing import LambdaContext
from layercake.dynamodb import DynamoDBPersistenceLayer, KeyPair
from boto3clients import dynamodb_client, sesv2_client
from config import (
EMAIL_SENDER,
ENROLLMENT_TABLE,
)
from .email_ import send_email
logger = Logger(__name__)
enrollment_layer = DynamoDBPersistenceLayer(ENROLLMENT_TABLE, dynamodb_client)
# 7 days after the user's last activity in the course
SUBJECT = 'Seu curso {course} está parado há 7 dias...'
MESSAGE = """
@@ -27,34 +9,3 @@ Não deixe seu período de acesso expirar! Retome seu aprendizado agora mesmo.<b
<a href="https://saladeaula.digital">👉 Clique aqui para acessar seu curso</a>
"""
@event_source(data_class=EventBridgeEvent)
@logger.inject_lambda_context
def lambda_handler(event: EventBridgeEvent, context: LambdaContext) -> bool:
"""7 days after the user's last activity in the course."""
old_image = event.detail['old_image']
# Post-migration: Remove the following lines
if 'email' not in old_image:
# If email is missing, use enrollment email
cur_image = enrollment_layer.get_item(KeyPair(old_image['id'], '0'))
old_image['name'] = cur_image['user']['name']
old_image['email'] = cur_image['user']['email']
old_image['course'] = cur_image['course']['name']
return send_email(
to=(old_image['name'], old_image['email']),
subject=SUBJECT,
message=MESSAGE,
context={
'course': old_image['course'],
},
sender=EMAIL_SENDER,
sesv2_client=sesv2_client,
event={
'id': old_image['id'],
'sk': 'SCHEDULE#REMINDER_NO_ACTIVITY_AFTER_7_DAYS',
},
dynamodb_persistence_layer=enrollment_layer,
)