phpcfdi / finkok
Librería para conectar con la API de servicios de FINKOK
Installs: 6 439
Dependents: 0
Suggesters: 0
Security: 0
Stars: 18
Watchers: 7
Forks: 9
Open Issues: 0
Requires
- php: >=7.3
- ext-dom: *
- ext-json: *
- ext-openssl: *
- ext-soap: *
- eclipxe/enum: ^0.2.0
- eclipxe/micro-catalog: ^0.1.0
- phpcfdi/cfdi-expresiones: ^3.2
- phpcfdi/credentials: ^1.0.1
- phpcfdi/xml-cancelacion: ^2.0.2
- psr/log: ^1.1 || ^2.0 || ^3.0
- robrichards/xmlseclibs: ^3.0.4
Requires (Dev)
- ext-fileinfo: *
- eclipxe/cfdiutils: ^2.23.2
- phpcfdi/rfc: ^1.1
- phpunit/phpunit: ^9.5.10
- symfony/dotenv: ^5.1 || ^6.0 || ^7.0
README
Librería para conectar con la API de servicios de FINKOK (México)
🇺🇸 The documentation of this project is in spanish as this is the natural language for intented audience.
Acerca de phpcfdi/finkok
Esta librería es un esfuerzo de la comunidad de https://www.phpcfdi.com/ para tener un cliente que explote las funcionalidades ofrecidas por el integrador https://www.finkok.com/.
No está relacionado con Finkok y Finkok es una marca registrada de FINKOK, SAPI DE CV.
Instalación
Usa composer
composer require phpcfdi/finkok
Ejemplo básico de uso
use PhpCfdi\Finkok\FinkokEnvironment; use PhpCfdi\Finkok\FinkokSettings; use PhpCfdi\Finkok\QuickFinkok; $settings = new FinkokSettings('user@host.com', 'secret', FinkokEnvironment::makeProduction()); $finkok = new QuickFinkok($settings); // el PreCFDI a firmar, podría venir de CfdiUtils ;) $creator->asXml() $precfdi = file_get_contents('precfdi-to-sign.xml'); $stampResult = $finkok->stamp($precfdi); // <- aquí contactamos a Finkok if ($stampResult->hasAlerts()) { // stamp es un objeto con propiedades nombradas foreach ($stampResult->alerts() as $alert) { echo $alert->id() . ' - ' . $alert->message() . PHP_EOL; } } else { file_put_contents($stampResult->uuid() . '.xml', $stampResult->xml()); // CFDI firmado }
Y también hay otros ejemplos explicados:
Y todos los test de integración, donde se prueba la comunicación y respuestas contra la plataforma de pruebas.
Se recomienda utilizar la clase PhpCfdi\Finkok\QuickFinkok
para un uso rápido de los comandos de finkok,
sin embargo, se pueden utilizar un modo totalmente explícito y granular por comando, servicio y resultado.
Métodos implementados
La librería utiliza un modelo basado en comando, servicio y resultado. El comando es la definición de la acción que queremos realizar, contiene todos los parámetros necesarios. El servicio es el encargado de usar ese comando como entrada, ejecutarlo en Finkok (vía SOAP) y construir un resultado a partir de la respuesta. El resultado son los datos que representa la respuesta.
No hemos implementado intencionalmente los comandos que requieren transmitir la llave privada de un CSD (Certificado de Sello Digital) o de la eFirma/FIEL (Firma electrónica). No creemos que vayamos a implementarlos porque a) No es necesario y b) Es inseguro.
Servicios de estampado
Finkok tiene dos métodos de firmado: stamp
y quickstamp
.
stamp(Stamping\StampingCommand $command): Stamping\StampingResult
quickstamp(Stamping\StampingCommand $command): Stamping\StampingResult
El servicio stamped
para revisar si previamente se generó un cfdi:
stamped(Stamping\StampingCommand $command): Stamping\StampingResult
El servicio stampQueryPending
por si estás usando pending buffer
(que te recomiendo no hacerlo):
stampQueryPending(Stamping\QueryPendingCommand $command): Stamping\QueryPendingResult
Servicios de cancelación
Solo se pueden cancelar cfdi con esta librería usando cancelSignature
porque es el único
donde no tienes que transmitir información confidencial.
cancelSignature(Cancel\CancelSignatureCommand $command): Cancel\CancelSignatureResult
Puedes consultar el estado de un CFDI usando getSatStatus
(antes o después de cancelarlo)
getSatStatus(Cancel\GetSatStatusCommand $command): Cancel\GetSatStatusResult
Y obtener el último acuse de recibo del SAT a una solicitud de cancelación con getCancelReceipt
.
Aunque recuerda que tener un acuse no significa que se haya cancelado, el acuse lo único que
contiene es la respuesta de cancelación presentada al SAT.
getCancelReceipt(Cancel\GetReceiptResult $command): Cancel\GetReceiptResult
Gracias a getPendingToCancel
se puede obtener el listado de CFDI pendientes por cancelar por
parte de un receptor.
getPendingToCancel(Cancel\GetPendingCommand $command): Cancel\GetPendingResult
Se pueden obtener los UUID relacionados hijos (que el UUID consultado relaciona)
y padres (que relacionan al UUID consultado) usando getRelatedSignature
.
Tal como el método cancelSignature
este método requiere de un mensaje firmado.
getRelatedSignature(Cancel\GetRelatedSignatureCommand $command): Cancel\GetRelatedSignatureResult
A su vez, se puede aceptar o rechazar una solicitud de cancelación usando acceptRejectSignature
.
Este método puede trabajar con varios UUID, pero Finkok recomienda que solo se realice uno a la vez.
Tal como el método cancelSignature
este método requiere de un mensaje firmado.
acceptRejectSignature(Cancel\AcceptRejectSignatureCommand $command): Cancel\AcceptRejectSignatureResult
Utilerías y manejo de clientes
Obtener la hora de Finkok (por si estás teniendo problemas de CFDI fuera de tiempo):
datetime(): Utilities\DatetimeResult
Obtener un CFDI firmado con Finkok en los últimos 3 meses:
downloadXml(Utilities\DownloadXmlCommand $command): Utilities\DownloadXmlResult
Obtener reportes de consumo de crédito y manejo de clientes:
reportCredit(Utilities\ReportCreditCommand $command): Utilities\ReportCreditResult
reportTotal(Utilities\ReportTotalCommand $command): Utilities\ReportTotalResult
reportUuid(Utilities\ReportUuidCommand $command): Utilities\ReportUuidResult
registrationAdd(Registration\AddCommand $command): Registration\AddResult
registrationAssign(Registration\AssignCommand $command): Registration\AssignResult
registrationEdit(Registration\EditCommand $command): Registration\EditResult
registrationSwitch(Registration\SwitchCommand $command): Registration\SwitchResult
registrationObtain(Registration\ObtainCommand $command): Registration\ObtainResult
registrationCustomers(Registration\ObtainCustomersCommand $command): Registration\ObtainCustomersResult
Manifiestos y contrato
Para obtener los contratos y enviar las firmas están getContracts
y signContracts
respectivamente.
getContracts(Manifest\GetContractsCommand $command): Manifest\GetContractsResult
signContracts(Manifest\SignContractsCommand $command): Manifest\SignContractsResult
Retenciones
Los CFDI de Retenciones e información de pagos (RET) siguen un estándar más parecido a CFDI 3.2. Su cancelación es inmediata (al contrario de la solicitud de cancelación actual).
stamp(Retentions\StampCommand $command): Retentions\StampResult
stamped(Retentions\StampedCommand $command): Retentions\StampedResult
cancelSignature(Retentions\CancelSignatureCommand $command): Retentions\StampedResult
Para descargar una retención debe usar el servicio Utilerias
, método get_xml
que está implementado previamente.
Igualmente, se ha creado el método QuickFinkok::retentionDownload($uuid, $rfc)
para simplificar su implementación.
Ayuda para firmado XML para SAT y Finkok
Esta librería implementa el firmado CSD de los mensajes con el SAT para Cancelar, Obtener UUID relacionados
y Aceptación o rechazo de solicitud de cancelación.
Toda la lógica involucrada en la creación de los XML firmados se encuentra en la librería
phpcfdi/xml-cancelacion
.
También implementa el firmado con FIEL de manifiestos con Finkok.
Para estas tareas se han creado los siguientes objetos que permiten realizar el firmado de la información:
Helpers\CancelSigner
: Ayuda a firmar una solicitud de cancelación.Helpers\GetRelatedSigner
: Ayuda a firmar una solicitud de información de UUID relacionados.Helpers\AcceptRejectSigner
: Ayuda a firmar una respuesta de cancelación de 1 UUID.Helpers\DocumentSigner
: Ayuda a firmar los documentos de manifiesto de Finkok.
A su vez, estos métodos utilizan la librería phpcfdi/credentials
para poder crear las firmas y la información requerida por el SAT o Finkok.
La clase QuickFinkok
ahorra el proceso de firmar peticiones y lo hace de forma automática, sin embargo,
se muestra el siguiente ejemplo de cancelación firmada de 1 UUID con certificado y llave privada en archivos.
use PhpCfdi\Credentials\Credential; use PhpCfdi\Finkok\Helpers\CancelSigner; use PhpCfdi\Finkok\Services\Cancel\CancelSignatureCommand; // el objeto con el que se van a firmar las solicitudes $credential = Credential::openFiles('certificate.cer', 'privateKey.pem', 'password'); // el firmador de datos $signer = new CancelSigner(['11111111-2222-3333-4444-000000000001']); $signedXml = $signer->sign($credential); // el comando a pasar al método Finkok::cancelSignature o al comando CancelSignatureService $cancelCommand = new CancelSignatureCommand($signedXml);
Notas de implementación
Durante el proceso de implementación he creado diversas notas y documentos:
-
Cancelación: Información del proceso de cancelación, métodos, acuses, pending buffer, etc.
-
Servicios: Documentación básica de servicios.
-
Listado de servicios: Listado de todos los servicios disponibles de Finkok y si están o no implementados, así como un listado de los servicios que no se implementarán.
-
Registro de clientes: Si vas a trabajar con la capacidad de Finkok de sub-distribuidor y así poder administrar los datos de clientes.
-
Pruebas de integración: Documentación de cómo funciona y como configurar el entorno de pruebas de integración.
-
Problemas encontrados:
- Cancelación de un CFDI recién creado
- Consumir
queryPending
con un CFDI recién creado - Consumir
stamp
para generar un doble estampado no devuelve los datos - Falta servicio que no requiera CSD/FIEL para aceptar o rechazar una solicitud de cancelación
- Falta servicio que no requiera CSD/FIEL para obtener los CFDI relacionados
- El acuse de cancelación entregado al cancelar y al solicitar el acuse no coinciden
- Error de cancelación de retenciones 1308 - Certificado revocado o caduco
- El valor
CodEstatus
está ausente en la cancelación de CFDI de Retenciones - El método
Registration#Get
contaxpayer_id
vacío no devuelve el listado de clientes - Al timbrar con un texto
&
devuelve705 - XML Estructura inválida
Capturar conversación HTTP
Algunas veces, al reportar a Finkok un problema, nos solicitan la conversación HTTP (Request y Response) para poder revisar el problema sobre la información enviada.
Esta librería genera mensajes utilizando PSR-3: Logger Interface, y se utiliza dentro del objeto SoapFactory
para crear un SoapCaller
. Este objeto envía dos tipos de mensajes: LogLevel::ERROR
cuando ocurre un error al
momento de establecer comunicación con los servicios, y LogLevel::DEBUG
cuando se ejecutó una llamada SOAP.
Ambos mensajes están representados como una cadena en formato JSON, por lo que, para leerla fácilmente
es importante decodificarla.
El formato JSON es mejor dado que permite analizar el texto y encontrar caracteres especiales, mientras que, al convertirlo a un texto más entendible para el humano, estos caracteres especiales se pueden esconder o interpretar de forma errónea.
Se ofrece la clase PhpCfdi\Finkok\Helpers\FileLogger
como una utilería de LoggerInterface
que manda los mensajes recibidos a la salida estándar o a un archivo.
También se ofrece la clase PhpCfdi\Finkok\Helpers\JsonDecoderLogger
como una utilería de LoggerInterface
que decodifica el mensaje JSON y luego lo convierte a cadena de caracteres usando la función print_r()
,
para después mandarlo a otro objeto LoggerInterface
.
En el siguiente ejemplo se muestra la forma recomendada para establecer el objeto Logger
,
también se muestra el uso de JsonDecoderLogger
para realizar la conversión de JSON a texto plano y
FileLogger
para enviar el mensaje a un archivo específico.
La clase JsonDecoderLogger
puede generar pérdida de información, pero los mensajes son más entendibles,
si deseas también incluir el mensaje JSON puedes usar JsonDecoderLogger::setAlsoLogJsonMessage(true)
.
use PhpCfdi\Finkok\FinkokEnvironment; use PhpCfdi\Finkok\FinkokSettings; use PhpCfdi\Finkok\Helpers\FileLogger; use PhpCfdi\Finkok\Helpers\JsonDecoderLogger; $logger = new JsonDecoderLogger(new FileLogger('/tmp/finkok.log')); $logger->setAlsoLogJsonMessage(true); // enviar en texto simple y también en formato JSON $settings = new FinkokSettings('user@host.com', 'secret', FinkokEnvironment::makeProduction()); $settings->soapFactory()->setLogger($logger);
Si estás usando Laravel, ya cuentas con una implementación de LoggerInterface
, por lo que te recomiendo usar:
/** @var \Psr\Log\LoggerInterface $logger */ $logger = app(\Psr\Log\LoggerInterface::class); // Encapsular el logger en el decodificador JSON: $logger = new \PhpCfdi\Finkok\Helpers\JsonDecoderLogger($logger);
Compatibilidad
Esta librería se mantendrá compatible con al menos la versión con soporte activo de PHP más reciente.
También utilizamos Versionado Semántico 2.0.0 por lo que puedes usar esta librería sin temor a romper tu aplicación.
Contribuciones
Las contribuciones con bienvenidas. Por favor lee CONTRIBUTING para más detalles y recuerda revisar el archivo de tareas pendientes TODO y el archivo CHANGELOG.
Copyright and License
The phpcfdi/finkok library is copyright © PhpCfdi and licensed for use under the MIT License (MIT). Please see LICENSE for more information.