@accounter/server
Version:
517 lines (495 loc) • 17.1 kB
text/typescript
import { GraphQLError } from 'graphql';
import { BusinessesProvider } from '@modules/financial-entities/providers/businesses.provider.js';
import { TransactionsProvider } from '@modules/transactions/providers/transactions.provider.js';
import {
ETANA_BUSINESS_ID,
ETHERSCAN_BUSINESS_ID,
KRAKEN_BUSINESS_ID,
POALIM_BUSINESS_ID,
} from '@shared/constants';
import { ChargeTypeEnum } from '@shared/enums';
import type {
ChargeResolvers,
Maybe,
Resolver,
ResolversParentTypes,
ResolversTypes,
} from '@shared/gql-types';
import { formatAmount } from '@shared/helpers';
import { getChargeType } from '../../helpers/charge-type.js';
import type { ChargesModule } from '../../types.js';
import { missingConversionInfoSuggestions } from './conversion-suggeestions.resolver.js';
type SuggestionData = {
description?: string;
tags: Array<{
name: string;
}>;
phrases: Array<string>;
};
export type Suggestion = Awaited<ResolversTypes['ChargeSuggestions']>;
const missingInfoSuggestions: Resolver<
Maybe<Suggestion>,
ResolversParentTypes['Charge'],
GraphQLModules.Context
> = async (DbCharge, _, context, __) => {
// if all required fields are filled, no need for suggestions
if (!!DbCharge.tags?.length && !!DbCharge.user_description?.trim()) {
return null;
}
const { injector } = context;
const chargeType = getChargeType(DbCharge);
if (chargeType === ChargeTypeEnum.Conversion) {
return missingConversionInfoSuggestions(DbCharge, _, context, __);
}
// if charge has a businesses, use it's suggestion data
if (DbCharge.business_id) {
const business = await injector
.get(BusinessesProvider)
.getBusinessByIdLoader.load(DbCharge.business_id);
if (business?.suggestion_data) {
const suggestionData = business.suggestion_data as SuggestionData;
return {
description: suggestionData.description,
tags: suggestionData.tags,
};
}
}
if (DbCharge.business_array && DbCharge.business_array.length > 1) {
const isKrakenIncluded = DbCharge.business_array.includes(KRAKEN_BUSINESS_ID);
const isEtherscanIncluded = DbCharge.business_array.includes(ETHERSCAN_BUSINESS_ID);
const isEtanaIncluded = DbCharge.business_array.includes(ETANA_BUSINESS_ID);
const isPoalimIncluded = DbCharge.business_array.includes(POALIM_BUSINESS_ID);
if (isKrakenIncluded && isEtherscanIncluded) {
return {
description: 'Etherscan to Kraken transfer',
tags: [],
};
}
if (isKrakenIncluded && isEtanaIncluded) {
return {
description: 'Kraken to Etana transfer',
tags: [],
};
}
if (isPoalimIncluded && isEtanaIncluded) {
return {
description: 'Etana to Poalim transfer',
tags: [],
};
}
}
const allBusinesses = await injector.get(BusinessesProvider).getAllBusinesses();
const suggestions: Record<string, Suggestion> = {};
for (const business of allBusinesses) {
if (!business.suggestion_data) continue;
const suggestionData = business.suggestion_data as SuggestionData;
if (business.id in (DbCharge.business_array ?? [])) {
return {
description: suggestionData.description,
tags: suggestionData.tags,
};
}
for (const phrase of suggestionData.phrases) {
suggestions[phrase] = {
description: suggestionData.description,
tags: suggestionData.tags,
};
}
}
const transactions = await injector
.get(TransactionsProvider)
.getTransactionsByChargeIDLoader.load(DbCharge.id);
const description = transactions.map(t => t.source_description).join(' ');
for (const [phrase, suggestion] of Object.entries(suggestions)) {
if (Array.isArray(phrase) && new RegExp(phrase.join('|')).test(description)) {
return suggestion;
}
if (description.includes(phrase)) {
return suggestion;
}
}
if (DbCharge.business_id === KRAKEN_BUSINESS_ID && transactions.length > 1) {
let fromCurrency: string | undefined;
let toCurrency: string | undefined;
for (const transaction of transactions) {
if (transaction.is_fee) continue;
const amount = formatAmount(transaction.amount);
if (amount > 0) {
if (toCurrency) {
throw new GraphQLError('Multiple destination currencies in Kraken conversion');
}
toCurrency = transaction.currency;
}
if (amount < 0) {
if (fromCurrency) {
throw new GraphQLError('Multiple source currencies in Kraken conversion');
}
fromCurrency = transaction.currency;
}
if (fromCurrency && toCurrency) {
return {
description: `${fromCurrency} to ${toCurrency} conversion`,
tags: [{ name: 'conversion' }],
};
}
}
}
if (
description.includes('ע\' העברת מט"ח') ||
(description.includes('העברת מט"ח') && Math.abs(formatAmount(DbCharge.event_amount)) < 400) ||
(description.includes('מטח') && Math.abs(formatAmount(DbCharge.event_amount)) < 400) ||
description.includes('F.C.COM') ||
description.includes('ע.מפעולות-ישיר') ||
description.includes('ריבית חובה') ||
description.includes('FEE')
) {
const sourceTransaction =
transactions.length === 0
? 'Missing'
: transactions.length === 1
? transactions[0].source_reference
: `['${transactions.map(t => t.source_reference).join("','")}']`;
return {
tags: [{ name: 'financial' }],
description: `Fees for source transaction=${sourceTransaction}`,
};
}
if (description.includes('דותן שמחה') || description.includes('שמחה דותן')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', {
month: '2-digit',
});
return {
description: `${previousMonth}/2022 Salary`,
tags: [{ name: 'business' }],
};
}
if (description.includes('גולדשטין אורי')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', {
month: '2-digit',
});
return {
description: `${previousMonth}/2022 Salary`,
tags: [{ name: 'business' }],
};
}
if (description.includes('גרדוש')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', {
month: '2-digit',
});
return {
description: `${previousMonth}/2022 Salary`,
tags: [{ name: 'business' }],
};
}
if (description.includes('תובל')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', {
month: '2-digit',
});
return {
description: `${previousMonth}/2022 Salary`,
tags: [{ name: 'business' }],
};
}
if (description.includes('מנורה מבטחים פנס')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', {
month: '2-digit',
});
return {
description: `Pension ${previousMonth}/2022`,
tags: [{ name: 'business' }],
};
}
if (description.includes('פניקס אקסלנס')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', {
month: '2-digit',
});
return {
description: `Training Fund ${previousMonth}/2022`,
tags: [{ name: 'business' }],
};
}
if (description.includes('מיטב דש גמל ופנס')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', {
month: '2-digit',
});
return {
description: `Pension ${previousMonth}/2022`,
tags: [{ name: 'business' }],
};
}
if (description.includes('מגדל מקפת')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', {
month: '2-digit',
});
return {
description: `Pension ${previousMonth}/2022`,
tags: [{ name: 'business' }],
};
}
if (description.includes('מגדל השתלמות')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', {
month: '2-digit',
});
return {
description: `Training Fund ${previousMonth}/2022`,
tags: [{ name: 'business' }],
};
}
if (description.includes('ביטוח לאומי')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', {
month: '2-digit',
});
return {
description: `Social Security Deductions for Salaries ${previousMonth}/2022`,
tags: [{ name: 'business' }],
};
}
if (description.includes('LANCE GLOBAL')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', { month: 'long' });
return {
description: `The Guild Enterprise Support - ${previousMonth} 2022`,
tags: [{ name: 'business' }],
};
}
if (
(description.includes('העברת מט"ח') &&
(description.includes('fbv') || description.includes('fv'))) ||
description.includes('kamil kisiela')
) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', {
month: '2-digit',
});
return {
description: `Software Development and Consulting ${previousMonth}/23`,
tags: [{ name: 'business' }],
};
}
if (description.includes('slava')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', { month: '2-digit' });
return {
description: `Web Development Services ${previousMonth}/23`,
tags: [{ name: 'business' }],
};
}
if (description.includes('COURIER PLUS INC')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', { month: 'long' });
return {
description: `GraphQL Hive Enterprise License - ${previousMonth} 2023`,
tags: [{ name: 'business' }],
};
}
if (description.includes('GOBRANDS')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', { month: 'long' });
return {
description: `GraphQL Hive Enterprise License - ${previousMonth} 2023`,
tags: [{ name: 'business' }],
};
}
if (description.includes('MEDIC FIRST AI')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', { month: 'long' });
return {
description: `GraphQL Hive Enterprise License - ${previousMonth} 2023`,
tags: [{ name: 'business' }],
};
}
if (description.includes('מס הכנסה')) {
const flag = description.includes('מס הכנסה ני');
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', {
month: '2-digit',
});
return {
tags: [{ name: 'business' }],
description: flag
? `Tax for employees for ${previousMonth}/2022`
: `Advance Tax for ${previousMonth}/2022`,
};
}
if (description.includes('גורניצקי')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', {
month: '2-digit',
});
return {
description: `${previousMonth}/2022 lawyer support`,
tags: [{ name: 'business' }],
};
}
if (description.includes('המכס ומעמ-גביי תשלום') || description.includes('CUSTOM + V.A.T')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', {
month: '2-digit',
});
return {
description: `VAT for ${previousMonth}/2022`,
tags: [{ name: 'business' }],
};
}
if (description.includes('חניון')) {
return {
description: 'Parking',
tags: [{ name: 'transportation' }],
};
}
if (description.includes('ETANA')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', { month: 'long' });
return {
description: `The Guild Enterprise Support - ${previousMonth} 2022`,
tags: [{ name: 'business' }],
};
}
if (description.includes('deel')) {
return {
description: 'Laurin Salary',
tags: [{ name: 'business' }],
};
}
if (description.includes('GITHUB')) {
const suggested = {
description: 'GitHub Actions',
tags: [{ name: 'business' }],
};
if (formatAmount(DbCharge.event_amount) <= -2000) {
suggested.description = 'Monthly Sponsor for Benjie, Code-Hex, hayes';
} else if (formatAmount(DbCharge.event_amount) <= -1000) {
suggested.description = 'Monthly Sponsor for Andarist, warrenday';
} else {
suggested.description = 'GitHub Actions';
}
return suggested;
}
if (formatAmount(DbCharge.event_amount) === -4329) {
return {
description: 'Office rent',
tags: [{ name: 'business' }],
};
}
if (description.includes('APPLE COM BILL/ITUNES.COM')) {
const flag = formatAmount(DbCharge.event_amount) === -109.9;
return {
taxCategory: 'אתר',
beneficiaaries: [], // NOTE: used to be ' '
description: flag ? 'LinkedIn' : 'Apple Services',
tags: [{ name: flag ? 'business' : 'computer' }],
};
}
if (
description.includes('ע\' העברת מט"ח') ||
(description.includes('העברת מט"ח') && Math.abs(formatAmount(DbCharge.event_amount)) < 400) ||
(description.includes('מטח') && Math.abs(formatAmount(DbCharge.event_amount)) < 400) ||
description.includes('F.C.COM') ||
description.includes('ע.מפעולות-ישיר') ||
description.includes('ריבית חובה') ||
description.includes('FEE')
) {
//NOTE: multiple suggestions business
return {
tags: [{ name: 'financial' }],
description: `Fees for bank_reference=${transactions[0].source_reference ?? 'Missing'}`,
};
}
if (description.includes('ריבית זכות')) {
//NOTE: multiple suggestions business
return {
description: 'Interest fees on Euro plus',
tags: [{ name: 'financial' }],
};
}
if (description.includes('פועלים- דמי כרטיס')) {
//NOTE: multiple suggestions business
return {
description: 'Bank creditcard fees',
tags: [{ name: 'financial' }],
};
}
if (description.includes('אריה קריסטל')) {
//NOTE: multiple suggestions business
return {
description: 'Water bill for 04-2022',
tags: [{ name: 'house' }],
};
}
if (description.includes('aleksandra')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', { month: '2-digit' });
return {
description: `Software Consulting Fees (${previousMonth}/2023)`,
tags: [{ name: 'business' }],
};
}
if (description.includes('denelop')) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', { month: '2-digit' });
return {
description: `Software Development and Consulting ${previousMonth}/2023`,
tags: [{ name: 'business' }],
};
}
if (formatAmount(DbCharge.event_amount) === -12_000) {
const current = new Date();
current.setMonth(current.getMonth() - 1);
const previousMonth = current.toLocaleString('default', { month: '2-digit' });
return {
description: `${previousMonth}/2022`,
tags: [{ name: 'business' }],
};
}
if (formatAmount(DbCharge.event_amount) === -600) {
return {
description: 'Matic Zavadlal - April 2021',
tags: [{ name: 'business' }],
};
}
return null;
};
const commonChargeFields: ChargesModule.ChargeResolvers = {
missingInfoSuggestions: missingInfoSuggestions as ChargeResolvers['missingInfoSuggestions'],
};
export const chargeSuggestionsResolvers: ChargesModule.Resolvers = {
CommonCharge: commonChargeFields,
ConversionCharge: commonChargeFields,
SalaryCharge: commonChargeFields,
InternalTransferCharge: commonChargeFields,
DividendCharge: commonChargeFields,
BusinessTripCharge: commonChargeFields,
MonthlyVatCharge: commonChargeFields,
};