add cancel

This commit is contained in:
2025-07-18 00:14:45 -03:00
parent 6acc4bde4d
commit 61e7ac0f4b
7 changed files with 107 additions and 44 deletions

View File

@@ -44,8 +44,8 @@ Quando o responsável é uma pessoa física (CPF).
### Vagas ### Vagas
```json ```json
{"id": "slots#org#100", "sk": "order#101#faa8a547-bb9b-4103-bd8c-8fbe96b4056f", "course": {"name": "pytest"}} {"id": "slots#org#100", "sk": "order#101#enrollment#faa8a547-bb9b-4103-bd8c-8fbe96b4056f", "course": {"name": "pytest"}}
{"id": "slots#org#100", "sk": "order#101#afffbdde-fe58-4df7-b4d5-7553a571d32a", "course": {"name": "pytest"}} {"id": "slots#org#100", "sk": "order#101#enrollment#afffbdde-fe58-4df7-b4d5-7553a571d32a", "course": {"name": "pytest"}}
``` ```
### Emails/eventos agendados ### Emails/eventos agendados
@@ -94,7 +94,7 @@ Se houver `metadata#parent_slot`, deve ser devolvido.
```json ```json
{"id": "9omWNKymwU5U4aeun6mWzZ", "sk": "0"} {"id": "9omWNKymwU5U4aeun6mWzZ", "sk": "0"}
{"id": "9omWNKymwU5U4aeun6mWzZ", "sk": "cancel_policy"} {"id": "9omWNKymwU5U4aeun6mWzZ", "sk": "cancel_policy"}
{"id": "9omWNKymwU5U4aeun6mWzZ", "sk": "metadata#parent_slot", "slot": {"id": "slots#org#123", "sk": "order#1221#f7120daf-96d2-4639-b8f4-d736fd99e4ee"}} {"id": "9omWNKymwU5U4aeun6mWzZ", "sk": "metadata#parent_slot", "slot": {"id": "slots#org#123", "sk": "order#1221#enrollment#9omWNKymwU5U4aeun6mWzZ"}}
``` ```
# Cursos # Cursos

View File

@@ -32,7 +32,7 @@ user_layer = DynamoDBPersistenceLayer(USER_TABLE, dynamodb_client)
def get_courses(): def get_courses():
event = router.current_event event = router.current_event
query = event.get_query_string_value('q', '') query = event.get_query_string_value('q', '')
sort = event.get_query_string_value('sort', 'create_date:desc') sort = event.get_query_string_value('sort', 'created_at:desc')
page = int(event.get_query_string_value('page', '1')) page = int(event.get_query_string_value('page', '1'))
hits_per_page = int(event.get_query_string_value('hitsPerPage', '25')) hits_per_page = int(event.get_query_string_value('hitsPerPage', '25'))
@@ -58,11 +58,13 @@ def get_courses():
) )
def post_course(payload: Course): def post_course(payload: Course):
tenant: Tenant = router.context['tenant'] tenant: Tenant = router.context['tenant']
create_course( create_course(
payload, payload,
Org(id=tenant.id, name=tenant.name), tenant_id=str(tenant.id),
persistence_layer=course_layer, persistence_layer=course_layer,
) )
return JSONResponse( return JSONResponse(
body=payload, body=payload,
status_code=HTTPStatus.CREATED, status_code=HTTPStatus.CREATED,

View File

@@ -129,7 +129,7 @@ def put_user(id: str, payload: UserData):
@router.get('/<id>', compress=True, tags=['User'], summary='Get user') @router.get('/<id>', compress=True, tags=['User'], summary='Get user')
def get_user(id: str): def get_user(id: str):
return user_collect.get_items( return user_collect.get_items(
TransactKey(id) + SortKey('0') + SortKey('last_profile_edit') TransactKey(id) + SortKey('0') + SortKey('rate_limit#user_update')
) )

View File

@@ -1,37 +1,24 @@
from layercake.dateutils import now from layercake.dateutils import now
from layercake.dynamodb import DynamoDBPersistenceLayer, KeyPair from layercake.dynamodb import DynamoDBPersistenceLayer, KeyPair
from models import Course, Org from models import Course
def create_course( def create_course(
course: Course, course: Course,
org: Org,
/, /,
tenant_id: str,
persistence_layer: DynamoDBPersistenceLayer, persistence_layer: DynamoDBPersistenceLayer,
): ):
now_ = now() now_ = now()
return persistence_layer.put_item(
with persistence_layer.transact_writer() as transact: item={
transact.put( 'sk': '0',
item={ 'tenant_id': tenant_id,
'sk': '0', 'created_at': now_,
'tenant_id': {org.id}, **course.model_dump(),
'created_at': now_, }
**course.model_dump(), )
}
)
transact.put(
item={
'id': course.id,
'sk': 'metadata#tenant',
'org_id': org.id,
'name': org.name,
'created_at': now_,
}
)
return True
def update_course( def update_course(

View File

@@ -0,0 +1,30 @@
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
import konviva
from boto3clients import dynamodb_client
from config import ENROLLMENT_TABLE
logger = Logger(__name__)
enrollment_layer = DynamoDBPersistenceLayer(ENROLLMENT_TABLE, dynamodb_client)
@event_source(data_class=EventBridgeEvent)
@logger.inject_lambda_context
def lambda_handler(event: EventBridgeEvent, context: LambdaContext) -> bool:
new_image = event.detail['new_image']
data = enrollment_layer.get_item(KeyPair(new_image['id'], 'konviva'))
try:
result = konviva.cancel_enrollment(data['enrollment_id'])
except Exception as exc:
logger.exception(exc)
return False
else:
logger.info('Enrollment canceled', result=result)
return True

View File

@@ -31,7 +31,7 @@ def create_user(
name: str, name: str,
email: str, email: str,
cpf: str | None, cpf: str | None,
) -> dict: ) -> int:
url = urlparse(KONVIVA_API_URL)._replace( url = urlparse(KONVIVA_API_URL)._replace(
path='/action/api/integrarUsuario', path='/action/api/integrarUsuario',
query='sendMail=false', query='sendMail=false',
@@ -58,9 +58,10 @@ def create_user(
}, },
) )
r.raise_for_status() r.raise_for_status()
data = r.json()
# Because Konviva does not return the proper HTTP status code # Because Konviva does not return the proper HTTP status code
if err := glom(r.json(), 'errors', default=None): if err := glom(data, 'errors', default=None):
err = err[0] if isinstance(err, list) else err err = err[0] if isinstance(err, list) else err
if err == 'Login já existente': if err == 'Login já existente':
@@ -68,7 +69,7 @@ def create_user(
else: else:
raise KonvivaError(err) raise KonvivaError(err)
return r.json().get('IDUsuario') return int(data.get('IDUsuario'))
def update_user(id: str, **kwargs) -> dict: def update_user(id: str, **kwargs) -> dict:
@@ -86,12 +87,13 @@ def update_user(id: str, **kwargs) -> dict:
}, },
) )
r.raise_for_status() r.raise_for_status()
data = r.json()
if err := glom(r.json(), 'errors', default=None): if err := glom(data, 'errors', default=None):
err = err[0] if isinstance(err, list) else err err = err[0] if isinstance(err, list) else err
raise KonvivaError(err) raise KonvivaError(err)
return r.json() return data
def get_users_by_email(email: str) -> list[dict]: def get_users_by_email(email: str) -> list[dict]:
@@ -104,29 +106,48 @@ def get_users_by_email(email: str) -> list[dict]:
headers=headers, headers=headers,
) )
r.raise_for_status() r.raise_for_status()
data = r.json()
if glom(r.json(), '0.errors', default=None): if glom(data, '0.errors', default=None):
return [] return []
return r.json() return data
def enroll(user_id: str, class_id: str) -> str: def _post_enrollment(json: dict) -> dict:
url = urlparse(KONVIVA_API_URL)._replace(path='/action/api/integrarMatricula') url = urlparse(KONVIVA_API_URL)._replace(path='/action/api/integrarMatricula')
r = requests.post( r = requests.post(
url=url.geturl(), url=url.geturl(),
headers=headers, headers=headers,
json={ json=json,
'IDUsuario': str(user_id),
'IDTurma': str(class_id),
'StatusMatricula': 'MATRICULADO',
},
) )
r.raise_for_status() r.raise_for_status()
data = r.json()
if err := glom(r.json(), 'errors', default=None): if err := glom(data, 'errors', default=None):
err = err[0] if isinstance(err, list) else err err = err[0] if isinstance(err, list) else err
raise KonvivaError(err) raise KonvivaError(err)
return r.json().get('IDMatricula') return data
def enroll(user_id: str, class_id: str) -> int:
r = _post_enrollment(
{
'IDUsuario': int(user_id),
'IDTurma': int(class_id),
'StatusMatricula': 'MATRICULADO',
}
)
return int(r.get('IDMatricula')) # type: ignore
def cancel_enrollment(id: str) -> dict:
return _post_enrollment(
{
'IDMatricula': str(id),
'StatusMatricula': 'CANCELADO',
}
)

View File

@@ -81,3 +81,26 @@ Resources:
sk: ["konviva"] sk: ["konviva"]
enrollment_id: enrollment_id:
- exists: false - exists: false
EventCancelFunction:
Type: AWS::Serverless::Function
Properties:
Handler: events.cancel.lambda_handler
LoggingConfig:
LogGroup: !Ref EventLog
Policies:
- DynamoDBReadPolicy:
TableName: !Ref EnrollmentTable
Events:
DynamoDBEvent:
Type: EventBridgeRule
Properties:
Pattern:
resources: [!Ref EnrollmentTable]
detail-type: [MODIFY]
detail:
new_image:
sk: ["0"]
status: [CANCELED]
old_image:
status: [PENDING]