phpcfdi/finkok

Librería para conectar con la API de servicios de FINKOK

v0.5.4 2024-04-13 04:55 UTC

README

Source Code Latest Version Software License Build Status Scrutinizer Coverage Status Total Downloads

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:

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 correctamente es importante decodificarla.

La clase PhpCfdi\Finkok\Tests\LoggerPrinter es un ejemplo de implementación de LoggerInterface que manda los mensajes recibidos a la salida estándar o a un archivo. Es importante notar que el objeto LoggerPrinter no está disponible en el paquete, sin embargo, lo puedes descargar y poner dentro de tu proyecto con tu espacio de nombres.

De igual forma, se puede utilizar cualquier objeto que implemente LoggerInterface, por ejemplo, en Laravel se puede usar $logger = app(Psr\Log\LoggerInterface::class). Pero recuerda que, una vez que tengas el mensaje, deberás decodificarlo de JSON a texto plano.

Para establecer el objeto Logger es recomendable hacerlo de la siguiente forma:

use PhpCfdi\Finkok\FinkokEnvironment;
use PhpCfdi\Finkok\FinkokSettings;
use PhpCfdi\Finkok\Tests\LoggerPrinter;

$logger = new LoggerPrinter('/tmp/finkok.log');

$settings = new FinkokSettings('user@host.com', 'secret', FinkokEnvironment::makeProduction());
$settings->soapFactory()->setLogger($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.

Versión de la librería Versión de PHP Fecha de lanzamiento
0.1.0 7.2, 7.3 y 7.4 2019-03-29
0.3.0 7.3, 7.4, 8.0, 8.1, 8.2 y 8.3 2021-03-18

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.