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.

266 lines 20.3 kB
import { BaseValidator } from './base.validator.js'; import { ValidationLevel } from '../interfaces.js'; import * as xpath from 'xpath'; import { DOMParser } from 'xmldom'; /** * Validator for Factur-X/ZUGFeRD invoice format * Implements validation rules according to EN16931 and Factur-X specification */ export class FacturXValidator extends BaseValidator { // XML namespaces for Factur-X/ZUGFeRD static { this.NS_RSMT = 'urn:un:unece:uncefact:data:standard:CrossIndustryInvoice:100'; } static { this.NS_RAM = 'urn:un:unece:uncefact:data:standard:ReusableAggregateBusinessInformationEntity:100'; } static { this.NS_UDT = 'urn:un:unece:uncefact:data:standard:UnqualifiedDataType:100'; } constructor(xml) { super(xml); // XML document for processing this.xmlDoc = null; // Factur-X profile (BASIC, EN16931, EXTENDED, etc.) this.profile = ''; try { // Parse XML document this.xmlDoc = new DOMParser().parseFromString(xml, 'application/xml'); // Determine Factur-X profile this.detectProfile(); } catch (error) { this.addError('FX-PARSE', `Failed to parse XML: ${error}`, '/'); } } /** * Validates the Factur-X invoice against the specified level * @param level Validation level * @returns Validation result */ validate(level = ValidationLevel.SYNTAX) { // Reset errors this.errors = []; // Check if document was parsed successfully if (!this.xmlDoc) { return { valid: false, errors: this.errors, level: level }; } // Perform validation based on level let valid = true; if (level === ValidationLevel.SYNTAX) { valid = this.validateSchema(); } else if (level === ValidationLevel.SEMANTIC) { valid = this.validateSchema() && this.validateStructure(); } else if (level === ValidationLevel.BUSINESS) { valid = this.validateSchema() && this.validateStructure() && this.validateBusinessRules(); } return { valid, errors: this.errors, level }; } /** * Validates XML against schema * @returns True if schema validation passed */ validateSchema() { // Basic schema validation (simplified for now) if (!this.xmlDoc) return false; // Check for root element const root = this.xmlDoc.documentElement; if (!root || root.nodeName !== 'rsm:CrossIndustryInvoice') { this.addError('FX-SCHEMA-1', 'Root element must be rsm:CrossIndustryInvoice', '/'); return false; } // Check for required namespaces if (!root.lookupNamespaceURI('rsm') || !root.lookupNamespaceURI('ram')) { this.addError('FX-SCHEMA-2', 'Required namespaces rsm and ram must be declared', '/'); return false; } return true; } /** * Validates structure of the XML document * @returns True if structure validation passed */ validateStructure() { if (!this.xmlDoc) return false; let valid = true; // Check for required main sections const sections = [ 'rsm:ExchangedDocumentContext', 'rsm:ExchangedDocument', 'rsm:SupplyChainTradeTransaction' ]; for (const section of sections) { if (!this.exists(section)) { this.addError('FX-STRUCT-1', `Required section ${section} is missing`, '/rsm:CrossIndustryInvoice'); valid = false; } } // Check for SupplyChainTradeTransaction sections if (this.exists('rsm:SupplyChainTradeTransaction')) { const tradeSubsections = [ 'ram:ApplicableHeaderTradeAgreement', 'ram:ApplicableHeaderTradeDelivery', 'ram:ApplicableHeaderTradeSettlement' ]; for (const subsection of tradeSubsections) { if (!this.exists(`rsm:SupplyChainTradeTransaction/ram:ApplicableHeaderTradeTransaction/${subsection}`)) { this.addError('FX-STRUCT-2', `Required subsection ${subsection} is missing`, '/rsm:CrossIndustryInvoice/rsm:SupplyChainTradeTransaction/ram:ApplicableHeaderTradeTransaction'); valid = false; } } } return valid; } /** * Validates business rules * @returns True if business rule validation passed */ validateBusinessRules() { if (!this.xmlDoc) return false; let valid = true; // BR-16: Amount due for payment (BT-115) = Invoice total amount with VAT (BT-112) - Paid amount (BT-113) valid = this.validateAmounts() && valid; // BR-CO-3: Value added tax point date (BT-7) and Value added tax point date code (BT-8) are mutually exclusive valid = this.validateMutuallyExclusiveFields() && valid; // BR-S-1: An Invoice that contains a line (BG-25) where the Invoiced item VAT category code (BT-151) is "Standard rated" // shall contain the Seller VAT Identifier (BT-31), the Seller tax registration identifier (BT-32) // and/or the Seller tax representative VAT identifier (BT-63). valid = this.validateSellerVatIdentifier() && valid; return valid; } /** * Detects Factur-X profile from the XML */ detectProfile() { if (!this.xmlDoc) return; // Look for profile identifier const profileNode = xpath.select1('string(//rsm:ExchangedDocumentContext/ram:GuidelineSpecifiedDocumentContextParameter/ram:ID)', this.xmlDoc); if (profileNode) { const profileText = profileNode.toString(); if (profileText.includes('BASIC')) { this.profile = 'BASIC'; } else if (profileText.includes('EN16931')) { this.profile = 'EN16931'; } else if (profileText.includes('EXTENDED')) { this.profile = 'EXTENDED'; } else if (profileText.includes('MINIMUM')) { this.profile = 'MINIMUM'; } } } /** * Validates amount calculations in the invoice * @returns True if amount validation passed */ validateAmounts() { if (!this.xmlDoc) return false; try { // Extract amounts const totalAmount = this.getNumberValue('//ram:ApplicableHeaderTradeSettlement/ram:SpecifiedTradeSettlementHeaderMonetarySummation/ram:GrandTotalAmount'); const paidAmount = this.getNumberValue('//ram:ApplicableHeaderTradeSettlement/ram:SpecifiedTradeSettlementHeaderMonetarySummation/ram:TotalPrepaidAmount') || 0; const dueAmount = this.getNumberValue('//ram:ApplicableHeaderTradeSettlement/ram:SpecifiedTradeSettlementHeaderMonetarySummation/ram:DuePayableAmount'); // Calculate expected due amount const expectedDueAmount = totalAmount - paidAmount; // Compare with a small tolerance for rounding errors if (Math.abs(dueAmount - expectedDueAmount) > 0.01) { this.addError('BR-16', `Amount due for payment (${dueAmount}) must equal Invoice total amount with VAT (${totalAmount}) - Paid amount (${paidAmount})`, '//ram:ApplicableHeaderTradeSettlement/ram:SpecifiedTradeSettlementHeaderMonetarySummation'); return false; } return true; } catch (error) { this.addError('FX-AMOUNT', `Error validating amounts: ${error}`, '/'); return false; } } /** * Validates mutually exclusive fields * @returns True if validation passed */ validateMutuallyExclusiveFields() { if (!this.xmlDoc) return false; try { // Check for VAT point date and code (BR-CO-3) const vatPointDate = this.exists('//ram:ApplicableHeaderTradeSettlement/ram:ApplicableTradeTax/ram:TaxPointDate'); const vatPointDateCode = this.exists('//ram:ApplicableHeaderTradeSettlement/ram:ApplicableTradeTax/ram:DueDateTypeCode'); if (vatPointDate && vatPointDateCode) { this.addError('BR-CO-3', 'Value added tax point date and Value added tax point date code are mutually exclusive', '//ram:ApplicableHeaderTradeSettlement/ram:ApplicableTradeTax'); return false; } return true; } catch (error) { this.addError('FX-MUTUAL', `Error validating mutually exclusive fields: ${error}`, '/'); return false; } } /** * Validates seller VAT identifier requirements * @returns True if validation passed */ validateSellerVatIdentifier() { if (!this.xmlDoc) return false; try { // Check if there are any standard rated line items const standardRatedItems = this.exists('//ram:SpecifiedLineTradeSettlement/ram:ApplicableTradeTax/ram:CategoryCode[text()="S"]'); if (standardRatedItems) { // Check for seller VAT identifier const sellerVatId = this.exists('//ram:ApplicableHeaderTradeAgreement/ram:SellerTradeParty/ram:SpecifiedTaxRegistration/ram:ID[@schemeID="VA"]'); const sellerTaxId = this.exists('//ram:ApplicableHeaderTradeAgreement/ram:SellerTradeParty/ram:SpecifiedTaxRegistration/ram:ID[@schemeID="FC"]'); const sellerTaxRepId = this.exists('//ram:ApplicableHeaderTradeAgreement/ram:SellerTaxRepresentativeTradeParty/ram:SpecifiedTaxRegistration/ram:ID[@schemeID="VA"]'); if (!sellerVatId && !sellerTaxId && !sellerTaxRepId) { this.addError('BR-S-1', 'An Invoice with standard rated items must contain the Seller VAT Identifier, Tax registration identifier or Tax representative VAT identifier', '//ram:ApplicableHeaderTradeAgreement/ram:SellerTradeParty'); return false; } } return true; } catch (error) { this.addError('FX-VAT', `Error validating seller VAT identifier: ${error}`, '/'); return false; } } /** * Helper method to check if a node exists * @param xpathExpression XPath to check * @returns True if node exists */ exists(xpathExpression) { if (!this.xmlDoc) return false; const nodes = xpath.select(xpathExpression, this.xmlDoc); // Handle different return types from xpath.select() if (Array.isArray(nodes)) { return nodes.length > 0; } return nodes ? true : false; } /** * Helper method to get a number value from XPath * @param xpathExpression XPath to get number from * @returns Number value or NaN if not found */ getNumberValue(xpathExpression) { if (!this.xmlDoc) return NaN; const node = xpath.select1(`string(${xpathExpression})`, this.xmlDoc); return node ? parseFloat(node.toString()) : NaN; } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"facturx.validator.js","sourceRoot":"","sources":["../../ts/formats/facturx.validator.ts"],"names":[],"mappings":"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"}