UNPKG

@fin.cx/einvoice

Version:

A TypeScript module for creating, manipulating, and embedding XML data within PDF files specifically tailored for electronic invoice (einvoice) packages.

249 lines 19.7 kB
import { InvoiceFormat } from '../../interfaces/common.js'; import { DOMParser, xpath } from '../../plugins.js'; import { CII_PROFILE_IDS, ZUGFERD_V1_NAMESPACES } from '../cii/cii.types.js'; /** * Utility class for detecting invoice formats */ export class FormatDetector { /** * Detects the format of an XML document * @param xml XML content to analyze * @returns Detected invoice format */ static detectFormat(xml) { try { // Quick check for empty or invalid XML if (!xml || typeof xml !== 'string' || xml.trim().length === 0) { return InvoiceFormat.UNKNOWN; } // Quick string-based pre-checks for performance const quickCheck = FormatDetector.quickFormatCheck(xml); if (quickCheck !== InvoiceFormat.UNKNOWN) { return quickCheck; } // More thorough parsing-based checks const doc = new DOMParser().parseFromString(xml, 'application/xml'); const root = doc.documentElement; if (!root) { return InvoiceFormat.UNKNOWN; } // UBL detection (Invoice or CreditNote root element) if (FormatDetector.isUBLFormat(root)) { // Check for XRechnung customization if (FormatDetector.isXRechnungFormat(doc)) { return InvoiceFormat.XRECHNUNG; } return InvoiceFormat.UBL; } // Factur-X/ZUGFeRD detection (CrossIndustryInvoice root element) if (FormatDetector.isCIIFormat(root)) { return FormatDetector.detectCIIFormat(doc, xml); } // ZUGFeRD v1 detection (CrossIndustryDocument root element) if (FormatDetector.isZUGFeRDV1Format(root)) { return InvoiceFormat.ZUGFERD; } // FatturaPA detection if (FormatDetector.isFatturaPAFormat(root)) { return InvoiceFormat.FATTURAPA; } return InvoiceFormat.UNKNOWN; } catch (error) { console.error('Error detecting format:', error); return InvoiceFormat.UNKNOWN; } } /** * Performs a quick format check based on string content * This is faster than full XML parsing for obvious cases * @param xml XML string * @returns Detected format or UNKNOWN if more analysis is needed */ static quickFormatCheck(xml) { const lowerXml = xml.toLowerCase(); // Check for obvious Factur-X indicators if (lowerXml.includes('factur-x.eu') || lowerXml.includes('factur-x.xml') || lowerXml.includes('factur-x:') || lowerXml.includes('urn:cen.eu:en16931:2017') && lowerXml.includes('factur-x')) { return InvoiceFormat.FACTURX; } // Check for obvious ZUGFeRD indicators if (lowerXml.includes('zugferd:') || lowerXml.includes('zugferd-invoice.xml') || lowerXml.includes('urn:ferd:') || lowerXml.includes('urn:zugferd')) { return InvoiceFormat.ZUGFERD; } // Check for obvious XRechnung indicators if (lowerXml.includes('xrechnung') || lowerXml.includes('urn:xoev-de:kosit:standard:xrechnung')) { return InvoiceFormat.XRECHNUNG; } // Check for obvious FatturaPA indicators if (lowerXml.includes('fatturapa') || lowerXml.includes('fattura elettronica') || lowerXml.includes('fatturaelettronica')) { return InvoiceFormat.FATTURAPA; } // Need more analysis return InvoiceFormat.UNKNOWN; } /** * Checks if the document is a UBL format * @param root Root element * @returns True if it's a UBL format */ static isUBLFormat(root) { return (root.nodeName === 'Invoice' || root.nodeName === 'CreditNote' || root.nodeName === 'ubl:Invoice' || root.nodeName === 'ubl:CreditNote' || root.nodeName.endsWith(':Invoice') || root.nodeName.endsWith(':CreditNote')); } /** * Checks if the document is an XRechnung format * @param doc XML document * @returns True if it's an XRechnung format */ static isXRechnungFormat(doc) { try { // Set up namespaces for XPath queries const namespaces = { 'cbc': 'urn:oasis:names:specification:ubl:schema:xsd:CommonBasicComponents-2', 'ubl': 'urn:oasis:names:specification:ubl:schema:xsd:Invoice-2' }; // Create XPath selector with namespaces const select = xpath.useNamespaces(namespaces); // Use getElementsByTagName directly for more reliable results const customizationNodes = doc.getElementsByTagName('cbc:CustomizationID'); // Check if any CustomizationID node contains "xrechnung" for (let i = 0; i < customizationNodes.length; i++) { const node = customizationNodes[i]; if (node.textContent && node.textContent.includes('xrechnung')) { return true; } } return false; } catch (error) { console.warn('Error checking for XRechnung format:', error); // If direct DOM access fails, try a string-based approach const xmlStr = new XMLSerializer().serializeToString(doc); return xmlStr.includes('xrechnung') || xmlStr.includes('XRechnung'); } } /** * Checks if the document is a CII format (Factur-X/ZUGFeRD v2+) * @param root Root element * @returns True if it's a CII format */ static isCIIFormat(root) { return (root.nodeName === 'rsm:CrossIndustryInvoice' || root.nodeName === 'CrossIndustryInvoice' || root.nodeName.endsWith(':CrossIndustryInvoice')); } /** * Checks if the document is a ZUGFeRD v1 format * @param root Root element * @returns True if it's a ZUGFeRD v1 format */ static isZUGFeRDV1Format(root) { return (root.nodeName === 'rsm:CrossIndustryDocument' || root.nodeName === 'CrossIndustryDocument' || root.nodeName === 'ram:CrossIndustryDocument' || root.nodeName.endsWith(':CrossIndustryDocument')); } /** * Checks if the document is a FatturaPA format * @param root Root element * @returns True if it's a FatturaPA format */ static isFatturaPAFormat(root) { return (root.nodeName === 'FatturaElettronica' || (root.getAttribute('xmlns') && root.getAttribute('xmlns').includes('fatturapa.gov.it'))); } /** * Detects the specific CII format (Factur-X vs ZUGFeRD) * @param doc XML document * @param xml Original XML string for fallback checks * @returns Detected format */ static detectCIIFormat(doc, xml) { try { // Use direct DOM traversal instead of XPath for more reliable behavior const contextNodes = doc.getElementsByTagNameNS('urn:un:unece:uncefact:data:standard:CrossIndustryInvoice:100', 'ExchangedDocumentContext'); if (contextNodes.length === 0) { // Try without namespace const noNsContextNodes = doc.getElementsByTagName('ExchangedDocumentContext'); if (noNsContextNodes.length === 0) { // Fallback to string-based detection return FormatDetector.detectCIIFormatFromString(xml); } } // Loop through all potential context nodes const allContextNodes = [...Array.from(contextNodes), ...Array.from(doc.getElementsByTagName('ExchangedDocumentContext'))]; for (const contextNode of allContextNodes) { // Find guideline parameter const guidelineNodes = contextNode.getElementsByTagName('ram:GuidelineSpecifiedDocumentContextParameter'); if (guidelineNodes.length === 0) { continue; } for (const guidelineNode of Array.from(guidelineNodes)) { // Find ID element const idNodes = guidelineNode.getElementsByTagName('ram:ID'); if (idNodes.length === 0) { continue; } for (const idNode of Array.from(idNodes)) { const profileText = idNode.textContent || ''; // Check for ZUGFeRD profiles (v1 and v2) if (profileText.includes('zugferd') || profileText.includes('urn:ferd:') || profileText === CII_PROFILE_IDS.ZUGFERD_BASIC || profileText === CII_PROFILE_IDS.ZUGFERD_COMFORT || profileText === CII_PROFILE_IDS.ZUGFERD_EXTENDED || profileText === CII_PROFILE_IDS.ZUGFERD_V1_BASIC || profileText === CII_PROFILE_IDS.ZUGFERD_V1_COMFORT || profileText === CII_PROFILE_IDS.ZUGFERD_V1_EXTENDED) { return InvoiceFormat.ZUGFERD; } // Check for Factur-X profiles if (profileText.includes('factur-x') || profileText === CII_PROFILE_IDS.FACTURX_MINIMUM || profileText === CII_PROFILE_IDS.FACTURX_BASIC || profileText === CII_PROFILE_IDS.FACTURX_EN16931) { return InvoiceFormat.FACTURX; } } } } // If we reach here, fall back to string checking return FormatDetector.detectCIIFormatFromString(xml); } catch (error) { console.warn('Error detecting CII format, falling back to generic CII:', error); return FormatDetector.detectCIIFormatFromString(xml); } } /** * Fallback method to detect CII format from string content * @param xml XML string * @returns Detected format */ static detectCIIFormatFromString(xml) { // Check for Factur-X indicators if (xml.includes('factur-x') || xml.includes('Factur-X')) { return InvoiceFormat.FACTURX; } // Check for ZUGFeRD indicators if (xml.includes('zugferd') || xml.includes('ZUGFeRD')) { return InvoiceFormat.ZUGFERD; } // Generic CII if we can't determine more specifically return InvoiceFormat.CII; } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"format.detector.js","sourceRoot":"","sources":["../../../ts/formats/utils/format.detector.ts"],"names":[],"mappings":"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"}