206 lines
6.9 KiB
TypeScript
206 lines
6.9 KiB
TypeScript
import { apiClient } from './client';
|
|
import type { CfdiListResponse, CfdiFilters, Cfdi } from '@horux/shared';
|
|
|
|
export interface CfdiConceptoRow {
|
|
fechaEmision: string;
|
|
uuid: string;
|
|
rfcEmisor: string;
|
|
rfcReceptor: string;
|
|
nombreEmisor: string;
|
|
nombreReceptor: string;
|
|
tipoComprobante: string;
|
|
status: string;
|
|
id: number;
|
|
cfdi_id: number;
|
|
clave_prod_serv: string;
|
|
no_identificacion: string | null;
|
|
descripcion: string;
|
|
cantidad: number;
|
|
clave_unidad: string;
|
|
unidad: string | null;
|
|
valor_unitario: number;
|
|
importe: number;
|
|
descuento: number | null;
|
|
// ...y todas las demás columnas de cfdi_conceptos (incluye las _mxn)
|
|
[key: string]: any;
|
|
}
|
|
|
|
export interface ConceptosListResponse {
|
|
data: CfdiConceptoRow[];
|
|
total: number;
|
|
page: number;
|
|
limit: number;
|
|
totalPages: number;
|
|
}
|
|
|
|
export interface ConceptosFilters extends CfdiFilters {
|
|
uuidLike?: string;
|
|
claveProdServ?: string;
|
|
descripcionConcepto?: string;
|
|
noIdentificacion?: string;
|
|
orderBy?: 'fecha' | 'importe';
|
|
orderDir?: 'asc' | 'desc';
|
|
}
|
|
|
|
export async function getConceptosList(filters: ConceptosFilters): Promise<ConceptosListResponse> {
|
|
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);
|
|
if (filters.uuidLike) params.set('uuidLike', filters.uuidLike);
|
|
if (filters.claveProdServ) params.set('claveProdServ', filters.claveProdServ);
|
|
if (filters.descripcionConcepto) params.set('descripcionConcepto', filters.descripcionConcepto);
|
|
if (filters.noIdentificacion) params.set('noIdentificacion', filters.noIdentificacion);
|
|
if (filters.orderBy) params.set('orderBy', filters.orderBy);
|
|
if (filters.orderDir) params.set('orderDir', filters.orderDir);
|
|
const response = await apiClient.get<ConceptosListResponse>(`/cfdi/conceptos?${params}`);
|
|
return response.data;
|
|
}
|
|
|
|
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 downloadXmlsZip(ids: number[]): Promise<Blob> {
|
|
const response = await apiClient.post('/cfdi/download-xmls', { ids }, { responseType: 'blob' });
|
|
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 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;
|
|
}
|