@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
JavaScript
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":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,4BAA4B,CAAC;AAC3D,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,kBAAkB,CAAC;AACpD,OAAO,EAAE,eAAe,EAAE,qBAAqB,EAAE,MAAM,qBAAqB,CAAC;AAE7E;;GAEG;AACH,MAAM,OAAO,cAAc;IACzB;;;;OAIG;IACI,MAAM,CAAC,YAAY,CAAC,GAAW;QACpC,IAAI,CAAC;YACH,uCAAuC;YACvC,IAAI,CAAC,GAAG,IAAI,OAAO,GAAG,KAAK,QAAQ,IAAI,GAAG,CAAC,IAAI,EAAE,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBAC/D,OAAO,aAAa,CAAC,OAAO,CAAC;YAC/B,CAAC;YAED,gDAAgD;YAChD,MAAM,UAAU,GAAG,cAAc,CAAC,gBAAgB,CAAC,GAAG,CAAC,CAAC;YACxD,IAAI,UAAU,KAAK,aAAa,CAAC,OAAO,EAAE,CAAC;gBACzC,OAAO,UAAU,CAAC;YACpB,CAAC;YAED,qCAAqC;YACrC,MAAM,GAAG,GAAG,IAAI,SAAS,EAAE,CAAC,eAAe,CAAC,GAAG,EAAE,iBAAiB,CAAC,CAAC;YACpE,MAAM,IAAI,GAAG,GAAG,CAAC,eAAe,CAAC;YAEjC,IAAI,CAAC,IAAI,EAAE,CAAC;gBACV,OAAO,aAAa,CAAC,OAAO,CAAC;YAC/B,CAAC;YAED,qDAAqD;YACrD,IAAI,cAAc,CAAC,WAAW,CAAC,IAAI,CAAC,EAAE,CAAC;gBACrC,oCAAoC;gBACpC,IAAI,cAAc,CAAC,iBAAiB,CAAC,GAAG,CAAC,EAAE,CAAC;oBAC1C,OAAO,aAAa,CAAC,SAAS,CAAC;gBACjC,CAAC;gBACD,OAAO,aAAa,CAAC,GAAG,CAAC;YAC3B,CAAC;YAED,iEAAiE;YACjE,IAAI,cAAc,CAAC,WAAW,CAAC,IAAI,CAAC,EAAE,CAAC;gBACrC,OAAO,cAAc,CAAC,eAAe,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;YAClD,CAAC;YAED,4DAA4D;YAC5D,IAAI,cAAc,CAAC,iBAAiB,CAAC,IAAI,CAAC,EAAE,CAAC;gBAC3C,OAAO,aAAa,CAAC,OAAO,CAAC;YAC/B,CAAC;YAED,sBAAsB;YACtB,IAAI,cAAc,CAAC,iBAAiB,CAAC,IAAI,CAAC,EAAE,CAAC;gBAC3C,OAAO,aAAa,CAAC,SAAS,CAAC;YACjC,CAAC;YAED,OAAO,aAAa,CAAC,OAAO,CAAC;QAC/B,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,yBAAyB,EAAE,KAAK,CAAC,CAAC;YAChD,OAAO,aAAa,CAAC,OAAO,CAAC;QAC/B,CAAC;IACH,CAAC;IAED;;;;;OAKG;IACK,MAAM,CAAC,gBAAgB,CAAC,GAAW;QACzC,MAAM,QAAQ,GAAG,GAAG,CAAC,WAAW,EAAE,CAAC;QAEnC,wCAAwC;QACxC,IACE,QAAQ,CAAC,QAAQ,CAAC,aAAa,CAAC;YAChC,QAAQ,CAAC,QAAQ,CAAC,cAAc,CAAC;YACjC,QAAQ,CAAC,QAAQ,CAAC,WAAW,CAAC;YAC9B,QAAQ,CAAC,QAAQ,CAAC,yBAAyB,CAAC,IAAI,QAAQ,CAAC,QAAQ,CAAC,UAAU,CAAC,EAC7E,CAAC;YACD,OAAO,aAAa,CAAC,OAAO,CAAC;QAC/B,CAAC;QAED,uCAAuC;QACvC,IACE,QAAQ,CAAC,QAAQ,CAAC,UAAU,CAAC;YAC7B,QAAQ,CAAC,QAAQ,CAAC,qBAAqB,CAAC;YACxC,QAAQ,CAAC,QAAQ,CAAC,WAAW,CAAC;YAC9B,QAAQ,CAAC,QAAQ,CAAC,aAAa,CAAC,EAChC,CAAC;YACD,OAAO,aAAa,CAAC,OAAO,CAAC;QAC/B,CAAC;QAED,yCAAyC;QACzC,IACE,QAAQ,CAAC,QAAQ,CAAC,WAAW,CAAC;YAC9B,QAAQ,CAAC,QAAQ,CAAC,sCAAsC,CAAC,EACzD,CAAC;YACD,OAAO,aAAa,CAAC,SAAS,CAAC;QACjC,CAAC;QAED,yCAAyC;QACzC,IACE,QAAQ,CAAC,QAAQ,CAAC,WAAW,CAAC;YAC9B,QAAQ,CAAC,QAAQ,CAAC,qBAAqB,CAAC;YACxC,QAAQ,CAAC,QAAQ,CAAC,oBAAoB,CAAC,EACvC,CAAC;YACD,OAAO,aAAa,CAAC,SAAS,CAAC;QACjC,CAAC;QAED,qBAAqB;QACrB,OAAO,aAAa,CAAC,OAAO,CAAC;IAC/B,CAAC;IAED;;;;OAIG;IACK,MAAM,CAAC,WAAW,CAAC,IAAa;QACtC,OAAO,CACL,IAAI,CAAC,QAAQ,KAAK,SAAS;YAC3B,IAAI,CAAC,QAAQ,KAAK,YAAY;YAC9B,IAAI,CAAC,QAAQ,KAAK,aAAa;YAC/B,IAAI,CAAC,QAAQ,KAAK,gBAAgB;YAClC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,UAAU,CAAC;YAClC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,aAAa,CAAC,CACtC,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACK,MAAM,CAAC,iBAAiB,CAAC,GAAa;QAC5C,IAAI,CAAC;YACH,sCAAsC;YACtC,MAAM,UAAU,GAAG;gBACjB,KAAK,EAAE,sEAAsE;gBAC7E,KAAK,EAAE,wDAAwD;aAChE,CAAC;YAEF,wCAAwC;YACxC,MAAM,MAAM,GAAG,KAAK,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC;YAE/C,8DAA8D;YAC9D,MAAM,kBAAkB,GAAG,GAAG,CAAC,oBAAoB,CAAC,qBAAqB,CAAC,CAAC;YAE3E,yDAAyD;YACzD,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,kBAAkB,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;gBACnD,MAAM,IAAI,GAAG,kBAAkB,CAAC,CAAC,CAAC,CAAC;gBACnC,IAAI,IAAI,CAAC,WAAW,IAAI,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,WAAW,CAAC,EAAE,CAAC;oBAC/D,OAAO,IAAI,CAAC;gBACd,CAAC;YACH,CAAC;YAED,OAAO,KAAK,CAAC;QACf,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,IAAI,CAAC,sCAAsC,EAAE,KAAK,CAAC,CAAC;YAC5D,0DAA0D;YAC1D,MAAM,MAAM,GAAG,IAAI,aAAa,EAAE,CAAC,iBAAiB,CAAC,GAAG,CAAC,CAAC;YAC1D,OAAO,MAAM,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,MAAM,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC;QACtE,CAAC;IACH,CAAC;IAED;;;;OAIG;IACK,MAAM,CAAC,WAAW,CAAC,IAAa;QACtC,OAAO,CACL,IAAI,CAAC,QAAQ,KAAK,0BAA0B;YAC5C,IAAI,CAAC,QAAQ,KAAK,sBAAsB;YACxC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,uBAAuB,CAAC,CAChD,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACK,MAAM,CAAC,iBAAiB,CAAC,IAAa;QAC5C,OAAO,CACL,IAAI,CAAC,QAAQ,KAAK,2BAA2B;YAC7C,IAAI,CAAC,QAAQ,KAAK,uBAAuB;YACzC,IAAI,CAAC,QAAQ,KAAK,2BAA2B;YAC7C,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,wBAAwB,CAAC,CACjD,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACK,MAAM,CAAC,iBAAiB,CAAC,IAAa;QAC5C,OAAO,CACL,IAAI,CAAC,QAAQ,KAAK,oBAAoB;YACtC,CAAC,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,IAAI,IAAI,CAAC,YAAY,CAAC,OAAO,CAAE,CAAC,QAAQ,CAAC,kBAAkB,CAAC,CAAC,CACzF,CAAC;IACJ,CAAC;IAED;;;;;OAKG;IACK,MAAM,CAAC,eAAe,CAAC,GAAa,EAAE,GAAW;QACvD,IAAI,CAAC;YACH,uEAAuE;YACvE,MAAM,YAAY,GAAG,GAAG,CAAC,sBAAsB,CAC7C,8DAA8D,EAC9D,0BAA0B,CAC3B,CAAC;YAEF,IAAI,YAAY,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBAC9B,wBAAwB;gBACxB,MAAM,gBAAgB,GAAG,GAAG,CAAC,oBAAoB,CAAC,0BAA0B,CAAC,CAAC;gBAC9E,IAAI,gBAAgB,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;oBAClC,qCAAqC;oBACrC,OAAO,cAAc,CAAC,yBAAyB,CAAC,GAAG,CAAC,CAAC;gBACvD,CAAC;YACH,CAAC;YAED,2CAA2C;YAC3C,MAAM,eAAe,GAAG,CAAC,GAAG,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,EAAE,GAAG,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,oBAAoB,CAAC,0BAA0B,CAAC,CAAC,CAAC,CAAC;YAE3H,KAAK,MAAM,WAAW,IAAI,eAAe,EAAE,CAAC;gBAC1C,2BAA2B;gBAC3B,MAAM,cAAc,GAAG,WAAW,CAAC,oBAAoB,CAAC,gDAAgD,CAAC,CAAC;gBAE1G,IAAI,cAAc,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;oBAChC,SAAS;gBACX,CAAC;gBAED,KAAK,MAAM,aAAa,IAAI,KAAK,CAAC,IAAI,CAAC,cAAc,CAAC,EAAE,CAAC;oBACvD,kBAAkB;oBAClB,MAAM,OAAO,GAAG,aAAa,CAAC,oBAAoB,CAAC,QAAQ,CAAC,CAAC;oBAE7D,IAAI,OAAO,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;wBACzB,SAAS;oBACX,CAAC;oBAED,KAAK,MAAM,MAAM,IAAI,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;wBACzC,MAAM,WAAW,GAAG,MAAM,CAAC,WAAW,IAAI,EAAE,CAAC;wBAE7C,yCAAyC;wBACzC,IACE,WAAW,CAAC,QAAQ,CAAC,SAAS,CAAC;4BAC/B,WAAW,CAAC,QAAQ,CAAC,WAAW,CAAC;4BACjC,WAAW,KAAK,eAAe,CAAC,aAAa;4BAC7C,WAAW,KAAK,eAAe,CAAC,eAAe;4BAC/C,WAAW,KAAK,eAAe,CAAC,gBAAgB;4BAChD,WAAW,KAAK,eAAe,CAAC,gBAAgB;4BAChD,WAAW,KAAK,eAAe,CAAC,kBAAkB;4BAClD,WAAW,KAAK,eAAe,CAAC,mBAAmB,EACnD,CAAC;4BACD,OAAO,aAAa,CAAC,OAAO,CAAC;wBAC/B,CAAC;wBAED,8BAA8B;wBAC9B,IACE,WAAW,CAAC,QAAQ,CAAC,UAAU,CAAC;4BAChC,WAAW,KAAK,eAAe,CAAC,eAAe;4BAC/C,WAAW,KAAK,eAAe,CAAC,aAAa;4BAC7C,WAAW,KAAK,eAAe,CAAC,eAAe,EAC/C,CAAC;4BACD,OAAO,aAAa,CAAC,OAAO,CAAC;wBAC/B,CAAC;oBACH,CAAC;gBACH,CAAC;YACH,CAAC;YAED,iDAAiD;YACjD,OAAO,cAAc,CAAC,yBAAyB,CAAC,GAAG,CAAC,CAAC;QACvD,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,IAAI,CAAC,0DAA0D,EAAE,KAAK,CAAC,CAAC;YAChF,OAAO,cAAc,CAAC,yBAAyB,CAAC,GAAG,CAAC,CAAC;QACvD,CAAC;IACH,CAAC;IAED;;;;OAIG;IACK,MAAM,CAAC,yBAAyB,CAAC,GAAW;QAClD,gCAAgC;QAChC,IAAI,GAAG,CAAC,QAAQ,CAAC,UAAU,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,UAAU,CAAC,EAAE,CAAC;YACzD,OAAO,aAAa,CAAC,OAAO,CAAC;QAC/B,CAAC;QAED,+BAA+B;QAC/B,IAAI,GAAG,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;YACvD,OAAO,aAAa,CAAC,OAAO,CAAC;QAC/B,CAAC;QAED,sDAAsD;QACtD,OAAO,aAAa,CAAC,GAAG,CAAC;IAC3B,CAAC;CACF"}