update event
This commit is contained in:
@@ -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
|
||||||
|
|||||||
@@ -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
|
|
||||||
@@ -1,22 +1,4 @@
|
|||||||
from aws_lambda_powertools import Logger
|
# 30 days before the course access period ends
|
||||||
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)
|
|
||||||
|
|
||||||
|
|
||||||
SUBJECT = 'Seu acesso ao curso {course} termina em 30 dias'
|
SUBJECT = 'Seu acesso ao curso {course} termina em 30 dias'
|
||||||
MESSAGE = """
|
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>
|
<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,
|
|
||||||
)
|
|
||||||
|
|||||||
@@ -1,22 +1,7 @@
|
|||||||
from aws_lambda_powertools import Logger
|
"""
|
||||||
from aws_lambda_powertools.utilities.data_classes import (
|
If a certificate exists, remind the user 30 days before
|
||||||
EventBridgeEvent,
|
the certificate expires
|
||||||
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)
|
|
||||||
|
|
||||||
|
|
||||||
SUBJECT = 'Seu certificado {course} vai expirar em breve'
|
SUBJECT = 'Seu certificado {course} vai expirar em breve'
|
||||||
MESSAGE = """
|
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>
|
<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,
|
|
||||||
)
|
|
||||||
|
|||||||
@@ -1,22 +1,7 @@
|
|||||||
from aws_lambda_powertools import Logger
|
"""
|
||||||
from aws_lambda_powertools.utilities.data_classes import (
|
If the user does not access the course within 3 days
|
||||||
EventBridgeEvent,
|
after enrollment creation
|
||||||
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)
|
|
||||||
|
|
||||||
|
|
||||||
SUBJECT = 'Seu curso {course} está esperando por você na EDUSEG®'
|
SUBJECT = 'Seu curso {course} está esperando por você na EDUSEG®'
|
||||||
MESSAGE = """
|
MESSAGE = """
|
||||||
@@ -28,35 +13,3 @@ ao máximo seu curso!<br/><br/>
|
|||||||
|
|
||||||
<a href="https://saladeaula.digital">👉 Acesse seu curso agora</a>
|
<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,
|
|
||||||
)
|
|
||||||
|
|||||||
@@ -1,22 +1,4 @@
|
|||||||
from aws_lambda_powertools import Logger
|
# 7 days after the user's last activity in the course
|
||||||
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)
|
|
||||||
|
|
||||||
|
|
||||||
SUBJECT = 'Seu curso {course} está parado há 7 dias...'
|
SUBJECT = 'Seu curso {course} está parado há 7 dias...'
|
||||||
MESSAGE = """
|
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>
|
<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,
|
|
||||||
)
|
|
||||||
|
|||||||
49
enrollments-events/app/events/send_reminder_emails.py
Normal file
49
enrollments-events/app/events/send_reminder_emails.py
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
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 .emails import TEMPLATES, send_email
|
||||||
|
|
||||||
|
logger = Logger(__name__)
|
||||||
|
dyn = DynamoDBPersistenceLayer(ENROLLMENT_TABLE, dynamodb_client)
|
||||||
|
|
||||||
|
|
||||||
|
@event_source(data_class=EventBridgeEvent)
|
||||||
|
@logger.inject_lambda_context
|
||||||
|
def lambda_handler(event: EventBridgeEvent, context: LambdaContext) -> bool:
|
||||||
|
old_image = event.detail['old_image']
|
||||||
|
event_name = old_image['sk'].removeprefix('SCHEDULE#').lower()
|
||||||
|
template = TEMPLATES[event_name]
|
||||||
|
|
||||||
|
# If email is missing, use enrollment email
|
||||||
|
if 'email' not in old_image:
|
||||||
|
r = dyn.get_item(KeyPair(pk=old_image['id'], sk='0'))
|
||||||
|
old_image['name'] = r['user']['name']
|
||||||
|
old_image['email'] = r['user']['email']
|
||||||
|
old_image['course'] = r['course']['name']
|
||||||
|
|
||||||
|
return send_email(
|
||||||
|
to=(old_image['name'], old_image['email']),
|
||||||
|
subject=template['subject'],
|
||||||
|
message=template['message'],
|
||||||
|
context={
|
||||||
|
'course': old_image['course'],
|
||||||
|
},
|
||||||
|
sender=EMAIL_SENDER,
|
||||||
|
sesv2_client=sesv2_client,
|
||||||
|
event={
|
||||||
|
'id': old_image['id'],
|
||||||
|
'sk': old_image['sk'],
|
||||||
|
},
|
||||||
|
dynamodb_persistence_layer=dyn,
|
||||||
|
)
|
||||||
@@ -9,15 +9,12 @@ from aws_lambda_powertools.utilities.data_classes import (
|
|||||||
)
|
)
|
||||||
from aws_lambda_powertools.utilities.typing import LambdaContext
|
from aws_lambda_powertools.utilities.typing import LambdaContext
|
||||||
from layercake.dateutils import now
|
from layercake.dateutils import now
|
||||||
from layercake.dynamodb import DynamoDBPersistenceLayer, KeyPair
|
from layercake.dynamodb import DynamoDBPersistenceLayer, KeyPair, SortKey
|
||||||
from sqlite_utils import Database
|
|
||||||
|
|
||||||
from boto3clients import dynamodb_client
|
from boto3clients import dynamodb_client
|
||||||
from config import (
|
from config import (
|
||||||
COURSE_TABLE,
|
COURSE_TABLE,
|
||||||
ENROLLMENT_TABLE,
|
ENROLLMENT_TABLE,
|
||||||
SQLITE_DATABASE,
|
|
||||||
SQLITE_TABLE,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
sqlite3.register_converter('json', json.loads)
|
sqlite3.register_converter('json', json.loads)
|
||||||
@@ -72,23 +69,24 @@ def lambda_handler(event: EventBridgeEvent, context: LambdaContext) -> bool:
|
|||||||
|
|
||||||
class CourseNotFoundError(Exception):
|
class CourseNotFoundError(Exception):
|
||||||
def __init__(self, *args):
|
def __init__(self, *args):
|
||||||
super().__init__('Course not found in SQLite')
|
super().__init__('Course not found')
|
||||||
|
|
||||||
|
|
||||||
def _get_course(course_id: str) -> dict:
|
def _get_course(id: str) -> dict:
|
||||||
course = course_layer.get_item(KeyPair(pk=course_id, sk='0'))
|
course = course_layer.get_item(KeyPair(pk=id, sk='0'))
|
||||||
if course:
|
if course:
|
||||||
return course
|
return course
|
||||||
|
|
||||||
with sqlite3.connect(
|
course_id = course_layer.collection.get_item(
|
||||||
database=SQLITE_DATABASE, detect_types=sqlite3.PARSE_DECLTYPES
|
KeyPair(
|
||||||
) as conn:
|
pk='MIGRATION',
|
||||||
db = Database(conn)
|
sk=SortKey(
|
||||||
rows = db[SQLITE_TABLE].rows_where(
|
f'COURSE#{id}',
|
||||||
"json->>'$.metadata__betaeducacao_id' = ?", [course_id]
|
path_spec='course_id',
|
||||||
)
|
),
|
||||||
|
),
|
||||||
for row in rows:
|
exc_cls=CourseNotFoundError,
|
||||||
return row['json']
|
)
|
||||||
|
return course_layer.collection.get_item(
|
||||||
raise CourseNotFoundError
|
KeyPair(pk=course_id, sk='0'),
|
||||||
|
)
|
||||||
|
|||||||
@@ -219,11 +219,10 @@ Resources:
|
|||||||
- !Sub arn:aws:ses:${AWS::Region}:${AWS::AccountId}:identity/eduseg.com.br
|
- !Sub arn:aws:ses:${AWS::Region}:${AWS::AccountId}:identity/eduseg.com.br
|
||||||
- !Sub arn:aws:ses:${AWS::Region}:${AWS::AccountId}:configuration-set/tracking
|
- !Sub arn:aws:ses:${AWS::Region}:${AWS::AccountId}:configuration-set/tracking
|
||||||
|
|
||||||
# If the user does not access the course within 3 days after enrollment creation
|
EventSendReminderEmailsFunction:
|
||||||
EventReminderNoAccessAfter3DaysFunction:
|
|
||||||
Type: AWS::Serverless::Function
|
Type: AWS::Serverless::Function
|
||||||
Properties:
|
Properties:
|
||||||
Handler: events.emails.reminder_no_access_after_3_days.lambda_handler
|
Handler: events.emails.send_reminder_emails.lambda_handler
|
||||||
LoggingConfig:
|
LoggingConfig:
|
||||||
LogGroup: !Ref EventLog
|
LogGroup: !Ref EventLog
|
||||||
Policies:
|
Policies:
|
||||||
@@ -239,17 +238,19 @@ Resources:
|
|||||||
detail-type: [EXPIRE]
|
detail-type: [EXPIRE]
|
||||||
detail:
|
detail:
|
||||||
keys:
|
keys:
|
||||||
sk: [SCHEDULE#REMINDER_NO_ACCESS_AFTER_3_DAYS]
|
sk:
|
||||||
|
- SCHEDULE#REMINDER_NO_ACCESS_AFTER_3_DAYS
|
||||||
|
- SCHEDULE#REMINDER_NO_ACTIVITY_AFTER_7_DAYS
|
||||||
|
- SCHEDULE#REMINDER_ACCESS_PERIOD_BEFORE_30_DAYS
|
||||||
|
- SCHEDULE#REMINDER_CERT_EXPIRATION_BEFORE_30_DAYS
|
||||||
|
|
||||||
# 7 days after the user's last activity in the course
|
EventScheduleRemindersFunction:
|
||||||
EventReminderNoActivityAfter7DaysFunction:
|
|
||||||
Type: AWS::Serverless::Function
|
Type: AWS::Serverless::Function
|
||||||
Properties:
|
Properties:
|
||||||
Handler: events.emails.reminder_no_activity_after_7_days.lambda_handler
|
Handler: events.schedule_reminders.lambda_handler
|
||||||
LoggingConfig:
|
LoggingConfig:
|
||||||
LogGroup: !Ref EventLog
|
LogGroup: !Ref EventLog
|
||||||
Policies:
|
Policies:
|
||||||
- !Ref SesPolicy
|
|
||||||
- DynamoDBCrudPolicy:
|
- DynamoDBCrudPolicy:
|
||||||
TableName: !Ref EnrollmentTable
|
TableName: !Ref EnrollmentTable
|
||||||
Events:
|
Events:
|
||||||
@@ -258,54 +259,11 @@ Resources:
|
|||||||
Properties:
|
Properties:
|
||||||
Pattern:
|
Pattern:
|
||||||
resources: [!Ref EnrollmentTable]
|
resources: [!Ref EnrollmentTable]
|
||||||
detail-type: [EXPIRE]
|
detail-type: [INSERT]
|
||||||
detail:
|
detail:
|
||||||
keys:
|
new_image:
|
||||||
sk: [SCHEDULE#REMINDER_NO_ACTIVITY_AFTER_7_DAYS]
|
sk: ["0"]
|
||||||
|
status: [PENDING]
|
||||||
# 30 days before the course access period ends.
|
|
||||||
EventReminderAccessPeriodBefore30DaysFunction:
|
|
||||||
Type: AWS::Serverless::Function
|
|
||||||
Properties:
|
|
||||||
Handler: events.emails.reminder_access_period_before_30_days.lambda_handler
|
|
||||||
LoggingConfig:
|
|
||||||
LogGroup: !Ref EventLog
|
|
||||||
Policies:
|
|
||||||
- !Ref SesPolicy
|
|
||||||
- DynamoDBCrudPolicy:
|
|
||||||
TableName: !Ref EnrollmentTable
|
|
||||||
Events:
|
|
||||||
DynamoDBEvent:
|
|
||||||
Type: EventBridgeRule
|
|
||||||
Properties:
|
|
||||||
Pattern:
|
|
||||||
resources: [!Ref EnrollmentTable]
|
|
||||||
detail-type: [EXPIRE]
|
|
||||||
detail:
|
|
||||||
keys:
|
|
||||||
sk: [SCHEDULE#REMINDER_ACCESS_PERIOD_BEFORE_30_DAYS]
|
|
||||||
|
|
||||||
# If a certificate exists, remind the user 30 days before the certificate expires
|
|
||||||
EventReminderCertExpirationBefore30DaysFunction:
|
|
||||||
Type: AWS::Serverless::Function
|
|
||||||
Properties:
|
|
||||||
Handler: events.emails.reminder_cert_expiration_before_30_days.lambda_handler
|
|
||||||
LoggingConfig:
|
|
||||||
LogGroup: !Ref EventLog
|
|
||||||
Policies:
|
|
||||||
- !Ref SesPolicy
|
|
||||||
- DynamoDBCrudPolicy:
|
|
||||||
TableName: !Ref EnrollmentTable
|
|
||||||
Events:
|
|
||||||
DynamoDBEvent:
|
|
||||||
Type: EventBridgeRule
|
|
||||||
Properties:
|
|
||||||
Pattern:
|
|
||||||
resources: [!Ref EnrollmentTable]
|
|
||||||
detail-type: [EXPIRE]
|
|
||||||
detail:
|
|
||||||
keys:
|
|
||||||
sk: [SCHEDULE#REMINDER_CERT_EXPIRATION_BEFORE_30_DAYS]
|
|
||||||
|
|
||||||
EventSetAccessExpiredFunction:
|
EventSetAccessExpiredFunction:
|
||||||
Type: AWS::Serverless::Function
|
Type: AWS::Serverless::Function
|
||||||
@@ -347,27 +305,6 @@ Resources:
|
|||||||
keys:
|
keys:
|
||||||
sk: [SCHEDULE#SET_CERT_EXPIRED, SCHEDULE#SET_AS_ARCHIVED]
|
sk: [SCHEDULE#SET_CERT_EXPIRED, SCHEDULE#SET_AS_ARCHIVED]
|
||||||
|
|
||||||
EventScheduleRemindersFunction:
|
|
||||||
Type: AWS::Serverless::Function
|
|
||||||
Properties:
|
|
||||||
Handler: events.schedule_reminders.lambda_handler
|
|
||||||
LoggingConfig:
|
|
||||||
LogGroup: !Ref EventLog
|
|
||||||
Policies:
|
|
||||||
- DynamoDBCrudPolicy:
|
|
||||||
TableName: !Ref EnrollmentTable
|
|
||||||
Events:
|
|
||||||
DynamoDBEvent:
|
|
||||||
Type: EventBridgeRule
|
|
||||||
Properties:
|
|
||||||
Pattern:
|
|
||||||
resources: [!Ref EnrollmentTable]
|
|
||||||
detail-type: [INSERT]
|
|
||||||
detail:
|
|
||||||
new_image:
|
|
||||||
sk: ["0"]
|
|
||||||
status: [PENDING]
|
|
||||||
|
|
||||||
EventIssueCertFunction:
|
EventIssueCertFunction:
|
||||||
Type: AWS::Serverless::Function
|
Type: AWS::Serverless::Function
|
||||||
Properties:
|
Properties:
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import app.events.emails.reminder_access_period_before_30_days as app
|
import app.events.send_reminder_emails as app
|
||||||
from aws_lambda_powertools.utilities.typing import LambdaContext
|
from aws_lambda_powertools.utilities.typing import LambdaContext
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import app.events.emails.reminder_cert_expiration_before_30_days as app
|
import app.events.send_reminder_emails as app
|
||||||
from aws_lambda_powertools.utilities.typing import LambdaContext
|
from aws_lambda_powertools.utilities.typing import LambdaContext
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import app.events.emails.reminder_no_access_after_3_days as app
|
import app.events.send_reminder_emails as app
|
||||||
from aws_lambda_powertools.utilities.typing import LambdaContext
|
from aws_lambda_powertools.utilities.typing import LambdaContext
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import app.events.emails.reminder_no_activity_after_7_days as app
|
import app.events.send_reminder_emails as app
|
||||||
from aws_lambda_powertools.utilities.typing import LambdaContext
|
from aws_lambda_powertools.utilities.typing import LambdaContext
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -16,6 +16,9 @@
|
|||||||
{"id": "6a403773-aeac-4e6a-ac39-dc958e4be52a", "sk": "0", "access_period": "360", "cert": {"exp_interval": 360}, "created_at": "2025-07-14T15:09:18.559528-03:00", "metadata__konviva_class_id": "281", "name": "Reciclagem em NR-11 - Operador de Empilhadeira", "tenant_id": "*"}
|
{"id": "6a403773-aeac-4e6a-ac39-dc958e4be52a", "sk": "0", "access_period": "360", "cert": {"exp_interval": 360}, "created_at": "2025-07-14T15:09:18.559528-03:00", "metadata__konviva_class_id": "281", "name": "Reciclagem em NR-11 - Operador de Empilhadeira", "tenant_id": "*"}
|
||||||
{"id": "e1c44881-2fe3-484e-ada2-12b6bf5b9398", "sk": "0", "name": "NR-35 Segurança nos Trabalhos em Altura (Teórico)", "updated_at": "2025-08-22T00:00:24.431267-03:00", "access_period": 360, "created_at": "2024-12-30T00:11:33.088916-03:00", "metadata__konviva_class_id": 1, "tenant_id": "*", "cert": {"exp_interval": 700}, "metadata__unit_price": 119}
|
{"id": "e1c44881-2fe3-484e-ada2-12b6bf5b9398", "sk": "0", "name": "NR-35 Segurança nos Trabalhos em Altura (Teórico)", "updated_at": "2025-08-22T00:00:24.431267-03:00", "access_period": 360, "created_at": "2024-12-30T00:11:33.088916-03:00", "metadata__konviva_class_id": 1, "tenant_id": "*", "cert": {"exp_interval": 700}, "metadata__unit_price": 119}
|
||||||
|
|
||||||
|
// Course migration
|
||||||
|
{"id": "MIGRATION", "sk": "COURSE#57", "course_id": "e1c44881-2fe3-484e-ada2-12b6bf5b9398"}
|
||||||
|
|
||||||
// User data
|
// User data
|
||||||
{"id": "5OxmMjL-ujoR5IMGegQz", "sk": "konviva", "konvivaId": 26943}
|
{"id": "5OxmMjL-ujoR5IMGegQz", "sk": "konviva", "konvivaId": 26943}
|
||||||
|
|
||||||
@@ -34,4 +37,5 @@
|
|||||||
{"id": "294e9864-8284-4287-b153-927b15d90900", "sk": "METADATA#COURSE", "access_period": 360, "created_at": "2025-09-09T09:11:29.292760-03:00", "cert": {"exp_interval": 360}}
|
{"id": "294e9864-8284-4287-b153-927b15d90900", "sk": "METADATA#COURSE", "access_period": 360, "created_at": "2025-09-09T09:11:29.292760-03:00", "cert": {"exp_interval": 360}}
|
||||||
{"id": "294e9864-8284-4287-b153-927b15d90900", "sk": "konviva", "class_id": 34, "user_id": 26943, "created_at": "2025-09-09T09:11:29.315247-03:00", "enrollment_id": 244488}
|
{"id": "294e9864-8284-4287-b153-927b15d90900", "sk": "konviva", "class_id": 34, "user_id": 26943, "created_at": "2025-09-09T09:11:29.315247-03:00", "enrollment_id": 244488}
|
||||||
{"id": "294e9864-8284-4287-b153-927b15d90900", "sk": "METADATA#DEDUPLICATION_WINDOW", "offset_days": 90, "created_at": "2025-09-11T09:00:45.923035-03:00"}
|
{"id": "294e9864-8284-4287-b153-927b15d90900", "sk": "METADATA#DEDUPLICATION_WINDOW", "offset_days": 90, "created_at": "2025-09-11T09:00:45.923035-03:00"}
|
||||||
{"id": "294e9864-8284-4287-b153-927b15d90900", "sk": "tenant", "org_id": "123", "name": "EDUSEG", "create_date": "2025-09-12T17:11:00.556907-03:00"}
|
{"id": "294e9864-8284-4287-b153-927b15d90900", "sk": "tenant", "org_id": "123", "name": "EDUSEG", "create_date": "2025-09-12T17:11:00.556907-03:00"}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user