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
This commit is contained in:
@@ -1,5 +1,5 @@
|
||||
import { apiClient } from './client';
|
||||
import type { CfdiListResponse, CfdiFilters, Cfdi } from '@horux/shared';
|
||||
import type { CfdiListResponse, CfdiFilters, Cfdi, CfdiConceptoFilters, CfdiConceptoListResponse } from '@horux/shared';
|
||||
|
||||
export async function getCfdis(filters: CfdiFilters): Promise<CfdiListResponse> {
|
||||
const params = new URLSearchParams();
|
||||
@@ -102,6 +102,28 @@ export async function getCfdiConceptos(id: number | string): Promise<any[]> {
|
||||
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}`);
|
||||
}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
|
||||
import * as cfdiApi from '@/lib/api/cfdi';
|
||||
import type { CfdiFilters } from '@horux/shared';
|
||||
import type { CfdiFilters, CfdiConceptoFilters } from '@horux/shared';
|
||||
import type { CreateCfdiData } from '@/lib/api/cfdi';
|
||||
import { useContribuyenteStore } from '@/stores/contribuyente-store';
|
||||
|
||||
@@ -58,6 +58,18 @@ export function useCreateManyCfdis() {
|
||||
});
|
||||
}
|
||||
|
||||
export function useCfdiConceptos(filters: CfdiConceptoFilters) {
|
||||
const { selectedContribuyenteId } = useContribuyenteStore();
|
||||
const filtersWithContribuyente: CfdiConceptoFilters = {
|
||||
...filters,
|
||||
contribuyenteId: selectedContribuyenteId || undefined,
|
||||
};
|
||||
return useQuery({
|
||||
queryKey: ['cfdi-conceptos', filters, selectedContribuyenteId],
|
||||
queryFn: () => cfdiApi.getAllCfdiConceptos(filtersWithContribuyente),
|
||||
});
|
||||
}
|
||||
|
||||
export function useDeleteCfdi() {
|
||||
const queryClient = useQueryClient();
|
||||
|
||||
|
||||
Reference in New Issue
Block a user