Files
HoruxDespachos/apps/web/lib/api/cfdi.ts
Horux Dev e7dbae1ab7 feat: conceptos tab, filters, backfill, facturapi live keys, fixes
- Add Conceptos tab in CFDI page with column filters, sorting, pagination
- Add GET /cfdi/conceptos endpoint with filters and orderBy
- Backfill cfdi_conceptos from legacy XMLs (824k concepts inserted)
- Fix CFDI delete button (bypass subscription check, add alerts)
- Fix export to Excel (fetch all filtered results, limit 10k)
- Fix facturacion page concepto delete bug (immutable updates, unique ids)
- Add Facturapi live key auto-generation and caching
- Fix SAT fechaPagoP parsing
- Add metrics cache support for current year
- Increase DB pool max to 15
2026-04-29 21:03:41 +00:00

158 lines
5.6 KiB
TypeScript

import { apiClient } from './client';
import type { CfdiListResponse, CfdiFilters, Cfdi, CfdiConceptoFilters, CfdiConceptoListResponse } from '@horux/shared';
export async function getCfdis(filters: CfdiFilters): Promise<CfdiListResponse> {
const params = new URLSearchParams();
if (filters.tipo) params.set('tipo', filters.tipo);
if (filters.tipoComprobante) params.set('tipoComprobante', filters.tipoComprobante);
if (filters.estado) params.set('estado', filters.estado);
if (filters.fechaInicio) params.set('fechaInicio', filters.fechaInicio);
if (filters.fechaFin) params.set('fechaFin', filters.fechaFin);
if (filters.rfc) params.set('rfc', filters.rfc);
if (filters.emisor) params.set('emisor', filters.emisor);
if (filters.receptor) params.set('receptor', filters.receptor);
if (filters.search) params.set('search', filters.search);
if (filters.page) params.set('page', filters.page.toString());
if (filters.limit) params.set('limit', filters.limit.toString());
if (filters.contribuyenteId) params.set('contribuyenteId', filters.contribuyenteId);
const response = await apiClient.get<CfdiListResponse>(`/cfdi?${params}`);
return response.data;
}
export async function getCfdiById(id: string): Promise<Cfdi> {
const response = await apiClient.get<Cfdi>(`/cfdi/${id}`);
return response.data;
}
export async function getResumenCfdi(año?: number, mes?: number, contribuyenteId?: string) {
const params = new URLSearchParams();
if (año) params.set('año', año.toString());
if (mes) params.set('mes', mes.toString());
if (contribuyenteId) params.set('contribuyenteId', contribuyenteId);
const response = await apiClient.get(`/cfdi/resumen?${params}`);
return response.data;
}
export interface CreateCfdiData {
uuid: string;
type: 'EMITIDO' | 'RECIBIDO';
serie?: string;
folio?: string;
status?: string;
fechaEmision: string;
rfcEmisor: string;
nombreEmisor: string;
rfcReceptor: string;
nombreReceptor: string;
subtotal: number;
subtotalMxn?: number;
descuento?: number;
descuentoMxn?: number;
total: number;
totalMxn?: number;
moneda?: string;
tipoCambio?: number;
tipoComprobante?: string;
metodoPago?: string;
formaPago?: string;
usoCfdi?: string;
ivaTraslado?: number;
ivaTrasladoMxn?: number;
isrRetencion?: number;
isrRetencionMxn?: number;
ivaRetencion?: number;
ivaRetencionMxn?: number;
}
export async function createCfdi(data: CreateCfdiData): Promise<Cfdi> {
const response = await apiClient.post<Cfdi>('/cfdi', data);
return response.data;
}
export interface BatchUploadResult {
message: string;
batchNumber: number;
totalBatches: number;
inserted: number;
duplicates: number;
errors: number;
errorMessages?: string[];
}
export async function createManyCfdis(
cfdis: CreateCfdiData[],
batchNumber?: number,
totalBatches?: number,
totalFiles?: number
): Promise<BatchUploadResult> {
const response = await apiClient.post<BatchUploadResult>('/cfdi/bulk', {
cfdis,
batchNumber: batchNumber || 1,
totalBatches: totalBatches || 1,
totalFiles: totalFiles || cfdis.length
});
return response.data;
}
export async function getCfdiConceptos(id: number | string): Promise<any[]> {
const response = await apiClient.get<any[]>(`/cfdi/${id}/conceptos`);
return response.data;
}
export async function getAllCfdiConceptos(filters: CfdiConceptoFilters): Promise<CfdiConceptoListResponse> {
const params = new URLSearchParams();
if (filters.fechaInicio) params.set('fechaInicio', filters.fechaInicio);
if (filters.fechaFin) params.set('fechaFin', filters.fechaFin);
if (filters.tipo) params.set('tipo', filters.tipo);
if (filters.tipoComprobante) params.set('tipoComprobante', filters.tipoComprobante);
if (filters.estado) params.set('estado', filters.estado);
if (filters.rfc) params.set('rfc', filters.rfc);
if (filters.search) params.set('search', filters.search);
if (filters.uuid) params.set('uuid', filters.uuid);
if (filters.claveProdServ) params.set('claveProdServ', filters.claveProdServ);
if (filters.descripcion) params.set('descripcion', filters.descripcion);
if (filters.orderBy) params.set('orderBy', filters.orderBy);
if (filters.orderDir) params.set('orderDir', filters.orderDir);
if (filters.contribuyenteId) params.set('contribuyenteId', filters.contribuyenteId);
if (filters.page) params.set('page', filters.page.toString());
if (filters.limit) params.set('limit', filters.limit.toString());
const response = await apiClient.get<CfdiConceptoListResponse>(`/cfdi/conceptos?${params}`);
return response.data;
}
export async function deleteCfdi(id: string): Promise<void> {
await apiClient.delete(`/cfdi/${id}`);
}
export async function getCfdiXml(id: string): Promise<string> {
const response = await apiClient.get<string>(`/cfdi/${id}/xml`, {
responseType: 'text'
});
return response.data;
}
export interface EmisorReceptor {
rfc: string;
nombre: string;
}
export async function searchEmisores(search: string, contribuyenteId?: string): Promise<EmisorReceptor[]> {
if (search.length < 2) return [];
const params = new URLSearchParams({ search });
if (contribuyenteId) params.set('contribuyenteId', contribuyenteId);
const response = await apiClient.get<EmisorReceptor[]>(`/cfdi/emisores?${params}`);
return response.data;
}
export async function searchReceptores(search: string, contribuyenteId?: string): Promise<EmisorReceptor[]> {
if (search.length < 2) return [];
const params = new URLSearchParams({ search });
if (contribuyenteId) params.set('contribuyenteId', contribuyenteId);
const response = await apiClient.get<EmisorReceptor[]>(`/cfdi/receptores?${params}`);
return response.data;
}