From b51d5ee54b64ca9b89b36a79e013eec2223e523c Mon Sep 17 00:00:00 2001 From: Attiya Ishaque Date: Wed, 20 Nov 2024 19:16:18 +0500 Subject: [PATCH] fix: Improve logging messages --- .../apps/commercetools/clients.py | 20 +++++--- .../apps/commercetools/sub_messages/tasks.py | 10 ++-- .../apps/commercetools/tasks.py | 36 ++++++------- .../apps/commercetools/tests/test_clients.py | 19 ++++--- .../apps/commercetools/tests/test_tasks.py | 50 +++++++------------ .../apps/commercetools/utils.py | 5 +- .../apps/commercetools/views.py | 6 +-- commerce_coordinator/apps/core/clients.py | 30 ++--------- commerce_coordinator/apps/core/views.py | 2 +- .../apps/ecommerce/clients.py | 3 +- commerce_coordinator/apps/lms/clients.py | 41 ++++++++------- commerce_coordinator/apps/lms/tasks.py | 11 ++-- commerce_coordinator/apps/titan/clients.py | 3 +- 13 files changed, 106 insertions(+), 130 deletions(-) diff --git a/commerce_coordinator/apps/commercetools/clients.py b/commerce_coordinator/apps/commercetools/clients.py index 99e96b5cf..31ad0f56c 100644 --- a/commerce_coordinator/apps/commercetools/clients.py +++ b/commerce_coordinator/apps/commercetools/clients.py @@ -433,7 +433,8 @@ def create_return_for_order(self, order_id: str, order_version: int, order_line_ ) return returned_order except CommercetoolsError as err: - handle_commercetools_error(err, f"Unable to create return for order {order_id}") + handle_commercetools_error("[CommercetoolsAPIClient.create_return_for_order]", + err, f"Unable to create return for order {order_id}") raise err def update_return_payment_state_after_successful_refund( @@ -498,7 +499,8 @@ def update_return_payment_state_after_successful_refund( logger.info("Updated transaction with return item id") return updated_order except CommercetoolsError as err: - handle_commercetools_error(err, f"Unable to update ReturnPaymentState of order {order_id}") + handle_commercetools_error("[CommercetoolsAPIClient.update_return_payment_state_after_successful_refund]", + err, f"Unable to update ReturnPaymentState of order {order_id}") raise OpenEdxFilterException(str(err)) from err def _preprocess_refund_object(self, refund: Refund, psp: str) -> Refund: @@ -560,7 +562,7 @@ def create_return_payment_transaction( f"Unable to create refund payment transaction for payment {payment_id}, refund {refund['id']} " f"with PSP: {psp}" ) - handle_commercetools_error(err, context) + handle_commercetools_error("[CommercetoolsAPIClient.create_return_payment_transaction]", err, context) raise err def update_line_item_transition_state_on_fulfillment( @@ -611,13 +613,15 @@ def update_line_item_transition_state_on_fulfillment( return updated_fulfillment_line_item_order else: logger.info( - f"The line item {line_item_id} already has the correct state {new_state_key}. " - "Not attempting to transition LineItemState" + f"[CommercetoolsAPIClient] - The line item {line_item_id} " + f"already has the correct state {new_state_key}. " + f"Not attempting to transition LineItemState for order id {order_id}" ) return self.get_order_by_id(order_id) except CommercetoolsError as err: + context_prefix = "[CommercetoolsAPIClient.update_line_item_state_on_fulfillment_completion]" # Logs & ignores version conflict errors due to duplicate Commercetools messages - handle_commercetools_error(err, f"Unable to update LineItemState of order {order_id}", True) + handle_commercetools_error(context_prefix, err, f"Unable to update LineItemState of order {order_id}") return None def update_line_items_transition_state( @@ -676,6 +680,7 @@ def update_line_items_transition_state( except CommercetoolsError as err: # Logs & ignores version conflict errors due to duplicate Commercetools messages handle_commercetools_error( + "[CommercetoolsAPIClient.update_line_items_transition_state]", err, f"Failed to update LineItemStates for order ID '{order_id}'. " f"Line Item IDs: {', '.join(item.id for item in line_items)}", @@ -770,6 +775,7 @@ def is_first_time_discount_eligible(self, email: str, code: str) -> bool: return True except CommercetoolsError as err: # pragma no cover # Logs & ignores version conflict errors due to duplicate Commercetools messages - handle_commercetools_error(err, f"Unable to check if user {email} is eligible for a " + handle_commercetools_error("[CommercetoolsAPIClient.is_first_time_discount_eligible]", + err, f"Unable to check if user {email} is eligible for a " f"first time discount", True) return True diff --git a/commerce_coordinator/apps/commercetools/sub_messages/tasks.py b/commerce_coordinator/apps/commercetools/sub_messages/tasks.py index 41a821f6e..b3d6232ea 100644 --- a/commerce_coordinator/apps/commercetools/sub_messages/tasks.py +++ b/commerce_coordinator/apps/commercetools/sub_messages/tasks.py @@ -78,11 +78,11 @@ def fulfill_order_placed_message_signal_task( return False if not (customer and order and is_edx_lms_order(order)): - logger.debug(f'[CT-{tag}] order {order_id} is not an edX order, message id: {message_id}') + logger.info(f'[CT-{tag}] order {order_id} is not an edX order, message id: {message_id}') return True - logger.debug(f'[CT-{tag}] processing edX order {order_id}, message id: {message_id}') + logger.info(f'[CT-{tag}] processing edX order {order_id}, message id: {message_id}') lms_user_id = get_edx_lms_user_id(customer) @@ -266,7 +266,7 @@ def _prepare_segment_event_properties(in_order, return_line_item_return_id): return False if not (customer and order and is_edx_lms_order(order)): # pragma no cover - logger.debug(f'[CT-{tag}] order {order_id} is not an edX order, message id: {message_id}') + logger.info(f'[CT-{tag}] order {order_id} is not an edX order, message id: {message_id}') return True payment_intent_id = get_edx_payment_intent_id(order) @@ -286,13 +286,13 @@ def _prepare_segment_event_properties(in_order, return_line_item_return_id): logger.info(f'[CT-{tag}] payment intent {payment_intent_id} already has refunded transaction, ' f'sending Slack notification, message id: {message_id}') else: - logger.debug(f'[CT-{tag}] payment intent {payment_intent_id} refunded. message id: {message_id}') + logger.info(f'[CT-{tag}] payment intent {payment_intent_id} refunded for message id: {message_id}') segment_event_properties = _prepare_segment_event_properties(order, return_line_item_return_id) for line_item in get_edx_items(order): course_run = get_edx_product_course_run_key(line_item) # TODO: Remove LMS Enrollment - logger.debug( + logger.info( f'[CT-{tag}] calling lms to unenroll user {lms_user_name} in {course_run}' f', message id: {message_id}' ) diff --git a/commerce_coordinator/apps/commercetools/tasks.py b/commerce_coordinator/apps/commercetools/tasks.py index 14dbccf71..1004c6ea5 100644 --- a/commerce_coordinator/apps/commercetools/tasks.py +++ b/commerce_coordinator/apps/commercetools/tasks.py @@ -31,20 +31,16 @@ def update_line_item_state_on_fulfillment_completion( Task for fulfillment completed and order line item state update via Commercetools API. """ client = CommercetoolsAPIClient() - try: - updated_order = client.update_line_item_transition_state_on_fulfillment( - order_id, - order_version, - line_item_id, - item_quantity, - from_state_id, - to_state_key - ) - return updated_order - except CommercetoolsError as err: - logger.error(f"Unable to update line item [ {line_item_id} ] state on fulfillment " - f"result with error {err.errors} and correlation id {err.correlation_id}") - return None + + updated_order = client.update_line_item_transition_state_on_fulfillment( + order_id, + order_version, + line_item_id, + item_quantity, + from_state_id, + to_state_key + ) + return updated_order @shared_task(autoretry_for=(CommercetoolsError,), retry_kwargs={'max_retries': 5, 'countdown': 3}) @@ -58,9 +54,12 @@ def refund_from_stripe_task( """ client = CommercetoolsAPIClient() try: + logger.info( + f"[refund_from_stripe_task] Initiating creation of CT payment's refund transaction object " + f"for payment Intent ID {payment_intent_id}.") payment = client.get_payment_by_key(payment_intent_id) if has_full_refund_transaction(payment): - logger.info(f"Stripe charge.refunded event received, but Payment with ID {payment.id} " + logger.info(f"[refund_from_stripe_task] Event 'charge.refunded' received, but Payment with ID {payment.id} " f"already has a full refund. Skipping task to add refund transaction") return None updated_payment = client.create_return_payment_transaction( @@ -70,8 +69,8 @@ def refund_from_stripe_task( ) return updated_payment except CommercetoolsError as err: - logger.error(f"Unable to create refund transaction for payment [ {payment.id} ] " - f"on Stripe refund {stripe_refund['id']} " + logger.error(f"[refund_from_stripe_task] Unable to create CT payment's refund transaction " + f"object for [ {payment.id} ] on Stripe refund {stripe_refund['id']} " f"with error {err.errors} and correlation id {err.correlation_id}") return None @@ -100,7 +99,8 @@ def refund_from_paypal_task( ) return updated_payment except CommercetoolsError as err: - logger.error(f"Unable to create refund transaction for payment {payment.key} " + logger.error(f"[refund_from_paypal_task] Unable to create CT payment's refund " + f"transaction object for payment {payment.key} " f"on PayPal refund {refund.get('id')} " f"with error {err.errors} and correlation id {err.correlation_id}") return None diff --git a/commerce_coordinator/apps/commercetools/tests/test_clients.py b/commerce_coordinator/apps/commercetools/tests/test_clients.py index f4b6fc714..51ddbcdae 100644 --- a/commerce_coordinator/apps/commercetools/tests/test_clients.py +++ b/commerce_coordinator/apps/commercetools/tests/test_clients.py @@ -470,8 +470,8 @@ def test_create_return_for_order_exception(self): exception = cm.exception expected_message = ( - f"[CommercetoolsError] Unable to create return for " - f"order mock_order_id " + f"[CommercetoolsError] [CommercetoolsAPIClient.create_return_for_order] " + f"Unable to create return for order mock_order_id " f"- Correlation ID: {exception.correlation_id}, Details: {exception.errors}" ) @@ -604,7 +604,8 @@ def test_create_refund_transaction_exception(self): exception = cm.exception expected_message = ( - f"[CommercetoolsError] Unable to create refund payment transaction for " + f"[CommercetoolsError] [CommercetoolsAPIClient.create_return_payment_transaction] " + f"Unable to create refund payment transaction for " f"payment mock_payment_id, refund {mock_stripe_refund.id} with PSP: stripe_edx " f"- Correlation ID: {exception.correlation_id}, Details: {exception.errors}" ) @@ -672,7 +673,7 @@ def test_update_line_item_state_exception(self, mock_state_by_id): status_code=409 ) - with patch('commerce_coordinator.apps.commercetools.clients.logging.Logger.info') as log_mock: + with patch('commerce_coordinator.apps.commercetools.clients.logging.Logger.error') as log_mock: self.client_set.client.update_line_item_transition_state_on_fulfillment( "mock_order_id", 1, @@ -683,7 +684,8 @@ def test_update_line_item_state_exception(self, mock_state_by_id): ) expected_message = ( - f"[CommercetoolsError] Unable to update LineItemState " + f"[CommercetoolsError] [CommercetoolsAPIClient.update_line_item_state_on_fulfillment_completion] " + f"Unable to update LineItemState " f"of order mock_order_id " f"- Correlation ID: {mock_error_response['correlation_id']}, " f"Details: {mock_error_response['errors']}" @@ -763,7 +765,8 @@ def test_update_all_line_items_state_exception(self, mock_state_by_id): ) expected_message = ( - f"[CommercetoolsError] Failed to update LineItemStates " + f"[CommercetoolsError] [CommercetoolsAPIClient.update_line_items_transition_state] " + f"Failed to update LineItemStates " f"for order ID 'mock_order_id'. Line Item IDs: {mock_order.line_items[0].id} " f"- Correlation ID: {mock_error_response['correlation_id']}, " f"Details: {mock_error_response['errors']}" @@ -795,8 +798,8 @@ def test_order_line_item_in_correct_state(self, mock_order_by_id, mock_state_by_ ) expected_message = ( - f"The line item {line_item_id} already has the correct state {mock_line_item_state.key}. " - f"Not attempting to transition LineItemState" + f"[CommercetoolsAPIClient] - The line item {line_item_id} already has the correct state " + f"{mock_line_item_state.key}. Not attempting to transition LineItemState for order id mock_order_id" ) log_mock.assert_called_with(expected_message) diff --git a/commerce_coordinator/apps/commercetools/tests/test_tasks.py b/commerce_coordinator/apps/commercetools/tests/test_tasks.py index 15dcd2419..6eff10347 100644 --- a/commerce_coordinator/apps/commercetools/tests/test_tasks.py +++ b/commerce_coordinator/apps/commercetools/tests/test_tasks.py @@ -3,7 +3,7 @@ """ import logging -from unittest.mock import patch +from unittest.mock import call, patch import stripe from commercetools import CommercetoolsError @@ -62,28 +62,6 @@ def test_correct_arguments_passed(self, mock_client): *list(EXAMPLE_UPDATE_LINE_ITEM_SIGNAL_PAYLOAD.values()) ) - @patch('commerce_coordinator.apps.commercetools.tasks.logger') - def test_exception_handling(self, mock_logger, mock_client): - ''' - Check if an error in the client results in a logged error - and None returned. - ''' - mock_client().update_line_item_transition_state_on_fulfillment.side_effect = CommercetoolsError( - message="Could not update ReturnPaymentState", - errors="Some error message", - response={}, - correlation_id="123456" - ) - - result = fulfillment_uut(*self.unpack_for_uut(EXAMPLE_UPDATE_LINE_ITEM_SIGNAL_PAYLOAD)) - - mock_logger.error.assert_called_once_with( - f"Unable to update line item [ {EXAMPLE_UPDATE_LINE_ITEM_SIGNAL_PAYLOAD['line_item_id']} ] " - "state on fulfillment result with error Some error message and correlation id 123456" - ) - - assert result is None - @patch('commerce_coordinator.apps.commercetools.tasks.CommercetoolsAPIClient') class ReturnedOrderfromStripeTaskTest(TestCase): @@ -123,10 +101,10 @@ def test_correct_arguments_passed(self, mock_client): ) def test_full_refund_already_exists(self, mock_client): - ''' + """ Check if the payment already has a full refund, the task logs the - appropriate message and skips creating a refund transaction. - ''' + appropriate messages and skips creating a refund transaction. + """ mock_payment = gen_payment_with_multiple_transactions( TransactionType.CHARGE, 4900, TransactionType.REFUND, 4900 @@ -135,13 +113,22 @@ def test_full_refund_already_exists(self, mock_client): mock_client.return_value.get_payment_by_key.return_value = mock_payment + payment_intent_id = EXAMPLE_RETURNED_ORDER_STRIPE_SIGNAL_PAYLOAD['payment_intent_id'] + with patch('commerce_coordinator.apps.commercetools.tasks.logger') as mock_logger: result = refund_from_stripe_task(*self.unpack_for_uut(EXAMPLE_RETURNED_ORDER_STRIPE_SIGNAL_PAYLOAD)) self.assertIsNone(result) - mock_logger.info.assert_called_once_with( - f"Stripe charge.refunded event received, but Payment with ID {mock_payment.id} " - f"already has a full refund. Skipping task to add refund transaction" - ) + + # Check that both info messages were logged in the expected order + mock_logger.info.assert_has_calls([ + call( + f"[refund_from_stripe_task] " + f"Initiating creation of CT payment's refund transaction object " + f"for payment Intent ID {payment_intent_id}."), + call(f"[refund_from_stripe_task] Event 'charge.refunded' received, " + f"but Payment with ID {mock_payment.id} " + f"already has a full refund. Skipping task to add refund transaction") + ]) @patch('commerce_coordinator.apps.commercetools.tasks.logger') def test_exception_handling(self, mock_logger, mock_client): @@ -162,7 +149,8 @@ def test_exception_handling(self, mock_logger, mock_client): returned_uut(*self.unpack_for_uut(EXAMPLE_RETURNED_ORDER_STRIPE_SIGNAL_PAYLOAD)) mock_logger.error.assert_called_once_with( - f"Unable to create refund transaction for payment [ {mock_payment.id} ] " + f"[refund_from_stripe_task] Unable to create CT payment's refund transaction " + f"object for [ {mock_payment.id} ] " f"on Stripe refund {EXAMPLE_RETURNED_ORDER_STRIPE_CLIENT_PAYLOAD['stripe_refund']['id']} " f"with error Some error message and correlation id 123456" ) diff --git a/commerce_coordinator/apps/commercetools/utils.py b/commerce_coordinator/apps/commercetools/utils.py index 955571283..5111a4208 100644 --- a/commerce_coordinator/apps/commercetools/utils.py +++ b/commerce_coordinator/apps/commercetools/utils.py @@ -35,9 +35,10 @@ def get_braze_client(): ) -def handle_commercetools_error(err: CommercetoolsError, context: str, is_info=False): +def handle_commercetools_error(context_prefix, err: CommercetoolsError, context: str, is_info=False): """Handles commercetools errors.""" - error_message = f"[CommercetoolsError] {context} - Correlation ID: {err.correlation_id}, Details: {err.errors}" + error_message = (f"[CommercetoolsError] {context_prefix} {context} " + f"- Correlation ID: {err.correlation_id}, Details: {err.errors}") if is_info: logger.info(error_message) else: diff --git a/commerce_coordinator/apps/commercetools/views.py b/commerce_coordinator/apps/commercetools/views.py index f159248a9..ad573b71d 100644 --- a/commerce_coordinator/apps/commercetools/views.py +++ b/commerce_coordinator/apps/commercetools/views.py @@ -41,7 +41,7 @@ def post(self, request): **request.data } - logger.debug(f'[CT-{tag}] Message received from commercetools with details: {input_data}') + logger.info(f'[CT-{tag}] Message received from commercetools with details: {input_data}') message_details = OrderLineItemMessageInputSerializer(data=input_data) message_details.is_valid(raise_exception=True) @@ -85,7 +85,7 @@ def post(self, request): **request.data } - logger.debug(f'[CT-{tag}] Message received from commercetools with details: {input_data}') + logger.info(f'[CT-{tag}] Message received from commercetools with details: {input_data}') message_details = OrderSanctionedViewMessageInputSerializer(data=input_data) message_details.is_valid(raise_exception=True) @@ -127,7 +127,7 @@ def post(self, request): **request.data } - logger.debug(f'[CT-{tag}] Message received from commercetools with details: {input_data}') + logger.info(f'[CT-{tag}] Message received from commercetools with details: {input_data}') message_details = OrderReturnedViewMessageInputSerializer(data=input_data) message_details.is_valid(raise_exception=True) diff --git a/commerce_coordinator/apps/core/clients.py b/commerce_coordinator/apps/core/clients.py index dc48d64bf..052eceb81 100644 --- a/commerce_coordinator/apps/core/clients.py +++ b/commerce_coordinator/apps/core/clients.py @@ -64,38 +64,18 @@ def normal_timeout(self): settings.REQUEST_READ_TIMEOUT_SECONDS ) - def log_request_response(self, ext_logger, response): - """ - Log requests.Request Response. - - Args: - ext_logger: The logger of the module the Client (or Client- - derived class) is running in. - response: A successful Response object from the requests library. - """ - ext_logger.debug('Request URL: %s', response.request.url) - ext_logger.debug('Request method: %s', response.request.method) - ext_logger.debug('Request body: %s', response.request.body) - ext_logger.debug('Request headers: %s', response.request.headers) - ext_logger.debug('Response status: %s %s', response.status_code, response.reason) - ext_logger.debug('Response text: %s', response.text) - - def log_request_exception(self, ext_logger, exc): + def log_request_exception(self, context_prefix, ext_logger, exc): """ Log requests.Request exceptions. Args: ext_logger: The logger of the module the Client (or Client- derived class) is running in. - response: A RequestException object from the requests library. + exc: A RequestException object from the requests library. """ - ext_logger.error(exc) - ext_logger.info('Request URL: %s', exc.request.url) - ext_logger.info('Request method: %s', exc.request.method) - ext_logger.info('Request body: %s', exc.request.body) - ext_logger.debug('Request headers: %s', exc.request.headers) - ext_logger.info('Response status: %s %s', exc.response.status_code, exc.response.reason) - ext_logger.info('Response body: %s', exc.response.text) + ext_logger.error( + f"{context_prefix} - The API call failed with status code: " + f"{exc.response.status_code if exc.response else ''}. Reason: {exc}.") class BaseEdxOAuthClient(Client): diff --git a/commerce_coordinator/apps/core/views.py b/commerce_coordinator/apps/core/views.py index eb0855913..1f97331f8 100644 --- a/commerce_coordinator/apps/core/views.py +++ b/commerce_coordinator/apps/core/views.py @@ -130,7 +130,7 @@ def _is_running(view: str, identifier: str): key = SingleInvocationAPIView._view_cache_key(view, identifier) cache_value = TieredCache.get_cached_response(key) if cache_value.is_found or cache_value.get_value_or_default(False): - logger.debug(f'[CT-{view}] Currently processing request for %s, ignoring invocation', identifier) + logger.info(f'[CT-{view}] Currently processing request for %s, ignoring invocation', identifier) return True return False diff --git a/commerce_coordinator/apps/ecommerce/clients.py b/commerce_coordinator/apps/ecommerce/clients.py index 69c1d78d4..a19da75c2 100644 --- a/commerce_coordinator/apps/ecommerce/clients.py +++ b/commerce_coordinator/apps/ecommerce/clients.py @@ -37,9 +37,8 @@ def get_orders(self, query_params): resource_url = urljoin_directory(self.api_base_url, '/orders') response = self.client.get(resource_url, params=query_params) response.raise_for_status() - self.log_request_response(logger, response) except RequestException as exc: - self.log_request_exception(logger, exc) + self.log_request_exception("[EcommerceAPIClient.get_orders]", logger, exc) raise return response.json() diff --git a/commerce_coordinator/apps/lms/clients.py b/commerce_coordinator/apps/lms/clients.py index 6e9d06757..a353be2de 100644 --- a/commerce_coordinator/apps/lms/clients.py +++ b/commerce_coordinator/apps/lms/clients.py @@ -40,18 +40,19 @@ def deactivate_user(self, username, ct_message_id): Intended use is on SDN check failure. """ try: - logger.info(f'Calling LMS to deactivate account for user with username {username}' - f'after receiving subsctiption message with ID: {ct_message_id}') + logger.info( + f"[LMSAPIClient.deactivate_user] Initiating account deactivation on LMS for user '{username}', " + f"triggered by CT subscription message ID: {ct_message_id}." + ) response = self.client.post( self.deactivate_user_api_url.format(username=username), timeout=self.normal_timeout, ) response.raise_for_status() except (ConnectionError, RequestException) as exc: - logger.info(f'Unsuccessful call to LMS to deactivate account for user with username {username}' - f'with details: [message_id: {ct_message_id}]') - logger.exception( - f'An error occurred while deactivating account for user with username {username}: {exc}' + logger.error( + f"[LMSAPIClient.deactivate_user] Failed to deactivate account for user '{username}', " + f"(triggered by CT subscription Message ID: {ct_message_id}). Error: {exc}" ) raise @@ -62,11 +63,11 @@ def enroll_user_in_course( fulfillment_logging_obj ): """ - Send a POST request to LMS Enrollment API endpoint + Send a POST request to LMS Enrollment API endpoint. Arguments: enrollment_data: dictionary to send to the API resource. Returns: - dict: Dictionary represention of JSON returned from API + dict: Dictionary representation of JSON returned from API. """ return self.post( url=self.api_enrollment_base_url, @@ -78,7 +79,7 @@ def enroll_user_in_course( def post(self, url, json, line_item_state_payload, logging_obj, timeout=None): """ - Send a POST request to a url with json payload. + Send a POST request to a URL with JSON payload. """ if not timeout: # pragma no cover timeout = self.normal_timeout @@ -97,33 +98,31 @@ def post(self, url, json, line_item_state_payload, logging_obj, timeout=None): timeout=timeout, ) response.raise_for_status() - self.log_request_response(logger, response) fulfill_line_item_state_payload = { **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']}]" + f"[LMSAPIClient.post] LMS fulfillment successful for user '{logging_obj['user']}'. " + f"Details: lms_user_id: {logging_obj['lms_user_id']}, CT order ID: {logging_obj['order_id']}, " + f"course ID: {logging_obj['course_id']}, CT subscription 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) + context_prefix = (f"[LMSAPIClient] lms_user_id:{logging_obj['lms_user_id']}, " + f"CT order ID: {logging_obj['order_id']}" + f"course ID: {logging_obj['course_id']}, celery task ID: {logging_obj['celery_task_id']}") + self.log_request_exception(context_prefix, logger, exc) + fulfill_line_item_state_payload = { **line_item_state_payload, 'is_fulfilled': False } + fulfillment_completed_signal.send_robust( 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( diff --git a/commerce_coordinator/apps/lms/tasks.py b/commerce_coordinator/apps/lms/tasks.py index c276f97ba..85407a84b 100644 --- a/commerce_coordinator/apps/lms/tasks.py +++ b/commerce_coordinator/apps/lms/tasks.py @@ -7,6 +7,7 @@ from celery import Task, shared_task from celery.utils.log import get_task_logger +from commercetools import CommercetoolsError from django.contrib.auth import get_user_model from requests import RequestException @@ -68,7 +69,7 @@ def on_failure(self, exc, task_id, args, kwargs, einfo): @shared_task( bind=True, - autoretry_for=(RequestException,), + autoretry_for=(RequestException, CommercetoolsError), retry_kwargs={'max_retries': 5, 'countdown': 3}, base=CourseEnrollTaskAfterReturn, ) @@ -95,10 +96,8 @@ def fulfill_order_placed_send_enroll_in_course_task( """ Celery task for order placed fulfillment and enrollment via LMS Enrollment API. """ - logger.info( - f'LMS fulfill_order_placed_send_enroll_in_course_task fired with {locals()},' - ) - + tag = "fulfill_order_placed_send_enroll_in_course_task" + logger.info(f"{tag} Starting task with details: {locals()}.") user = User.objects.get(lms_user_id=edx_lms_user_id) enrollment_data = { @@ -147,6 +146,8 @@ def fulfill_order_placed_send_enroll_in_course_task( # Updating the order version and stateID after the transition to 'Fulfillment Failure' if self.request.retries > 0: + logger.warning(f"{tag} " + f"Task retry count# {self.request.retries} for CT order ID {order_id}.") client = CommercetoolsAPIClient() # A retry means the current line item state on the order would be a failure state line_item_state_id = client.get_state_by_key(TwoUKeys.FAILURE_FULFILMENT_STATE).id diff --git a/commerce_coordinator/apps/titan/clients.py b/commerce_coordinator/apps/titan/clients.py index 1059b14eb..69734e06a 100644 --- a/commerce_coordinator/apps/titan/clients.py +++ b/commerce_coordinator/apps/titan/clients.py @@ -61,9 +61,8 @@ def _request(self, request_method, resource_path, params=None, json=None, header headers=headers, ) response.raise_for_status() - self.log_request_response(logger, response) except RequestException as exc: - self.log_request_exception(logger, exc) + self.log_request_exception("[TitanAPIClient]", logger, exc) raise return response.json()