phpcfdi/finkok

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

v0.5.5 2024-05-24 19:18 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 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.