Skip to content

Commit

Permalink
feat: Added logging for fulfillment events
Browse files Browse the repository at this point in the history
  • Loading branch information
MushtaqRossier committed Jul 3, 2024
1 parent 6e8c266 commit 5014534
Show file tree
Hide file tree
Showing 13 changed files with 72 additions and 11 deletions.
3 changes: 3 additions & 0 deletions commerce_coordinator/apps/commercetools/serializers.py
Original file line number Diff line number Diff line change
Expand Up @@ -36,12 +36,14 @@ class OrderLineItemMessageDetailSerializer(CoordinatorSerializer):
"""
Serializer for CommerceTools message 'detail'
"""
id = serializers.CharField()
resource = serializers.DictField(child=serializers.CharField())
fromState = serializers.DictField(child=serializers.CharField())
toState = serializers.DictField(child=serializers.CharField())

def to_representation(self, instance):
representation = super().to_representation(instance)
representation['message_id'] = representation.pop('id')
order_id = representation['resource'].get('id')
if order_id: # pragma no cover
representation['order_id'] = order_id
Expand Down Expand Up @@ -80,6 +82,7 @@ class OrderFulfillViewInputSerializer(CoordinatorSerializer):
source_system = serializers.CharField(allow_null=False)
line_item_state_id = serializers.CharField(allow_null=False)
edx_lms_user_id = serializers.IntegerField(allow_null=False)
message_id = serializers.CharField(allow_null=False)


class OrderReturnedViewMessageLineItemReturnItemSerializer(CoordinatorSerializer):
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ def fulfill_order_placed_message_signal(**kwargs):
order_id=kwargs['order_id'],
line_item_state_id=kwargs['line_item_state_id'],
source_system=kwargs['source_system'],
message_id=kwargs['message_id']
)
return async_result.id

Expand Down
6 changes: 4 additions & 2 deletions commerce_coordinator/apps/commercetools/sub_messages/tasks.py
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,7 @@ def fulfill_order_placed_message_signal_task(
order_id,
line_item_state_id,
source_system,
message_id
):
"""Celery task for fulfilling an order placed message."""

Expand All @@ -55,7 +56,7 @@ def fulfill_order_placed_message_signal_task(
except CommercetoolsError as err: # pragma no cover
logger.error(f'[CT-{tag}] Order not found: {order_id} with CT error {err}, {err.errors}')
return False

# breakpoint()
try:
customer = client.get_customer_by_id(order.customer_id)
except CommercetoolsError as err: # pragma no cover
Expand Down Expand Up @@ -107,7 +108,8 @@ def fulfill_order_placed_message_signal_task(
'course_id': get_edx_product_course_run_key(item), # likely not correct
'line_item_id': item.id,
'item_quantity': item.quantity,
'line_item_state_id': line_item_state_id
'line_item_state_id': line_item_state_id,
'message_id': message_id
})

# the following throws and thus doesn't need to be a conditional
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ class FulfillOrderPlacedMessageSignalTest(CoordinatorSignalReceiverTestCase):
'order_id': uuid4_str(),
'line_item_state_id': uuid4_str(),
'source_system': SOURCE_SYSTEM,
'message_id': uuid4_str()
}

def test_correct_arguments_passed(self, mock_task):
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@ def gen_example_fulfill_payload():
'line_item_state_id': uuid4_str(),
'order_line_id': uuid4_str(),
'source_system': SOURCE_SYSTEM,
'message_id': uuid4_str()
}


Expand Down Expand Up @@ -105,7 +106,8 @@ def unpack_for_uut(values):
return (
values['order_id'],
values['line_item_state_id'],
values['source_system']
values['source_system'],
values['message_id']
)

@staticmethod
Expand Down
4 changes: 3 additions & 1 deletion commerce_coordinator/apps/commercetools/views.py
Original file line number Diff line number Diff line change
Expand Up @@ -71,6 +71,7 @@ def post(self, request):

order_id = message_details.data['order_id']
line_item_state_id = message_details.data['to_state']['id']
message_id = message_details.data['message_id']

if self._is_running(tag, order_id): # pragma no cover
self.meta_should_mark_not_running = False
Expand All @@ -82,7 +83,8 @@ def post(self, request):
sender=self,
order_id=order_id,
line_item_state_id=line_item_state_id,
source_system=SOURCE_SYSTEM
source_system=SOURCE_SYSTEM,
message_id=message_id
)

return Response(status=status.HTTP_200_OK)
Expand Down
22 changes: 20 additions & 2 deletions commerce_coordinator/apps/lms/clients.py
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,12 @@ def api_enrollment_base_url(self):
"""
return urljoin_directory(settings.LMS_URL_ROOT, '/api/enrollment/v1/enrollment')

def enroll_user_in_course(self, enrollment_data, line_item_state_payload):
def enroll_user_in_course(
self,
enrollment_data,
line_item_state_payload,
fulfillment_logging_obj
):
"""
Send a POST request to LMS Enrollment API endpoint
Arguments:
Expand All @@ -36,10 +41,11 @@ def enroll_user_in_course(self, enrollment_data, line_item_state_payload):
url=self.api_enrollment_base_url,
json=enrollment_data,
line_item_state_payload=line_item_state_payload,
logging_obj=fulfillment_logging_obj,
timeout=settings.FULFILLMENT_TIMEOUT
)

def post(self, url, json, line_item_state_payload, timeout=None):
def post(self, url, json, line_item_state_payload, logging_obj, timeout=None):
"""
Send a POST request to a url with json payload.
"""
Expand All @@ -65,6 +71,12 @@ def post(self, url, json, line_item_state_payload, timeout=None):
**line_item_state_payload,
'is_fulfilled': True
}
logger.info(
f"Successful fulfillment for user: {logging_obj['user']} with details: "
f"[lms user id: {logging_obj['lms_user_id']}, order id: {logging_obj['order_id']}, "
f"course id: {logging_obj['course_id']}, message_id: {logging_obj['message_id']}, "
f"celery_task_id: {logging_obj['celery_task_id']}]"
)
except RequestException as exc:
self.log_request_exception(logger, exc)
fulfill_line_item_state_payload = {
Expand All @@ -75,6 +87,12 @@ def post(self, url, json, line_item_state_payload, timeout=None):
sender=self.__class__,
**fulfill_line_item_state_payload
)
logger.info(
f"Unsuccessful fulfillment for user: {logging_obj['user']} with details: "
f"[lms user id: {logging_obj['lms_user_id']}, order id: {logging_obj['order_id']}, "
f"course id: {logging_obj['course_id']}, message_id: {logging_obj['message_id']}, "
f"celery_task_id: {logging_obj['celery_task_id']}]"
)
raise

fulfillment_completed_signal.send_robust(
Expand Down
1 change: 1 addition & 0 deletions commerce_coordinator/apps/lms/signal_handlers.py
Original file line number Diff line number Diff line change
Expand Up @@ -28,5 +28,6 @@ def fulfill_order_placed_send_enroll_in_course(**kwargs):
line_item_id=kwargs['line_item_id'],
item_quantity=kwargs['item_quantity'],
line_item_state_id=kwargs['line_item_state_id'],
message_id=kwargs['message_id']
)
return async_result.id
12 changes: 11 additions & 1 deletion commerce_coordinator/apps/lms/tasks.py
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ def fulfill_order_placed_send_enroll_in_course_task(
line_item_id,
item_quantity,
line_item_state_id,
message_id
):
"""
Celery task for order placed fulfillment and enrollment via LMS Enrollment API.
Expand Down Expand Up @@ -100,4 +101,13 @@ def fulfill_order_placed_send_enroll_in_course_task(
'line_item_state_id': line_item_state_id,
}

return LMSAPIClient().enroll_user_in_course(enrollment_data, line_item_state_payload)
fulfillment_logging_obj = {
'user': user.username,
'lms_user_id': user.lms_user_id,
'order_id': order_id,
'course_id': course_id,
'message_id': message_id,
'celery_task_id': self.request.id
}

return LMSAPIClient().enroll_user_in_course(enrollment_data, line_item_state_payload, fulfillment_logging_obj)
10 changes: 10 additions & 0 deletions commerce_coordinator/apps/lms/tests/constants.py
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@
'line_item_id': '822d77c4-00a6-4fb9-909b-094ef0b8c4b9',
'item_quantity': 1,
'line_item_state_id': '8f2e888e-9777-4557-9a7f-c649153770c2',
'message_id': '1063f19c-08f3-41a4-a952-a8577374373c'
}

EXAMPLE_FULFILLMENT_REQUEST_PAYLOAD = {
Expand Down Expand Up @@ -104,3 +105,12 @@
'pacing_type': 'Instructor Paced'
},
}

EXAMPLE_FULFILLMENT_LOGGING_OBJ = {
'user': 'test-user',
'lms_user_id': 4,
'order_id': '61ec1afa-1b0e-4234-ae28-f997728054fa',
'course_id': 'course-v1:edX+DemoX+Demo_Course',
'message_id': '1063f19c-08f3-41a4-a952-a8577374373c',
'celery_task_id': None
}
8 changes: 7 additions & 1 deletion commerce_coordinator/apps/lms/tests/test_clients.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@
from commerce_coordinator.apps.core.tests.utils import CoordinatorOAuthClientTestCase
from commerce_coordinator.apps.lms.clients import LMSAPIClient
from commerce_coordinator.apps.lms.tests.constants import (
EXAMPLE_FULFILLMENT_LOGGING_OBJ,
EXAMPLE_FULFILLMENT_REQUEST_PAYLOAD,
EXAMPLE_FULFILLMENT_RESPONSE_PAYLOAD,
EXAMPLE_LINE_ITEM_STATE_PAYLOAD
Expand Down Expand Up @@ -55,6 +56,7 @@ def test_order_create_success(self, mock_signal):
input_kwargs={
'enrollment_data': EXAMPLE_FULFILLMENT_REQUEST_PAYLOAD,
'line_item_state_payload': EXAMPLE_LINE_ITEM_STATE_PAYLOAD,
'fulfillment_logging_obj': EXAMPLE_FULFILLMENT_LOGGING_OBJ
},
expected_request=EXAMPLE_FULFILLMENT_REQUEST_PAYLOAD,
mock_url=self.url,
Expand All @@ -68,7 +70,11 @@ def test_order_create_failure(self, mock_signal):
with self.assertRaises(HTTPError):
self.assertJSONClientResponse(
uut=self.client.enroll_user_in_course,
input_kwargs={'enrollment_data': '', 'line_item_state_payload': EXAMPLE_LINE_ITEM_STATE_PAYLOAD},
input_kwargs={
'enrollment_data': '',
'line_item_state_payload': EXAMPLE_LINE_ITEM_STATE_PAYLOAD,
'fulfillment_logging_obj': EXAMPLE_FULFILLMENT_LOGGING_OBJ
},
mock_url=self.url,
mock_status=400,
)
Expand Down
1 change: 1 addition & 0 deletions commerce_coordinator/apps/lms/tests/test_signals.py
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@ class FulfillOrderPlacedSendEnrollInCourseTest(CoordinatorSignalReceiverTestCase
'line_item_id': 11,
'item_quantity': 1,
'line_item_state_id': 12,
'message_id': 13
}

def test_correct_arguments_passed(self, mock_task):
Expand Down
10 changes: 7 additions & 3 deletions commerce_coordinator/apps/lms/tests/test_tasks.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@
from commerce_coordinator.apps.core.models import User
from commerce_coordinator.apps.lms.tasks import fulfill_order_placed_send_enroll_in_course_task
from commerce_coordinator.apps.lms.tests.constants import (
EXAMPLE_FULFILLMENT_LOGGING_OBJ,
EXAMPLE_FULFILLMENT_REQUEST_PAYLOAD,
EXAMPLE_FULFILLMENT_SIGNAL_PAYLOAD,
EXAMPLE_LINE_ITEM_STATE_PAYLOAD
Expand Down Expand Up @@ -44,7 +45,8 @@ def unpack_for_uut(values):
values['source_system'],
values['line_item_id'],
values['item_quantity'],
values['line_item_state_id']
values['line_item_state_id'],
values['message_id']
)

def setUp(self):
Expand All @@ -59,7 +61,8 @@ def test_correct_arguments_passed(self, mock_client):
logger.info('mock_client().mock_calls: %s', mock_client().mock_calls)
mock_client().enroll_user_in_course.assert_called_once_with(
EXAMPLE_FULFILLMENT_REQUEST_PAYLOAD,
EXAMPLE_LINE_ITEM_STATE_PAYLOAD
EXAMPLE_LINE_ITEM_STATE_PAYLOAD,
EXAMPLE_FULFILLMENT_LOGGING_OBJ
)

def test_correct_arguments_passed_credit(self, mock_client):
Expand All @@ -86,7 +89,8 @@ def test_correct_arguments_passed_credit(self, mock_client):
logger.info('mock_client().mock_calls: %s', mock_client().mock_calls)
mock_client().enroll_user_in_course.assert_called_once_with(
credit_expected_data,
EXAMPLE_LINE_ITEM_STATE_PAYLOAD
EXAMPLE_LINE_ITEM_STATE_PAYLOAD,
EXAMPLE_FULFILLMENT_LOGGING_OBJ
)

def test_passes_through_client_return(self, mock_client):
Expand Down

0 comments on commit 5014534

Please sign in to comment.