@accounter/server
Version:
Accounter GraphQL server
532 lines (506 loc) • 19.2 kB
text/typescript
import { GraphQLError } from 'graphql';
import { Injectable, Scope } from 'graphql-modules';
import { sql } from '@pgtyped/runtime';
import { formatCurrency, optionalDateToTimelessDateString } from '../../../shared/helpers/index.js';
import type { AuthContext } from '../../../shared/types/auth.js';
import { TenantAwareDBClient } from '../../app-providers/tenant-db-client.js';
import { AuthContextProvider } from '../../auth/providers/auth-context.provider.js';
import type {
AdminContext,
IGetAdminContextsQuery,
IGetAdminContextsResult,
IUpdateAdminContextParams,
IUpdateAdminContextQuery,
} from '../types.js';
const getAdminContexts = sql<IGetAdminContextsQuery>`
SELECT *
FROM accounter_schema.user_context
WHERE owner_id IN $$ownerIds;
`;
const updateAdminContext = sql<IUpdateAdminContextQuery>`
UPDATE accounter_schema.user_context
SET
default_local_currency = COALESCE(
$defaultLocalCurrency,
default_local_currency
),
default_fiat_currency_for_crypto_conversions = COALESCE(
$defaultFiatCurrencyForCryptoConversions,
default_fiat_currency_for_crypto_conversions
),
default_tax_category_id = COALESCE(
$defaultTaxCategoryId,
default_tax_category_id
),
vat_business_id = COALESCE(
$vatBusinessId,
vat_business_id
),
input_vat_tax_category_id = COALESCE(
$inputVatTaxCategoryId,
input_vat_tax_category_id
),
output_vat_tax_category_id = COALESCE(
$outputVatTaxCategoryId,
output_vat_tax_category_id
),
property_output_vat_tax_category_id = COALESCE(
$propertyOutputVatTaxCategoryId,
property_output_vat_tax_category_id
),
tax_business_id = COALESCE(
$taxBusinessId,
tax_business_id
),
tax_expenses_tax_category_id = COALESCE(
$taxExpensesTaxCategoryId,
tax_expenses_tax_category_id
),
social_security_business_id = COALESCE(
$socialSecurityBusinessId,
social_security_business_id
),
exchange_rate_tax_category_id = COALESCE(
$exchangeRateTaxCategoryId,
exchange_rate_tax_category_id
),
income_exchange_rate_tax_category_id = COALESCE(
$incomeExchangeRateTaxCategoryId,
income_exchange_rate_tax_category_id
),
exchange_rate_revaluation_tax_category_id = COALESCE(
$exchangeRateRevaluationTaxCategoryId,
exchange_rate_revaluation_tax_category_id
),
fee_tax_category_id = COALESCE(
$feeTaxCategoryId,
fee_tax_category_id
),
general_fee_tax_category_id = COALESCE(
$generalFeeTaxCategoryId,
general_fee_tax_category_id
),
fine_tax_category_id = COALESCE(
$fineTaxCategoryId,
fine_tax_category_id
),
untaxable_gifts_tax_category_id = COALESCE(
$untaxableGiftsTaxCategoryId,
untaxable_gifts_tax_category_id
),
balance_cancellation_tax_category_id = COALESCE(
$balanceCancellationTaxCategoryId,
balance_cancellation_tax_category_id
),
development_foreign_tax_category_id = COALESCE(
$developmentForeignTaxCategoryId,
development_foreign_tax_category_id
),
development_local_tax_category_id = COALESCE(
$developmentLocalTaxCategoryId,
development_local_tax_category_id
),
accumulated_depreciation_tax_category_id = COALESCE(
$accumulatedDepreciationTaxCategoryId,
accumulated_depreciation_tax_category_id
),
rnd_depreciation_expenses_tax_category_id = COALESCE(
$rndDepreciationExpensesTaxCategoryId,
rnd_depreciation_expenses_tax_category_id
),
gnm_depreciation_expenses_tax_category_id = COALESCE(
$gnmDepreciationExpensesTaxCategoryId,
gnm_depreciation_expenses_tax_category_id
),
marketing_depreciation_expenses_tax_category_id = COALESCE(
$marketingDepreciationExpensesTaxCategoryId,
marketing_depreciation_expenses_tax_category_id
),
bank_deposit_interest_income_tax_category_id = COALESCE(
$bankDepositInterestIncomeTaxCategoryId,
bank_deposit_interest_income_tax_category_id
),
business_trip_tax_category_id = COALESCE(
$businessTripTaxCategoryId,
business_trip_tax_category_id
),
business_trip_tag_id = COALESCE(
$businessTripTagId,
business_trip_tag_id
),
expenses_to_pay_tax_category_id = COALESCE(
$expensesToPayTaxCategoryId,
expenses_to_pay_tax_category_id
),
expenses_in_advance_tax_category_id = COALESCE(
$expensesInAdvanceTaxCategoryId,
expenses_in_advance_tax_category_id
),
income_to_collect_tax_category_id = COALESCE(
$incomeToCollectTaxCategoryId,
income_to_collect_tax_category_id
),
income_in_advance_tax_category_id = COALESCE(
$incomeInAdvanceTaxCategoryId,
income_in_advance_tax_category_id
),
zkufot_expenses_tax_category_id = COALESCE(
$zkufotExpensesTaxCategoryId,
zkufot_expenses_tax_category_id
),
zkufot_income_tax_category_id = COALESCE(
$zkufotIncomeTaxCategoryId,
zkufot_income_tax_category_id
),
social_security_expenses_tax_category_id = COALESCE(
$socialSecurityExpensesTaxCategoryId,
social_security_expenses_tax_category_id
),
salary_expenses_tax_category_id = COALESCE(
$salaryExpensesTaxCategoryId,
salary_expenses_tax_category_id
),
training_fund_expenses_tax_category_id = COALESCE(
$trainingFundExpensesTaxCategoryId,
training_fund_expenses_tax_category_id
),
pension_fund_expenses_tax_category_id = COALESCE(
$pensionFundExpensesTaxCategoryId,
pension_fund_expenses_tax_category_id
),
compensation_fund_expenses_tax_category_id = COALESCE(
$compensationFundExpensesTaxCategoryId,
compensation_fund_expenses_tax_category_id
),
batched_employees_business_id = COALESCE(
$batchedEmployeesBusinessId,
batched_employees_business_id
),
batched_funds_business_id = COALESCE(
$batchedFundsBusinessId,
batched_funds_business_id
),
tax_deductions_business_id = COALESCE(
$taxDeductionsBusinessId,
tax_deductions_business_id
),
recovery_reserve_expenses_tax_category_id = COALESCE(
$recoveryReserveExpensesTaxCategoryId,
recovery_reserve_expenses_tax_category_id
),
recovery_reserve_tax_category_id = COALESCE(
$recoveryReserveTaxCategoryId,
recovery_reserve_tax_category_id
),
vacation_reserve_expenses_tax_category_id = COALESCE(
$vacationReserveExpensesTaxCategoryId,
vacation_reserve_expenses_tax_category_id
),
vacation_reserve_tax_category_id = COALESCE(
$vacationReserveTaxCategoryId,
vacation_reserve_tax_category_id
),
poalim_business_id = COALESCE(
$poalimBusinessId,
poalim_business_id
),
discount_business_id = COALESCE(
$discountBusinessId,
discount_business_id
),
isracard_business_id = COALESCE(
$isracardBusinessId,
isracard_business_id
),
amex_business_id = COALESCE(
$amexBusinessId,
amex_business_id
),
cal_business_id = COALESCE(
$calBusinessId,
cal_business_id
),
etana_business_id = COALESCE(
$etanaBusinessId,
etana_business_id
),
kraken_business_id = COALESCE(
$krakenBusinessId,
kraken_business_id
),
etherscan_business_id = COALESCE(
$etherscanBusinessId,
etherscan_business_id
),
swift_business_id = COALESCE(
$swiftBusinessId,
swift_business_id
),
bank_deposit_business_id = COALESCE(
$bankDepositBusinessId,
bank_deposit_business_id
),
dividend_withholding_tax_business_id = COALESCE(
$dividendWithholdingTaxBusinessId,
dividend_withholding_tax_business_id
),
dividend_tax_category_id = COALESCE(
$dividendTaxCategoryId,
dividend_tax_category_id
),
salary_excess_expenses_tax_category_id = COALESCE(
$salaryExcessExpensesTaxCategoryId,
salary_excess_expenses_tax_category_id
),
ledger_lock = COALESCE(
$ledgerLock,
ledger_lock
),
foreign_securities_business_id = COALESCE(
$foreignSecuritiesBusinessId,
foreign_securities_business_id
),
foreign_securities_fees_category_id = COALESCE(
$foreignSecuritiesFeesCategoryId,
foreign_securities_fees_category_id
),
date_established = COALESCE(
$dateEstablished,
date_established
),
initial_accounter_year = COALESCE(
$initialAccounterYear,
initial_accounter_year
),
locality = COALESCE(
$locality,
locality
)
WHERE owner_id = $ownerId
RETURNING *;
`;
@Injectable({
scope: Scope.Operation,
global: true,
})
export class AdminContextProvider {
private authContext: AuthContext | null = null;
private authContextInitialized = false;
private cachedContext: AdminContext | null = null;
private cachedContextInitializing: Promise<AdminContext | null> | null = null;
constructor(
private authContextProvider: AuthContextProvider,
private db: TenantAwareDBClient,
) {}
public normalizeContext(rawContext: IGetAdminContextsResult): AdminContext {
const dividendPaymentBusinessIds = [
'4bcca705-5b47-41c5-ba26-1e42c69cbf0d', // Uri Dividend
'909fbe3c-0419-44ed-817d-ab774e93748a', // Dotan Dividend
// TODO: fetch those IDs from DB somehow
];
const bankAccountIds = [rawContext.poalim_business_id, rawContext.discount_business_id].filter(
Boolean,
) as string[];
const creditCardIds = [
rawContext.isracard_business_id,
rawContext.amex_business_id,
rawContext.cal_business_id,
].filter(Boolean) as string[];
const salaryBatchedBusinessIds = [
rawContext.batched_employees_business_id,
rawContext.batched_funds_business_id,
].filter(Boolean) as string[];
const vatReportExcludedBusinessNames = [
rawContext.vat_business_id,
rawContext.tax_business_id,
rawContext.social_security_business_id,
];
return {
defaultLocalCurrency: formatCurrency(rawContext.default_local_currency),
defaultCryptoConversionFiatCurrency: formatCurrency(
rawContext.default_fiat_currency_for_crypto_conversions,
),
ownerId: rawContext.owner_id,
defaultTaxCategoryId: rawContext.default_tax_category_id,
locality: rawContext.locality,
ledgerLock: optionalDateToTimelessDateString(rawContext.ledger_lock) ?? undefined,
dateEstablished: optionalDateToTimelessDateString(rawContext.date_established) ?? undefined,
initialAccounterYear: rawContext.initial_accounter_year ?? undefined,
authorities: {
vatBusinessId: rawContext.vat_business_id,
inputVatTaxCategoryId: rawContext.input_vat_tax_category_id,
outputVatTaxCategoryId: rawContext.output_vat_tax_category_id,
propertyOutputVatTaxCategoryId: rawContext.property_output_vat_tax_category_id,
taxBusinessId: rawContext.tax_business_id,
taxExpensesTaxCategoryId: rawContext.tax_expenses_tax_category_id,
socialSecurityBusinessId: rawContext.social_security_business_id,
vatReportExcludedBusinessNames,
},
depreciation: {
accumulatedDepreciationTaxCategoryId: rawContext.accumulated_depreciation_tax_category_id,
rndDepreciationExpensesTaxCategoryId: rawContext.rnd_depreciation_expenses_tax_category_id,
gnmDepreciationExpensesTaxCategoryId: rawContext.gnm_depreciation_expenses_tax_category_id,
marketingDepreciationExpensesTaxCategoryId:
rawContext.marketing_depreciation_expenses_tax_category_id,
},
bankDeposits: {
bankDepositBusinessId: rawContext.bank_deposit_business_id,
bankDepositInterestIncomeTaxCategoryId:
rawContext.bank_deposit_interest_income_tax_category_id,
},
foreignSecurities: {
foreignSecuritiesBusinessId: rawContext.foreign_securities_business_id,
foreignSecuritiesFeesCategoryId: rawContext.foreign_securities_fees_category_id,
},
dividends: {
dividendWithholdingTaxBusinessId: rawContext.dividend_withholding_tax_business_id,
dividendTaxCategoryId: rawContext.dividend_tax_category_id,
dividendPaymentBusinessIds,
dividendBusinessIds: [
...(rawContext.dividend_withholding_tax_business_id
? [rawContext.dividend_withholding_tax_business_id]
: []),
...dividendPaymentBusinessIds,
],
},
businessTrips: {
businessTripTaxCategoryId: rawContext.business_trip_tax_category_id,
businessTripTagId: rawContext.business_trip_tag_id,
},
financialAccounts: {
poalimBusinessId: rawContext.poalim_business_id,
discountBusinessId: rawContext.discount_business_id,
swiftBusinessId: rawContext.swift_business_id,
isracardBusinessId: rawContext.isracard_business_id,
amexBusinessId: rawContext.amex_business_id,
calBusinessId: rawContext.cal_business_id,
etanaBusinessId: rawContext.etana_business_id,
krakenBusinessId: rawContext.kraken_business_id,
etherScanBusinessId: rawContext.etherscan_business_id,
foreignSecuritiesBusinessId: rawContext.foreign_securities_business_id,
bankAccountIds,
creditCardIds,
internalWalletsIds: [
...bankAccountIds,
...creditCardIds,
rawContext.etana_business_id,
rawContext.kraken_business_id,
rawContext.etherscan_business_id,
rawContext.foreign_securities_business_id,
].filter(Boolean) as string[],
},
salaries: {
zkufotExpensesTaxCategoryId: rawContext.zkufot_expenses_tax_category_id,
zkufotIncomeTaxCategoryId: rawContext.zkufot_income_tax_category_id,
socialSecurityExpensesTaxCategoryId: rawContext.social_security_expenses_tax_category_id,
salaryExpensesTaxCategoryId: rawContext.salary_expenses_tax_category_id,
trainingFundExpensesTaxCategoryId: rawContext.training_fund_expenses_tax_category_id,
compensationFundExpensesTaxCategoryId:
rawContext.compensation_fund_expenses_tax_category_id,
pensionExpensesTaxCategoryId: rawContext.pension_fund_expenses_tax_category_id,
batchedEmployeesBusinessId: rawContext.batched_employees_business_id,
batchedFundsBusinessId: rawContext.batched_funds_business_id,
salaryBatchedBusinessIds,
taxDeductionsBusinessId: rawContext.tax_deductions_business_id,
recoveryReserveExpensesTaxCategoryId: rawContext.recovery_reserve_expenses_tax_category_id,
recoveryReserveTaxCategoryId: rawContext.recovery_reserve_tax_category_id,
vacationReserveExpensesTaxCategoryId: rawContext.vacation_reserve_expenses_tax_category_id,
vacationReserveTaxCategoryId: rawContext.vacation_reserve_tax_category_id,
},
crossYear: {
expensesToPayTaxCategoryId: rawContext.expenses_to_pay_tax_category_id,
expensesInAdvanceTaxCategoryId: rawContext.expenses_in_advance_tax_category_id,
incomeToCollectTaxCategoryId: rawContext.income_to_collect_tax_category_id,
incomeInAdvanceTaxCategoryId:
rawContext.income_in_advance_tax_category_id ?? rawContext.default_tax_category_id,
},
general: {
taxCategories: {
exchangeRateTaxCategoryId: rawContext.exchange_rate_tax_category_id,
incomeExchangeRateTaxCategoryId: rawContext.income_exchange_rate_tax_category_id,
exchangeRevaluationTaxCategoryId: rawContext.exchange_rate_revaluation_tax_category_id,
feeTaxCategoryId: rawContext.fee_tax_category_id,
generalFeeTaxCategoryId: rawContext.general_fee_tax_category_id,
fineTaxCategoryId: rawContext.fine_tax_category_id,
untaxableGiftsTaxCategoryId: rawContext.untaxable_gifts_tax_category_id,
balanceCancellationTaxCategoryId: rawContext.balance_cancellation_tax_category_id,
developmentForeignTaxCategoryId: rawContext.development_foreign_tax_category_id,
developmentLocalTaxCategoryId: rawContext.development_local_tax_category_id,
salaryExcessExpensesTaxCategoryId: rawContext.salary_excess_expenses_tax_category_id,
},
},
};
}
public async getAdminContext(): Promise<AdminContext | null> {
if (this.cachedContext) {
return this.cachedContext;
}
await this.ensureAuthContext();
if (!this.authContext) {
throw new GraphQLError(
'Auth context not available. AdminContextProvider requires active authentication.',
{ extensions: { code: 'UNAUTHENTICATED' } },
);
}
const ownerId = this.authContext.tenant.businessId;
if (!ownerId) {
throw new Error('AdminContextProvider: ownerId not found in context (currentUser)');
}
const contexts = await getAdminContexts.run({ ownerIds: [ownerId] }, this.db);
const context = contexts[0] ? this.normalizeContext(contexts[0]) : null;
this.cachedContext = context;
return context;
}
public async getVerifiedAdminContext() {
if (this.cachedContext) {
return this.cachedContext;
}
this.cachedContextInitializing ??= this.getAdminContext();
const context = await this.cachedContextInitializing;
if (!context) {
throw new GraphQLError('Admin context not found for the authenticated user.', {
extensions: { code: 'FORBIDDEN' },
});
}
return context;
}
public async updateAdminContext(params: IUpdateAdminContextParams): Promise<AdminContext | null> {
await this.ensureAuthContext();
if (!this.authContext) {
throw new GraphQLError(
'Auth context not available. AdminContextProvider requires active authentication.',
{ extensions: { code: 'UNAUTHENTICATED' } },
);
}
const ownerId = this.authContext.tenant.businessId;
if (!ownerId) {
throw new Error('AdminContextProvider: ownerId not found in context (currentUser)');
}
this.cachedContext = null;
const updatedContexts = await updateAdminContext.run({ ...params, ownerId }, this.db);
if (updatedContexts.length >= 1) {
const normalizedContext = this.normalizeContext(updatedContexts[0]);
this.cachedContext = normalizedContext;
return normalizedContext;
}
return null;
}
public async getAdminContextByOwnerId(ownerId: string): Promise<AdminContext | null> {
const contexts = await getAdminContexts.run({ ownerIds: [ownerId] }, this.db);
return contexts[0] ? this.normalizeContext(contexts[0]) : null;
}
public clearCache() {
this.cachedContext = null;
}
/**
* Lazy initialization of auth context on first use.
* This ensures the async provider is called only when needed.
*/
private async ensureAuthContext(): Promise<void> {
if (this.authContextInitialized) {
return;
}
this.authContext = await this.authContextProvider.getAuthContext();
this.authContextInitialized = true;
}
}