From e55e6ce852a1026e30a0a85889728cc033798187 Mon Sep 17 00:00:00 2001 From: elitonzky Date: Wed, 13 Nov 2024 10:03:31 -0300 Subject: [PATCH 01/12] feat: add new fields to vtex integration --- retail/api/features/views.py | 6 ++++ ..._vtex_integrate_feature_config_and_more.py | 28 +++++++++++++++++++ retail/features/models.py | 4 +++ 3 files changed, 38 insertions(+) create mode 100644 retail/features/migrations/0015_feature_can_vtex_integrate_feature_config_and_more.py diff --git a/retail/api/features/views.py b/retail/api/features/views.py index 3240342..9dc2a51 100644 --- a/retail/api/features/views.py +++ b/retail/api/features/views.py @@ -13,6 +13,7 @@ class FeaturesView(BaseServiceView): def get(self, request, project_uuid: str): try: category = request.query_params.get("category", None) + can_vtex_integrate = request.query_params.get("can_vtex_integrate", None) integrated_features = IntegratedFeature.objects.filter( project__uuid=project_uuid @@ -26,11 +27,16 @@ def get(self, request, project_uuid: str): for email in settings.EMAILS_CAN_TESTING: if email in request.user.email: can_testing = True + if not can_testing: features = features.exclude(status="testing") + if category: features = features.filter(category=category) + if can_vtex_integrate: + features = features.filter(can_vtex_integrate=can_vtex_integrate) + serializer = FeaturesSerializer(features, many=True) usecase = RemoveGlobalsKeysUsecase( diff --git a/retail/features/migrations/0015_feature_can_vtex_integrate_feature_config_and_more.py b/retail/features/migrations/0015_feature_can_vtex_integrate_feature_config_and_more.py new file mode 100644 index 0000000..a4b86a8 --- /dev/null +++ b/retail/features/migrations/0015_feature_can_vtex_integrate_feature_config_and_more.py @@ -0,0 +1,28 @@ +# Generated by Django 5.1.1 on 2024-11-13 13:02 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ("features", "0014_feature_status_alter_featureversion_action_types"), + ] + + operations = [ + migrations.AddField( + model_name="feature", + name="can_vtex_integrate", + field=models.BooleanField(default=False), + ), + migrations.AddField( + model_name="feature", + name="config", + field=models.JSONField(default=dict), + ), + migrations.AddField( + model_name="integratedfeature", + name="created_by_vtex", + field=models.BooleanField(default=False), + ), + ] diff --git a/retail/features/models.py b/retail/features/models.py index 87d7f99..023b658 100644 --- a/retail/features/models.py +++ b/retail/features/models.py @@ -43,6 +43,9 @@ class Feature(models.Model): blank=True ) + can_vtex_integrate = models.BooleanField(default=False) + config = models.JSONField(default=dict) + def __str__(self): return self.name @@ -139,6 +142,7 @@ class IntegratedFeature(models.Model): User, on_delete=models.CASCADE, related_name="integrated_features" ) integrated_on = models.DateField(auto_now_add=True) + created_by_vtex = models.BooleanField(default=False) # def save(self, *args) -> None: # self.feature = self.feature_version.feature From 1219f9177d3cd05f7172b5c4ce03bc331fed9d1c Mon Sep 17 00:00:00 2001 From: elitonzky Date: Wed, 13 Nov 2024 15:49:06 -0300 Subject: [PATCH 02/12] chore: convert string to boolean --- retail/api/features/views.py | 2 ++ 1 file changed, 2 insertions(+) diff --git a/retail/api/features/views.py b/retail/api/features/views.py index 9dc2a51..2058597 100644 --- a/retail/api/features/views.py +++ b/retail/api/features/views.py @@ -35,6 +35,8 @@ def get(self, request, project_uuid: str): features = features.filter(category=category) if can_vtex_integrate: + # Convert "true"/"false" to boolean + can_vtex_integrate = can_vtex_integrate == 'true' features = features.filter(can_vtex_integrate=can_vtex_integrate) serializer = FeaturesSerializer(features, many=True) From fe76915637395dc05e8f9cc7da4be820bfc2d5b9 Mon Sep 17 00:00:00 2001 From: elitonzky Date: Thu, 14 Nov 2024 18:21:17 -0300 Subject: [PATCH 03/12] chore: add PopulateDefaultsUseCase --- retail/api/integrated_feature/views.py | 18 ++++++ .../populate_globals_with_defaults.py | 58 +++++++++++++++++++ 2 files changed, 76 insertions(+) create mode 100644 retail/api/usecases/populate_globals_with_defaults.py diff --git a/retail/api/integrated_feature/views.py b/retail/api/integrated_feature/views.py index cb5f0a0..6468c37 100644 --- a/retail/api/integrated_feature/views.py +++ b/retail/api/integrated_feature/views.py @@ -7,6 +7,7 @@ from retail.api.integrated_feature.serializers import IntegratedFeatureSerializer from retail.api.usecases.populate_globals_values import PopulateGlobalsValuesUsecase +from retail.api.usecases.populate_globals_with_defaults import PopulateDefaultsUseCase from retail.features.models import Feature, IntegratedFeature from retail.features.integrated_feature_eda import IntegratedFeatureEDA from retail.projects.models import Project @@ -15,6 +16,9 @@ class IntegratedFeatureView(BaseServiceView): def post(self, request, *args, **kwargs): feature = Feature.objects.get(uuid=kwargs["feature_uuid"]) + # Checks if the integration came from vtex + created_by_vtex = request.data.get("created_by_vtex", False) + try: project = Project.objects.get(uuid=request.data["project_uuid"]) except Project.DoesNotExist: @@ -67,6 +71,20 @@ def post(self, request, *args, **kwargs): for globals_key, globals_value in treated_globals_values.items(): integrated_feature.globals_values[globals_key] = globals_value + if created_by_vtex: + integrated_feature.created_by_vtex = created_by_vtex + + populate_defaults_use_case = PopulateDefaultsUseCase() + default_globals_values = populate_defaults_use_case.execute( + feature, globals_values_request + ) + + # Add default globals + for df_globals_key, df_globals_value in default_globals_values.items(): + integrated_feature.globals_values[df_globals_key] = df_globals_value + + integrated_feature.save() + for sector in integrated_feature.sectors: sectors_data.append( { diff --git a/retail/api/usecases/populate_globals_with_defaults.py b/retail/api/usecases/populate_globals_with_defaults.py new file mode 100644 index 0000000..be66879 --- /dev/null +++ b/retail/api/usecases/populate_globals_with_defaults.py @@ -0,0 +1,58 @@ +from retail.features.models import Feature +from typing import Dict, Any + + +class PopulateDefaultsUseCase: + """ + Use case to populate default values from a feature's configuration. + This allows for setting default global values and potentially other configurations + as specified within the feature's config field. + """ + + def execute( + self, feature: Feature, globals_values: Dict[str, Any] + ) -> Dict[str, Any]: + """ + Executes the population of default values for globals, updating + only with keys that have defined default values in the feature config. + + Args: + feature (Feature): The feature object containing configuration. + globals_values (dict): Current global values to be populated with defaults. + + Returns: + dict: Updated globals_values containing defaults for applicable keys. + """ + globals_values = self._populate_globals(feature, globals_values) + return globals_values + + def _populate_globals( + self, feature: Feature, globals_values: Dict[str, Any] + ) -> Dict[str, Any]: + """ + Populates globals with default values from the feature config, + only for keys present in default settings. + + Args: + feature (Feature): The feature object containing configuration. + globals_values (dict): Current global values. + + Returns: + dict: Dictionary with globals populated only with default keys found in feature config. + """ + # Load default values for 'globals' from the feature's `config` field + default_globals = ( + feature.config.get("vtex_config", {}) + .get("default_params", {}) + .get("globals_values", {}) + ) + + # Create a dictionary with only the keys present in `default_globals` + populated_globals = { + key: default_value + for key, default_value in default_globals.items() + if key in globals_values + or globals_values.setdefault(key, default_value) is not None + } + + return populated_globals From e509adef945631b6e8ac49ba7a65ca8e647e1c7e Mon Sep 17 00:00:00 2001 From: elitonzky Date: Mon, 18 Nov 2024 17:09:56 -0300 Subject: [PATCH 04/12] feat: refactoring integrated_feature view --- retail/api/integrated_feature/views.py | 126 +---------- .../create_integrated_feature_usecase.py | 211 ++++++++++++++++++ retail/api/usecases/remove_globals_keys.py | 4 +- 3 files changed, 221 insertions(+), 120 deletions(-) create mode 100644 retail/api/usecases/create_integrated_feature_usecase.py diff --git a/retail/api/integrated_feature/views.py b/retail/api/integrated_feature/views.py index 6468c37..a069845 100644 --- a/retail/api/integrated_feature/views.py +++ b/retail/api/integrated_feature/views.py @@ -5,6 +5,9 @@ from retail.api.base_service_view import BaseServiceView from retail.api.integrated_feature.serializers import IntegratedFeatureSerializer +from retail.api.usecases.create_integrated_feature_usecase import ( + CreateIntegratedFeatureUseCase, +) from retail.api.usecases.populate_globals_values import PopulateGlobalsValuesUsecase from retail.api.usecases.populate_globals_with_defaults import PopulateDefaultsUseCase @@ -15,128 +18,17 @@ class IntegratedFeatureView(BaseServiceView): def post(self, request, *args, **kwargs): - feature = Feature.objects.get(uuid=kwargs["feature_uuid"]) - # Checks if the integration came from vtex - created_by_vtex = request.data.get("created_by_vtex", False) - - try: - project = Project.objects.get(uuid=request.data["project_uuid"]) - except Project.DoesNotExist: - return Response( - status=status.HTTP_404_NOT_FOUND, - data={ - "error": f"Project with uuid equals {request.data['project_uuid']} does not exist!" - }, - ) - user, _ = User.objects.get_or_create(email=request.user.email) - feature_version = feature.last_version - - integrated_feature = IntegratedFeature.objects.create( - project=project, feature=feature, feature_version=feature_version, user=user - ) + request_data = request.data.copy() + request_data["feature_uuid"] = kwargs.get("feature_uuid") - sectors_data = [] - integrated_feature.sectors = [] - if feature_version.sectors is not None: - for sector in feature_version.sectors: - for r_sector in request.data.get("sectors", []): - if r_sector.get("name") == sector.get("name"): - new_sector = { - "name": r_sector.get("name"), - "tags": r_sector.get("tags"), - "queues": sector.get("queues"), - } - integrated_feature.sectors.append(new_sector) - break - - # Treat and fill specific globals - fill_globals_usecase = PopulateGlobalsValuesUsecase( + use_case = CreateIntegratedFeatureUseCase( self.integrations_service, self.flows_service ) - globals_values_request = {} - for globals_values in feature_version.globals_values: - globals_values_request[globals_values] = "" - - for key, value in request.data.get("globals_values", {}).items(): - globals_values_request[key] = value - - treated_globals_values = fill_globals_usecase.execute( - globals_values_request, - request.user.email, - request.data["project_uuid"], - ) - - # Add all globals from the request, including treated ones - for globals_key, globals_value in treated_globals_values.items(): - integrated_feature.globals_values[globals_key] = globals_value - - if created_by_vtex: - integrated_feature.created_by_vtex = created_by_vtex - - populate_defaults_use_case = PopulateDefaultsUseCase() - default_globals_values = populate_defaults_use_case.execute( - feature, globals_values_request - ) - - # Add default globals - for df_globals_key, df_globals_value in default_globals_values.items(): - integrated_feature.globals_values[df_globals_key] = df_globals_value - - integrated_feature.save() - - for sector in integrated_feature.sectors: - sectors_data.append( - { - "name": sector.get("name", ""), - "tags": sector.get("tags", ""), - "service_limit": 4, - "working_hours": {"init": "08:00", "close": "18:00"}, - "queues": sector.get("queues", []), - } - ) - - actions = [] - for function in feature.functions.all(): - function_last_version = function.last_version - if function_last_version.action_base_flow_uuid is not None: - actions.append( - { - "name": function_last_version.action_name, - "prompt": function_last_version.action_prompt, - "root_flow_uuid": str( - function_last_version.action_base_flow_uuid - ), - "type": "", - } - ) - if feature_version.action_base_flow_uuid: - actions.append( - { - "name": feature_version.action_name, - "prompt": feature_version.action_prompt, - "root_flow_uuid": str(feature_version.action_base_flow_uuid), - "type": "", - } - ) - - body = { - "definition": integrated_feature.feature_version.definition, - "user_email": integrated_feature.user.email, - "project_uuid": str(integrated_feature.project.uuid), - "parameters": integrated_feature.globals_values, - "feature_version": str(integrated_feature.feature_version.uuid), - "feature_uuid": str(integrated_feature.feature.uuid), - "sectors": sectors_data, - "action": actions, - } - - IntegratedFeatureEDA().publisher(body=body, exchange="integrated-feature.topic") - print(f"message sent `integrated feature` - body: {body}") - + integrated_feature = use_case.execute(request_data, user) serializer = IntegratedFeatureSerializer(integrated_feature.feature) - response = { + response_data = { "status": 200, "data": { "feature": integrated_feature.feature.uuid, @@ -147,7 +39,7 @@ def post(self, request, *args, **kwargs): **serializer.data, }, } - return Response(response) + return Response(response_data, status=status.HTTP_200_OK) def get(self, request, project_uuid): try: diff --git a/retail/api/usecases/create_integrated_feature_usecase.py b/retail/api/usecases/create_integrated_feature_usecase.py new file mode 100644 index 0000000..1c668c2 --- /dev/null +++ b/retail/api/usecases/create_integrated_feature_usecase.py @@ -0,0 +1,211 @@ +from rest_framework.exceptions import ValidationError, NotFound + +from retail.api.usecases.populate_globals_values import PopulateGlobalsValuesUsecase +from retail.api.usecases.populate_globals_with_defaults import PopulateDefaultsUseCase + +from retail.features.integrated_feature_eda import IntegratedFeatureEDA + +from retail.features.models import Feature, IntegratedFeature +from retail.projects.models import Project + + +class CreateIntegratedFeatureUseCase: + """ + Use case to handle the creation and configuration of an IntegratedFeature. + """ + + def __init__(self, integrations_service, flows_service): + self.integrations_service = integrations_service + self.flows_service = flows_service + + def execute(self, request_data, user): + """ + Execute the use case to create and configure an IntegratedFeature. + + Args: + request_data (dict): Data from the request. + user (User): The user performing the action. + + Returns: + IntegratedFeature: The created and configured IntegratedFeature instance. + + Raises: + ValidationError: If the feature is already integrated with the project. + NotFound: If the feature or project does not exist. + """ + # Validation and object retrieval + feature = self._get_feature(request_data["feature_uuid"]) + project = self._get_project(request_data["project_uuid"]) + + # Check if the feature is already integrated with the project + if self._is_feature_already_integrated(feature, project): + raise ValidationError( + f"Feature '{feature.uuid}' is already integrated with project '{project.uuid}'." + ) + + # Create IntegratedFeature + integrated_feature = self._create_integrated_feature( + feature, project, user, request_data.get("created_by_vtex", False) + ) + + # Process sectors and globals + self._process_sectors( + integrated_feature, feature, request_data.get("sectors", []) + ) + self._process_globals( + integrated_feature, feature, request_data.get("globals_values", {}) + ) + + # Publish integration event + self._publish_integration_event(integrated_feature) + + return integrated_feature + + def _get_feature(self, feature_uuid): + try: + return Feature.objects.get(uuid=feature_uuid) + except Feature.DoesNotExist: + raise NotFound(f"Feature with uuid '{feature_uuid}' does not exist.") + + def _get_project(self, project_uuid): + try: + return Project.objects.get(uuid=project_uuid) + except Project.DoesNotExist: + raise NotFound(f"Project with uuid '{project_uuid}' does not exist.") + + def _is_feature_already_integrated(self, feature, project): + """ + Check if the feature is already integrated with the given project. + + Args: + feature (Feature): The feature to check. + project (Project): The project to check. + + Returns: + bool: True if the feature is already integrated with the project, False otherwise. + """ + return IntegratedFeature.objects.filter( + feature=feature, project=project + ).exists() + + def _create_integrated_feature(self, feature, project, user, created_by_vtex): + feature_version = feature.last_version + integrated_feature = IntegratedFeature.objects.create( + project=project, + feature=feature, + feature_version=feature_version, + user=user, + created_by_vtex=created_by_vtex, + ) + return integrated_feature + + def _process_sectors(self, integrated_feature, feature, sectors_request): + integrated_feature.sectors = [] + if feature.last_version.sectors: + for sector in feature.last_version.sectors: + matching_sector = next( + (s for s in sectors_request if s.get("name") == sector.get("name")), + None, + ) + if matching_sector: + new_sector = { + "name": matching_sector.get("name"), + "tags": matching_sector.get("tags"), + "queues": sector.get("queues"), + } + integrated_feature.sectors.append(new_sector) + integrated_feature.save() + + def _process_globals(self, integrated_feature, feature, globals_values_request): + """ + Process and populate the global variables for the integrated feature. + + Args: + integrated_feature (IntegratedFeature): The integrated feature instance. + feature (Feature): The feature being integrated. + globals_values_request (dict): Global values provided in the request. + """ + # Initialize full_globals_values with all globals from feature_version, setting them to empty strings + feature_version = feature.last_version + full_globals_values = { + global_var: "" for global_var in feature_version.globals_values + } + + # Update with any provided values from the request + full_globals_values.update(globals_values_request) + + # Treat and fill specific globals + fill_globals_usecase = PopulateGlobalsValuesUsecase( + self.integrations_service, self.flows_service + ) + treated_globals_values = fill_globals_usecase.execute( + full_globals_values, + integrated_feature.user.email, + str(integrated_feature.project.uuid), + ) + + # If created by VTEX, populate default globals from config + if integrated_feature.created_by_vtex: + populate_defaults_use_case = PopulateDefaultsUseCase() + default_globals_values = populate_defaults_use_case.execute( + feature, full_globals_values + ) + # Merge default_globals_values into treated_globals_values + treated_globals_values.update(default_globals_values) + + # Ensure all globals are included + integrated_feature.globals_values = treated_globals_values + integrated_feature.save() + + def _publish_integration_event(self, integrated_feature): + # Prepare data for publishing + sectors_data = [ + { + "name": sector.get("name", ""), + "tags": sector.get("tags", ""), + "service_limit": 4, + "working_hours": {"init": "08:00", "close": "18:00"}, + "queues": sector.get("queues", []), + } + for sector in integrated_feature.sectors + ] + + actions = [] + feature_version = integrated_feature.feature_version + for function in integrated_feature.feature.functions.all(): + function_last_version = function.last_version + if function_last_version.action_base_flow_uuid: + actions.append( + { + "name": function_last_version.action_name, + "prompt": function_last_version.action_prompt, + "root_flow_uuid": str( + function_last_version.action_base_flow_uuid + ), + "type": "", + } + ) + + if feature_version.action_base_flow_uuid: + actions.append( + { + "name": feature_version.action_name, + "prompt": feature_version.action_prompt, + "root_flow_uuid": str(feature_version.action_base_flow_uuid), + "type": "", + } + ) + + body = { + "definition": feature_version.definition, + "user_email": integrated_feature.user.email, + "project_uuid": str(integrated_feature.project.uuid), + "parameters": integrated_feature.globals_values, + "feature_version": str(feature_version.uuid), + "feature_uuid": str(feature_version.feature.uuid), + "sectors": sectors_data, + "action": actions, + } + + IntegratedFeatureEDA().publisher(body=body, exchange="integrated-feature.topic") + print(f"message sent `integrated feature` - body: {body}") diff --git a/retail/api/usecases/remove_globals_keys.py b/retail/api/usecases/remove_globals_keys.py index 5190d0f..e899ef6 100644 --- a/retail/api/usecases/remove_globals_keys.py +++ b/retail/api/usecases/remove_globals_keys.py @@ -42,9 +42,7 @@ def execute( # Check and mark globals for removal based on flows data if available if flows_data: - if "api_token" in feature["globals"] and flows_data.get( - "api_token" - ): + if "api_token" in feature["globals"] and flows_data.get("api_token"): globals_to_remove.append("api_token") # Remove the marked globals From b735f6657d194b33403b18c2bf44650c8d3e7b2f Mon Sep 17 00:00:00 2001 From: elitonzky Date: Fri, 22 Nov 2024 14:59:56 -0300 Subject: [PATCH 05/12] feat: add usecases to integrated feature --- retail/api/integrated_feature/views.py | 19 ++----- .../create_integrated_feature_usecase.py | 51 +++++++++++++++++-- 2 files changed, 52 insertions(+), 18 deletions(-) diff --git a/retail/api/integrated_feature/views.py b/retail/api/integrated_feature/views.py index a069845..2085c2c 100644 --- a/retail/api/integrated_feature/views.py +++ b/retail/api/integrated_feature/views.py @@ -23,22 +23,11 @@ def post(self, request, *args, **kwargs): request_data["feature_uuid"] = kwargs.get("feature_uuid") use_case = CreateIntegratedFeatureUseCase( - self.integrations_service, self.flows_service + integrations_service=self.integrations_service, + flows_service=self.flows_service, ) - integrated_feature = use_case.execute(request_data, user) - serializer = IntegratedFeatureSerializer(integrated_feature.feature) - - response_data = { - "status": 200, - "data": { - "feature": integrated_feature.feature.uuid, - "feature_version": integrated_feature.feature_version.uuid, - "project": integrated_feature.project.uuid, - "user": integrated_feature.user.email, - "integrated_on": integrated_feature.integrated_on, - **serializer.data, - }, - } + + response_data = use_case.execute(request_data, user) return Response(response_data, status=status.HTTP_200_OK) def get(self, request, project_uuid): diff --git a/retail/api/usecases/create_integrated_feature_usecase.py b/retail/api/usecases/create_integrated_feature_usecase.py index 1c668c2..5d00484 100644 --- a/retail/api/usecases/create_integrated_feature_usecase.py +++ b/retail/api/usecases/create_integrated_feature_usecase.py @@ -1,5 +1,7 @@ from rest_framework.exceptions import ValidationError, NotFound +from django.conf import settings +from retail.api.integrated_feature.serializers import IntegratedFeatureSerializer from retail.api.usecases.populate_globals_values import PopulateGlobalsValuesUsecase from retail.api.usecases.populate_globals_with_defaults import PopulateDefaultsUseCase @@ -27,7 +29,7 @@ def execute(self, request_data, user): user (User): The user performing the action. Returns: - IntegratedFeature: The created and configured IntegratedFeature instance. + dict: Response data including integration details. Raises: ValidationError: If the feature is already integrated with the project. @@ -59,7 +61,10 @@ def execute(self, request_data, user): # Publish integration event self._publish_integration_event(integrated_feature) - return integrated_feature + # Prepare and return the response data + response_data = self._prepare_response_data(integrated_feature) + + return response_data def _get_feature(self, feature_uuid): try: @@ -100,6 +105,14 @@ def _create_integrated_feature(self, feature, project, user, created_by_vtex): return integrated_feature def _process_sectors(self, integrated_feature, feature, sectors_request): + """ + Process and set the sectors for the integrated feature. + + Args: + integrated_feature (IntegratedFeature): The integrated feature instance. + feature (Feature): The feature being integrated. + sectors_request (list): List of sectors provided in the request. + """ integrated_feature.sectors = [] if feature.last_version.sectors: for sector in feature.last_version.sectors: @@ -150,7 +163,7 @@ def _process_globals(self, integrated_feature, feature, globals_values_request): default_globals_values = populate_defaults_use_case.execute( feature, full_globals_values ) - # Merge default_globals_values into treated_globals_values + # Merge default globals into treated_globals_values treated_globals_values.update(default_globals_values) # Ensure all globals are included @@ -158,6 +171,12 @@ def _process_globals(self, integrated_feature, feature, globals_values_request): integrated_feature.save() def _publish_integration_event(self, integrated_feature): + """ + Publish the integration event to the message broker. + + Args: + integrated_feature (IntegratedFeature): The integrated feature instance. + """ # Prepare data for publishing sectors_data = [ { @@ -209,3 +228,29 @@ def _publish_integration_event(self, integrated_feature): IntegratedFeatureEDA().publisher(body=body, exchange="integrated-feature.topic") print(f"message sent `integrated feature` - body: {body}") + + def _prepare_response_data(self, integrated_feature): + """ + Prepare the response data to be sent back to the client. + + Args: + integrated_feature (IntegratedFeature): The integrated feature instance. + + Returns: + dict: Response data including additional info if necessary. + """ + serializer = IntegratedFeatureSerializer(integrated_feature.feature) + response_data = { + "status": 200, + "data": { + "feature": str(integrated_feature.feature.uuid), + "integrated_feature": str(integrated_feature.uuid), + "feature_version": str(integrated_feature.feature_version.uuid), + "project": str(integrated_feature.project.uuid), + "user": integrated_feature.user.email, + "integrated_on": integrated_feature.integrated_on.isoformat(), + **serializer.data, + }, + } + + return response_data From e1ede51890f8c7de50b1bf4f176554c7c2b14b17 Mon Sep 17 00:00:00 2001 From: elitonzky Date: Fri, 22 Nov 2024 19:35:29 -0300 Subject: [PATCH 06/12] feat: map cart actions --- .../migrations/0004_project_vtex_account.py | 18 +++ retail/projects/models.py | 1 + retail/settings.py | 3 + retail/urls.py | 3 + retail/vtex/__init__.py | 0 retail/vtex/apps.py | 6 + retail/vtex/migrations/0001_initial.py | 59 +++++++ retail/vtex/migrations/__init__.py | 0 retail/vtex/models.py | 34 ++++ retail/webhooks/__init__.py | 0 retail/webhooks/urls.py | 5 + retail/webhooks/vtex/dtos/cart_dto.py | 13 ++ retail/webhooks/vtex/serializers.py | 12 ++ retail/webhooks/vtex/urls.py | 11 ++ retail/webhooks/vtex/usecases/cart.py | 152 ++++++++++++++++++ .../vtex/views/abandoned_cart_notification.py | 44 +++++ 16 files changed, 361 insertions(+) create mode 100644 retail/projects/migrations/0004_project_vtex_account.py create mode 100644 retail/vtex/__init__.py create mode 100644 retail/vtex/apps.py create mode 100644 retail/vtex/migrations/0001_initial.py create mode 100644 retail/vtex/migrations/__init__.py create mode 100644 retail/vtex/models.py create mode 100644 retail/webhooks/__init__.py create mode 100644 retail/webhooks/urls.py create mode 100644 retail/webhooks/vtex/dtos/cart_dto.py create mode 100644 retail/webhooks/vtex/serializers.py create mode 100644 retail/webhooks/vtex/urls.py create mode 100644 retail/webhooks/vtex/usecases/cart.py create mode 100644 retail/webhooks/vtex/views/abandoned_cart_notification.py diff --git a/retail/projects/migrations/0004_project_vtex_account.py b/retail/projects/migrations/0004_project_vtex_account.py new file mode 100644 index 0000000..2b19d5e --- /dev/null +++ b/retail/projects/migrations/0004_project_vtex_account.py @@ -0,0 +1,18 @@ +# Generated by Django 5.1.1 on 2024-11-22 17:42 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ("projects", "0003_project_organization_uuid"), + ] + + operations = [ + migrations.AddField( + model_name="project", + name="vtex_account", + field=models.CharField(max_length=100, null=True), + ), + ] diff --git a/retail/projects/models.py b/retail/projects/models.py index 22bd96a..c3af7f9 100644 --- a/retail/projects/models.py +++ b/retail/projects/models.py @@ -5,6 +5,7 @@ class Project(models.Model): name = models.CharField(max_length=256) uuid = models.UUIDField() organization_uuid = models.UUIDField(null=True) + vtex_account = models.CharField(max_length=100, null=True) def __str__(self) -> str: return self.name diff --git a/retail/settings.py b/retail/settings.py index 560460d..7bd3420 100644 --- a/retail/settings.py +++ b/retail/settings.py @@ -69,6 +69,7 @@ "retail.healthcheck", "retail.internal", "rest_framework", + "retail.vtex" ] MIDDLEWARE = [ @@ -192,3 +193,5 @@ FLOWS_REST_ENDPOINT = env.str("FLOWS_REST_ENDPOINT") EMAILS_CAN_TESTING = env.str("EMAILS_CAN_TESTING", "").split(",") + +ABANDONED_CART_FEATURE_UUID = env.str("ABANDONED_CART_FEATURE_UUID", "") diff --git a/retail/urls.py b/retail/urls.py index 0f28618..f70487d 100644 --- a/retail/urls.py +++ b/retail/urls.py @@ -26,6 +26,8 @@ from retail.healthcheck import views from retail.projects import views as project_views from retail.api import routers as feature_routers +from retail.webhooks import urls as webhooks_urls + router = routers.SimpleRouter() router.register("projects", project_views.ProjectViewSet, basename="project") @@ -37,6 +39,7 @@ path("healthcheck/", views.healthcheck, name="healthcheck"), path("api/", include(router.urls)), path("v2/", include(feature_routers)), + path("", include(webhooks_urls)), ] urlpatterns.append( diff --git a/retail/vtex/__init__.py b/retail/vtex/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/retail/vtex/apps.py b/retail/vtex/apps.py new file mode 100644 index 0000000..4a8ee87 --- /dev/null +++ b/retail/vtex/apps.py @@ -0,0 +1,6 @@ +from django.apps import AppConfig + + +class VtexConfig(AppConfig): + default_auto_field = "django.db.models.BigAutoField" + name = "retail.vtex" diff --git a/retail/vtex/migrations/0001_initial.py b/retail/vtex/migrations/0001_initial.py new file mode 100644 index 0000000..90d5fc3 --- /dev/null +++ b/retail/vtex/migrations/0001_initial.py @@ -0,0 +1,59 @@ +# Generated by Django 5.1.1 on 2024-11-22 17:53 + +import django.db.models.deletion +import uuid +from django.db import migrations, models + + +class Migration(migrations.Migration): + initial = True + + dependencies = [ + ("projects", "0004_project_vtex_account"), + ] + + operations = [ + migrations.CreateModel( + name="Cart", + fields=[ + ( + "uuid", + models.UUIDField( + default=uuid.uuid4, + editable=False, + primary_key=True, + serialize=False, + verbose_name="UUID", + ), + ), + ("created_on", models.DateTimeField(auto_now_add=True)), + ("modified_on", models.DateTimeField(auto_now_add=True)), + ( + "status", + models.CharField( + choices=[ + ("created", "Created"), + ("purchased", "Purchased"), + ("abandoned", "Abandoned"), + ("delivered success", "Delivered Success"), + ("delivered error", "Delivered Error"), + ("empty", "Empty"), + ], + default="created", + max_length=20, + verbose_name="Status of Cart", + ), + ), + ("phone_number", models.CharField(max_length=256)), + ("config", models.JSONField(default=dict)), + ( + "project", + models.ForeignKey( + on_delete=django.db.models.deletion.CASCADE, + related_name="vtex_cart", + to="projects.project", + ), + ), + ], + ), + ] diff --git a/retail/vtex/migrations/__init__.py b/retail/vtex/migrations/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/retail/vtex/models.py b/retail/vtex/models.py new file mode 100644 index 0000000..805be6a --- /dev/null +++ b/retail/vtex/models.py @@ -0,0 +1,34 @@ +import uuid + +from django.db import models +from retail.projects.models import Project + + +class Cart(models.Model): + STATUS_CHOICES = [ + ("created", "Created"), + ("purchased", "Purchased"), + ("abandoned", "Abandoned"), + ("delivered success", "Delivered Success"), + ("delivered error", "Delivered Error"), + ("empty", "Empty"), + ] + uuid = models.UUIDField( + "UUID", primary_key=True, default=uuid.uuid4, editable=False + ) + created_on = models.DateTimeField(auto_now_add=True) + modified_on = models.DateTimeField(auto_now_add=True) + status = models.CharField( + max_length=20, + choices=STATUS_CHOICES, + default="created", + verbose_name="Status of Cart", + ) + phone_number = models.CharField(max_length=256) + config = models.JSONField(default=dict) + project = models.ForeignKey( + Project, on_delete=models.CASCADE, related_name="vtex_cart" + ) + + def __str__(self): + return f"{self.phone_number} - {self.status} on {self.modified_on}" diff --git a/retail/webhooks/__init__.py b/retail/webhooks/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/retail/webhooks/urls.py b/retail/webhooks/urls.py new file mode 100644 index 0000000..2308523 --- /dev/null +++ b/retail/webhooks/urls.py @@ -0,0 +1,5 @@ +from django.urls import path, include + +urlpatterns = [ + path("webhook/", include("retail.webhooks.vtex.urls")), +] diff --git a/retail/webhooks/vtex/dtos/cart_dto.py b/retail/webhooks/vtex/dtos/cart_dto.py new file mode 100644 index 0000000..deab2b4 --- /dev/null +++ b/retail/webhooks/vtex/dtos/cart_dto.py @@ -0,0 +1,13 @@ +from typing import Dict +from dataclasses import dataclass + + +@dataclass +class CartDTO: + """ + Data Transfer Object (DTO) for handling cart data. + """ + action: str # The action to perform, e.g., 'create', 'update', etc. + account: str # The VTEX account identifier + home_phone: str # The user's phone number + data: Dict # The cart details to be stored in the model diff --git a/retail/webhooks/vtex/serializers.py b/retail/webhooks/vtex/serializers.py new file mode 100644 index 0000000..f50f6c5 --- /dev/null +++ b/retail/webhooks/vtex/serializers.py @@ -0,0 +1,12 @@ +from rest_framework import serializers + + +class CartSerializer(serializers.Serializer): + """ + Serializer to validate cart data received from VTEX. + """ + + action = serializers.ChoiceField(choices=["create", "update", "purchased", "empty"]) + account = serializers.CharField() + homePhone = serializers.CharField() + data = serializers.JSONField() diff --git a/retail/webhooks/vtex/urls.py b/retail/webhooks/vtex/urls.py new file mode 100644 index 0000000..cdbba75 --- /dev/null +++ b/retail/webhooks/vtex/urls.py @@ -0,0 +1,11 @@ +from django.urls import path +from .views.abandoned_cart_notification import AbandonedCartNotification + + +urlpatterns = [ + path( + "vtex/abandoned-cart/api/notification/", + AbandonedCartNotification.as_view(), + name="abandoned-cart", + ), +] diff --git a/retail/webhooks/vtex/usecases/cart.py b/retail/webhooks/vtex/usecases/cart.py new file mode 100644 index 0000000..13f84de --- /dev/null +++ b/retail/webhooks/vtex/usecases/cart.py @@ -0,0 +1,152 @@ +from rest_framework.exceptions import ValidationError, NotFound +from retail.projects.models import Project +from retail.vtex.models import Cart +from retail.webhooks.vtex.dtos.cart_dto import CartDTO + + +class CartUseCase: + """ + Centralized use case for handling cart actions. + """ + + def __init__(self, account: str): + self.account = account + self.project = self._get_project_by_account() + + def _get_project_by_account(self) -> Project: + """ + Fetch the project associated with the account. + + Raises: + NotFound: If the project is not found. + + Returns: + Project: The associated project instance. + """ + try: + return Project.objects.get(vtex_account=self.account) + except Project.DoesNotExist: + raise NotFound(f"Project with account '{self.account}' does not exist.") + + def handle_action(self, action: str, cart_dto: CartDTO) -> Cart: + """ + Handle the specified cart action. + + Args: + action (str): The action to handle (create, update, purchased, empty). + cart_dto (CartDTO): The validated cart data. + + Returns: + Cart: The updated or created cart instance. + """ + action_methods = { + "create": self._create_cart, + "update": self._update_cart, + "purchased": self._mark_cart_purchased, + "empty": self._mark_cart_empty, + } + + if action not in action_methods: + raise ValidationError({"action": f"Invalid action: {action}"}) + + # Call the appropriate method dynamically + return action_methods[action](cart_dto) + + def _ensure_single_cart(self, home_phone: str): + """ + Ensure that a user has only one cart with the status "created". + + Args: + home_phone (str): The user's phone number. + + Raises: + ValidationError: If a cart with the "created" status already exists. + """ + # Check if there's an existing cart with the status "created" + cart_exists = Cart.objects.filter( + phone_number=home_phone, project=self.project, status="created" + ).exists() + + if cart_exists: + raise ValidationError( + {"cart": f"User with phone '{home_phone}' already has an active cart."} + ) + + def _create_cart(self, dto: CartDTO) -> Cart: + """ + Create a new cart entry. + + Args: + dto (CartDTO): The cart DTO. + + Returns: + Cart: The created cart instance. + """ + self._ensure_single_cart(dto.home_phone) + + return Cart.objects.create( + phone_number=dto.home_phone, + config=dto.data, + status="created", + project=self.project, + ) + + def _update_cart(self, dto: CartDTO) -> Cart: + """ + Update an existing cart. + + Args: + dto (CartDTO): The cart DTO. + + Returns: + Cart: The updated cart instance. + """ + try: + cart = Cart.objects.filter( + phone_number=dto.home_phone, project=self.project + ).latest("created_on") + cart.config.update(dto.data) + cart.save() + return cart + except Cart.DoesNotExist: + raise NotFound(f"Cart for phone '{dto.home_phone}' does not exist.") + + def _mark_cart_purchased(self, dto: CartDTO) -> Cart: + """ + Mark a cart as purchased. + + Args: + dto (CartDTO): The cart DTO. + + Returns: + Cart: The updated cart instance. + """ + try: + cart = Cart.objects.filter( + phone_number=dto.home_phone, project=self.project + ).latest("created_on") + cart.status = "purchased" + cart.save() + return cart + except Cart.DoesNotExist: + raise NotFound(f"Cart for phone '{dto.home_phone}' does not exist.") + + def _mark_cart_empty(self, dto: CartDTO) -> Cart: + """ + Mark a cart as empty. + + Args: + dto (CartDTO): The cart DTO. + + Returns: + Cart: The updated cart instance. + """ + try: + cart = Cart.objects.filter( + phone_number=dto.home_phone, project=self.project + ).latest("created_on") + cart.status = "empty" + cart.save() + return cart + except Cart.DoesNotExist: + raise NotFound(f"Cart for phone '{dto.home_phone}' does not exist.") diff --git a/retail/webhooks/vtex/views/abandoned_cart_notification.py b/retail/webhooks/vtex/views/abandoned_cart_notification.py new file mode 100644 index 0000000..5bf0b8a --- /dev/null +++ b/retail/webhooks/vtex/views/abandoned_cart_notification.py @@ -0,0 +1,44 @@ +from rest_framework.views import APIView +from rest_framework.response import Response +from rest_framework import status +from rest_framework.permissions import AllowAny + +from retail.webhooks.vtex.serializers import CartSerializer +from retail.webhooks.vtex.dtos.cart_dto import CartDTO +from retail.webhooks.vtex.usecases.cart import CartUseCase + + +class AbandonedCartNotification(APIView): + """ + View to handle abandoned cart notifications. + + This view receives data from the VTEX IO middleware, + processes it, and performs necessary actions. + """ + + authentication_classes = [] + permission_classes = [AllowAny] + + def post(self, request): + serializer = CartSerializer(data=request.data) + serializer.is_valid(raise_exception=True) + + validated_data = serializer.validated_data + cart_dto = CartDTO( + action=validated_data["action"], + account=validated_data["account"], + home_phone=validated_data["homePhone"], + data=validated_data["data"], + ) + + cart_use_case = CartUseCase(account=cart_dto.account) + result = cart_use_case.handle_action(cart_dto.action, cart_dto) + + return Response( + { + "message": f"Cart action '{cart_dto.action}' processed successfully.", + "cart_id": str(result.uuid), + "status": result.status, + }, + status=status.HTTP_200_OK, + ) From 29ed2f7e4476f07b74efad0b7143ecaeb505c839 Mon Sep 17 00:00:00 2001 From: elitonzky Date: Mon, 25 Nov 2024 14:36:04 -0300 Subject: [PATCH 07/12] chore: adjust cart models and add CartNotificationQueue --- retail/vtex/migrations/0001_initial.py | 60 +++++++++++++++++++++++--- retail/vtex/models.py | 42 ++++++++++++++++-- 2 files changed, 93 insertions(+), 9 deletions(-) diff --git a/retail/vtex/migrations/0001_initial.py b/retail/vtex/migrations/0001_initial.py index 90d5fc3..89c47c2 100644 --- a/retail/vtex/migrations/0001_initial.py +++ b/retail/vtex/migrations/0001_initial.py @@ -1,4 +1,4 @@ -# Generated by Django 5.1.1 on 2024-11-22 17:53 +# Generated by Django 5.1.1 on 2024-11-25 14:07 import django.db.models.deletion import uuid @@ -34,9 +34,8 @@ class Migration(migrations.Migration): choices=[ ("created", "Created"), ("purchased", "Purchased"), - ("abandoned", "Abandoned"), - ("delivered success", "Delivered Success"), - ("delivered error", "Delivered Error"), + ("delivered_success", "Delivered Success"), + ("delivered_error", "Delivered Error"), ("empty", "Empty"), ], default="created", @@ -44,8 +43,9 @@ class Migration(migrations.Migration): verbose_name="Status of Cart", ), ), - ("phone_number", models.CharField(max_length=256)), + ("phone_number", models.CharField(max_length=15)), ("config", models.JSONField(default=dict)), + ("abandoned", models.BooleanField(default=False)), ( "project", models.ForeignKey( @@ -56,4 +56,54 @@ class Migration(migrations.Migration): ), ], ), + migrations.CreateModel( + name="CartNotificationQueue", + fields=[ + ( + "uuid", + models.UUIDField( + default=uuid.uuid4, + editable=False, + primary_key=True, + serialize=False, + verbose_name="UUID", + ), + ), + ("created_on", models.DateTimeField(auto_now_add=True)), + ( + "status", + models.CharField( + choices=[ + ("pending", "Pending"), + ("processed", "Processed"), + ("failed", "Failed"), + ], + default="pending", + max_length=20, + verbose_name="Notification Status", + ), + ), + ("error_message", models.TextField(blank=True, null=True)), + ( + "cart", + models.OneToOneField( + on_delete=django.db.models.deletion.CASCADE, + related_name="notification_queue", + to="vtex.cart", + ), + ), + ], + ), + migrations.AddIndex( + model_name="cart", + index=models.Index( + fields=["project", "status"], name="vtex_cart_project_da4d1d_idx" + ), + ), + migrations.AddIndex( + model_name="cartnotificationqueue", + index=models.Index( + fields=["status", "created_on"], name="vtex_cartno_status_1942e9_idx" + ), + ), ] diff --git a/retail/vtex/models.py b/retail/vtex/models.py index 805be6a..0fe36a9 100644 --- a/retail/vtex/models.py +++ b/retail/vtex/models.py @@ -8,9 +8,8 @@ class Cart(models.Model): STATUS_CHOICES = [ ("created", "Created"), ("purchased", "Purchased"), - ("abandoned", "Abandoned"), - ("delivered success", "Delivered Success"), - ("delivered error", "Delivered Error"), + ("delivered_success", "Delivered Success"), + ("delivered_error", "Delivered Error"), ("empty", "Empty"), ] uuid = models.UUIDField( @@ -24,11 +23,46 @@ class Cart(models.Model): default="created", verbose_name="Status of Cart", ) - phone_number = models.CharField(max_length=256) + phone_number = models.CharField(max_length=15) config = models.JSONField(default=dict) project = models.ForeignKey( Project, on_delete=models.CASCADE, related_name="vtex_cart" ) + abandoned = models.BooleanField(default=False) def __str__(self): return f"{self.phone_number} - {self.status} on {self.modified_on}" + + class Meta: + indexes = [models.Index(fields=["project", "status"])] + + +class CartNotificationQueue(models.Model): + """ + Queue for carts that need to be processed for abandoned notifications. + """ + + uuid = models.UUIDField( + "UUID", primary_key=True, default=uuid.uuid4, editable=False + ) + cart = models.OneToOneField( + Cart, on_delete=models.CASCADE, related_name="notification_queue" + ) + created_on = models.DateTimeField(auto_now_add=True) + status = models.CharField( + max_length=20, + choices=[ + ("pending", "Pending"), + ("processed", "Processed"), + ("failed", "Failed"), + ], + default="pending", + verbose_name="Notification Status", + ) + error_message = models.TextField(blank=True, null=True) + + def __str__(self): + return f"Queue for cart {self.cart.uuid} - {self.status}" + + class Meta: + indexes = [models.Index(fields=["status", "created_on"])] From fc42dfc72c02a6455a0867f3be4d6ecb5434d103 Mon Sep 17 00:00:00 2001 From: elitonzky Date: Mon, 25 Nov 2024 14:37:48 -0300 Subject: [PATCH 08/12] feat: add Redis and Celery to project --- poetry.lock | 250 ++++++++++++++++++++++++++++++++++++++++++++- pyproject.toml | 3 +- retail/celery.py | 11 ++ retail/settings.py | 23 +++++ 4 files changed, 284 insertions(+), 3 deletions(-) create mode 100644 retail/celery.py diff --git a/poetry.lock b/poetry.lock index c890d94..16ec0eb 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,4 +1,4 @@ -# This file is automatically @generated by Poetry 1.8.3 and should not be changed by hand. +# This file is automatically @generated by Poetry 1.6.1 and should not be changed by hand. [[package]] name = "amqp" @@ -31,6 +31,28 @@ typing-extensions = {version = ">=4", markers = "python_version < \"3.11\""} [package.extras] tests = ["mypy (>=0.800)", "pytest", "pytest-asyncio"] +[[package]] +name = "async-timeout" +version = "5.0.1" +description = "Timeout context manager for asyncio programs" +optional = false +python-versions = ">=3.8" +files = [ + {file = "async_timeout-5.0.1-py3-none-any.whl", hash = "sha256:39e3809566ff85354557ec2398b55e096c8364bacac9405a7a1fa429e77fe76c"}, + {file = "async_timeout-5.0.1.tar.gz", hash = "sha256:d9321a7a3d5a6a5e187e824d2fa0793ce379a202935782d555d6e9d2735677d3"}, +] + +[[package]] +name = "billiard" +version = "4.2.1" +description = "Python multiprocessing fork with improvements and bugfixes" +optional = false +python-versions = ">=3.7" +files = [ + {file = "billiard-4.2.1-py3-none-any.whl", hash = "sha256:40b59a4ac8806ba2c2369ea98d876bc6108b051c227baffd928c644d15d8f3cb"}, + {file = "billiard-4.2.1.tar.gz", hash = "sha256:12b641b0c539073fc8d3f5b8b7be998956665c4233c7c1fcd66a7e677c4fb36f"}, +] + [[package]] name = "black" version = "24.8.0" @@ -77,6 +99,62 @@ d = ["aiohttp (>=3.7.4)", "aiohttp (>=3.7.4,!=3.9.0)"] jupyter = ["ipython (>=7.8.0)", "tokenize-rt (>=3.2.0)"] uvloop = ["uvloop (>=0.15.2)"] +[[package]] +name = "celery" +version = "5.4.0" +description = "Distributed Task Queue." +optional = false +python-versions = ">=3.8" +files = [ + {file = "celery-5.4.0-py3-none-any.whl", hash = "sha256:369631eb580cf8c51a82721ec538684994f8277637edde2dfc0dacd73ed97f64"}, + {file = "celery-5.4.0.tar.gz", hash = "sha256:504a19140e8d3029d5acad88330c541d4c3f64c789d85f94756762d8bca7e706"}, +] + +[package.dependencies] +billiard = ">=4.2.0,<5.0" +click = ">=8.1.2,<9.0" +click-didyoumean = ">=0.3.0" +click-plugins = ">=1.1.1" +click-repl = ">=0.2.0" +kombu = ">=5.3.4,<6.0" +python-dateutil = ">=2.8.2" +tzdata = ">=2022.7" +vine = ">=5.1.0,<6.0" + +[package.extras] +arangodb = ["pyArango (>=2.0.2)"] +auth = ["cryptography (==42.0.5)"] +azureblockblob = ["azure-storage-blob (>=12.15.0)"] +brotli = ["brotli (>=1.0.0)", "brotlipy (>=0.7.0)"] +cassandra = ["cassandra-driver (>=3.25.0,<4)"] +consul = ["python-consul2 (==0.1.5)"] +cosmosdbsql = ["pydocumentdb (==2.3.5)"] +couchbase = ["couchbase (>=3.0.0)"] +couchdb = ["pycouchdb (==1.14.2)"] +django = ["Django (>=2.2.28)"] +dynamodb = ["boto3 (>=1.26.143)"] +elasticsearch = ["elastic-transport (<=8.13.0)", "elasticsearch (<=8.13.0)"] +eventlet = ["eventlet (>=0.32.0)"] +gcs = ["google-cloud-storage (>=2.10.0)"] +gevent = ["gevent (>=1.5.0)"] +librabbitmq = ["librabbitmq (>=2.0.0)"] +memcache = ["pylibmc (==1.6.3)"] +mongodb = ["pymongo[srv] (>=4.0.2)"] +msgpack = ["msgpack (==1.0.8)"] +pymemcache = ["python-memcached (>=1.61)"] +pyro = ["pyro4 (==4.82)"] +pytest = ["pytest-celery[all] (>=1.0.0)"] +redis = ["redis (>=4.5.2,!=4.5.5,<6.0.0)"] +s3 = ["boto3 (>=1.26.143)"] +slmq = ["softlayer-messaging (>=1.0.3)"] +solar = ["ephem (==4.1.5)"] +sqlalchemy = ["sqlalchemy (>=1.4.48,<2.1)"] +sqs = ["boto3 (>=1.26.143)", "kombu[sqs] (>=5.3.4)", "pycurl (>=7.43.0.5)", "urllib3 (>=1.26.16)"] +tblib = ["tblib (>=1.3.0)", "tblib (>=1.5.0)"] +yaml = ["PyYAML (>=3.10)"] +zookeeper = ["kazoo (>=1.3.1)"] +zstd = ["zstandard (==0.22.0)"] + [[package]] name = "certifi" version = "2024.8.30" @@ -280,6 +358,55 @@ files = [ [package.dependencies] colorama = {version = "*", markers = "platform_system == \"Windows\""} +[[package]] +name = "click-didyoumean" +version = "0.3.1" +description = "Enables git-like *did-you-mean* feature in click" +optional = false +python-versions = ">=3.6.2" +files = [ + {file = "click_didyoumean-0.3.1-py3-none-any.whl", hash = "sha256:5c4bb6007cfea5f2fd6583a2fb6701a22a41eb98957e63d0fac41c10e7c3117c"}, + {file = "click_didyoumean-0.3.1.tar.gz", hash = "sha256:4f82fdff0dbe64ef8ab2279bd6aa3f6a99c3b28c05aa09cbfc07c9d7fbb5a463"}, +] + +[package.dependencies] +click = ">=7" + +[[package]] +name = "click-plugins" +version = "1.1.1" +description = "An extension module for click to enable registering CLI commands via setuptools entry-points." +optional = false +python-versions = "*" +files = [ + {file = "click-plugins-1.1.1.tar.gz", hash = "sha256:46ab999744a9d831159c3411bb0c79346d94a444df9a3a3742e9ed63645f264b"}, + {file = "click_plugins-1.1.1-py2.py3-none-any.whl", hash = "sha256:5d262006d3222f5057fd81e1623d4443e41dcda5dc815c06b442aa3c02889fc8"}, +] + +[package.dependencies] +click = ">=4.0" + +[package.extras] +dev = ["coveralls", "pytest (>=3.6)", "pytest-cov", "wheel"] + +[[package]] +name = "click-repl" +version = "0.3.0" +description = "REPL plugin for Click" +optional = false +python-versions = ">=3.6" +files = [ + {file = "click-repl-0.3.0.tar.gz", hash = "sha256:17849c23dba3d667247dc4defe1757fff98694e90fe37474f3feebb69ced26a9"}, + {file = "click_repl-0.3.0-py3-none-any.whl", hash = "sha256:fb7e06deb8da8de86180a33a9da97ac316751c094c6899382da7feeeeb51b812"}, +] + +[package.dependencies] +click = ">=7.0" +prompt-toolkit = ">=3.0.36" + +[package.extras] +testing = ["pytest (>=7.2.1)", "pytest-cov (>=4.0.0)", "tox (>=4.4.3)"] + [[package]] name = "colorama" version = "0.4.6" @@ -391,6 +518,24 @@ develop = ["coverage[toml] (>=5.0a4)", "furo (>=2021.8.17b43,<2021.9.dev0)", "py docs = ["furo (>=2021.8.17b43,<2021.9.dev0)", "sphinx (>=3.5.0)", "sphinx-notfound-page"] testing = ["coverage[toml] (>=5.0a4)", "pytest (>=4.6.11)"] +[[package]] +name = "django-redis" +version = "5.4.0" +description = "Full featured redis cache backend for Django." +optional = false +python-versions = ">=3.6" +files = [ + {file = "django-redis-5.4.0.tar.gz", hash = "sha256:6a02abaa34b0fea8bf9b707d2c363ab6adc7409950b2db93602e6cb292818c42"}, + {file = "django_redis-5.4.0-py3-none-any.whl", hash = "sha256:ebc88df7da810732e2af9987f7f426c96204bf89319df4c6da6ca9a2942edd5b"}, +] + +[package.dependencies] +Django = ">=3.2" +redis = ">=3,<4.0.0 || >4.0.0,<4.0.1 || >4.0.1" + +[package.extras] +hiredis = ["redis[hiredis] (>=3,!=4.0.0,!=4.0.1)"] + [[package]] name = "djangorestframework" version = "3.15.2" @@ -509,6 +654,39 @@ pyopenssl = ">=0.13" [package.extras] docs = ["sphinx (>=4.3.0)", "sphinx-rtd-theme (>=1.0)"] +[[package]] +name = "kombu" +version = "5.4.2" +description = "Messaging library for Python." +optional = false +python-versions = ">=3.8" +files = [ + {file = "kombu-5.4.2-py3-none-any.whl", hash = "sha256:14212f5ccf022fc0a70453bb025a1dcc32782a588c49ea866884047d66e14763"}, + {file = "kombu-5.4.2.tar.gz", hash = "sha256:eef572dd2fd9fc614b37580e3caeafdd5af46c1eff31e7fba89138cdb406f2cf"}, +] + +[package.dependencies] +amqp = ">=5.1.1,<6.0.0" +tzdata = {version = "*", markers = "python_version >= \"3.9\""} +vine = "5.1.0" + +[package.extras] +azureservicebus = ["azure-servicebus (>=7.10.0)"] +azurestoragequeues = ["azure-identity (>=1.12.0)", "azure-storage-queue (>=12.6.0)"] +confluentkafka = ["confluent-kafka (>=2.2.0)"] +consul = ["python-consul2 (==0.1.5)"] +librabbitmq = ["librabbitmq (>=2.0.0)"] +mongodb = ["pymongo (>=4.1.1)"] +msgpack = ["msgpack (==1.1.0)"] +pyro = ["pyro4 (==4.82)"] +qpid = ["qpid-python (>=0.26)", "qpid-tools (>=0.26)"] +redis = ["redis (>=4.5.2,!=4.5.5,!=5.0.2)"] +slmq = ["softlayer-messaging (>=1.0.3)"] +sqlalchemy = ["sqlalchemy (>=1.4.48,<2.1)"] +sqs = ["boto3 (>=1.26.143)", "pycurl (>=7.43.0.5)", "urllib3 (>=1.26.16)"] +yaml = ["PyYAML (>=3.10)"] +zookeeper = ["kazoo (>=2.8.0)"] + [[package]] name = "mccabe" version = "0.7.0" @@ -586,6 +764,20 @@ docs = ["furo (>=2024.8.6)", "proselint (>=0.14)", "sphinx (>=8.0.2)", "sphinx-a test = ["appdirs (==1.4.4)", "covdefaults (>=2.3)", "pytest (>=8.3.2)", "pytest-cov (>=5)", "pytest-mock (>=3.14)"] type = ["mypy (>=1.11.2)"] +[[package]] +name = "prompt-toolkit" +version = "3.0.48" +description = "Library for building powerful interactive command lines in Python" +optional = false +python-versions = ">=3.7.0" +files = [ + {file = "prompt_toolkit-3.0.48-py3-none-any.whl", hash = "sha256:f49a827f90062e411f1ce1f854f2aedb3c23353244f8108b89283587397ac10e"}, + {file = "prompt_toolkit-3.0.48.tar.gz", hash = "sha256:d6623ab0477a80df74e646bdbc93621143f5caf104206aa29294d53de1a03d90"}, +] + +[package.dependencies] +wcwidth = "*" + [[package]] name = "psycopg2" version = "2.9.9" @@ -740,6 +932,20 @@ cryptography = ">=41.0.5,<44" docs = ["sphinx (!=5.2.0,!=5.2.0.post0,!=7.2.5)", "sphinx-rtd-theme"] test = ["pretend", "pytest (>=3.0.1)", "pytest-rerunfailures"] +[[package]] +name = "python-dateutil" +version = "2.9.0.post0" +description = "Extensions to the standard Python datetime module" +optional = false +python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,>=2.7" +files = [ + {file = "python-dateutil-2.9.0.post0.tar.gz", hash = "sha256:37dd54208da7e1cd875388217d5e00ebd4179249f90fb72437e91a35459a0ad3"}, + {file = "python_dateutil-2.9.0.post0-py2.py3-none-any.whl", hash = "sha256:a8b2bc7bffae282281c8140a97d3aa9c14da0b136dfe83f850eea9a5f7470427"}, +] + +[package.dependencies] +six = ">=1.5" + [[package]] name = "pytz" version = "2024.2" @@ -813,6 +1019,24 @@ files = [ {file = "pyyaml-6.0.2.tar.gz", hash = "sha256:d584d9ec91ad65861cc08d42e834324ef890a082e591037abe114850ff7bbc3e"}, ] +[[package]] +name = "redis" +version = "5.2.0" +description = "Python client for Redis database and key-value store" +optional = false +python-versions = ">=3.8" +files = [ + {file = "redis-5.2.0-py3-none-any.whl", hash = "sha256:ae174f2bb3b1bf2b09d54bf3e51fbc1469cf6c10aa03e21141f51969801a7897"}, + {file = "redis-5.2.0.tar.gz", hash = "sha256:0b1087665a771b1ff2e003aa5bdd354f15a70c9e25d5a7dbf9c722c16528a7b0"}, +] + +[package.dependencies] +async-timeout = {version = ">=4.0.3", markers = "python_full_version < \"3.11.3\""} + +[package.extras] +hiredis = ["hiredis (>=3.0.0)"] +ocsp = ["cryptography (>=36.0.1)", "pyopenssl (==23.2.1)", "requests (>=2.31.0)"] + [[package]] name = "requests" version = "2.32.3" @@ -834,6 +1058,17 @@ urllib3 = ">=1.21.1,<3" socks = ["PySocks (>=1.5.6,!=1.5.7)"] use-chardet-on-py3 = ["chardet (>=3.0.2,<6)"] +[[package]] +name = "six" +version = "1.16.0" +description = "Python 2 and 3 compatibility utilities" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*" +files = [ + {file = "six-1.16.0-py2.py3-none-any.whl", hash = "sha256:8abb2f1d86890a2dfb989f9a77cfcfd3e47c2a354b01111771326f8aa26e0254"}, + {file = "six-1.16.0.tar.gz", hash = "sha256:1e61c37477a1626458e36f7b1d82aa5c9b094fa4802892072e49de9c60c4c926"}, +] + [[package]] name = "sqlparse" version = "0.5.1" @@ -921,6 +1156,17 @@ files = [ {file = "vine-5.1.0.tar.gz", hash = "sha256:8b62e981d35c41049211cf62a0a1242d8c1ee9bd15bb196ce38aefd6799e61e0"}, ] +[[package]] +name = "wcwidth" +version = "0.2.13" +description = "Measures the displayed width of unicode strings in a terminal" +optional = false +python-versions = "*" +files = [ + {file = "wcwidth-0.2.13-py2.py3-none-any.whl", hash = "sha256:3da69048e4540d84af32131829ff948f1e022c1c6bdb8d6102117aac784f6859"}, + {file = "wcwidth-0.2.13.tar.gz", hash = "sha256:72ea0c06399eb286d978fdedb6923a9eb47e1c486ce63e9b4e64fc18303972b5"}, +] + [[package]] name = "weni-eda" version = "0.1.1" @@ -938,4 +1184,4 @@ amqp = ">=5.2.0,<6.0.0" [metadata] lock-version = "2.0" python-versions = "^3.10" -content-hash = "a2bd9312db8b52add39d9386a1cbd9bb0a8231fddc5157559ed58a5aec2b181e" +content-hash = "29f0b4ccf36ce0c81c1fa689d8e6e1e96d793b05713d6dc7e3d617ae89d76b91" diff --git a/pyproject.toml b/pyproject.toml index f5e374b..8ae9d6d 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -3,7 +3,6 @@ name = "retail-setup" version = "0.0.1" description = "" authors = ["Weni"] -package-mode = false [tool.poetry.dependencies] python = "^3.10" @@ -17,6 +16,8 @@ mozilla-django-oidc = "^4.0.1" djangorestframework = "^3.15.2" drf-yasg = "^1.21.7" django-cors-headers = "^4.4.0" +celery = "^5.1.2" +django-redis = "^5.2.0" [tool.poetry.group.dev.dependencies] flake8 = "^7.1.0" diff --git a/retail/celery.py b/retail/celery.py new file mode 100644 index 0000000..585565c --- /dev/null +++ b/retail/celery.py @@ -0,0 +1,11 @@ +from __future__ import absolute_import + +import os +from celery import Celery + +os.environ.setdefault("DJANGO_SETTINGS_MODULE", "retail.settings") + +app = Celery("retail") + +app.config_from_object("django.conf:settings", namespace="CELERY") +app.autodiscover_tasks() diff --git a/retail/settings.py b/retail/settings.py index 7bd3420..4bf26e8 100644 --- a/retail/settings.py +++ b/retail/settings.py @@ -195,3 +195,26 @@ EMAILS_CAN_TESTING = env.str("EMAILS_CAN_TESTING", "").split(",") ABANDONED_CART_FEATURE_UUID = env.str("ABANDONED_CART_FEATURE_UUID", "") + + +# Redis +REDIS_URL = env.str("REDIS_URL", default="redis://localhost:6379") + + +# Celery +CELERY_BROKER_URL = env.str("CELERY_BROKER_URL", default=REDIS_URL) +CELERY_RESULT_BACKEND = None +CELERY_ACCEPT_CONTENT = ["application/json"] +CELERY_TASK_SERIALIZER = "json" +CELERY_RESULT_SERIALIZER = "json" +CELERY_TIMEZONE = TIME_ZONE + + +# Cache +CACHES = { + "default": { + "BACKEND": "django_redis.cache.RedisCache", + "LOCATION": REDIS_URL, + "OPTIONS": {"CLIENT_CLASS": "django_redis.client.DefaultClient"}, + } +} From c8809843de39ae45987b7c480a233b723961e01c Mon Sep 17 00:00:00 2001 From: elitonzky Date: Fri, 29 Nov 2024 19:05:08 -0300 Subject: [PATCH 09/12] feat: add abandoned cart task --- retail/clients/flows/client.py | 38 ++++ .../0016_integratedfeature_config.py | 17 ++ retail/features/models.py | 1 + retail/interfaces/clients/flows/interface.py | 18 ++ retail/services/flows/service.py | 27 ++- retail/settings.py | 2 + retail/vtex/migrations/0001_initial.py | 84 +++----- retail/vtex/models.py | 48 +---- retail/vtex/tasks.py | 14 ++ retail/vtex/usecases/cart_abandonment.py | 194 ++++++++++++++++++ .../vtex/usecases/phone_number_normalizer.py | 40 ++++ retail/webhooks/vtex/serializers.py | 2 +- retail/webhooks/vtex/usecases/cart.py | 123 ++++++++++- .../vtex/views/abandoned_cart_notification.py | 2 +- 14 files changed, 508 insertions(+), 102 deletions(-) create mode 100644 retail/features/migrations/0016_integratedfeature_config.py create mode 100644 retail/vtex/tasks.py create mode 100644 retail/vtex/usecases/cart_abandonment.py create mode 100644 retail/vtex/usecases/phone_number_normalizer.py diff --git a/retail/clients/flows/client.py b/retail/clients/flows/client.py index d1f8060..32b473a 100644 --- a/retail/clients/flows/client.py +++ b/retail/clients/flows/client.py @@ -12,6 +12,16 @@ def __init__(self): self.authentication_instance = InternalAuthentication() def get_user_api_token(self, user_email: str, project_uuid: str): + """ + Fetch a user API token from the Flows service. + + Args: + user_email (str): Email of the user. + project_uuid (str): UUID of the project. + + Returns: + str: API token for the user. + """ url = f"{self.base_url}/api/v2/internals/users/api-token/" params = dict(user=user_email, project=str(project_uuid)) response = self.make_request( @@ -21,3 +31,31 @@ def get_user_api_token(self, user_email: str, project_uuid: str): headers=self.authentication_instance.headers, ) return response.json() + + def send_whatsapp_broadcast(self, payload: dict, token: str) -> dict: + """ + Sends a WhatsApp broadcast message using the Flows API. + + Args: + payload (dict): The full body of the request as a pre-built payload. + token (str): Authorization token for the API. + + Returns: + dict: Response from the API. + """ + if not token: + raise ValueError("Authorization token is required to send a broadcast.") + + url = f"{self.base_url}/api/v2/whatsapp_broadcasts.json" + headers = { + "Authorization": f"Token {token}", + "Content-Type": "application/json", + } + + response = self.make_request( + url, + method="POST", + json=payload, + headers=headers, + ) + return response.json() diff --git a/retail/features/migrations/0016_integratedfeature_config.py b/retail/features/migrations/0016_integratedfeature_config.py new file mode 100644 index 0000000..e519c92 --- /dev/null +++ b/retail/features/migrations/0016_integratedfeature_config.py @@ -0,0 +1,17 @@ +# Generated by Django 5.1.1 on 2024-11-29 20:44 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + dependencies = [ + ("features", "0015_feature_can_vtex_integrate_feature_config_and_more"), + ] + + operations = [ + migrations.AddField( + model_name="integratedfeature", + name="config", + field=models.JSONField(default=dict), + ), + ] diff --git a/retail/features/models.py b/retail/features/models.py index 023b658..964e43a 100644 --- a/retail/features/models.py +++ b/retail/features/models.py @@ -143,6 +143,7 @@ class IntegratedFeature(models.Model): ) integrated_on = models.DateField(auto_now_add=True) created_by_vtex = models.BooleanField(default=False) + config = models.JSONField(default=dict) # def save(self, *args) -> None: # self.feature = self.feature_version.feature diff --git a/retail/interfaces/clients/flows/interface.py b/retail/interfaces/clients/flows/interface.py index 1907123..f836c53 100644 --- a/retail/interfaces/clients/flows/interface.py +++ b/retail/interfaces/clients/flows/interface.py @@ -1,7 +1,25 @@ from abc import ABC, abstractmethod +from typing import List, Optional, Dict class FlowsClientInterface(ABC): @abstractmethod def get_user_api_token(self, user_email: str, project_uuid: str): + """ + Retrieve the user API token for a given email and project UUID. + """ + pass + + @abstractmethod + def send_whatsapp_broadcast(self, payload: Dict, token: str) -> Dict: + """ + Sends a WhatsApp broadcast message. + + Args: + payload (dict): The pre-built payload containing all necessary data for the broadcast. + token (str): Authorization token for the API. + + Returns: + dict: API response containing the broadcast information. + """ pass diff --git a/retail/services/flows/service.py b/retail/services/flows/service.py index 2204733..75906d6 100644 --- a/retail/services/flows/service.py +++ b/retail/services/flows/service.py @@ -9,10 +9,33 @@ def __init__(self, client: FlowsClientInterface): def get_user_api_token(self, user_email: str, project_uuid: str) -> dict: """ Retrieve the user API token for a given email and project UUID. - Handles communication errors and returns None in case of failure. """ try: return self.client.get_user_api_token(user_email, project_uuid) except CustomAPIException as e: - print(f"Error {e.status_code} when retrieving user API token for project {project_uuid}.") + print( + f"Error {e.status_code} when retrieving user API token for project {project_uuid}." + ) return None + + def send_whatsapp_broadcast( + self, payload: dict, project_uuid: str, user_email: str + ) -> dict: + """ + Send a WhatsApp broadcast message. + + Args: + payload (dict): The full body of the request as a pre-built payload. + project_uuid (str): The UUID of the project. + user_email (str): Email of the user for authentication. + + Returns: + dict: API response from the Flows service. + """ + # Retrieve the API token + token = self.client.get_user_api_token(user_email, project_uuid) + if not token: + raise CustomAPIException("Failed to retrieve API token.") + + # Send the broadcast using the token + return self.client.send_whatsapp_broadcast(payload=payload, token=token) diff --git a/retail/settings.py b/retail/settings.py index 4bf26e8..025e29a 100644 --- a/retail/settings.py +++ b/retail/settings.py @@ -196,6 +196,7 @@ ABANDONED_CART_FEATURE_UUID = env.str("ABANDONED_CART_FEATURE_UUID", "") +FLOWS_USER_CRM_EMAIL = env.str("FLOWS_USER_CRM_EMAIL", "") # Redis REDIS_URL = env.str("REDIS_URL", default="redis://localhost:6379") @@ -204,6 +205,7 @@ # Celery CELERY_BROKER_URL = env.str("CELERY_BROKER_URL", default=REDIS_URL) CELERY_RESULT_BACKEND = None +CELERY_TASK_IGNORE_RESULT = True CELERY_ACCEPT_CONTENT = ["application/json"] CELERY_TASK_SERIALIZER = "json" CELERY_RESULT_SERIALIZER = "json" diff --git a/retail/vtex/migrations/0001_initial.py b/retail/vtex/migrations/0001_initial.py index 89c47c2..5f5bddf 100644 --- a/retail/vtex/migrations/0001_initial.py +++ b/retail/vtex/migrations/0001_initial.py @@ -1,4 +1,4 @@ -# Generated by Django 5.1.1 on 2024-11-25 14:07 +# Generated by Django 5.1.1 on 2024-11-29 19:23 import django.db.models.deletion import uuid @@ -6,9 +6,11 @@ class Migration(migrations.Migration): + initial = True dependencies = [ + ("features", "0015_feature_can_vtex_integrate_feature_config_and_more"), ("projects", "0004_project_vtex_account"), ] @@ -17,17 +19,20 @@ class Migration(migrations.Migration): name="Cart", fields=[ ( - "uuid", - models.UUIDField( - default=uuid.uuid4, - editable=False, + "id", + models.BigAutoField( + auto_created=True, primary_key=True, serialize=False, - verbose_name="UUID", + verbose_name="ID", ), ), + ( + "uuid", + models.UUIDField(default=uuid.uuid4, editable=False, unique=True), + ), ("created_on", models.DateTimeField(auto_now_add=True)), - ("modified_on", models.DateTimeField(auto_now_add=True)), + ("modified_on", models.DateTimeField(auto_now=True)), ( "status", models.CharField( @@ -46,64 +51,31 @@ class Migration(migrations.Migration): ("phone_number", models.CharField(max_length=15)), ("config", models.JSONField(default=dict)), ("abandoned", models.BooleanField(default=False)), + ("error_message", models.TextField(blank=True, null=True)), ( - "project", + "integrated_feature", models.ForeignKey( on_delete=django.db.models.deletion.CASCADE, - related_name="vtex_cart", - to="projects.project", + related_name="carts_by_feature", + to="features.integratedfeature", ), ), - ], - ), - migrations.CreateModel( - name="CartNotificationQueue", - fields=[ ( - "uuid", - models.UUIDField( - default=uuid.uuid4, - editable=False, - primary_key=True, - serialize=False, - verbose_name="UUID", - ), - ), - ("created_on", models.DateTimeField(auto_now_add=True)), - ( - "status", - models.CharField( - choices=[ - ("pending", "Pending"), - ("processed", "Processed"), - ("failed", "Failed"), - ], - default="pending", - max_length=20, - verbose_name="Notification Status", - ), - ), - ("error_message", models.TextField(blank=True, null=True)), - ( - "cart", - models.OneToOneField( + "project", + models.ForeignKey( on_delete=django.db.models.deletion.CASCADE, - related_name="notification_queue", - to="vtex.cart", + related_name="carts_by_project", + to="projects.project", ), ), ], - ), - migrations.AddIndex( - model_name="cart", - index=models.Index( - fields=["project", "status"], name="vtex_cart_project_da4d1d_idx" - ), - ), - migrations.AddIndex( - model_name="cartnotificationqueue", - index=models.Index( - fields=["status", "created_on"], name="vtex_cartno_status_1942e9_idx" - ), + options={ + "indexes": [ + models.Index( + fields=["project", "status"], + name="vtex_cart_project_da4d1d_idx", + ) + ], + }, ), ] diff --git a/retail/vtex/models.py b/retail/vtex/models.py index 0fe36a9..9505d6e 100644 --- a/retail/vtex/models.py +++ b/retail/vtex/models.py @@ -2,6 +2,7 @@ from django.db import models from retail.projects.models import Project +from retail.features.models import IntegratedFeature class Cart(models.Model): @@ -12,11 +13,10 @@ class Cart(models.Model): ("delivered_error", "Delivered Error"), ("empty", "Empty"), ] - uuid = models.UUIDField( - "UUID", primary_key=True, default=uuid.uuid4, editable=False - ) + + uuid = models.UUIDField(default=uuid.uuid4, editable=False, unique=True) created_on = models.DateTimeField(auto_now_add=True) - modified_on = models.DateTimeField(auto_now_add=True) + modified_on = models.DateTimeField(auto_now=True) status = models.CharField( max_length=20, choices=STATUS_CHOICES, @@ -26,43 +26,17 @@ class Cart(models.Model): phone_number = models.CharField(max_length=15) config = models.JSONField(default=dict) project = models.ForeignKey( - Project, on_delete=models.CASCADE, related_name="vtex_cart" - ) - abandoned = models.BooleanField(default=False) - - def __str__(self): - return f"{self.phone_number} - {self.status} on {self.modified_on}" - - class Meta: - indexes = [models.Index(fields=["project", "status"])] - - -class CartNotificationQueue(models.Model): - """ - Queue for carts that need to be processed for abandoned notifications. - """ - - uuid = models.UUIDField( - "UUID", primary_key=True, default=uuid.uuid4, editable=False + Project, on_delete=models.CASCADE, related_name="carts_by_project" ) - cart = models.OneToOneField( - Cart, on_delete=models.CASCADE, related_name="notification_queue" - ) - created_on = models.DateTimeField(auto_now_add=True) - status = models.CharField( - max_length=20, - choices=[ - ("pending", "Pending"), - ("processed", "Processed"), - ("failed", "Failed"), - ], - default="pending", - verbose_name="Notification Status", + integrated_feature = models.ForeignKey( + IntegratedFeature, on_delete=models.CASCADE, related_name="carts_by_feature" ) + abandoned = models.BooleanField(default=False) error_message = models.TextField(blank=True, null=True) def __str__(self): - return f"Queue for cart {self.cart.uuid} - {self.status}" + status = "Abandoned" if self.abandoned else self.status.capitalize() + return f"Cart: {self.phone_number}, Status: {status}, Last Modified: {self.modified_on:%Y-%m-%d %H:%M:%S}" class Meta: - indexes = [models.Index(fields=["status", "created_on"])] + indexes = [models.Index(fields=["project", "status"])] diff --git a/retail/vtex/tasks.py b/retail/vtex/tasks.py new file mode 100644 index 0000000..6221f18 --- /dev/null +++ b/retail/vtex/tasks.py @@ -0,0 +1,14 @@ +from celery import shared_task +from retail.vtex.usecases.cart_abandonment import CartAbandonmentUseCase + + +@shared_task +def mark_cart_as_abandoned(cart_uuid: str): + """ + Mark a cart as abandoned and trigger the broadcast notification process. + + Args: + cart_uuid (str): The UUID of the cart to process. + """ + use_case = CartAbandonmentUseCase() + use_case.process_abandoned_cart(cart_uuid) diff --git a/retail/vtex/usecases/cart_abandonment.py b/retail/vtex/usecases/cart_abandonment.py new file mode 100644 index 0000000..3d134a5 --- /dev/null +++ b/retail/vtex/usecases/cart_abandonment.py @@ -0,0 +1,194 @@ +import logging + +from django.conf import settings + +from retail.vtex.models import Cart +from retail.services.flows.service import FlowsService +from retail.clients.flows.client import FlowsClient +from retail.clients.exceptions import CustomAPIException + + +logger = logging.getLogger(__name__) + + +class CartAbandonmentUseCase: + """ + Use case for handling cart abandonment and notifications. + """ + + def __init__(self): + self.flows_service = FlowsService(FlowsClient()) + self.message_builder = MessageBuilder() + + def process_abandoned_cart(self, cart_uuid: str): + """ + Process a cart marked as abandoned. + + Args: + cart_uuid (str): The UUID of the cart to process. + """ + try: + # Fetch the cart + cart = Cart.objects.get(uuid=cart_uuid, status="created") + self._mark_cart_as_abandoned(cart) + + # Prepare and send the notification + payload = self.message_builder.build_abandonment_message(cart) + response = self.flows_service.send_whatsapp_broadcast( + payload=payload, + project_uuid=cart.project.uuid, + user_email=settings.FLOWS_USER_CRM_EMAIL, + ) + # Update cart status based on the response + self._update_cart_status(cart, "delivered_success", response) + except Cart.DoesNotExist: + logger.warning( + f"Cart with UUID {cart_uuid} does not exist or is already processed." + ) + except CustomAPIException as e: + logger.error( + f"Unexpected error while processing cart {cart_uuid}: {str(e)}" + ) + self._handle_error(cart_uuid, str(e)) + + def _mark_cart_as_abandoned(self, cart: Cart): + """ + Mark a cart as abandoned. + + Args: + cart (Cart): The cart to mark as abandoned. + """ + cart.abandoned = True + cart.save() + + def _send_broadcast(self, cart: Cart, msg_payload: dict, token: str) -> dict: + """ + Send a broadcast notification for the abandoned cart. + + Args: + cart (Cart): The cart for which to send the notification. + msg_payload (dict): The message payload. + token (str): The API token for authentication. + + Returns: + dict: The response from the broadcast service. + """ + phone_number = f"tel:{cart.phone_number}" + return self.flows_service.send_whatsapp_broadcast( + urns=[phone_number], + text="Cart abandoned notification", + msg=msg_payload, + token=token, + ) + + def _update_cart_status(self, cart: Cart, status: str, response=None): + """ + Update the cart's status and log errors if applicable. + + Args: + cart (Cart): The cart to update. + status (str): The new status to set. + response (dict, optional): The response from the broadcast service. Defaults to None. + """ + cart.status = status + if status == "delivered_error" and response: + cart.error_message = f"Broadcast failed: {response}" + cart.save() + + def _handle_error(self, cart_uuid: str, error_message: str): + """ + Handle unexpected errors by logging them to the cart. + + Args: + cart_uuid (str): The UUID of the cart. + error_message (str): The error message to log. + """ + try: + cart = Cart.objects.get(uuid=cart_uuid) + self._update_cart_status(cart, "delivered_error", error_message) + except Cart.DoesNotExist: + logger.error(f"Cart not found during error handling: {error_message}") + + +class MessageBuilder: + """ + Helper to build broadcast message payloads for abandoned cart notifications. + """ + + def build_abandonment_message(self, cart: Cart) -> dict: + """ + Build the message payload for an abandoned cart notification. + + Args: + cart (Cart): The cart for which to build the message. + + Returns: + dict: The message payload. + + Raises: + ValueError: If required data is missing in the cart or feature. + """ + # Fetch required data from the cart's integrated feature + template_uuid = self._get_feature_config_value(cart, "template_message") + channel_uuid = self._get_feature_config_value(cart, "flow_channel_uuid") + + # Fetch cart-specific data + cart_link = self._get_cart_config_value(cart, "cart_url") + + # Build the payload + return { + "urns": [f"whatsapp:{cart.phone_number}"], + "channel": channel_uuid, + "msg": { + "template": { + "uuid": template_uuid, + "variables": ["@contact.name"], + }, + "buttons": [ + { + "sub_type": "url", + "parameters": [{"type": "text", "text": cart_link}], + } + ], + }, + } + + def _get_feature_config_value(self, cart: Cart, key: str) -> str: + """ + Helper method to retrieve a configuration value from the integrated feature. + + Args: + cart (Cart): The cart containing the integrated feature. + key (str): The key to fetch from the feature's configuration. + + Returns: + str: The value associated with the key. + + Raises: + ValueError: If the key is missing. + """ + value = cart.integrated_feature.config.get(key) + if not value: + raise ValueError( + f"Failed to retrieve '{key}' from feature '{cart.integrated_feature.feature.name}'." + ) + return value + + def _get_cart_config_value(self, cart: Cart, key: str) -> str: + """ + Helper method to retrieve a configuration value from the cart. + + Args: + cart (Cart): The cart to fetch the value from. + key (str): The key to fetch from the cart's configuration. + + Returns: + str: The value associated with the key. + + Raises: + ValueError: If the key is missing. + """ + value = cart.config.get(key) + if not value: + raise ValueError(f"Failed to retrieve '{key}' from the cart configuration.") + return value diff --git a/retail/vtex/usecases/phone_number_normalizer.py b/retail/vtex/usecases/phone_number_normalizer.py new file mode 100644 index 0000000..63da265 --- /dev/null +++ b/retail/vtex/usecases/phone_number_normalizer.py @@ -0,0 +1,40 @@ +import re + + +class PhoneNumberNormalizer: + """ + Helper class to normalize phone numbers into the format: CC DDD NUMBER + """ + + @staticmethod + def normalize(phone_number: str) -> str: + """ + Normalize a phone number to the format CC DDD NUMBER (e.g., 5584987654321). + + Args: + phone_number (str): The phone number to normalize. + + Returns: + str: The normalized phone number. + + Raises: + ValueError: If the phone number cannot be normalized. + """ + if not phone_number: + raise ValueError("Phone number cannot be empty.") + + # Remove non-numeric characters except the leading "+" + phone_number = re.sub(r"[^\d+]", "", phone_number) + + # Ensure there is only one "+" at the beginning (if any) + if phone_number.startswith("++"): + phone_number = phone_number.lstrip("+") + + # Remove "+" and ensure only digits are left + phone_number = phone_number.lstrip("+") + + # Validate the resulting number length (minimum CC + DDD + NUMBER) + if len(phone_number) < 10: + raise ValueError(f"Invalid phone number: {phone_number}") + + return phone_number diff --git a/retail/webhooks/vtex/serializers.py b/retail/webhooks/vtex/serializers.py index f50f6c5..3551ed9 100644 --- a/retail/webhooks/vtex/serializers.py +++ b/retail/webhooks/vtex/serializers.py @@ -9,4 +9,4 @@ class CartSerializer(serializers.Serializer): action = serializers.ChoiceField(choices=["create", "update", "purchased", "empty"]) account = serializers.CharField() homePhone = serializers.CharField() - data = serializers.JSONField() + cart_url = serializers.CharField() diff --git a/retail/webhooks/vtex/usecases/cart.py b/retail/webhooks/vtex/usecases/cart.py index 13f84de..96d2240 100644 --- a/retail/webhooks/vtex/usecases/cart.py +++ b/retail/webhooks/vtex/usecases/cart.py @@ -1,8 +1,27 @@ +import logging + +from django.conf import settings + from rest_framework.exceptions import ValidationError, NotFound +from retail.features.models import Feature, IntegratedFeature from retail.projects.models import Project from retail.vtex.models import Cart +from retail.vtex.tasks import mark_cart_as_abandoned +from retail.vtex.usecases.phone_number_normalizer import PhoneNumberNormalizer from retail.webhooks.vtex.dtos.cart_dto import CartDTO +from retail.celery import app as celery_app + + +logger = logging.getLogger(__name__) + + +def generate_task_key(cart_uuid: str) -> str: + """ + Generate a deterministic task key using the cart UUID. + """ + return f"abandonment-task-{cart_uuid}" + class CartUseCase: """ @@ -12,6 +31,7 @@ class CartUseCase: def __init__(self, account: str): self.account = account self.project = self._get_project_by_account() + self.feature = self._get_feature() def _get_project_by_account(self) -> Project: """ @@ -28,6 +48,42 @@ def _get_project_by_account(self) -> Project: except Project.DoesNotExist: raise NotFound(f"Project with account '{self.account}' does not exist.") + def _get_feature(self) -> IntegratedFeature: + """ + Retrieve the IntegratedFeature for the abandoned cart notification feature + associated with the current project. + + This method fetches the `IntegratedFeature` associated with the abandoned cart + functionality for the project linked to this use case. + + Raises: + NotFound: If the feature or the integration is not found for the project. + ValidationError: For any other unexpected errors. + + Returns: + IntegratedFeature: The integrated feature instance for the abandoned cart. + """ + try: + abandoned_cart_feature_uuid = settings.ABANDONED_CART_FEATURE_UUID + feature = Feature.objects.get(uuid=abandoned_cart_feature_uuid) + return IntegratedFeature.objects.get(project=self.project, feature=feature) + except Feature.DoesNotExist: + error_message = ( + f"Feature with UUID {abandoned_cart_feature_uuid} not found." + ) + logger.error(error_message, exc_info=True) + raise NotFound(error_message) + except IntegratedFeature.DoesNotExist: + error_message = f"IntegratedFeature for project '{self.project}' and feature '{feature}' not found." + logger.error(error_message, exc_info=True) + raise NotFound(error_message) + except Exception as e: + error_message = ( + f"An unexpected error occurred while retrieving the feature: {str(e)}" + ) + logger.error(error_message, exc_info=True) # Captura o traceback completo + raise ValidationError(error_message) + def handle_action(self, action: str, cart_dto: CartDTO) -> Cart: """ Handle the specified cart action. @@ -54,7 +110,7 @@ def handle_action(self, action: str, cart_dto: CartDTO) -> Cart: def _ensure_single_cart(self, home_phone: str): """ - Ensure that a user has only one cart with the status "created". + Ensure that a user has only one cart with the status "created" and not abandoned. Args: home_phone (str): The user's phone number. @@ -62,9 +118,11 @@ def _ensure_single_cart(self, home_phone: str): Raises: ValidationError: If a cart with the "created" status already exists. """ - # Check if there's an existing cart with the status "created" cart_exists = Cart.objects.filter( - phone_number=home_phone, project=self.project, status="created" + phone_number=home_phone, + project=self.project, + status="created", + abandoned=False, ).exists() if cart_exists: @@ -72,6 +130,36 @@ def _ensure_single_cart(self, home_phone: str): {"cart": f"User with phone '{home_phone}' already has an active cart."} ) + def _schedule_abandonment_task(self, cart_uuid: str): + """ + Schedule a task to mark a cart as abandoned after 25 minutes. + + Args: + cart_uuid (str): The UUID of the cart. + + Returns: + AsyncResult: The result object for the scheduled task. + """ + task_key = generate_task_key(cart_uuid) + + # Schedule the task and capture the AsyncResult + mark_cart_as_abandoned.apply_async( + (cart_uuid,), countdown=25 * 60, task_id=task_key + ) + + # Log task details for debugging + print(f"Scheduled task with ID: {task_key}") + + def _cancel_abandonment_task(self, cart_uuid: str): + """ + Cancel a previously scheduled abandonment task. + + Args: + cart_uuid (str): The UUID of the cart. + """ + task_key = generate_task_key(cart_uuid) + celery_app.control.revoke(task_key, terminate=True) + def _create_cart(self, dto: CartDTO) -> Cart: """ Create a new cart entry. @@ -84,13 +172,24 @@ def _create_cart(self, dto: CartDTO) -> Cart: """ self._ensure_single_cart(dto.home_phone) - return Cart.objects.create( - phone_number=dto.home_phone, + try: + normalized_phone = PhoneNumberNormalizer.normalize(dto.home_phone) + except ValueError as e: + raise ValidationError({"phone_number": str(e)}) + + integrated_feature = self._get_feature() + cart = Cart.objects.create( + phone_number=normalized_phone, config=dto.data, status="created", project=self.project, + integrated_feature=integrated_feature, ) + # Schedule abandonment task + self._schedule_abandonment_task(str(cart.uuid)) + return cart + def _update_cart(self, dto: CartDTO) -> Cart: """ Update an existing cart. @@ -105,8 +204,12 @@ def _update_cart(self, dto: CartDTO) -> Cart: cart = Cart.objects.filter( phone_number=dto.home_phone, project=self.project ).latest("created_on") + cart.config.update(dto.data) cart.save() + + # Reschedule abandonment task + self._schedule_abandonment_task(str(cart.uuid)) return cart except Cart.DoesNotExist: raise NotFound(f"Cart for phone '{dto.home_phone}' does not exist.") @@ -125,8 +228,13 @@ def _mark_cart_purchased(self, dto: CartDTO) -> Cart: cart = Cart.objects.filter( phone_number=dto.home_phone, project=self.project ).latest("created_on") + cart.status = "purchased" + cart.abandoned = False cart.save() + + # Cancel abandonment task + self._cancel_abandonment_task(str(cart.uuid)) return cart except Cart.DoesNotExist: raise NotFound(f"Cart for phone '{dto.home_phone}' does not exist.") @@ -145,8 +253,13 @@ def _mark_cart_empty(self, dto: CartDTO) -> Cart: cart = Cart.objects.filter( phone_number=dto.home_phone, project=self.project ).latest("created_on") + cart.status = "empty" + cart.abandoned = False cart.save() + + # Cancel abandonment task + self._cancel_abandonment_task(str(cart.uuid)) return cart except Cart.DoesNotExist: raise NotFound(f"Cart for phone '{dto.home_phone}' does not exist.") diff --git a/retail/webhooks/vtex/views/abandoned_cart_notification.py b/retail/webhooks/vtex/views/abandoned_cart_notification.py index 5bf0b8a..248341c 100644 --- a/retail/webhooks/vtex/views/abandoned_cart_notification.py +++ b/retail/webhooks/vtex/views/abandoned_cart_notification.py @@ -28,7 +28,7 @@ def post(self, request): action=validated_data["action"], account=validated_data["account"], home_phone=validated_data["homePhone"], - data=validated_data["data"], + data=request.data, ) cart_use_case = CartUseCase(account=cart_dto.account) From bfa1ebf01fc7a60aafb2667a0fd30fa346cb95bd Mon Sep 17 00:00:00 2001 From: elitonzky Date: Mon, 30 Dec 2024 19:45:23 -0300 Subject: [PATCH 10/12] feat: adjust abandonment task and add vtex clients --- retail/clients/vtex_io/client.py | 56 ++++++ .../interfaces/clients/vtex_io/interface.py | 11 ++ retail/services/vtex_io/service.py | 46 +++++ retail/settings.py | 5 +- retail/vtex/migrations/0001_initial.py | 6 +- retail/vtex/models.py | 1 + retail/vtex/usecases/cart_abandonment.py | 181 ++++++++++++++---- retail/webhooks/vtex/serializers.py | 4 +- retail/webhooks/vtex/usecases/cart.py | 176 +++-------------- .../vtex/views/abandoned_cart_notification.py | 32 ++-- 10 files changed, 308 insertions(+), 210 deletions(-) create mode 100644 retail/clients/vtex_io/client.py create mode 100644 retail/interfaces/clients/vtex_io/interface.py create mode 100644 retail/services/vtex_io/service.py diff --git a/retail/clients/vtex_io/client.py b/retail/clients/vtex_io/client.py new file mode 100644 index 0000000..7c6f7b0 --- /dev/null +++ b/retail/clients/vtex_io/client.py @@ -0,0 +1,56 @@ +"""Client for connection with Vtex IO""" + +from django.conf import settings + +from retail.clients.base import RequestClient +from retail.interfaces.clients.vtex_io.interface import VtexIOClientInterface + + +class InternalVtexIOAuthentication(RequestClient): + def __get_module_token(self): + data = { + "client_id": settings.VTEX_IO_OIDC_RP_CLIENT_ID, + "client_secret": settings.VTEX_IO_OIDC_RP_CLIENT_SECRET, + "grant_type": "client_credentials", + } + request = self.make_request( + url=settings.OIDC_OP_TOKEN_ENDPOINT, method="POST", data=data + ) + + token = request.json().get("access_token") + + return f"Bearer {token}" + + @property + def headers(self): + return { + "Content-Type": "application/json; charset: utf-8", + "Authorization": self.__get_module_token(), + } + + +class VtexIOClient(RequestClient, VtexIOClientInterface): + def __init__(self): + self.authentication_instance = InternalVtexIOAuthentication() + + def get_order_form_details(self, account_domain: str, order_form_id: str) -> dict: + url = f"https://{account_domain}/_v/order-form-details" + params = {"orderFormId": order_form_id} + response = self.make_request( + url, + method="GET", + params=params, + headers=self.authentication_instance.headers, + ) + return response.json() + + def get_order_details(self, account_domain: str, user_email: str) -> dict: + url = f"https://{account_domain}/_v/orders-by-email" + params = {"user_email": user_email} + response = self.make_request( + url, + method="GET", + params=params, + headers=self.authentication_instance.headers, + ) + return response.json() diff --git a/retail/interfaces/clients/vtex_io/interface.py b/retail/interfaces/clients/vtex_io/interface.py new file mode 100644 index 0000000..402ea62 --- /dev/null +++ b/retail/interfaces/clients/vtex_io/interface.py @@ -0,0 +1,11 @@ +from abc import ABC, abstractmethod + + +class VtexIOClientInterface(ABC): + @abstractmethod + def get_order_form_details(self, account_domain: str, order_form_id: str) -> dict: + pass + + @abstractmethod + def get_order_details(self, account_domain: str, user_email: str) -> dict: + pass diff --git a/retail/services/vtex_io/service.py b/retail/services/vtex_io/service.py new file mode 100644 index 0000000..db0a025 --- /dev/null +++ b/retail/services/vtex_io/service.py @@ -0,0 +1,46 @@ +from retail.interfaces.clients.vtex_io.interface import VtexIOClientInterface + + +class VtexIOService: + """ + Service for interacting with VTEX IO APIs. + Provides methods to fetch order form details and order history based on email. + """ + + def __init__(self, client: VtexIOClientInterface): + """ + Initialize the VTEX IO service with the provided client. + + Args: + client (VtexIOClientInterface): The client interface for VTEX IO. + """ + self.client = client + + def get_order_form_details(self, account_domain: str, order_form_id: str) -> dict: + """ + Retrieve order form details from VTEX IO. + + Args: + account_domain (str): The domain of the VTEX account. + order_form_id (str): The unique identifier of the order form. + + Returns: + dict: The order form details if successful + + """ + return self.client.get_order_form_details(account_domain, order_form_id) + + def get_order_details(self, account_domain: str, user_email: str) -> dict: + """ + Retrieve order details by user email from VTEX IO. + + Args: + account_domain (str): The domain of the VTEX account. + user_email (str): The email address of the user. + + Returns: + dict: The order details if successful + + """ + + return self.client.get_order_details(account_domain, user_email) diff --git a/retail/settings.py b/retail/settings.py index 31e5bd5..f9fe446 100644 --- a/retail/settings.py +++ b/retail/settings.py @@ -69,7 +69,7 @@ "retail.healthcheck", "retail.internal", "rest_framework", - "retail.vtex" + "retail.vtex", ] MIDDLEWARE = [ @@ -227,3 +227,6 @@ OIDC_CACHE_TTL = env.int( "OIDC_CACHE_TTL", default=600 ) # Time-to-live for cached user tokens (default: 600 seconds). + +VTEX_IO_OIDC_RP_CLIENT_SECRET = env.str("VTEX_IO_OIDC_RP_CLIENT_SECRET", "") +VTEX_IO_OIDC_RP_CLIENT_ID = env.str("VTEX_IO_OIDC_RP_CLIENT_ID", "") diff --git a/retail/vtex/migrations/0001_initial.py b/retail/vtex/migrations/0001_initial.py index 5f5bddf..69bb413 100644 --- a/retail/vtex/migrations/0001_initial.py +++ b/retail/vtex/migrations/0001_initial.py @@ -1,4 +1,4 @@ -# Generated by Django 5.1.1 on 2024-11-29 19:23 +# Generated by Django 5.1.1 on 2024-12-30 14:18 import django.db.models.deletion import uuid @@ -6,11 +6,10 @@ class Migration(migrations.Migration): - initial = True dependencies = [ - ("features", "0015_feature_can_vtex_integrate_feature_config_and_more"), + ("features", "0016_integratedfeature_config"), ("projects", "0004_project_vtex_account"), ] @@ -31,6 +30,7 @@ class Migration(migrations.Migration): "uuid", models.UUIDField(default=uuid.uuid4, editable=False, unique=True), ), + ("cart_id", models.CharField(blank=True, null=True)), ("created_on", models.DateTimeField(auto_now_add=True)), ("modified_on", models.DateTimeField(auto_now=True)), ( diff --git a/retail/vtex/models.py b/retail/vtex/models.py index 9505d6e..490c0c2 100644 --- a/retail/vtex/models.py +++ b/retail/vtex/models.py @@ -15,6 +15,7 @@ class Cart(models.Model): ] uuid = models.UUIDField(default=uuid.uuid4, editable=False, unique=True) + cart_id = models.CharField(null=True, blank=True) created_on = models.DateTimeField(auto_now_add=True) modified_on = models.DateTimeField(auto_now=True) status = models.CharField( diff --git a/retail/vtex/usecases/cart_abandonment.py b/retail/vtex/usecases/cart_abandonment.py index 3d134a5..0901f33 100644 --- a/retail/vtex/usecases/cart_abandonment.py +++ b/retail/vtex/usecases/cart_abandonment.py @@ -2,10 +2,13 @@ from django.conf import settings +from retail.clients.vtex_io.client import VtexIOClient +from retail.services.vtex_io.service import VtexIOService from retail.vtex.models import Cart from retail.services.flows.service import FlowsService from retail.clients.flows.client import FlowsClient from retail.clients.exceptions import CustomAPIException +from retail.vtex.usecases.phone_number_normalizer import PhoneNumberNormalizer logger = logging.getLogger(__name__) @@ -16,9 +19,23 @@ class CartAbandonmentUseCase: Use case for handling cart abandonment and notifications. """ - def __init__(self): - self.flows_service = FlowsService(FlowsClient()) - self.message_builder = MessageBuilder() + def __init__( + self, + flows_service: FlowsService = None, + vtex_client: VtexIOService = None, + message_builder=None, + ): + """ + Initialize dependencies for the CartAbandonmentUseCase. + + Args: + flows_service (FlowsService): Service to handle notification flows. + vtex_client (VtexIOService): Client for interacting with VTEX API. + message_builder (MessageBuilder): Builder for constructing notification messages. + """ + self.flows_service = flows_service or FlowsService(FlowsClient()) + self.vtex_client = vtex_client or VtexIOService(VtexIOClient()) + self.message_builder = message_builder or MessageBuilder() def process_abandoned_cart(self, cart_uuid: str): """ @@ -29,18 +46,23 @@ def process_abandoned_cart(self, cart_uuid: str): """ try: # Fetch the cart - cart = Cart.objects.get(uuid=cart_uuid, status="created") - self._mark_cart_as_abandoned(cart) + cart = self._get_cart(cart_uuid) + + # Fetch order form details from VTEX IO + order_form = self._fetch_order_form(cart) + + # Process and update cart information + client_profile = self._extract_client_profile(cart, order_form) + + if not order_form.get("items", []): + # Mark cart as empty if no items are found + self._update_cart_status(cart, "empty") + return + + # Check orders by email + orders = self._fetch_orders_by_email(cart, client_profile["email"]) + self._evaluate_orders(cart, orders) - # Prepare and send the notification - payload = self.message_builder.build_abandonment_message(cart) - response = self.flows_service.send_whatsapp_broadcast( - payload=payload, - project_uuid=cart.project.uuid, - user_email=settings.FLOWS_USER_CRM_EMAIL, - ) - # Update cart status based on the response - self._update_cart_status(cart, "delivered_success", response) except Cart.DoesNotExist: logger.warning( f"Cart with UUID {cart_uuid} does not exist or is already processed." @@ -51,35 +73,123 @@ def process_abandoned_cart(self, cart_uuid: str): ) self._handle_error(cart_uuid, str(e)) - def _mark_cart_as_abandoned(self, cart: Cart): + def _get_cart(self, cart_uuid: str) -> Cart: """ - Mark a cart as abandoned. + Retrieve the cart instance by UUID. Args: - cart (Cart): The cart to mark as abandoned. + cart_uuid (str): The UUID of the cart. + + Returns: + Cart: The cart instance. + + Raises: + Cart.DoesNotExist: If no cart is found with the given UUID. + """ + return Cart.objects.get(uuid=cart_uuid, status="created") + + def _fetch_order_form(self, cart: Cart) -> dict: + """ + Retrieve order form details from VTEX API. + + Args: + cart (Cart): The cart instance. + + Returns: + dict: The order form details. + + Raises: + CustomAPIException: If the API request fails. """ - cart.abandoned = True + + order_form = self.vtex_client.get_order_form_details( + account_domain=self._get_account_domain(cart), order_form_id=cart.cart_id + ) + if not order_form: + logger.warning( + f"Order form for {cart.project.vtex_account}-{cart.uuid} is empty." + ) + raise CustomAPIException("Empty order form.") + + return order_form + + def _extract_client_profile(self, cart: Cart, order_form: dict) -> dict: + """ + Extract and normalize client profile data from order form. + + Args: + cart (Cart): The cart instance. + order_form (dict): Order form details. + + Returns: + dict: Normalized client profile data. + """ + client_profile = order_form.get("clientProfileData", {}) + phone = client_profile.get("phone") + + if phone: + normalized_phone = PhoneNumberNormalizer.normalize(phone) + cart.phone_number = normalized_phone + + # Update cart configuration and phone number + cart.config["client_profile"] = client_profile cart.save() - def _send_broadcast(self, cart: Cart, msg_payload: dict, token: str) -> dict: + return client_profile + + def _fetch_orders_by_email(self, cart: Cart, email: str) -> dict: """ - Send a broadcast notification for the abandoned cart. + Fetch orders associated with a given email. Args: - cart (Cart): The cart for which to send the notification. - msg_payload (dict): The message payload. - token (str): The API token for authentication. + cart (Cart): The cart instance. + email (str): The client email address. Returns: - dict: The response from the broadcast service. - """ - phone_number = f"tel:{cart.phone_number}" - return self.flows_service.send_whatsapp_broadcast( - urns=[phone_number], - text="Cart abandoned notification", - msg=msg_payload, - token=token, + dict: List of orders associated with the email. + """ + orders = self.vtex_client.get_order_details( + account_domain=self._get_account_domain(cart), user_email=email + ) + return orders or {"list": []} + + def _evaluate_orders(self, cart: Cart, orders: dict): + """ + Evaluate orders and determine the status of the cart. + + Args: + cart (Cart): The cart instance. + orders (dict): List of orders retrieved. + """ + if not orders.get("list"): + self._mark_cart_as_abandoned(cart) + return + + recent_orders = orders.get("list", [])[:3] + for order in recent_orders: + if order.get("orderFormId") == cart.cart_id: + self._update_cart_status(cart, "purchased") + return + + self._mark_cart_as_abandoned(cart) + + def _mark_cart_as_abandoned(self, cart: Cart): + """ + Mark a cart as abandoned and send notification. + + Args: + cart (Cart): The cart to process. + """ + self._update_cart_status(cart, "abandoned") + + # Prepare and send the notification + payload = self.message_builder.build_abandonment_message(cart) + response = self.flows_service.send_whatsapp_broadcast( + payload=payload, + project_uuid=cart.project.uuid, + user_email=settings.FLOWS_USER_CRM_EMAIL, ) + self._update_cart_status(cart, "delivered_success", response) def _update_cart_status(self, cart: Cart, status: str, response=None): """ @@ -88,7 +198,7 @@ def _update_cart_status(self, cart: Cart, status: str, response=None): Args: cart (Cart): The cart to update. status (str): The new status to set. - response (dict, optional): The response from the broadcast service. Defaults to None. + response (dict, optional): The response from the broadcast service. """ cart.status = status if status == "delivered_error" and response: @@ -97,11 +207,11 @@ def _update_cart_status(self, cart: Cart, status: str, response=None): def _handle_error(self, cart_uuid: str, error_message: str): """ - Handle unexpected errors by logging them to the cart. + Handle errors by logging and updating the cart status. Args: cart_uuid (str): The UUID of the cart. - error_message (str): The error message to log. + error_message (str): Error message to log. """ try: cart = Cart.objects.get(uuid=cart_uuid) @@ -109,6 +219,9 @@ def _handle_error(self, cart_uuid: str, error_message: str): except Cart.DoesNotExist: logger.error(f"Cart not found during error handling: {error_message}") + def _get_account_domain(self, cart: Cart) -> str: + return f"dev5--{cart.project.vtex_account}.myvtex.com" + class MessageBuilder: """ diff --git a/retail/webhooks/vtex/serializers.py b/retail/webhooks/vtex/serializers.py index 3551ed9..616df1e 100644 --- a/retail/webhooks/vtex/serializers.py +++ b/retail/webhooks/vtex/serializers.py @@ -6,7 +6,5 @@ class CartSerializer(serializers.Serializer): Serializer to validate cart data received from VTEX. """ - action = serializers.ChoiceField(choices=["create", "update", "purchased", "empty"]) account = serializers.CharField() - homePhone = serializers.CharField() - cart_url = serializers.CharField() + cart_id = serializers.CharField() diff --git a/retail/webhooks/vtex/usecases/cart.py b/retail/webhooks/vtex/usecases/cart.py index 96d2240..f0dd1c7 100644 --- a/retail/webhooks/vtex/usecases/cart.py +++ b/retail/webhooks/vtex/usecases/cart.py @@ -84,182 +84,58 @@ def _get_feature(self) -> IntegratedFeature: logger.error(error_message, exc_info=True) # Captura o traceback completo raise ValidationError(error_message) - def handle_action(self, action: str, cart_dto: CartDTO) -> Cart: + def process_cart_notification(self, cart_id: str) -> Cart: """ - Handle the specified cart action. + Process incoming cart notification, renewing task or creating new cart. Args: - action (str): The action to handle (create, update, purchased, empty). - cart_dto (CartDTO): The validated cart data. + cart_id (str): The unique identifier for the cart. Returns: - Cart: The updated or created cart instance. + Cart: The created or updated cart instance. """ - action_methods = { - "create": self._create_cart, - "update": self._update_cart, - "purchased": self._mark_cart_purchased, - "empty": self._mark_cart_empty, - } - - if action not in action_methods: - raise ValidationError({"action": f"Invalid action: {action}"}) - - # Call the appropriate method dynamically - return action_methods[action](cart_dto) - - def _ensure_single_cart(self, home_phone: str): - """ - Ensure that a user has only one cart with the status "created" and not abandoned. - - Args: - home_phone (str): The user's phone number. - - Raises: - ValidationError: If a cart with the "created" status already exists. - """ - cart_exists = Cart.objects.filter( - phone_number=home_phone, - project=self.project, - status="created", - abandoned=False, - ).exists() - - if cart_exists: - raise ValidationError( - {"cart": f"User with phone '{home_phone}' already has an active cart."} + try: + # Check if the cart already exists + cart = Cart.objects.get( + cart_id=cart_id, project=self.project, status="created" ) + # Renew abandonment task + self._schedule_abandonment_task(str(cart.uuid)) + return cart + except Cart.DoesNotExist: + # Create new cart if it doesn't exist + return self._create_cart(cart_id) - def _schedule_abandonment_task(self, cart_uuid: str): - """ - Schedule a task to mark a cart as abandoned after 25 minutes. - - Args: - cart_uuid (str): The UUID of the cart. - - Returns: - AsyncResult: The result object for the scheduled task. - """ - task_key = generate_task_key(cart_uuid) - - # Schedule the task and capture the AsyncResult - mark_cart_as_abandoned.apply_async( - (cart_uuid,), countdown=25 * 60, task_id=task_key - ) - - # Log task details for debugging - print(f"Scheduled task with ID: {task_key}") - - def _cancel_abandonment_task(self, cart_uuid: str): - """ - Cancel a previously scheduled abandonment task. - - Args: - cart_uuid (str): The UUID of the cart. - """ - task_key = generate_task_key(cart_uuid) - celery_app.control.revoke(task_key, terminate=True) - - def _create_cart(self, dto: CartDTO) -> Cart: + def _create_cart(self, cart_id: str) -> Cart: """ - Create a new cart entry. + Create a new cart entry and schedule an abandonment task. Args: - dto (CartDTO): The cart DTO. + cart_id (str): The UUID of the cart. Returns: Cart: The created cart instance. """ - self._ensure_single_cart(dto.home_phone) - - try: - normalized_phone = PhoneNumberNormalizer.normalize(dto.home_phone) - except ValueError as e: - raise ValidationError({"phone_number": str(e)}) - - integrated_feature = self._get_feature() cart = Cart.objects.create( - phone_number=normalized_phone, - config=dto.data, + cart_id=cart_id, status="created", project=self.project, - integrated_feature=integrated_feature, + integrated_feature=self._get_feature(), ) # Schedule abandonment task self._schedule_abandonment_task(str(cart.uuid)) return cart - def _update_cart(self, dto: CartDTO) -> Cart: - """ - Update an existing cart. - - Args: - dto (CartDTO): The cart DTO. - - Returns: - Cart: The updated cart instance. - """ - try: - cart = Cart.objects.filter( - phone_number=dto.home_phone, project=self.project - ).latest("created_on") - - cart.config.update(dto.data) - cart.save() - - # Reschedule abandonment task - self._schedule_abandonment_task(str(cart.uuid)) - return cart - except Cart.DoesNotExist: - raise NotFound(f"Cart for phone '{dto.home_phone}' does not exist.") - - def _mark_cart_purchased(self, dto: CartDTO) -> Cart: - """ - Mark a cart as purchased. - - Args: - dto (CartDTO): The cart DTO. - - Returns: - Cart: The updated cart instance. - """ - try: - cart = Cart.objects.filter( - phone_number=dto.home_phone, project=self.project - ).latest("created_on") - - cart.status = "purchased" - cart.abandoned = False - cart.save() - - # Cancel abandonment task - self._cancel_abandonment_task(str(cart.uuid)) - return cart - except Cart.DoesNotExist: - raise NotFound(f"Cart for phone '{dto.home_phone}' does not exist.") - - def _mark_cart_empty(self, dto: CartDTO) -> Cart: + def _schedule_abandonment_task(self, cart_uuid: str): """ - Mark a cart as empty. + Schedule a task to mark a cart as abandoned after 25 minutes. Args: - dto (CartDTO): The cart DTO. - - Returns: - Cart: The updated cart instance. + cart_uuid (str): The UUID of the cart. """ - try: - cart = Cart.objects.filter( - phone_number=dto.home_phone, project=self.project - ).latest("created_on") - - cart.status = "empty" - cart.abandoned = False - cart.save() + task_key = generate_task_key(cart_uuid) - # Cancel abandonment task - self._cancel_abandonment_task(str(cart.uuid)) - return cart - except Cart.DoesNotExist: - raise NotFound(f"Cart for phone '{dto.home_phone}' does not exist.") + mark_cart_as_abandoned.apply_async( + (cart_uuid,), countdown=25 * 60, task_id=task_key + ) diff --git a/retail/webhooks/vtex/views/abandoned_cart_notification.py b/retail/webhooks/vtex/views/abandoned_cart_notification.py index 248341c..146192c 100644 --- a/retail/webhooks/vtex/views/abandoned_cart_notification.py +++ b/retail/webhooks/vtex/views/abandoned_cart_notification.py @@ -1,43 +1,37 @@ from rest_framework.views import APIView from rest_framework.response import Response from rest_framework import status -from rest_framework.permissions import AllowAny +from retail.internal.permissions import CanCommunicateInternally from retail.webhooks.vtex.serializers import CartSerializer -from retail.webhooks.vtex.dtos.cart_dto import CartDTO from retail.webhooks.vtex.usecases.cart import CartUseCase class AbandonedCartNotification(APIView): """ - View to handle abandoned cart notifications. - - This view receives data from the VTEX IO middleware, - processes it, and performs necessary actions. + Handle abandoned cart notifications. """ - - authentication_classes = [] - permission_classes = [AllowAny] + permission_classes = [CanCommunicateInternally] def post(self, request): + # Validação dos dados recebidos serializer = CartSerializer(data=request.data) serializer.is_valid(raise_exception=True) + # Extrai dados validados validated_data = serializer.validated_data - cart_dto = CartDTO( - action=validated_data["action"], - account=validated_data["account"], - home_phone=validated_data["homePhone"], - data=request.data, - ) + account = validated_data["account"] + cart_id = validated_data["cart_id"] - cart_use_case = CartUseCase(account=cart_dto.account) - result = cart_use_case.handle_action(cart_dto.action, cart_dto) + # Processa a notificação + cart_use_case = CartUseCase(account=account) + result = cart_use_case.process_cart_notification(cart_id) return Response( { - "message": f"Cart action '{cart_dto.action}' processed successfully.", - "cart_id": str(result.uuid), + "message": "Cart processed successfully.", + "cart_uuid": str(result.uuid), + "cart_id": str(result.cart_id), "status": result.status, }, status=status.HTTP_200_OK, From 39f12b665bda0163c17391d586d348692cb88bfa Mon Sep 17 00:00:00 2001 From: elitonzky Date: Fri, 3 Jan 2025 19:32:43 -0300 Subject: [PATCH 11/12] fix: move authentication to params and adjust phone number serializer --- retail/clients/vtex_io/client.py | 83 +++++++++++++------ .../vtex/usecases/phone_number_normalizer.py | 18 ++-- retail/webhooks/vtex/usecases/cart.py | 2 - 3 files changed, 69 insertions(+), 34 deletions(-) diff --git a/retail/clients/vtex_io/client.py b/retail/clients/vtex_io/client.py index 7c6f7b0..4d020f5 100644 --- a/retail/clients/vtex_io/client.py +++ b/retail/clients/vtex_io/client.py @@ -7,50 +7,85 @@ class InternalVtexIOAuthentication(RequestClient): - def __get_module_token(self): + """ + Handles authentication with VTEX IO using client credentials. + """ + + def __get_module_token(self) -> str: + """ + Retrieves the access token from the VTEX IO OIDC endpoint. + """ + # Authentication payload data = { "client_id": settings.VTEX_IO_OIDC_RP_CLIENT_ID, "client_secret": settings.VTEX_IO_OIDC_RP_CLIENT_SECRET, "grant_type": "client_credentials", } - request = self.make_request( + response = self.make_request( url=settings.OIDC_OP_TOKEN_ENDPOINT, method="POST", data=data ) + # Extracts the token + token = response.json().get("access_token") + if not token: + raise ValueError("Failed to retrieve access token.") - token = request.json().get("access_token") - - return f"Bearer {token}" + return token @property - def headers(self): - return { - "Content-Type": "application/json; charset: utf-8", - "Authorization": self.__get_module_token(), - } + def token(self) -> str: + """ + Returns the access token to be used in requests. + """ + return self.__get_module_token() class VtexIOClient(RequestClient, VtexIOClientInterface): + """ + Handles API communication with VTEX IO. + """ + def __init__(self): + """ + Initializes the authentication instance. + """ self.authentication_instance = InternalVtexIOAuthentication() def get_order_form_details(self, account_domain: str, order_form_id: str) -> dict: + """ + Fetches order form details by ID. + + Args: + account_domain (str): VTEX account domain. + order_form_id (str): Unique identifier for the order form. + + Returns: + dict: Order form details. + """ url = f"https://{account_domain}/_v/order-form-details" - params = {"orderFormId": order_form_id} - response = self.make_request( - url, - method="GET", - params=params, - headers=self.authentication_instance.headers, - ) + params = { + "orderFormId": order_form_id, + "token": self.authentication_instance.token, + } + response = self.make_request(url, method="GET", params=params) + return response.json() def get_order_details(self, account_domain: str, user_email: str) -> dict: + """ + Fetches order details by user email. + + Args: + account_domain (str): VTEX account domain. + user_email (str): Email address of the user. + + Returns: + dict: Order details. + """ url = f"https://{account_domain}/_v/orders-by-email" - params = {"user_email": user_email} - response = self.make_request( - url, - method="GET", - params=params, - headers=self.authentication_instance.headers, - ) + params = { + "user_email": user_email, + "token": self.authentication_instance.token, + } + response = self.make_request(url, method="GET", params=params) + return response.json() diff --git a/retail/vtex/usecases/phone_number_normalizer.py b/retail/vtex/usecases/phone_number_normalizer.py index 63da265..5fde5b4 100644 --- a/retail/vtex/usecases/phone_number_normalizer.py +++ b/retail/vtex/usecases/phone_number_normalizer.py @@ -1,4 +1,5 @@ import re +from rest_framework.exceptions import ValidationError class PhoneNumberNormalizer: @@ -18,23 +19,24 @@ def normalize(phone_number: str) -> str: str: The normalized phone number. Raises: - ValueError: If the phone number cannot be normalized. + ValidationError: If the phone number cannot be normalized. """ - if not phone_number: - raise ValueError("Phone number cannot be empty.") + # Check if the number is empty or censored (contains '*') + if not phone_number or "*" in phone_number: + raise ValidationError(f"Invalid or censored phone number: {phone_number}") - # Remove non-numeric characters except the leading "+" + # Remove all non-numeric characters except the leading "+" phone_number = re.sub(r"[^\d+]", "", phone_number) - # Ensure there is only one "+" at the beginning (if any) + # Remove multiple "+" and keep only one at the beginning (if present) if phone_number.startswith("++"): phone_number = phone_number.lstrip("+") - # Remove "+" and ensure only digits are left + # Remove any remaining "+" and keep only digits phone_number = phone_number.lstrip("+") - # Validate the resulting number length (minimum CC + DDD + NUMBER) + # Validate if the number has at least 10 digits (CC + DDD + Number) if len(phone_number) < 10: - raise ValueError(f"Invalid phone number: {phone_number}") + raise ValidationError(f"Invalid phone number length: {phone_number}") return phone_number diff --git a/retail/webhooks/vtex/usecases/cart.py b/retail/webhooks/vtex/usecases/cart.py index f0dd1c7..02a268d 100644 --- a/retail/webhooks/vtex/usecases/cart.py +++ b/retail/webhooks/vtex/usecases/cart.py @@ -7,8 +7,6 @@ from retail.projects.models import Project from retail.vtex.models import Cart from retail.vtex.tasks import mark_cart_as_abandoned -from retail.vtex.usecases.phone_number_normalizer import PhoneNumberNormalizer -from retail.webhooks.vtex.dtos.cart_dto import CartDTO from retail.celery import app as celery_app From 6c8591d5b0f7870dc3acdbeacf3382f86deef211 Mon Sep 17 00:00:00 2001 From: Eliton Jorge Date: Wed, 8 Jan 2025 16:02:52 -0300 Subject: [PATCH 12/12] feat: add service to create template message on integrations engine (#70) --- retail/clients/integrations/client.py | 36 ++++++ .../clients/integrations/interface.py | 12 ++ retail/services/integrations/service.py | 116 +++++++++++++++++- 3 files changed, 163 insertions(+), 1 deletion(-) diff --git a/retail/clients/integrations/client.py b/retail/clients/integrations/client.py index 1297ba1..41dd5ab 100644 --- a/retail/clients/integrations/client.py +++ b/retail/clients/integrations/client.py @@ -18,3 +18,39 @@ def get_vtex_integration_detail(self, project_uuid): url, method="GET", headers=self.authentication_instance.headers ) return response.json() + + def create_template_message( + self, app_uuid: str, project_uuid: str, name: str, category: str + ) -> str: + url = f"{self.base_url}/api/v1/apps/{app_uuid}/templates/" + + payload = { + "name": name, + "category": category, + "text_preview": name, + "project_uuid": project_uuid, + } + + response = self.make_request( + url, + method="POST", + json=payload, + headers=self.authentication_instance.headers, + ) + template_uuid = response.json().get("uuid") + return template_uuid + + def create_template_translation( + self, app_uuid: str, project_uuid: str, template_uuid: str, payload: dict + ): + payload["project_uuid"] = project_uuid + + url = f"{self.base_url}/api/v1/apps/{app_uuid}/templates/{template_uuid}/translations/" + + response = self.make_request( + url, + method="POST", + json=payload, + headers=self.authentication_instance.headers, + ) + return response diff --git a/retail/interfaces/clients/integrations/interface.py b/retail/interfaces/clients/integrations/interface.py index 1df03f8..53ae25d 100644 --- a/retail/interfaces/clients/integrations/interface.py +++ b/retail/interfaces/clients/integrations/interface.py @@ -5,3 +5,15 @@ class IntegrationsClientInterface(ABC): @abstractmethod def get_vtex_integration_detail(self, project_uuid: str): pass + + @abstractmethod + def create_template_message( + self, app_uuid: str, project_uuid: str, name: str, category: str + ) -> str: + pass + + @abstractmethod + def create_template_translation( + self, app_uuid: str, project_uuid: str, template_uuid: str, payload: dict + ) -> dict: + pass diff --git a/retail/services/integrations/service.py b/retail/services/integrations/service.py index 53fed70..834aebd 100644 --- a/retail/services/integrations/service.py +++ b/retail/services/integrations/service.py @@ -14,5 +14,119 @@ def get_vtex_integration_detail(self, project_uuid: str) -> dict: try: return self.client.get_vtex_integration_detail(project_uuid) except CustomAPIException as e: - print(f"Error {e.status_code} when retrieving VTEX integration for project {project_uuid}.") + print( + f"Error {e.status_code} when retrieving VTEX integration for project {project_uuid}." + ) return None + + def create_abandoned_cart_template( + self, app_uuid: str, project_uuid: str, store: str + ) -> dict: + """ + Creates an abandoned cart template and translations for multiple languages. + """ + try: + # Create Template + template_uuid = self.client.create_template_message( + app_uuid=app_uuid, + project_uuid=project_uuid, + name="weni_abandoned_cart_notification", + category="MARKETING", + ) + + # Prepare translations for multiple languages + translations = [ + { + "language": "pt_BR", + "body": { + "type": "BODY", + "text": ( + "Olá, {{1}} vimos que você deixou itens no seu carrinho 🛒. " + "\nVamos fechar o pedido e garantir essas ofertas? " + "\n\nClique em Finalizar Pedido para concluir sua compra 👇" + ), + "example": {"body_text": [["João"]]}, + }, + "footer": {"type": "FOOTER", "text": "Finalizar Pedido"}, + "buttons": [ + { + "button_type": "URL", + "text": "Finalizar Pedido", + "url": f"https://{store}/checkout/cart/add?sc=1{{1}}", + "example": ["&sku=1&qty=1"], + }, + { + "button_type": "QUICK_REPLY", + "text": "Parar Promoções", + }, + ], + }, + { + "language": "es", + "body": { + "type": "BODY", + "text": ( + "Hola, {{1}} notamos que dejaste artículos en tu carrito 🛒. " + "\n¿Listo para completar tu pedido y asegurar estas ofertas? " + "\n\nHaz clic en Finalizar Pedido para completar tu compra 👇" + ), + "example": {"body_text": [["Juan"]]}, + }, + "footer": {"type": "FOOTER", "text": "Finalizar Pedido"}, + "buttons": [ + { + "button_type": "URL", + "text": "Finalizar Pedido", + "url": f"https://{store}/checkout/cart/add?sc=1{{1}}", + "example": ["&sku=1&qty=1"], + }, + { + "button_type": "QUICK_REPLY", + "text": "Parar Promociones", + }, + ], + }, + { + "language": "en", + "body": { + "type": "BODY", + "text": ( + "Hello, {{1}} we noticed you left items in your cart 🛒. " + "\nReady to complete your order and secure these deals? " + "\n\nClick Finish Order to complete your purchase 👇" + ), + "example": {"body_text": [["John"]]}, + }, + "footer": {"type": "FOOTER", "text": "Finish Order"}, + "buttons": [ + { + "button_type": "URL", + "text": "Finish Order", + "url": f"https://{store}/checkout/cart/add?sc=1{{1}}", + "example": ["&sku=1&qty=1"], + }, + { + "button_type": "QUICK_REPLY", + "text": "Stop Promotions", + }, + ], + }, + ] + + # Create translations for each language + for translation in translations: + self.client.create_template_translation( + app_uuid=app_uuid, + project_uuid=project_uuid, + template_uuid=template_uuid, + payload=translation, + ) + print(f"Translation created for language {translation['language']}.") + + return {"template_uuid": template_uuid} + + except CustomAPIException as e: + print( + f"Error {e.status_code} during template or translation creation: {str(e)}" + ) + raise