Leer un CFDI

El problema de leer un CFDI es que la información entre versiones 4.0, 3.3, 3.2 y previas no es compatible. Por ello es necesario primero averiguar la versión del archivo que deseamos interpretar.

En esta sección se describe la lectura formal de la librería para leer nodos. Si tu intensión es solamente leer CFDI entonces te convendría brincar a la lectura rápida usando QuickReader.

Procesar un CFDI

Esta librería almacena la información de un CFDI en una estructura interna llamada Nodes. Por lo que, al leer un CFDI lo que en realidad sucede es una conversión del contenido XML a esta estructura interna.

El objeto CfdiUtils\Cfdi

Con este objeto podrás leer CFDI 3.2, CFDI 3.3 y CFDI 4.0.

Este es un ejemplo básico de lectura de un contenido XML a un objeto de tipo CfdiUtils\Cfdi. Por lo general se utiliza el método estático CfdiUtils\Cfdi::newFromString.

La clase ofrece cuatro principales getters para trabajo, siendo el más importante el método CfdiUtils\Cfdi::getNode que devuelve la instancia del objeto de tipo NodeInterface del elemento principal <cfdi:Comprobante>.

<?php
$xmlContents = '<cfdi:Comprobante Version="3.3">...</cfdi:Comprobante>';
$cfdi = \CfdiUtils\Cfdi::newFromString($xmlContents);
$cfdi->getVersion(); // (string) 3.3
$cfdi->getDocument(); // clon del objeto DOMDocument
$cfdi->getSource(); // (string) <cfdi:Comprobante...
$comprobante = $cfdi->getNode(); // Nodo de trabajo del nodo cfdi:Comprobante

Uso de CfdiUtils\Cfdi::newFromString

El método estático CfdiUtils\Cfdi::newFromString verifica que el contenido XML no esté vacío y no contenga errores (se pueda crear un DOMDocument a partir de este contenido). Posteriormente, invoca la creación de un objeto de tipo CfdiUtils\Cfdi pasando el objeto DOMDocument como parámetro.

Constructor de CfdiUtils\Cfdi

Al crear un objeto de tipo CfdiUtils\Cfdi se verifican las siguientes reglas del objeto DOMDocument:

  • El documento implementa el espacio de nombres del CFDI http://www.sat.gob.mx/cfd/3 o http://www.sat.gob.mx/cfd/4.
  • El espacio de nombres tiene el prefijo cfdi.
  • El elemento raíz tiene el nombre cfdi:Comprobante.
  • El atributo de versión coincide con el espacio de nombres.

No realiza ninguna validación. La validación de un CFDI está fuera de los límites de esta clase.

Ejemplos básicos de uso de NodeInterface

Para obtener el atributo Serie de un complemento, sin importar que el atributo fuera definido originalmente, si no se definió entonces devolverá una cadena de caracteres vacía.

<?php
/** @var \CfdiUtils\Cfdi $cfdi */
$complemento = $cfdi->getNode();
echo $complemento['Serie'];

Para verificar si está especificado el atributo MetodoPago

<?php
/** @var \CfdiUtils\Cfdi $cfdi */
$complemento = $cfdi->getNode();
if (isset($complemento['MetodoPago'])) {
    echo $complemento['MetodoPago'];
}

Para obtener todos los nodos cfdi:Concepto (que está dentro de cfdi:Conceptos) se puede utilizar el método CfdiUtils\Nodes\NodeInterface::searchNodes que devuelve una colección de nodos iterable (que se puede utilizar dentro de un foreach).

<?php
/** @var \CfdiUtils\Cfdi $cfdi */
$complemento = $cfdi->getNode();
$conceptos = $complemento->searchNodes('cfdi:Conceptos', 'cfdi:Concepto');
foreach ($conceptos as $concepto) {
    echo $concepto['Unidad'];
}

Para obtener la primera ocurrencia de un nodo de determinado nombre se puede usar el método CfdiUtils\Nodes\NodeInterface::searchNode. Este método devolverá el nodo si fue encontrado o devolverá null si no se encontró.

No confundir con el método anterior que devuelve una colección de nodos.

<?php
/** @var \CfdiUtils\Cfdi $cfdi */
$complemento = $cfdi->getNode();
$tfd = $complemento->searchNode('cfdi:Complemento', 'tfd:TimbreFiscalDigital');
if (null === $tfd) {
    echo 'No existe el timbre fiscal digital';
} else {
    echo 'UUID: ', $tfd['UUID'], PHP_EOL;
}

Recuerde consultar la entrada completa relacionada con la Estructura de datos Nodes.

Obteniendo la versión de un CFDI sin la clase CfdiUtils\Cfdi

Obtener la versión de un CFDI es sencillo con la clase CfdiUtils\CfdiVersion.

El método que usarás para obtener la versión depende de la información que ya tengas cargada:

  • getFromXmlString(): Cuando ya tienes el contenido del XML en una variable
  • getFromNode(): Cuando tienes el nodo principal en un objeto de tipo CfdiUtils\Nodes\NodeInterface
  • getFromDOMDocument() y getFromDOMElement(): Cuando tienes el contenido XML cargado en un objeto de tipo DOM.

El resultado de estos métodos será un string con el número de versión y vacío en caso de no encontrarse un número de versión compatible.

<?php
$cfdiFile = '/cfdi/archivo-cfdi.xml';
$xmlContents = file_get_contents($cfdiFile);
$cfdiVersion = new \CfdiUtils\CfdiVersion();
$version = $cfdiVersion->getFromXmlString($xmlContents);

Nota: la clase CfdiUtils\Cfdi ya realiza este proceso por lo que no es recomendado duplicar el trabajo de averiguar la versión.

Limpieza de CFDI

Es frecuente que los archivos CFDI contengan errores. Para entender más el tema vea el artículo de Limpieza de un CFDI.

Si está leyendo un CFDI recibido o no confiable este es un ejemplo de cómo limpiar y crear el objeto CFDI:

<?php
// obtener el contenido del archivo CFDI
$cfdiFile = '/cfdi/recibidos/2018/FEI-456823.xml';
$xmlContents = file_get_contents($cfdiFile);

// limpiar el cfdi
$xmlContents = \CfdiUtils\Cleaner\Cleaner::staticClean($xmlContents);

// crear la instancia del objeto CFDI
$cfdi = \CfdiUtils\Cfdi::newFromString($xmlContents);

Lector rápido

Ve la documentación del lector rápido, si tu intensión no es editar el documento y confías en el contenido (no te importa si está bien escrito el XML) entonces puedes usar el lector rápido.