diff --git a/commerce_coordinator/apps/commercetools/clients.py b/commerce_coordinator/apps/commercetools/clients.py index ed7c38b26..1d2ec44cb 100644 --- a/commerce_coordinator/apps/commercetools/clients.py +++ b/commerce_coordinator/apps/commercetools/clients.py @@ -13,17 +13,9 @@ from commercetools import Client as CTClient from commercetools import CommercetoolsError from commercetools.platform.models import Customer as CTCustomer -from commercetools.platform.models import ( - CustomerChangeEmailAction, - CustomerSetCustomFieldAction, -) -from commercetools.platform.models import ( - CustomerSetCustomTypeAction as CTCustomerSetCustomTypeAction, -) -from commercetools.platform.models import ( - CustomerSetFirstNameAction, - CustomerSetLastNameAction, -) +from commercetools.platform.models import CustomerChangeEmailAction, CustomerSetCustomFieldAction +from commercetools.platform.models import CustomerSetCustomTypeAction as CTCustomerSetCustomTypeAction +from commercetools.platform.models import CustomerSetFirstNameAction, CustomerSetLastNameAction from commercetools.platform.models import FieldContainer as CTFieldContainer from commercetools.platform.models import Money as CTMoney from commercetools.platform.models import Order as CTOrder @@ -31,13 +23,10 @@ OrderAddReturnInfoAction, OrderSetReturnItemCustomTypeAction, OrderSetReturnPaymentStateAction, - OrderTransitionLineItemStateAction, + OrderTransitionLineItemStateAction ) from commercetools.platform.models import Payment as CTPayment -from commercetools.platform.models import ( - PaymentAddTransactionAction, - PaymentSetTransactionCustomTypeAction, -) +from commercetools.platform.models import PaymentAddTransactionAction, PaymentSetTransactionCustomTypeAction from commercetools.platform.models import ProductVariant as CTProductVariant from commercetools.platform.models import ( ReturnItemDraft, @@ -45,35 +34,24 @@ ReturnShipmentState, StateResourceIdentifier, TransactionDraft, - TransactionType, + TransactionType ) from commercetools.platform.models import Type as CTType from commercetools.platform.models import TypeDraft as CTTypeDraft -from commercetools.platform.models import ( - TypeResourceIdentifier as CTTypeResourceIdentifier, -) +from commercetools.platform.models import TypeResourceIdentifier as CTTypeResourceIdentifier from commercetools.platform.models.state import State as CTLineItemState from django.conf import settings from openedx_filters.exceptions import OpenEdxFilterException -from commerce_coordinator.apps.commercetools.catalog_info.constants import ( - DEFAULT_ORDER_EXPANSION, - EDX_PAYPAL_PAYMENT_INTERFACE_NAME, - EdXFieldNames, -) -from commerce_coordinator.apps.commercetools.catalog_info.foundational_types import ( - TwoUCustomTypes, -) +from commerce_coordinator.apps.commercetools.catalog_info.constants import DEFAULT_ORDER_EXPANSION, EdXFieldNames +from commerce_coordinator.apps.commercetools.catalog_info.foundational_types import TwoUCustomTypes from commerce_coordinator.apps.commercetools.utils import ( find_refund_transaction, handle_commercetools_error, - translate_refund_status_to_transaction_status, + translate_refund_status_to_transaction_status ) from commerce_coordinator.apps.core.constants import ORDER_HISTORY_PER_SYSTEM_REQ_LIMIT -from typing import TypedDict, Union -from commerce_coordinator.apps.commercetools.catalog_info.constants import EDX_STRIPE_PAYMENT_INTERFACE_NAME - logger = logging.getLogger(__name__) T = TypeVar("T") @@ -82,8 +60,7 @@ class PaginatedResult(Generic[T]): - """Planned paginated response wrapper""" - + """ Planned paginated response wrapper """ results: List[T] total: int offset: int @@ -107,17 +84,8 @@ def rebuild(self, results: List[T]): return PaginatedResult(results, total=self.total, offset=self.offset) -class Refund(TypedDict): - id: str - amount: Union[str, int] - currency: str - created: Union[str, int] - status: str - - class CommercetoolsAPIClient: - """Commercetools API Client""" - + """ Commercetools API Client """ base_client = None def __init__(self): @@ -133,10 +101,10 @@ def __init__(self): self.base_client = CTClient( client_id=config["clientId"], client_secret=config["clientSecret"], - scope=config["scopes"].split(" "), + scope=config["scopes"].split(' '), url=config["apiUrl"], token_url=config["authUrl"], - project_key=config["projectKey"], + project_key=config["projectKey"] ) def ensure_custom_type_exists(self, type_def: CTTypeDraft) -> Optional[CTType]: @@ -156,9 +124,7 @@ def ensure_custom_type_exists(self, type_def: CTTypeDraft) -> Optional[CTType]: except CommercetoolsError as _: # pragma: no cover # commercetools.exceptions.CommercetoolsError: The Resource with key 'edx-user_information' was not found. pass - except ( - requests.exceptions.HTTPError - ) as _: # The test framework doesn't wrap to CommercetoolsError + except requests.exceptions.HTTPError as _: # The test framework doesn't wrap to CommercetoolsError pass if not type_exists: @@ -166,9 +132,7 @@ def ensure_custom_type_exists(self, type_def: CTTypeDraft) -> Optional[CTType]: return type_object - def tag_customer_with_lms_user_info( - self, customer: CTCustomer, lms_user_id: int, lms_user_name: str - ) -> CTCustomer: + def tag_customer_with_lms_user_info(self, customer: CTCustomer, lms_user_id: int, lms_user_name: str) -> CTCustomer: """ Updates a CoCo Customer Object with what we are currently using for LMS Identifiers Args: @@ -182,38 +146,28 @@ def tag_customer_with_lms_user_info( # All updates to CT Core require the version of the object you are working on as protection from out of band # updates; this does mean we have to fetch every (primary) object we want to chain. - type_object = self.ensure_custom_type_exists( - TwoUCustomTypes.CUSTOMER_TYPE_DRAFT - ) + type_object = self.ensure_custom_type_exists(TwoUCustomTypes.CUSTOMER_TYPE_DRAFT) # A customer can only have one custom type associated to it, and thus only one set of custom fields. THUS... # They can't be required, and shouldn't entirely be relied upon; Once a proper Type is changed, the old values # are LOST. if customer.custom and not customer.custom.type.id == type_object.id: - raise ValueError( - "User already has a custom type, and its not the one were expecting, Refusing to update. " - "(Updating will eradicate the values from the other type, as an object may only have one " - "Custom Type)" - ) + raise ValueError("User already has a custom type, and its not the one were expecting, Refusing to update. " + "(Updating will eradicate the values from the other type, as an object may only have one " + "Custom Type)") - ret = self.base_client.customers.update_by_id( - customer.id, - customer.version, - actions=[ - CTCustomerSetCustomTypeAction( - type=CTTypeResourceIdentifier( - key=TwoUCustomTypes.CUSTOMER_TYPE_DRAFT.key, - ), - fields=CTFieldContainer( - { - EdXFieldNames.LMS_USER_ID: f"{lms_user_id}", - EdXFieldNames.LMS_USER_NAME: lms_user_name, - } - ), + ret = self.base_client.customers.update_by_id(customer.id, customer.version, actions=[ + CTCustomerSetCustomTypeAction( + type=CTTypeResourceIdentifier( + key=TwoUCustomTypes.CUSTOMER_TYPE_DRAFT.key, ), - ], - ) + fields=CTFieldContainer({ + EdXFieldNames.LMS_USER_ID: f"{lms_user_id}", + EdXFieldNames.LMS_USER_NAME: lms_user_name + }) + ), + ]) return ret @@ -229,17 +183,15 @@ def get_customer_by_lms_user_id(self, lms_user_id: int) -> Optional[CTCustomer]: is returned. """ - logger.info( - f"[CommercetoolsAPIClient] - Attempting to get customer with LMS user id: {lms_user_id}" - ) + logger.info(f"[CommercetoolsAPIClient] - Attempting to get customer with LMS user id: {lms_user_id}") edx_lms_user_id_key = EdXFieldNames.LMS_USER_ID start_time = datetime.datetime.now() results = self.base_client.customers.query( - where=f"custom(fields({edx_lms_user_id_key}=:id))", + where=f'custom(fields({edx_lms_user_id_key}=:id))', limit=2, - predicate_var={"id": f"{lms_user_id}"}, + predicate_var={'id': f"{lms_user_id}"} ) duration = (datetime.datetime.now() - start_time).total_seconds() logger.info(f"[Performance Check] - customerId query took {duration} seconds") @@ -247,29 +199,20 @@ def get_customer_by_lms_user_id(self, lms_user_id: int) -> Optional[CTCustomer]: if results.count > 1: # We are unable due to CT Limitations to enforce unique LMS ID values on Customers on the catalog side, so # let's do a backhanded check by trying to pull 2 users and erroring if we find a discrepancy. - logger.info( - f"[CommercetoolsAPIClient] - More than one customer found with LMS " - f"user id: {lms_user_id}, raising error" - ) - raise ValueError( - "More than one user was returned from the catalog with this edX LMS User ID, these must " - "be unique." - ) + logger.info(f"[CommercetoolsAPIClient] - More than one customer found with LMS " + f"user id: {lms_user_id}, raising error") + raise ValueError("More than one user was returned from the catalog with this edX LMS User ID, these must " + "be unique.") if results.count == 0: - logger.info( - f"[CommercetoolsAPIClient] - No customer found with LMS user id: {lms_user_id}" - ) + logger.info(f"[CommercetoolsAPIClient] - No customer found with LMS user id: {lms_user_id}") return None else: - logger.info( - f"[CommercetoolsAPIClient] - Customer found with LMS user id: {lms_user_id}" - ) + logger.info(f"[CommercetoolsAPIClient] - Customer found with LMS user id: {lms_user_id}") return results.results[0] - def get_order_by_id( - self, order_id: str, expand: ExpandList = DEFAULT_ORDER_EXPANSION - ) -> CTOrder: + def get_order_by_id(self, order_id: str, expand: ExpandList = DEFAULT_ORDER_EXPANSION) \ + -> CTOrder: """ Fetch an order by the Order ID (UUID) @@ -279,14 +222,11 @@ def get_order_by_id( Returns (CTOrder): Order with Expanded Properties """ - logger.info( - f"[CommercetoolsAPIClient] - Attempting to find order with id: {order_id}" - ) + logger.info(f"[CommercetoolsAPIClient] - Attempting to find order with id: {order_id}") return self.base_client.orders.get_by_id(order_id, expand=list(expand)) - def get_order_by_number( - self, order_number: str, expand: ExpandList = DEFAULT_ORDER_EXPANSION - ) -> CTOrder: + def get_order_by_number(self, order_number: str, expand: ExpandList = DEFAULT_ORDER_EXPANSION) \ + -> CTOrder: """ Fetch an order by the Order Number (Human readable order number) @@ -296,21 +236,14 @@ def get_order_by_number( Returns (CTOrder): Order with Expanded Properties """ - logger.info( - f"[CommercetoolsAPIClient] - Attempting to find order with number {order_number}" - ) - return self.base_client.orders.get_by_order_number( - order_number, expand=list(expand) - ) + logger.info(f"[CommercetoolsAPIClient] - Attempting to find order with number {order_number}") + return self.base_client.orders.get_by_order_number(order_number, expand=list(expand)) + + def get_orders(self, customer_id: str, offset=0, + limit=ORDER_HISTORY_PER_SYSTEM_REQ_LIMIT, + expand: ExpandList = DEFAULT_ORDER_EXPANSION, + order_state="Complete") -> PaginatedResult[CTOrder]: - def get_orders( - self, - customer_id: str, - offset=0, - limit=ORDER_HISTORY_PER_SYSTEM_REQ_LIMIT, - expand: ExpandList = DEFAULT_ORDER_EXPANSION, - order_state="Complete", - ) -> PaginatedResult[CTOrder]: """ Call commercetools API overview endpoint for data about historical orders. @@ -326,35 +259,27 @@ def get_orders( See sample response in tests.py """ - logger.info( - f"[CommercetoolsAPIClient] - Attempting to find all completed orders for " - f"customer with ID {customer_id}" - ) - order_where_clause = f'orderState="{order_state}"' + logger.info(f"[CommercetoolsAPIClient] - Attempting to find all completed orders for " + f"customer with ID {customer_id}") + order_where_clause = f"orderState=\"{order_state}\"" start_time = datetime.datetime.now() values = self.base_client.orders.query( where=["customerId=:cid", order_where_clause], - predicate_var={"cid": customer_id}, + predicate_var={'cid': customer_id}, sort=["completedAt desc", "lastModifiedAt desc"], limit=limit, offset=offset, - expand=list(expand), + expand=list(expand) ) duration = (datetime.datetime.now() - start_time).total_seconds() logger.info(f"[Performance Check] get_orders call took {duration} seconds") return PaginatedResult(values.results, values.total, values.offset) - def get_orders_for_customer( - self, - edx_lms_user_id: int, - offset=0, - limit=ORDER_HISTORY_PER_SYSTEM_REQ_LIMIT, - customer_id=None, - email=None, - username=None, - ) -> (PaginatedResult[CTOrder], CTCustomer): + def get_orders_for_customer(self, edx_lms_user_id: int, offset=0, limit=ORDER_HISTORY_PER_SYSTEM_REQ_LIMIT, + customer_id=None, email=None, + username=None) -> (PaginatedResult[CTOrder], CTCustomer): """ Args: @@ -366,21 +291,21 @@ def get_orders_for_customer( customer = self.get_customer_by_lms_user_id(edx_lms_user_id) if customer is None: # pragma: no cover - raise ValueError( - f"Unable to locate customer with ID #{edx_lms_user_id}" - ) + raise ValueError(f'Unable to locate customer with ID #{edx_lms_user_id}') customer_id = customer.id else: if email is None or username is None: # pragma: no cover - raise ValueError( - "If customer_id is provided, both email and username must be provided" - ) + raise ValueError("If customer_id is provided, both email and username must be provided") customer = SimpleNamespace( id=customer_id, email=email, - custom=SimpleNamespace(fields={EdXFieldNames.LMS_USER_NAME: username}), + custom=SimpleNamespace( + fields={ + EdXFieldNames.LMS_USER_NAME: username + } + ) ) orders = self.get_orders(customer_id, offset, limit) @@ -388,80 +313,58 @@ def get_orders_for_customer( return orders, customer def get_customer_by_id(self, customer_id: str) -> CTCustomer: - logger.info( - f"[CommercetoolsAPIClient] - Attempting to find customer with ID {customer_id}" - ) + logger.info(f"[CommercetoolsAPIClient] - Attempting to find customer with ID {customer_id}") return self.base_client.customers.get_by_id(customer_id) def get_state_by_id(self, state_id: str) -> CTLineItemState: - logger.info( - f"[CommercetoolsAPIClient] - Attempting to find state with id {state_id}" - ) + logger.info(f"[CommercetoolsAPIClient] - Attempting to find state with id {state_id}") return self.base_client.states.get_by_id(state_id) def get_state_by_key(self, state_key: str) -> CTLineItemState: - logger.info( - f"[CommercetoolsAPIClient] - Attempting to find state with key {state_key}" - ) + logger.info(f"[CommercetoolsAPIClient] - Attempting to find state with key {state_key}") return self.base_client.states.get_by_key(state_key) def get_payment_by_key(self, payment_key: str) -> CTPayment: - logger.info( - f"[CommercetoolsAPIClient] - Attempting to find payment with key {payment_key}" - ) + logger.info(f"[CommercetoolsAPIClient] - Attempting to find payment with key {payment_key}") return self.base_client.payments.get_by_key(payment_key) - def get_payment_by_transaction_interaction_id( - self, interaction_id: str - ) -> CTPayment: - logger.info( - f"[CommercetoolsAPIClient] - Attempting to find payment with interaction ID {interaction_id}" - ) - return self.base_client.payments.query( - where=f'transactions(interactionId="{interaction_id}")' - ).results[0] - - def get_product_variant_by_course_run( - self, cr_id: str - ) -> Optional[CTProductVariant]: + def get_product_variant_by_course_run(self, cr_id: str) -> Optional[CTProductVariant]: """ Args: cr_id: variant course run key """ start_time = datetime.datetime.now() - results = self.base_client.product_projections.search( - False, filter=f'variants.sku:"{cr_id}"' - ).results + results = self.base_client.product_projections.search(False, filter=f"variants.sku:\"{cr_id}\"").results duration = (datetime.datetime.now() - start_time).total_seconds() logger.info( - f"[Performance Check] get_product_variant_by_course_run took {duration} seconds" - ) + f"[Performance Check] get_product_variant_by_course_run took {duration} seconds") if len(results) < 1: # pragma no cover return None # Make 2D List of all variants from all results, and then flatten - all_variants = [ - listitem - for sublist in list( - map( - lambda selection: [selection.master_variant, *selection.variants], - results, - ) + all_variants = [listitem for sublist in + list( + map( + lambda selection: [selection.master_variant, *selection.variants], + results + ) + ) + for listitem in sublist] + + matching_variant_list = list( + filter( + lambda v: v.sku == cr_id, + all_variants ) - for listitem in sublist - ] - - matching_variant_list = list(filter(lambda v: v.sku == cr_id, all_variants)) + ) if len(matching_variant_list) < 1: # pragma no cover return None return matching_variant_list[0] - def create_return_for_order( - self, order_id: str, order_version: int, order_line_item_id: str - ) -> CTOrder: + def create_return_for_order(self, order_id: str, order_version: int, order_line_item_id: str) -> CTOrder: """ Creates refund/return for Commercetools order Args: @@ -473,10 +376,8 @@ def create_return_for_order( """ try: - return_item_draft_comment = ( - f"Creating return item for order {order_id} with " - f"order line item ID {order_line_item_id}" - ) + return_item_draft_comment = f'Creating return item for order {order_id} with ' \ + f'order line item ID {order_line_item_id}' logger.info(f"[CommercetoolsAPIClient] - {return_item_draft_comment}") @@ -487,26 +388,25 @@ def create_return_for_order( shipment_state=ReturnShipmentState.RETURNED, ) - add_return_info_action = OrderAddReturnInfoAction(items=[return_item_draft]) + add_return_info_action = OrderAddReturnInfoAction( + items=[return_item_draft] + ) returned_order = self.base_client.orders.update_by_id( - id=order_id, version=order_version, actions=[add_return_info_action] + id=order_id, + version=order_version, + actions=[add_return_info_action] ) return returned_order except CommercetoolsError as err: - handle_commercetools_error( - err, f"Unable to create return for order {order_id}" - ) + handle_commercetools_error(err, f"Unable to create return for order {order_id}") raise err - def update_return_payment_state_after_successful_refund( - self, - order_id: str, - order_version: int, - return_line_item_return_id: str, - payment_intent_id: str, - amount_in_cents: decimal, - ) -> Union[CTOrder, None]: + def update_return_payment_state_after_successful_refund(self, order_id: str, + order_version: int, + return_line_item_return_id: str, + payment_intent_id: str, + amount_in_cents: decimal) -> Union[CTOrder, None]: """ Update paymentState on the LineItemReturnItem attached to the order. Updated by the Order ID (UUID) @@ -520,110 +420,80 @@ def update_return_payment_state_after_successful_refund( Raises Exception: Error if update was unsuccessful. """ try: - logger.info( - f"[CommercetoolsAPIClient] - Updating payment state for return " - f"with id {return_line_item_return_id} to '{ReturnPaymentState.REFUNDED}'." - ) + logger.info(f"[CommercetoolsAPIClient] - Updating payment state for return " + f"with id {return_line_item_return_id} to '{ReturnPaymentState.REFUNDED}'.") return_payment_state_action = OrderSetReturnPaymentStateAction( return_item_id=return_line_item_return_id, - payment_state=ReturnPaymentState.REFUNDED, + payment_state=ReturnPaymentState.REFUNDED ) if not payment_intent_id: - payment_intent_id = "" - logger.info( - f"Creating return for order - payment_intent_id: {payment_intent_id}" - ) + payment_intent_id = '' + logger.info(f'Creating return for order - payment_intent_id: {payment_intent_id}') payment = self.get_payment_by_key(payment_intent_id) logger.info(f"Payment found: {payment}") transaction_id = find_refund_transaction(payment, amount_in_cents) update_transaction_id_action = OrderSetReturnItemCustomTypeAction( return_item_id=return_line_item_return_id, type=CTTypeResourceIdentifier( - key="returnItemCustomType", + key='returnItemCustomType', ), - fields=CTFieldContainer({"transactionId": transaction_id}), - ) - return_transaction_return_item_action = ( - PaymentSetTransactionCustomTypeAction( - transaction_id=transaction_id, - type=CTTypeResourceIdentifier(key="transactionCustomType"), - fields=CTFieldContainer( - {"returnItemId": return_line_item_return_id} - ), - ) + fields=CTFieldContainer({ + 'transactionId': transaction_id + }) ) - logger.info( - f"Update return payment state after successful refund - payment_intent_id: {payment_intent_id}" + return_transaction_return_item_action = PaymentSetTransactionCustomTypeAction( + transaction_id=transaction_id, + type=CTTypeResourceIdentifier(key='transactionCustomType'), + fields=CTFieldContainer({ + 'returnItemId': return_line_item_return_id + }) ) + logger.info(f"Update return payment state after successful refund - payment_intent_id: {payment_intent_id}") updated_order = self.base_client.orders.update_by_id( id=order_id, version=order_version, - actions=[return_payment_state_action, update_transaction_id_action], + actions=[return_payment_state_action, update_transaction_id_action] ) self.base_client.payments.update_by_id( id=payment.id, version=payment.version, - actions=[return_transaction_return_item_action], + actions=[return_transaction_return_item_action] ) 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(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: - """ - Pre process refund object based on PSP - """ - if psp == EDX_PAYPAL_PAYMENT_INTERFACE_NAME: - refund["amount"] = float(refund["amount"]) * 100 - refund["created"] = datetime.datetime.fromisoformat(refund["created"]) - else: - refund["created"] = datetime.datetime.utcfromtimestamp(refund["created"]) - - refund["status"] = translate_refund_status_to_transaction_status(refund["status"]) - refund["currency"] = refund["currency"].upper() - return refund def create_return_payment_transaction( - self, - payment_id: str, - payment_version: int, - refund: Refund, - psp=EDX_STRIPE_PAYMENT_INTERFACE_NAME - ) -> CTPayment: + self, payment_id: str, + payment_version: int, + stripe_refund: stripe.Refund) -> CTPayment: """ Create Commercetools payment transaction for refund Args: payment_id (str): Payment ID (UUID) payment_version (int): Current version of payment - refund (stripe.Refund): Refund object + stripe_refund (stripe.Refund): Stripe's refund object Returns (CTPayment): Updated payment object or Raises Exception: Error if creation was unsuccessful. """ try: - logger.info( - f"[CommercetoolsAPIClient] - Creating refund transaction for payment with ID {payment_id} " - f"following successful refund {refund['id']}" - ) - refund = self._preprocess_refund_object(refund, psp) + logger.info(f"[CommercetoolsAPIClient] - Creating refund transaction for payment with ID {payment_id} " + f"following successful Stripe refund {stripe_refund.id}") amount_as_money = CTMoney( - cent_amount=float( - refund["amount"] - ), - currency_code=refund["currency"], + cent_amount=stripe_refund.amount, + currency_code=stripe_refund.currency.upper() ) transaction_draft = TransactionDraft( type=TransactionType.REFUND, amount=amount_as_money, - timestamp=refund["created"], - state=refund["status"], - interaction_id=refund["id"], + timestamp=datetime.datetime.utcfromtimestamp(stripe_refund.created), + state=translate_refund_status_to_transaction_status(stripe_refund.status), + interaction_id=stripe_refund.id ) add_transaction_action = PaymentAddTransactionAction( @@ -631,27 +501,21 @@ def create_return_payment_transaction( ) returned_payment = self.base_client.payments.update_by_id( - id=payment_id, version=payment_version, actions=[add_transaction_action] + id=payment_id, + version=payment_version, + actions=[add_transaction_action] ) return returned_payment except CommercetoolsError as err: - context = ( - f"Unable to create refund payment transaction for " - f"payment {payment_id} and refund {refund['id']}" - ) + context = f"Unable to create refund payment transaction for "\ + f"payment {payment_id} and stripe refund {stripe_refund.id}" handle_commercetools_error(err, context) raise err - def update_line_item_transition_state_on_fulfillment( - self, - order_id: str, - order_version: int, - line_item_id: str, - item_quantity: int, - from_state_id: str, - new_state_key: str, - ) -> CTOrder: + def update_line_item_transition_state_on_fulfillment(self, order_id: str, order_version: int, + line_item_id: str, item_quantity: int, + from_state_id: str, new_state_key: str) -> CTOrder: """ Update Commercetools order line item state Args: @@ -668,10 +532,8 @@ def update_line_item_transition_state_on_fulfillment( from_state_key = self.get_state_by_id(from_state_id).key - logger.info( - f"[CommercetoolsAPIClient] - Transitioning line item state for order with ID {order_id}" - f"from {from_state_key} to {new_state_key}" - ) + logger.info(f"[CommercetoolsAPIClient] - Transitioning line item state for order with ID {order_id}" + f"from {from_state_key} to {new_state_key}") try: if new_state_key != from_state_key: @@ -679,40 +541,28 @@ def update_line_item_transition_state_on_fulfillment( line_item_id=line_item_id, quantity=item_quantity, from_state=StateResourceIdentifier(key=from_state_key), - to_state=StateResourceIdentifier(key=new_state_key), + to_state=StateResourceIdentifier(key=new_state_key) ) - updated_fulfillment_line_item_order = ( - self.base_client.orders.update_by_id( - id=order_id, - version=order_version, - actions=[transition_line_item_state_action], - ) + updated_fulfillment_line_item_order = self.base_client.orders.update_by_id( + id=order_id, + version=order_version, + actions=[transition_line_item_state_action], ) 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" - ) + logger.info(f"The line item {line_item_id} already has the correct state {new_state_key}. " + "Not attempting to transition LineItemState") return self.get_order_by_id(order_id) except CommercetoolsError as err: # 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(err, f"Unable to update LineItemState of order {order_id}", True) return None - def retire_customer_anonymize_fields( - self, - customer_id: str, - customer_version: int, - retired_first_name: str, - retired_last_name: str, - retired_email: str, - retired_lms_username: str, - ) -> CTCustomer: + def retire_customer_anonymize_fields(self, customer_id: str, customer_version: int, + retired_first_name: str, retired_last_name: str, + retired_email: str, retired_lms_username: str) -> CTCustomer: """ Update Commercetools customer with anonymized fields Args: @@ -735,30 +585,31 @@ def retire_customer_anonymize_fields( last_name=retired_last_name ) - update_retired_email_action = CustomerChangeEmailAction(email=retired_email) + update_retired_email_action = CustomerChangeEmailAction( + email=retired_email + ) update_retired_lms_username_action = CustomerSetCustomFieldAction( - name="edx-lms_user_name", value=retired_lms_username + name="edx-lms_user_name", + value=retired_lms_username ) - actions.extend( - [ - update_retired_first_name_action, - update_retired_last_name_action, - update_retired_email_action, - update_retired_lms_username_action, - ] - ) + actions.extend([ + update_retired_first_name_action, + update_retired_last_name_action, + update_retired_email_action, + update_retired_lms_username_action + ]) try: retired_customer = self.base_client.customers.update_by_id( - id=customer_id, version=customer_version, actions=actions + id=customer_id, + version=customer_version, + actions=actions ) return retired_customer except CommercetoolsError as err: - logger.error( - f"[CommercetoolsError] Unable to anonymize customer fields for customer " - f"with ID: {customer_id}, after LMS retirement with " - f"error correlation id {err.correlation_id} and error/s: {err.errors}" - ) + logger.error(f"[CommercetoolsError] Unable to anonymize customer fields for customer " + f"with ID: {customer_id}, after LMS retirement with " + f"error correlation id {err.correlation_id} and error/s: {err.errors}") raise err diff --git a/commerce_coordinator/apps/commercetools/tests/test_utils.py b/commerce_coordinator/apps/commercetools/tests/test_utils.py index 43c7e53b8..efe6e417b 100644 --- a/commerce_coordinator/apps/commercetools/tests/test_utils.py +++ b/commerce_coordinator/apps/commercetools/tests/test_utils.py @@ -29,7 +29,7 @@ has_refund_transaction, send_order_confirmation_email, send_unsupported_mode_fulfillment_error_email, - translate_stripe_refund_status_to_transaction_status + translate_refund_status_to_transaction_status ) @@ -313,17 +313,17 @@ class TestTranslateStripeRefundStatus(unittest.TestCase): """ def test_translate_stripe_refund_status_succeeded(self): - self.assertEqual(translate_stripe_refund_status_to_transaction_status('succeeded'), TransactionState.SUCCESS) + self.assertEqual(translate_refund_status_to_transaction_status('succeeded'), TransactionState.SUCCESS) def test_translate_stripe_refund_status_pending(self): - self.assertEqual(translate_stripe_refund_status_to_transaction_status('pending'), TransactionState.PENDING) + self.assertEqual(translate_refund_status_to_transaction_status('pending'), TransactionState.PENDING) def test_translate_stripe_refund_status_failed(self): - self.assertEqual(translate_stripe_refund_status_to_transaction_status('failed'), TransactionState.FAILURE) + self.assertEqual(translate_refund_status_to_transaction_status('failed'), TransactionState.FAILURE) def test_translate_stripe_refund_status_other(self): # Test for an unknown status - self.assertEqual(translate_stripe_refund_status_to_transaction_status('unknown_status'), 'unknown_status') + self.assertEqual(translate_refund_status_to_transaction_status('unknown_status'), 'unknown_status') class TestRetirementAnonymizingTestCase(unittest.TestCase):