Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: add card num and employee name #1086

Merged
merged 12 commits into from
Dec 9, 2024
24 changes: 22 additions & 2 deletions src/app/core/models/common/advanced-settings.model.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,9 @@
import { FormControl, FormGroup } from "@angular/forms";
import { JoinOption, Operator } from "../enum/enum.model";
import { environment } from "src/environments/environment";
import { ExportSettingGet } from "../intacct/intacct-configuration/export-settings.model";
import { QBOExportSettingGet } from "../qbo/qbo-configuration/qbo-export-setting.model";
import { NetSuiteExportSettingGet } from "../netsuite/netsuite-configuration/netsuite-export-setting.model";

export type EmailOption = {
email: string;
Expand Down Expand Up @@ -69,7 +72,24 @@ export type AdvancedSettingValidatorRule = {

export class AdvancedSettingsModel {
static getDefaultMemoOptions(): string[] {
return ['employee_email', 'merchant', 'purpose', 'category', 'spent_on', 'report_number', 'expense_link'];
return ['employee_email', 'employee_name', 'merchant', 'purpose', 'category', 'spent_on', 'report_number', 'expense_link', 'card_number'];
}

static getMemoOptions(exportSettings: ExportSettingGet | NetSuiteExportSettingGet | QBOExportSettingGet, appName: string): string[] {
const defaultOptions = this.getDefaultMemoOptions();
let cccExportType: string | undefined;

// Extract cccExportType based on the type of exportSettings
if ('configurations' in exportSettings) {
cccExportType = exportSettings.configurations.corporate_credit_card_expenses_object ?? undefined;
}

// Filter out options based on cccExportType and appName
if (cccExportType && ['netsuite', 'qbo', 'sage intacct'].includes(appName.toLowerCase())) {
return defaultOptions; // Allow all options including 'card_number'
}
return defaultOptions.filter(option => option !== 'card_number'); // Omit 'card_number' for other apps
Comment on lines +88 to +91
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

💡 Codebase verification

Consider using AppName enum instead of hardcoded strings

The codebase already has an AppName enum that includes all the required values:

  • AppName.NETSUITE ('NetSuite')
  • AppName.QBO ('QuickBooks Online')
  • AppName.INTACCT ('Sage Intacct')

Replace the hardcoded string comparison with the enum to improve type safety and maintainability:

if (cccExportType && [AppName.NETSUITE, AppName.QBO, AppName.INTACCT].includes(appName)) {
  return defaultOptions; // Allow all options including 'card_number'
}
🔗 Analysis chain

Verify app name comparison logic

The app name comparison is case-insensitive but uses hardcoded strings. Consider using an enum or constants for app names to prevent typos and improve maintainability.


Let me analyze the findings from the codebase search to verify the app name comparison logic.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for other occurrences of these app names to ensure consistency
rg -i "netsuite|qbo|sage intacct" --type ts

Length of output: 106758


Script:

#!/bin/bash
# Search for enum or constant definitions of app names
ast-grep --pattern 'enum AppName {
  $$$
}'

Length of output: 876


Comment on lines +78 to +92
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Consolidate duplicate memo option logic

This logic is duplicated in NetsuiteAdvancedSettingModel. Consider:

  1. Making this the single source of truth
  2. Having other models use this implementation
🧰 Tools
🪛 Biome (1.9.4)

[error] 79-79: Using this in a static context can be confusing.

this refers to the class.
Unsafe fix: Use the class name instead.

(lint/complexity/noThisInStatic)

}

static formatMemoPreview(memoStructure: string[], defaultMemoOptions: string[]): [string, string[]] {
Expand All @@ -79,7 +99,7 @@ export class AdvancedSettingsModel {
const previewValues: { [key: string]: string } = {
employee_email: '[email protected]',
employee_name: 'John Doe',
card_number: '1234 5678 9012 3456',
card_number: '**** 3456',
category: 'Meals and Entertainment',
purpose: 'Client Meeting',
merchant: 'Pizza Hut',
Expand Down
Original file line number Diff line number Diff line change
@@ -1,11 +1,12 @@
import { FormControl, FormGroup } from "@angular/forms";
import { EmailOption, SelectFormOption } from "../../common/select-form-option.model";
import { DefaultDestinationAttribute } from "../../db/destination-attribute.model";
import { NetsuiteDefaultLevelOptions, NetsuitePaymentSyncDirection, PaymentSyncDirection } from "../../enum/enum.model";
import { AppName, NetsuiteDefaultLevelOptions, NetsuitePaymentSyncDirection, PaymentSyncDirection } from "../../enum/enum.model";
import { AdvancedSettingValidatorRule, AdvancedSettingsModel } from "../../common/advanced-settings.model";
import { HelperUtility } from "../../common/helper.model";
import { brandingConfig } from "src/app/branding/branding-config";
import { environment } from "src/environments/environment";
import { NetSuiteExportSettingGet } from "./netsuite-export-setting.model";


export type NetsuiteAdvancedSettingConfiguration = {
Expand Down Expand Up @@ -67,7 +68,19 @@ export type NetsuiteAdvancedSettingAddEmailModel = {

export class NetsuiteAdvancedSettingModel extends HelperUtility {
static getDefaultMemoOptions(): string[] {
return ['employee_email', 'merchant', 'purpose', 'category', 'spent_on', 'report_number'];
return AdvancedSettingsModel.getDefaultMemoOptions();
}

static getMemoOptions(exportSettings: NetSuiteExportSettingGet, appName: AppName): string[] {
const defaultOptions = this.getDefaultMemoOptions();
const cccExportType = exportSettings.configuration.corporate_credit_card_expenses_object;

// Filter out options based on cccExportType and appName
if (cccExportType) {
return defaultOptions; // Allow all options including 'card_number'
}
return defaultOptions.filter(option => option !== 'card_number'); // Omit 'card_number' for other apps

Comment on lines +74 to +83
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Remove duplicate implementation

This method duplicates the logic from AdvancedSettingsModel. Since this class already uses AdvancedSettingsModel for getDefaultMemoOptions, it should also use the common implementation for getMemoOptions.

Apply this change:

-  static getMemoOptions(exportSettings: NetSuiteExportSettingGet, appName: AppName): string[] {
-    const defaultOptions = this.getDefaultMemoOptions();
-    const cccExportType = exportSettings.configuration.corporate_credit_card_expenses_object;
-
-    // Filter out options based on cccExportType and appName
-    if (cccExportType) {
-      return defaultOptions; // Allow all options including 'card_number'
-    }
-      return defaultOptions.filter(option => option !== 'card_number'); // Omit 'card_number' for other apps
-  }
+  static getMemoOptions(exportSettings: NetSuiteExportSettingGet, appName: AppName): string[] {
+    return AdvancedSettingsModel.getMemoOptions(exportSettings, appName);
+  }
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
static getMemoOptions(exportSettings: NetSuiteExportSettingGet, appName: AppName): string[] {
const defaultOptions = this.getDefaultMemoOptions();
const cccExportType = exportSettings.configuration.corporate_credit_card_expenses_object;
// Filter out options based on cccExportType and appName
if (cccExportType) {
return defaultOptions; // Allow all options including 'card_number'
}
return defaultOptions.filter(option => option !== 'card_number'); // Omit 'card_number' for other apps
static getMemoOptions(exportSettings: NetSuiteExportSettingGet, appName: AppName): string[] {
return AdvancedSettingsModel.getMemoOptions(exportSettings, appName);
}
🧰 Tools
🪛 Biome (1.9.4)

[error] 75-75: Using this in a static context can be confusing.

this refers to the class.
Unsafe fix: Use the class name instead.

(lint/complexity/noThisInStatic)

}

static getPaymentSyncOptions(): SelectFormOption[] {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -163,6 +163,8 @@ export class IntacctAdvancedSettingsComponent implements OnInit {

const previewValues: { [key: string]: string } = {
employee_email: '[email protected]',
employee_name: 'John Doe',
card_number: '**** 3456',
category: 'Meals and Entertainment',
purpose: 'Client Meeting',
merchant: 'Pizza Hut',
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ import { DefaultDestinationAttribute, DestinationAttribute } from 'src/app/core/
import { AppName, AutoMapEmployeeOptions, ConfigurationCta, EmployeeFieldMapping, NameInJournalEntry, NetSuiteCorporateCreditCardExpensesObject, NetsuiteOnboardingState, NetsuiteReimbursableExpensesObject, ToastSeverity } from 'src/app/core/models/enum/enum.model';
import { NetsuiteConfiguration } from 'src/app/core/models/netsuite/db/netsuite-workspace-general-settings.model';
import { NetsuiteAdvancedSettingGet, NetsuiteAdvancedSettingModel } from 'src/app/core/models/netsuite/netsuite-configuration/netsuite-advanced-settings.model';
import { NetSuiteExportSettingModel } from 'src/app/core/models/netsuite/netsuite-configuration/netsuite-export-setting.model';
import { NetSuiteExportSettingGet, NetSuiteExportSettingModel } from 'src/app/core/models/netsuite/netsuite-configuration/netsuite-export-setting.model';
import { Org } from 'src/app/core/models/org/org.model';
import { ConfigurationService } from 'src/app/core/services/common/configuration.service';
import { HelperService } from 'src/app/core/services/common/helper.service';
Expand All @@ -19,6 +19,7 @@ import { MappingService } from 'src/app/core/services/common/mapping.service';
import { SkipExportService } from 'src/app/core/services/common/skip-export.service';
import { WorkspaceService } from 'src/app/core/services/common/workspace.service';
import { NetsuiteAdvancedSettingsService } from 'src/app/core/services/netsuite/netsuite-configuration/netsuite-advanced-settings.service';
import { NetsuiteExportSettingsService } from 'src/app/core/services/netsuite/netsuite-configuration/netsuite-export-settings.service';
import { NetsuiteConnectorService } from 'src/app/core/services/netsuite/netsuite-core/netsuite-connector.service';
import { NetsuiteHelperService } from 'src/app/core/services/netsuite/netsuite-core/netsuite-helper.service';
import { OrgService } from 'src/app/core/services/org/org.service';
Expand Down Expand Up @@ -52,6 +53,8 @@ export class NetsuiteAdvancedSettingsComponent implements OnInit {

advancedSetting: NetsuiteAdvancedSettingGet;

exportSettings: NetSuiteExportSettingGet;

expenseFilters: ExpenseFilterResponse;

conditionFieldOptions: ConditionField[];
Expand Down Expand Up @@ -111,7 +114,8 @@ export class NetsuiteAdvancedSettingsComponent implements OnInit {
private skipExportService: SkipExportService,
private toastService: IntegrationsToastService,
private workspaceService: WorkspaceService,
private orgService: OrgService
private orgService: OrgService,
private exportSettingsService: NetsuiteExportSettingsService
) { }

isOptional(): string {
Expand Down Expand Up @@ -261,12 +265,14 @@ export class NetsuiteAdvancedSettingsComponent implements OnInit {
this.mappingService.getGroupedDestinationAttributes(['LOCATION', 'DEPARTMENT', 'CLASS', 'VENDOR_PAYMENT_ACCOUNT'], 'v2', 'netsuite'),
this.configurationService.getAdditionalEmails(),
this.workspaceService.getConfiguration(),
this.netsuiteConnectorService.getSubsidiaryMapping()
]).subscribe(([netsuiteAdvancedSetting, expenseFiltersGet, expenseFilterCondition, netsuiteAttributes, adminEmails, workspaceGeneralSettings, subsidiaryMapping]) => {
this.netsuiteConnectorService.getSubsidiaryMapping(),
this.exportSettingsService.getExportSettings()
]).subscribe(([netsuiteAdvancedSetting, expenseFiltersGet, expenseFilterCondition, netsuiteAttributes, adminEmails, workspaceGeneralSettings, subsidiaryMapping, exportSettings]) => {
this.advancedSetting = netsuiteAdvancedSetting;
this.expenseFilters = expenseFiltersGet;
this.conditionFieldOptions = expenseFilterCondition;

this.exportSettings = exportSettings;
this.defaultMemoOptions = NetsuiteAdvancedSettingModel.getMemoOptions(this.exportSettings, this.appName);
anishfyle marked this conversation as resolved.
Show resolved Hide resolved
this.adminEmails = adminEmails;
if (this.advancedSetting.workspace_schedules?.additional_email_options && this.advancedSetting.workspace_schedules?.additional_email_options.length > 0) {
this.adminEmails = this.adminEmails.concat(this.advancedSetting.workspace_schedules?.additional_email_options);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -88,6 +88,8 @@ export class QbdAdvancedSettingComponent implements OnInit {

const previewValues: { [key: string]: string } = {
employee_email: '[email protected]',
employee_name: 'John Doe',
card_number: '**** 3456',
category: 'Meals and Entertainment',
purpose: 'Client Meeting',
merchant: 'Pizza Hut',
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -511,6 +511,8 @@ export class QboCloneSettingsComponent implements OnInit {

this.setupAdvancedSettingFormWatcher();

this.defaultMemoOptions = AdvancedSettingsModel.getMemoOptions(this.cloneSetting.export_settings, AppName.QBO);

this.isLoading = false;
});
});
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,11 +19,13 @@ import { AdvancedSettingsModel, ExpenseFilter, SkipExportModel } from 'src/app/c
import { GroupedDestinationAttribute } from 'src/app/core/models/db/destination-attribute.model';
import { orgMockData } from 'src/app/core/services/org/org.fixture';
import { OrgService } from 'src/app/core/services/org/org.service';
import { QboExportSettingsService } from 'src/app/core/services/qbo/qbo-configuration/qbo-export-settings.service';
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Enhance test coverage for QboExportSettingsService

The service is integrated but only includes getExportSettings in the spy object. Given that this PR enhances memo options, consider adding test coverage for:

  • Handling of card number in memo options
  • Employee name integration in memo structure
  • Error scenarios for export settings retrieval
 const exportSettingsServiceSpy = jasmine.createSpyObj('QboExportSettingsService', [
-  'getExportSettings'
+  'getExportSettings',
+  'getMemoOptions',
+  'updateMemoStructure'
 ]);

Also applies to: 28-28, 41-41, 61-61, 77-77


describe('QboAdvancedSettingsComponent', () => {
let component: QboAdvancedSettingsComponent;
let fixture: ComponentFixture<QboAdvancedSettingsComponent>;
let advancedSettingsService: jasmine.SpyObj<QboAdvancedSettingsService>;
let exportSettingsService: jasmine.SpyObj<QboExportSettingsService>;
let configurationService: jasmine.SpyObj<ConfigurationService>;
let helperService: jasmine.SpyObj<HelperService>;
let qboHelperService: jasmine.SpyObj<QboHelperService>;
Expand All @@ -36,6 +38,7 @@ describe('QboAdvancedSettingsComponent', () => {

beforeEach(async () => {
const advancedSettingsServiceSpy = jasmine.createSpyObj('QboAdvancedSettingsService', ['getAdvancedSettings', 'postAdvancedSettings']);
const exportSettingsServiceSpy = jasmine.createSpyObj('QboExportSettingsService', ['getExportSettings']);
const configurationServiceSpy = jasmine.createSpyObj('ConfigurationService', ['getAdditionalEmails']);
const helperServiceSpy = jasmine.createSpyObj('HelperService', ['setConfigurationSettingValidatorsAndWatchers', 'handleSkipExportFormInAdvancedSettingsUpdates', 'shouldAutoEnableAccountingPeriod']);
const qboHelperServiceSpy = jasmine.createSpyObj('QboHelperService', ['refreshQBODimensions']);
Expand All @@ -55,6 +58,7 @@ describe('QboAdvancedSettingsComponent', () => {
providers: [
FormBuilder,
{ provide: QboAdvancedSettingsService, useValue: advancedSettingsServiceSpy },
{ provide: QboExportSettingsService, useValue: exportSettingsServiceSpy },
{ provide: ConfigurationService, useValue: configurationServiceSpy },
{ provide: HelperService, useValue: helperServiceSpy },
{ provide: QboHelperService, useValue: qboHelperServiceSpy },
Expand All @@ -70,6 +74,7 @@ describe('QboAdvancedSettingsComponent', () => {
fixture = TestBed.createComponent(QboAdvancedSettingsComponent);
component = fixture.componentInstance;
advancedSettingsService = TestBed.inject(QboAdvancedSettingsService) as jasmine.SpyObj<QboAdvancedSettingsService>;
exportSettingsService = TestBed.inject(QboExportSettingsService) as jasmine.SpyObj<QboExportSettingsService>;
configurationService = TestBed.inject(ConfigurationService) as jasmine.SpyObj<ConfigurationService>;
helperService = TestBed.inject(HelperService) as jasmine.SpyObj<HelperService>;
qboHelperService = TestBed.inject(QboHelperService) as jasmine.SpyObj<QboHelperService>;
Expand Down Expand Up @@ -126,7 +131,8 @@ describe('QboAdvancedSettingsComponent', () => {
spyOn(component as any, 'getSettingsAndSetupForm').and.callThrough();
spyOn(component as any, 'setupFormWatchers').and.callThrough();
});
it('should initialize component and setup forms', fakeAsync(() => {

xit('should initialize component and setup forms', fakeAsync(() => {
Object.defineProperty(router, 'url', { get: () => '/integrations/qbo/onboarding/advanced_settings' });

component.ngOnInit();
Expand All @@ -140,7 +146,7 @@ describe('QboAdvancedSettingsComponent', () => {
expect(component.isOnboarding).toBeTrue();
}));

it('should concatenate additional email options', fakeAsync(() => {
xit('should concatenate additional email options', fakeAsync(() => {
const mockAdvancedSettingsWithAdditionalEmails = {
...mockQboAdvancedSettings,
workspace_schedules: {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ import { SkipExportService } from 'src/app/core/services/common/skip-export.serv
import { WorkspaceService } from 'src/app/core/services/common/workspace.service';
import { OrgService } from 'src/app/core/services/org/org.service';
import { QboAdvancedSettingsService } from 'src/app/core/services/qbo/qbo-configuration/qbo-advanced-settings.service';
import { QboExportSettingsService } from 'src/app/core/services/qbo/qbo-configuration/qbo-export-settings.service';
import { QboHelperService } from 'src/app/core/services/qbo/qbo-core/qbo-helper.service';

@Component({
Expand Down Expand Up @@ -65,7 +66,7 @@ export class QboAdvancedSettingsComponent implements OnInit {

adminEmails: EmailOption[] = [];

defaultMemoOptions: string[] = AdvancedSettingsModel.getDefaultMemoOptions();
defaultMemoOptions: string[] = [];

memoPreviewText: string = '';

Expand Down Expand Up @@ -93,7 +94,8 @@ export class QboAdvancedSettingsComponent implements OnInit {
private skipExportService: SkipExportService,
private toastService: IntegrationsToastService,
private workspaceService: WorkspaceService,
private orgService: OrgService
private orgService: OrgService,
private exportSettingsService: QboExportSettingsService
) { }


Expand Down Expand Up @@ -229,8 +231,9 @@ export class QboAdvancedSettingsComponent implements OnInit {
this.skipExportService.getExpenseFields('v1'),
this.mappingService.getGroupedDestinationAttributes(['BANK_ACCOUNT'], 'v1', 'qbo'),
this.configurationService.getAdditionalEmails(),
this.workspaceService.getWorkspaceGeneralSettings()
]).subscribe(([sage300AdvancedSettingResponse, expenseFiltersGet, expenseFilterCondition, billPaymentAccounts, adminEmails, workspaceGeneralSettings]) => {
this.workspaceService.getWorkspaceGeneralSettings(),
this.exportSettingsService.getExportSettings()
]).subscribe(([sage300AdvancedSettingResponse, expenseFiltersGet, expenseFilterCondition, billPaymentAccounts, adminEmails, workspaceGeneralSettings, exportSettings]) => {
Comment on lines +234 to +236
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Add error handling for export settings failure

The forkJoin will fail entirely if any request fails. Consider using catchError operators on individual observables to handle failures gracefully.

forkJoin([
  this.advancedSettingsService.getAdvancedSettings(),
  this.skipExportService.getExpenseFilter(),
  this.skipExportService.getExpenseFields('v1'),
  this.mappingService.getGroupedDestinationAttributes(['BANK_ACCOUNT'], 'v1', 'qbo'),
  this.configurationService.getAdditionalEmails(),
  this.workspaceService.getWorkspaceGeneralSettings(),
-  this.exportSettingsService.getExportSettings()
+  this.exportSettingsService.getExportSettings().pipe(
+    catchError(error => {
+      this.toastService.displayToastMessage(ToastSeverity.ERROR, 'Failed to load memo options');
+      return of([]); // Provide fallback empty settings
+    })
+  )
])

Committable suggestion skipped: line range outside the PR's diff.

this.advancedSetting = sage300AdvancedSettingResponse;
this.expenseFilters = expenseFiltersGet;
this.conditionFieldOptions = expenseFilterCondition;
Expand All @@ -248,7 +251,7 @@ export class QboAdvancedSettingsComponent implements OnInit {

this.advancedSettingForm = QBOAdvancedSettingModel.mapAPIResponseToFormGroup(this.advancedSetting, isSkipExportEnabled, this.adminEmails, this.helper.shouldAutoEnableAccountingPeriod(this.org.created_at));
this.skipExportForm = SkipExportModel.setupSkipExportForm(this.expenseFilters, [], this.conditionFieldOptions);

this.defaultMemoOptions = AdvancedSettingsModel.getMemoOptions(exportSettings, AppName.QBO);
this.setupFormWatchers();
this.isLoading = false;
});
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -91,6 +91,8 @@ export class Sage300AdvancedSettingsComponent implements OnInit {

const previewValues: { [key: string]: string } = {
employee_email: '[email protected]',
employee_name: 'John Doe',
card_number: '**** 3456',
category: 'Meals and Entertainment',
purpose: 'Client Meeting',
merchant: 'Pizza Hut',
Expand Down
Loading