Uecko_ERP/modules/customer-invoices/src/api/infrastructure/mappers/domain/customer-invoice.mapper.ts

480 lines
14 KiB
TypeScript
Raw Normal View History

2025-09-26 18:09:14 +00:00
import { ISequelizeDomainMapper, MapperParamsType, SequelizeDomainMapper } from "@erp/core/api";
2025-09-04 17:57:04 +00:00
import {
CurrencyCode,
LanguageCode,
Percentage,
TextValue,
UniqueID,
UtcDate,
2025-09-16 11:29:45 +00:00
ValidationErrorCollection,
ValidationErrorDetail,
extractOrPushError,
2025-09-04 17:57:04 +00:00
maybeFromNullableVO,
2025-09-26 15:00:11 +00:00
toNullable,
2025-09-04 17:57:04 +00:00
} from "@repo/rdx-ddd";
2025-09-26 18:09:14 +00:00
import { Collection, Maybe, Result } from "@repo/rdx-utils";
2025-06-12 06:55:17 +00:00
import {
CustomerInvoice,
2025-09-05 11:23:45 +00:00
CustomerInvoiceItems,
2025-06-12 06:55:17 +00:00
CustomerInvoiceNumber,
2025-09-03 10:41:12 +00:00
CustomerInvoiceProps,
2025-06-12 06:55:17 +00:00
CustomerInvoiceSerie,
CustomerInvoiceStatus,
2025-09-22 17:31:49 +00:00
InvoicePaymentMethod,
2025-09-11 12:05:50 +00:00
} from "../../../domain";
import { CustomerInvoiceCreationAttributes, CustomerInvoiceModel } from "../../sequelize";
2025-09-13 18:45:55 +00:00
import { CustomerInvoiceItemDomainMapper as CustomerInvoiceItemFullMapper } from "./customer-invoice-item.mapper";
import { InvoiceRecipientDomainMapper as InvoiceRecipientFullMapper } from "./invoice-recipient.mapper";
2025-09-26 15:00:11 +00:00
import { TaxesDomainMapper as TaxesFullMapper } from "./invoice-taxes.mapper";
2025-09-11 12:05:50 +00:00
2025-09-13 18:45:55 +00:00
export interface ICustomerInvoiceDomainMapper
2025-09-11 12:05:50 +00:00
extends ISequelizeDomainMapper<
2025-06-12 06:55:17 +00:00
CustomerInvoiceModel,
CustomerInvoiceCreationAttributes,
CustomerInvoice
> {}
2025-06-11 15:13:44 +00:00
2025-09-13 18:45:55 +00:00
export class CustomerInvoiceDomainMapper
2025-09-11 12:05:50 +00:00
extends SequelizeDomainMapper<
CustomerInvoiceModel,
CustomerInvoiceCreationAttributes,
CustomerInvoice
>
2025-09-13 18:45:55 +00:00
implements ICustomerInvoiceDomainMapper
2025-06-11 15:13:44 +00:00
{
2025-09-11 12:05:50 +00:00
private _itemsMapper: CustomerInvoiceItemFullMapper;
private _recipientMapper: InvoiceRecipientFullMapper;
private _taxesMapper: TaxesFullMapper;
2025-06-11 15:13:44 +00:00
2025-09-10 16:06:29 +00:00
constructor(params: MapperParamsType) {
2025-06-11 15:13:44 +00:00
super();
2025-09-10 16:06:29 +00:00
2025-09-11 12:05:50 +00:00
this._itemsMapper = new CustomerInvoiceItemFullMapper(params); // Instanciar el mapper de items
this._recipientMapper = new InvoiceRecipientFullMapper();
this._taxesMapper = new TaxesFullMapper(params);
2025-09-09 18:13:54 +00:00
}
2025-09-22 17:31:49 +00:00
private _mapPaymentMethodToDomain(source: CustomerInvoiceModel, params?: MapperParamsType) {
const { errors } = params as {
errors: ValidationErrorDetail[];
};
const paymentId = extractOrPushError(
maybeFromNullableVO(source.payment_method_id, (value) => UniqueID.create(value)),
"payment_method_id",
errors
);
const paymentDescription = extractOrPushError(
maybeFromNullableVO(source.payment_method_description, (value) => Result.ok(String(value))),
"payment_method_description",
errors
);
if (errors.length > 0) {
return Result.fail(new ValidationErrorCollection("Invoice payment mapping failed", errors));
}
if (paymentDescription!.isNone() || paymentId!.isNone()) {
return Result.ok(Maybe.none<InvoicePaymentMethod>());
}
const paymentResult = InvoicePaymentMethod.create(
{
paymentDescription: paymentDescription?.getOrUndefined()!,
},
paymentId?.getOrUndefined()!
);
if (paymentResult.isFailure) {
return Result.fail(
new ValidationErrorCollection("Invoice payment method creation failed", [
{ path: "paymentMethod", message: paymentResult.error.message },
])
);
}
return Result.ok(Maybe.some<InvoicePaymentMethod>(paymentResult.data));
}
private _mapAttributesToDomain(source: CustomerInvoiceModel, params?: MapperParamsType) {
2025-09-09 18:13:54 +00:00
const { errors } = params as {
errors: ValidationErrorDetail[];
};
const invoiceId = extractOrPushError(UniqueID.create(source.id), "id", errors);
const companyId = extractOrPushError(UniqueID.create(source.company_id), "company_id", errors);
const customerId = extractOrPushError(
UniqueID.create(source.customer_id),
"customer_id",
errors
);
const isProforma = Boolean(source.is_proforma);
const status = extractOrPushError(
CustomerInvoiceStatus.create(source.status),
"status",
errors
);
const series = extractOrPushError(
maybeFromNullableVO(source.series, (value) => CustomerInvoiceSerie.create(value)),
"serie",
errors
);
const invoiceNumber = extractOrPushError(
2025-09-26 18:09:14 +00:00
maybeFromNullableVO(source.invoice_number, (value) => CustomerInvoiceNumber.create(value)),
2025-09-09 18:13:54 +00:00
"invoice_number",
errors
);
const invoiceDate = extractOrPushError(
UtcDate.createFromISO(source.invoice_date),
"invoice_date",
errors
);
const operationDate = extractOrPushError(
maybeFromNullableVO(source.operation_date, (value) => UtcDate.createFromISO(value)),
"operation_date",
errors
);
2025-09-26 18:09:14 +00:00
const reference = extractOrPushError(
maybeFromNullableVO(source.reference, (value) => Result.ok(String(value))),
"reference",
errors
);
2025-09-30 11:57:21 +00:00
const description = extractOrPushError(
maybeFromNullableVO(source.description, (value) => Result.ok(String(value))),
"description",
errors
);
2025-09-09 18:13:54 +00:00
const notes = extractOrPushError(
maybeFromNullableVO(source.notes, (value) => TextValue.create(value)),
"notes",
errors
);
const languageCode = extractOrPushError(
LanguageCode.create(source.language_code),
"language_code",
errors
);
const currencyCode = extractOrPushError(
CurrencyCode.create(source.currency_code),
"currency_code",
errors
);
const discountPercentage = extractOrPushError(
2025-09-10 18:14:19 +00:00
Percentage.create({
2025-09-18 08:49:32 +00:00
value: source.discount_percentage_value,
2025-09-10 18:14:19 +00:00
scale: source.discount_percentage_scale,
}),
"discount_percentage_value",
2025-09-09 18:13:54 +00:00
errors
);
return {
invoiceId,
companyId,
customerId,
isProforma,
status,
series,
invoiceNumber,
invoiceDate,
operationDate,
2025-09-26 18:09:14 +00:00
reference,
2025-09-30 11:57:21 +00:00
description,
2025-09-09 18:13:54 +00:00
notes,
languageCode,
currencyCode,
discountPercentage,
};
2025-06-11 15:13:44 +00:00
}
2025-06-12 06:55:17 +00:00
public mapToDomain(
source: CustomerInvoiceModel,
params?: MapperParamsType
): Result<CustomerInvoice, Error> {
2025-09-04 17:57:04 +00:00
try {
const errors: ValidationErrorDetail[] = [];
2025-09-10 16:06:29 +00:00
// 1) Valores escalares (atributos generales)
2025-09-22 17:31:49 +00:00
const attributes = this._mapAttributesToDomain(source, { errors, ...params });
2025-09-09 18:13:54 +00:00
2025-09-10 18:14:19 +00:00
// 2) Recipient (snapshot en la factura o include)
2025-09-09 18:13:54 +00:00
const recipientResult = this._recipientMapper.mapToDomain(source, {
2025-09-09 15:48:12 +00:00
errors,
2025-09-09 18:13:54 +00:00
attributes,
2025-09-09 15:48:12 +00:00
...params,
});
2025-09-04 17:57:04 +00:00
2025-09-09 18:13:54 +00:00
if (recipientResult.isFailure) {
errors.push({
path: "recipient",
2025-09-10 16:06:29 +00:00
2025-09-09 18:13:54 +00:00
message: recipientResult.error.message,
});
}
2025-09-10 18:14:19 +00:00
// 3) Items (colección)
2025-09-11 12:05:50 +00:00
const itemsResults = this._itemsMapper.mapToDomainCollection(
source.items,
source.items.length,
{
errors,
attributes,
...params,
}
);
2025-09-04 17:57:04 +00:00
2025-09-09 18:13:54 +00:00
if (itemsResults.isFailure) {
errors.push({
path: "items",
message: recipientResult.error.message,
});
}
2025-09-10 18:14:19 +00:00
// 4) Taxes (colección a nivel factura)
2025-09-11 12:05:50 +00:00
const taxesResults = this._taxesMapper.mapToDomainCollection(
source.taxes,
source.taxes.length,
{
errors,
attributes,
...params,
}
);
2025-09-08 17:24:38 +00:00
2025-09-09 18:13:54 +00:00
if (taxesResults.isFailure) {
errors.push({
path: "taxes",
2025-09-10 16:06:29 +00:00
message: taxesResults.error.message,
2025-09-09 18:13:54 +00:00
});
}
2025-09-22 17:31:49 +00:00
// Payment method
const paymentMethodResult = this._mapPaymentMethodToDomain(source, { errors, ...params });
if (paymentMethodResult.isFailure) {
errors.push({
path: "paymentMethod",
message: paymentMethodResult.error.message,
});
}
2025-09-10 18:14:19 +00:00
// 5) Si hubo errores de mapeo, devolvemos colección de validación
2025-09-04 17:57:04 +00:00
if (errors.length > 0) {
2025-10-04 17:43:03 +00:00
return Result.fail(new ValidationErrorCollection(errors));
2025-09-04 17:57:04 +00:00
}
2025-09-10 18:14:19 +00:00
// 6) Construcción del agregado (Dominio)
2025-09-09 18:13:54 +00:00
const recipient = recipientResult.data;
2025-09-22 17:31:49 +00:00
const paymentMethod = paymentMethodResult.data;
2025-09-09 18:13:54 +00:00
const items = CustomerInvoiceItems.create({
languageCode: attributes.languageCode!,
currencyCode: attributes.currencyCode!,
items: itemsResults.data.getAll(),
});
2025-09-04 17:57:04 +00:00
const invoiceProps: CustomerInvoiceProps = {
2025-09-09 18:13:54 +00:00
companyId: attributes.companyId!,
2025-09-09 15:48:12 +00:00
2025-09-09 18:13:54 +00:00
isProforma: attributes.isProforma,
status: attributes.status!,
series: attributes.series!,
invoiceNumber: attributes.invoiceNumber!,
invoiceDate: attributes.invoiceDate!,
operationDate: attributes.operationDate!,
2025-09-04 17:57:04 +00:00
2025-09-09 18:13:54 +00:00
customerId: attributes.customerId!,
2025-09-09 15:48:12 +00:00
recipient: recipient,
2025-09-08 17:24:38 +00:00
2025-09-26 18:09:14 +00:00
reference: attributes.reference!,
2025-09-30 11:57:21 +00:00
description: attributes.description!,
2025-09-09 18:13:54 +00:00
notes: attributes.notes!,
2025-09-04 17:57:04 +00:00
2025-09-09 18:13:54 +00:00
languageCode: attributes.languageCode!,
currencyCode: attributes.currencyCode!,
2025-09-04 17:57:04 +00:00
2025-09-10 18:14:19 +00:00
discountPercentage: attributes.discountPercentage!,
2025-09-04 17:57:04 +00:00
2025-09-22 17:31:49 +00:00
paymentMethod: paymentMethod!,
2025-09-09 18:13:54 +00:00
items,
2025-09-04 17:57:04 +00:00
};
2025-09-09 18:13:54 +00:00
const createResult = CustomerInvoice.create(invoiceProps, attributes.invoiceId);
if (createResult.isFailure) {
return Result.fail(
2025-10-04 17:43:03 +00:00
new ValidationErrorCollection([{ path: "invoice", message: createResult.error.message }])
2025-09-09 18:13:54 +00:00
);
}
return Result.ok(createResult.data);
2025-09-04 17:57:04 +00:00
} catch (err: unknown) {
return Result.fail(err as Error);
}
2025-06-11 15:13:44 +00:00
}
2025-06-12 06:55:17 +00:00
public mapToPersistence(
source: CustomerInvoice,
params?: MapperParamsType
2025-09-11 12:05:50 +00:00
): Result<CustomerInvoiceCreationAttributes, Error> {
2025-09-26 15:00:11 +00:00
const errors: ValidationErrorDetail[] = [];
// 1) Items
const itemsResult = this._itemsMapper.mapToPersistenceArray(source.items, {
errors,
parent: source,
...params,
});
if (itemsResult.isFailure) {
errors.push({
path: "items",
message: itemsResult.error.message,
});
}
2025-09-08 17:24:38 +00:00
2025-09-26 18:09:14 +00:00
const items = itemsResult.data;
2025-09-26 15:00:11 +00:00
// 1) Taxes
2025-09-26 18:09:14 +00:00
const taxesResult = this._taxesMapper.mapToPersistenceArray(new Collection(source.taxes), {
2025-09-26 15:00:11 +00:00
errors,
parent: source,
...params,
});
if (taxesResult.isFailure) {
errors.push({
path: "taxes",
message: taxesResult.error.message,
});
}
2025-09-26 18:09:14 +00:00
const taxes = taxesResult.data;
2025-09-26 15:00:11 +00:00
// 3) Calcular totales
const allAmounts = source.getAllAmounts();
2025-09-26 18:09:14 +00:00
// 4) Cliente
2025-10-03 19:01:38 +00:00
const recipient = this._mapRecipientToPersistence(source, {
errors,
parent: source,
...params,
});
2025-09-26 18:09:14 +00:00
// 7) Si hubo errores de mapeo, devolvemos colección de validación
if (errors.length > 0) {
return Result.fail(
new ValidationErrorCollection("Customer invoice mapping to persistence failed", errors)
);
}
const invoiceValues: CustomerInvoiceCreationAttributes = {
2025-09-05 11:23:45 +00:00
id: source.id.toPrimitive(),
company_id: source.companyId.toPrimitive(),
2025-09-26 15:00:11 +00:00
is_proforma: source.isProforma,
2025-09-05 11:23:45 +00:00
status: source.status.toPrimitive(),
series: toNullable(source.series, (series) => series.toPrimitive()),
2025-09-26 18:09:14 +00:00
invoice_number: toNullable(source.invoiceNumber, (invoiceNumber) =>
invoiceNumber.toPrimitive()
),
2025-09-04 17:57:04 +00:00
invoice_date: source.invoiceDate.toPrimitive(),
2025-09-05 11:23:45 +00:00
operation_date: toNullable(source.operationDate, (operationDate) =>
operationDate.toPrimitive()
),
2025-09-26 15:00:11 +00:00
language_code: source.languageCode.toPrimitive(),
currency_code: source.currencyCode.toPrimitive(),
2025-09-05 11:23:45 +00:00
2025-09-26 18:09:14 +00:00
reference: toNullable(source.reference, (reference) => reference),
2025-09-30 11:57:21 +00:00
description: toNullable(source.description, (description) => description),
2025-09-05 11:23:45 +00:00
notes: toNullable(source.notes, (notes) => notes.toPrimitive()),
2025-09-26 15:00:11 +00:00
subtotal_amount_value: allAmounts.subtotalAmount.value,
subtotal_amount_scale: allAmounts.subtotalAmount.scale,
discount_percentage_value: source.discountPercentage.toPrimitive().value,
discount_percentage_scale: source.discountPercentage.toPrimitive().scale,
2025-09-05 11:23:45 +00:00
2025-09-26 15:00:11 +00:00
discount_amount_value: allAmounts.discountAmount.value,
discount_amount_scale: allAmounts.discountAmount.scale,
2025-09-05 11:23:45 +00:00
2025-09-26 15:00:11 +00:00
taxable_amount_value: allAmounts.taxableAmount.value,
taxable_amount_scale: allAmounts.taxableAmount.scale,
2025-09-05 11:23:45 +00:00
2025-09-26 15:00:11 +00:00
taxes_amount_value: allAmounts.taxesAmount.value,
taxes_amount_scale: allAmounts.taxesAmount.scale,
2025-09-08 17:24:38 +00:00
2025-09-26 15:00:11 +00:00
total_amount_value: allAmounts.totalAmount.value,
total_amount_scale: allAmounts.totalAmount.scale,
2025-09-09 18:13:54 +00:00
2025-09-26 15:00:11 +00:00
payment_method_id: toNullable(source.paymentMethod, (payment) => payment.toObjectString().id),
payment_method_description: toNullable(
source.paymentMethod,
(payment) => payment.toObjectString().payment_description
),
2025-09-26 18:09:14 +00:00
customer_id: source.customerId.toPrimitive(),
...recipient,
taxes,
items,
2025-09-26 15:00:11 +00:00
};
2025-09-26 18:09:14 +00:00
return Result.ok<CustomerInvoiceCreationAttributes>(invoiceValues);
}
2025-09-26 15:00:11 +00:00
2025-09-26 18:09:14 +00:00
protected _mapRecipientToPersistence(source: CustomerInvoice, params?: MapperParamsType) {
const { errors } = params as {
errors: ValidationErrorDetail[];
};
2025-09-26 15:00:11 +00:00
2025-09-26 18:09:14 +00:00
const recipient = source.recipient.getOrUndefined();
if (!source.isProforma && !recipient) {
errors.push({
path: "recipient",
message: "[CustomerInvoiceDomainMapper] Issued customer invoice w/o recipient data",
});
2025-09-26 15:00:11 +00:00
}
2025-06-11 15:13:44 +00:00
2025-09-26 18:09:14 +00:00
const recipientValues = {
customer_tin: !source.isProforma ? recipient!.tin.toPrimitive() : null,
customer_name: !source.isProforma ? recipient!.name.toPrimitive() : null,
customer_street: !source.isProforma
? toNullable(recipient!.street, (v) => v.toPrimitive())
: null,
customer_street2: !source.isProforma
? toNullable(recipient!.street2, (v) => v.toPrimitive())
: null,
customer_city: !source.isProforma
? toNullable(recipient!.city, (v) => v.toPrimitive())
: null,
customer_province: !source.isProforma
? toNullable(recipient!.province, (v) => v.toPrimitive())
: null,
customer_postal_code: !source.isProforma
? toNullable(recipient!.postalCode, (v) => v.toPrimitive())
: null,
customer_country: !source.isProforma
? toNullable(recipient!.country, (v) => v.toPrimitive())
: null,
};
return recipientValues;
2025-06-11 15:13:44 +00:00
}
}