From cbdece0403c11fec057ab423ba5f6e89d6648c18 Mon Sep 17 00:00:00 2001 From: Shwetabh Kumar Date: Mon, 18 Mar 2024 11:47:31 +0530 Subject: [PATCH 1/2] Fix: Updating the same QBD Mapping in case of name case change --- apps/mappings/actions.py | 9 +++++---- apps/mappings/connector.py | 3 +++ .../migrations/0002_auto_20240318_0616.py | 18 ++++++++++++++++++ apps/mappings/models.py | 10 +++++----- 4 files changed, 31 insertions(+), 9 deletions(-) create mode 100644 apps/mappings/migrations/0002_auto_20240318_0616.py diff --git a/apps/mappings/actions.py b/apps/mappings/actions.py index a04a3c6..04ca50d 100644 --- a/apps/mappings/actions.py +++ b/apps/mappings/actions.py @@ -5,17 +5,18 @@ def get_qbd_mapping_stat(source_type: str, workspace_id: int): """ get qbd mapping stat will return the count of total mappings available and unmapped mappings """ - total_attributes_count = QBDMapping.objects.filter( workspace_id=workspace_id, - attribute_type = source_type).count() + attribute_type = source_type + ).count() unmapped_attributes_count = QBDMapping.objects.filter( workspace_id=workspace_id, attribute_type = source_type, - destination_value__isnull=True).count() + destination_value__isnull=True + ).count() return { 'all_attributes_count': total_attributes_count, 'unmapped_attributes_count': unmapped_attributes_count - } + } diff --git a/apps/mappings/connector.py b/apps/mappings/connector.py index fc6f3b6..08b6c2a 100644 --- a/apps/mappings/connector.py +++ b/apps/mappings/connector.py @@ -30,7 +30,9 @@ def sync_corporate_card(self): query = { 'order': 'updated_at.desc', } + generator = self.platform.v1beta.admin.corporate_cards.list_all(query) + for items in generator: card_attributes = [] unique_card_numbers = [] @@ -47,5 +49,6 @@ def sync_corporate_card(self): 'value': value, 'source_id': card['id'], }) + if len(card_attributes) > 0: QBDMapping.update_or_create_mapping_objects(card_attributes, self.workspace_id) diff --git a/apps/mappings/migrations/0002_auto_20240318_0616.py b/apps/mappings/migrations/0002_auto_20240318_0616.py new file mode 100644 index 0000000..2ec12f2 --- /dev/null +++ b/apps/mappings/migrations/0002_auto_20240318_0616.py @@ -0,0 +1,18 @@ +# Generated by Django 3.1.14 on 2024-03-18 06:16 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('mappings', '0001_initial'), + ] + + operations = [ + migrations.AlterField( + model_name='qbdmapping', + name='source_id', + field=models.CharField(help_text='Fyle ID', max_length=255, unique=True), + ), + ] diff --git a/apps/mappings/models.py b/apps/mappings/models.py index 0f77889..86276d2 100644 --- a/apps/mappings/models.py +++ b/apps/mappings/models.py @@ -10,7 +10,7 @@ class QBDMapping(models.Model): id = models.AutoField(primary_key=True) attribute_type = models.CharField(max_length=255, help_text='Type of expense attribute') source_value = models.CharField(max_length=1000, help_text='Value of expense attribute') - source_id = models.CharField(max_length=255, help_text='Fyle ID') + source_id = models.CharField(max_length=255, help_text='Fyle ID', unique=True) destination_value = models.CharField(max_length=1000, null=True, blank=True, help_text='Value of destination attribute') workspace = models.ForeignKey(Workspace, on_delete=models.PROTECT, help_text='Reference to Workspace model') @@ -24,10 +24,10 @@ class Meta: def update_or_create_mapping_objects(qbd_mapping_objects: List[Dict],workspace_id: int): for qbd_mapping_object in qbd_mapping_objects: QBDMapping.objects.update_or_create( - workspace_id= workspace_id, - source_value= qbd_mapping_object['value'], - attribute_type= qbd_mapping_object['attribute_type'], + workspace_id=workspace_id, + source_id=qbd_mapping_object['source_id'], + attribute_type=qbd_mapping_object['attribute_type'], defaults={ - 'source_id': qbd_mapping_object['source_id'], + 'source_value': qbd_mapping_object['value'], } ) From 6c8a26ec0af8397c3900ade3abb9815387f03ee6 Mon Sep 17 00:00:00 2001 From: Shwetabh Kumar Date: Mon, 18 Mar 2024 12:25:57 +0530 Subject: [PATCH 2/2] Fixing tests --- apps/mappings/serializers.py | 6 +++++ apps/mappings/views.py | 2 +- tests/conftest.py | 2 +- tests/test_fyle/test_view.py | 6 ++--- .../test_mapping/{fixture.py => fixtures.py} | 6 ++--- tests/test_mapping/test_connector.py | 8 +++--- tests/test_mapping/test_view.py | 25 +++++++++++-------- tests/test_qbd/test_tasks.py | 11 ++++---- 8 files changed, 37 insertions(+), 29 deletions(-) rename tests/test_mapping/{fixture.py => fixtures.py} (96%) diff --git a/apps/mappings/serializers.py b/apps/mappings/serializers.py index da3289c..9fef871 100644 --- a/apps/mappings/serializers.py +++ b/apps/mappings/serializers.py @@ -8,9 +8,15 @@ class Meta: model = QBDMapping fields = '__all__' read_only_fields = ('workspace', 'created_at', 'updated_at') + extra_kwargs = { + 'source_id': { + 'validators': [] + } + } def create(self, validated_data): workspace_id = self.context['request'].parser_context.get('kwargs').get('workspace_id') + qbd_mapping, _ = QBDMapping.objects.update_or_create( source_id=validated_data['source_id'], workspace_id=workspace_id, diff --git a/apps/mappings/views.py b/apps/mappings/views.py index 264dec4..613637a 100644 --- a/apps/mappings/views.py +++ b/apps/mappings/views.py @@ -24,7 +24,7 @@ class QBDMappingView(LookupFieldMixin, generics.ListCreateAPIView): ordering_fields = ('source_value',) -#mapping stats view +# Mapping stats view class QBDMappingStatsView(generics.RetrieveAPIView): """ Stats for total mapped and unmapped count for a given attribute type diff --git a/tests/conftest.py b/tests/conftest.py index 317df7a..7838d25 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -22,7 +22,7 @@ from quickbooks_desktop_api.tests import settings from .test_fyle.fixtures import fixtures as fyle_fixtures -from .test_mapping.fixture import fixture as mapping_fixtures +from .test_mapping.fixtures import fixture as mapping_fixtures @pytest.fixture diff --git a/tests/test_fyle/test_view.py b/tests/test_fyle/test_view.py index bb3a7d2..21b30e4 100644 --- a/tests/test_fyle/test_view.py +++ b/tests/test_fyle/test_view.py @@ -3,7 +3,7 @@ from apps.mappings.models import QBDMapping -from tests.test_mapping.fixture import fixture +from tests.test_mapping.fixtures import fixture @pytest.mark.django_db(databases=['default']) def test_sync_fyle_dimension_view(api_client, test_connection, mocker): @@ -31,5 +31,5 @@ def test_sync_fyle_dimension_view(api_client, test_connection, mocker): qbd_mapping = QBDMapping.objects.filter(workspace_id=workspace_id) assert response.status_code == 200 - assert len(qbd_mapping) == fixture['get_qbd_CCC_mapping']['count'] - assert qbd_mapping[0].source_value == fixture['get_qbd_CCC_mapping']['results'][0]['source_value'] + assert len(qbd_mapping) == fixture['get_qbd_ccc_mapping']['count'] + assert qbd_mapping[0].source_value == fixture['get_qbd_ccc_mapping']['results'][0]['source_value'] diff --git a/tests/test_mapping/fixture.py b/tests/test_mapping/fixtures.py similarity index 96% rename from tests/test_mapping/fixture.py rename to tests/test_mapping/fixtures.py index 53b912d..c9cb252 100644 --- a/tests/test_mapping/fixture.py +++ b/tests/test_mapping/fixtures.py @@ -20,7 +20,7 @@ ] } ], - 'post_qbd_CCC_mapping': { + 'post_qbd_ccc_mapping': { "id": 1, "attribute_type": "CORPORATE_CARD", "source_value": "AMERICAN EXPRESS - 4818", @@ -30,7 +30,7 @@ "updated_at": "2023-08-28T10:07:30.524503Z", "workspace": 1 }, - 'get_qbd_CCC_mapping_state': { + 'get_qbd_ccc_mapping_state': { "all_attributes_count":2, "unmapped_attributes_count":2 }, @@ -46,7 +46,7 @@ 'source_id': 'baccEg1AzugNxZ', } ], - 'get_qbd_CCC_mapping': { + 'get_qbd_ccc_mapping': { "count": 2, "next": "http://localhost:8008/api/workspaces/1/qbd_mappings/?attribute_type=CORPORATE_CARD", "previous": '', diff --git a/tests/test_mapping/test_connector.py b/tests/test_mapping/test_connector.py index 35ce020..365823a 100644 --- a/tests/test_mapping/test_connector.py +++ b/tests/test_mapping/test_connector.py @@ -1,7 +1,7 @@ import pytest from apps.mappings.connector import PlatformConnector from apps.mappings.models import QBDMapping -from .fixture import fixture +from .fixtures import fixture @pytest.mark.django_db(databases=['default'], transaction=True) @@ -17,7 +17,7 @@ def test_sync_corporate_card(create_temp_workspace, qbd_connection = PlatformConnector(workspace_id=workspace_id) qbd_connection.sync_corporate_card() qbd_mappings = QBDMapping.objects.filter(workspace_id=workspace_id, attribute_type = 'CORPORATE_CARD') - assert len(qbd_mappings) == len(fixture['get_qbd_CCC_mapping']['results']) - for i, item in enumerate(qbd_mappings): - assert qbd_mappings[i].source_value == fixture['get_qbd_CCC_mapping']['results'][i]['source_value'] + assert len(qbd_mappings) == len(fixture['get_qbd_ccc_mapping']['results']) + for i, _ in enumerate(qbd_mappings): + assert qbd_mappings[i].source_value == fixture['get_qbd_ccc_mapping']['results'][i]['source_value'] diff --git a/tests/test_mapping/test_view.py b/tests/test_mapping/test_view.py index 7a6440e..497f372 100644 --- a/tests/test_mapping/test_view.py +++ b/tests/test_mapping/test_view.py @@ -3,7 +3,7 @@ from apps.mappings.models import QBDMapping -from .fixture import fixture +from .fixtures import fixture @pytest.mark.django_db(databases=['default']) def test_qbd_mapping_view(api_client, test_connection): @@ -28,8 +28,8 @@ def test_qbd_mapping_view(api_client, test_connection): response = api_client.get(url, {'attribute_type': 'CORPORATE_CARD', 'limit':10, 'offset':0}) assert response.status_code == 200 - assert len(response.data['results']) == len(fixture['get_qbd_CCC_mapping']['results']) - assert response.data['count'] == fixture['get_qbd_CCC_mapping']['count'] + assert len(response.data['results']) == len(fixture['get_qbd_ccc_mapping']['results']) + assert response.data['count'] == fixture['get_qbd_ccc_mapping']['count'] # post qbd mapping url = reverse( @@ -50,6 +50,8 @@ def test_qbd_mapping_view(api_client, test_connection): post_value = QBDMapping.objects.filter(workspace_id=workspace_id, attribute_type = payload['attribute_type'], source_id = payload['source_id']) + + print(response.data) assert response.status_code == 201 assert post_value[0].destination_value == payload['destination_value'] @@ -59,9 +61,9 @@ def test_qbd_mapping_view(api_client, test_connection): response = api_client.get(url, param) assert response.status_code == 200 - assert len(response.data['results']) == len(fixture['get_qbd_CCC_mapping']['results'])-1 - assert response.data['count'] == fixture['get_qbd_CCC_mapping']['count']-1 - assert response.data['results'][0]['source_value'] == fixture['get_qbd_CCC_mapping']['results'][0]['source_value'] + assert len(response.data['results']) == len(fixture['get_qbd_ccc_mapping']['results'])-1 + assert response.data['count'] == fixture['get_qbd_ccc_mapping']['count']-1 + assert response.data['results'][0]['source_value'] == fixture['get_qbd_ccc_mapping']['results'][0]['source_value'] # get all unmapped rows (destination_value__isnull is true) @@ -69,9 +71,10 @@ def test_qbd_mapping_view(api_client, test_connection): response = api_client.get(url, param) assert response.status_code == 200 - assert len(response.data['results']) == len(fixture['get_qbd_CCC_mapping']['results'])-1 - assert response.data['count'] == fixture['get_qbd_CCC_mapping']['count']-1 - assert response.data['results'][0]['source_value'] == fixture['get_qbd_CCC_mapping']['results'][1]['source_value'] + assert len(response.data['results']) == len(fixture['get_qbd_ccc_mapping']['results'])-1 + assert response.data['count'] == fixture['get_qbd_ccc_mapping']['count']-1 + assert response.data['results'][0]['source_value'] == fixture['get_qbd_ccc_mapping']['results'][1]['source_value'] + @pytest.mark.django_db(databases=['default']) def test_qbd_mapping_stats_view(api_client, test_connection): @@ -95,5 +98,5 @@ def test_qbd_mapping_stats_view(api_client, test_connection): response = api_client.get(url, {'source_type': 'CORPORATE_CARD'}) assert response.status_code==200 - assert response.data['all_attributes_count']==fixture['get_qbd_CCC_mapping_state']['all_attributes_count'] - assert response.data['unmapped_attributes_count']==fixture['get_qbd_CCC_mapping_state']['unmapped_attributes_count'] + assert response.data['all_attributes_count']==fixture['get_qbd_ccc_mapping_state']['all_attributes_count'] + assert response.data['unmapped_attributes_count']==fixture['get_qbd_ccc_mapping_state']['unmapped_attributes_count'] diff --git a/tests/test_qbd/test_tasks.py b/tests/test_qbd/test_tasks.py index 95709f3..694ff30 100644 --- a/tests/test_qbd/test_tasks.py +++ b/tests/test_qbd/test_tasks.py @@ -268,7 +268,7 @@ def test_create_credit_card_purchases_iif_file_expense_vendor( create_temp_workspace, add_accounting_export_bills, add_accounting_export_expenses, add_fyle_credentials, add_export_settings, add_field_mappings, add_advanced_settings, - add_ccc_mapping, mocker + mocker ): """ Test create credit card purchases iif file @@ -331,7 +331,7 @@ def test_create_credit_card_purchases_iif_file_expense_employee( create_temp_workspace, add_accounting_export_bills, add_accounting_export_expenses, add_fyle_credentials, add_export_settings, add_field_mappings, add_advanced_settings, - add_ccc_mapping, mocker + mocker ): """ Test create credit card purchases iif file @@ -399,8 +399,7 @@ def test_create_credit_card_purchases_iif_file_expense_employee( def test_create_credit_card_purchases_iif_file_expense_fail( create_temp_workspace, add_accounting_export_bills, add_accounting_export_expenses, add_fyle_credentials, - add_export_settings, add_advanced_settings, - add_ccc_mapping, mocker + add_export_settings, add_advanced_settings, mocker ): """ Test create credit card purchases iif file @@ -455,7 +454,7 @@ def test_create_credit_card_purchases_iif_file_expense_fatal( create_temp_workspace, add_accounting_export_bills, add_accounting_export_expenses, add_fyle_credentials, add_export_settings, add_field_mappings, add_advanced_settings, - add_ccc_mapping, mocker + mocker ): """ Test create credit card purchases iif file @@ -822,7 +821,7 @@ def test_email_failure( create_temp_workspace, add_accounting_export_bills, add_accounting_export_expenses, add_fyle_credentials, add_export_settings, add_field_mappings, add_advanced_settings, - add_ccc_mapping, mocker + mocker ): """ Test create journals iif file Failed for email failure