BREAKING CHANGE(core): Refactor contact and PDF handling across the library by replacing IContact with TContact and updating PDF processing to use a structured IPdf object. These changes ensure that empty contact objects include registration details, founded/closed dates, and status, and that PDF loading/exporting uniformly wraps buffers in a proper object.

This commit is contained in:
Philipp Kunz 2025-03-20 13:57:45 +00:00
parent 75b720a98d
commit 6906e2f778
8 changed files with 164 additions and 31 deletions

View File

@ -1,5 +1,13 @@
# Changelog # Changelog
## 2025-03-20 - 2.0.0 - BREAKING CHANGE(core)
Refactor contact and PDF handling across the library by replacing IContact with TContact and updating PDF processing to use a structured IPdf object. These changes ensure that empty contact objects include registration details, founded/closed dates, and status, and that PDF loading/exporting uniformly wraps buffers in a proper object.
- Updated createEmptyContact (renamed in documentation to reflect TContact) to return a complete TContact object with registrationDetails, foundedDate, closedDate, and status.
- Modified loadPdf and exportPdf in XInvoice to wrap PDF buffers in an IPdf object with name, id, and metadata instead of using a raw Uint8Array.
- Replaced IContact with TContact in FacturXEncoder, FacturXDecoder, and XInvoiceDecoder to standardize contact structure.
- Aligned address and contact data across decoders and encoders for consistency.
## 2025-03-17 - 1.3.3 - fix(commitinfo) ## 2025-03-17 - 1.3.3 - fix(commitinfo)
Synchronize commit info version with package.json version Synchronize commit info version with package.json version

View File

@ -3,6 +3,6 @@
*/ */
export const commitinfo = { export const commitinfo = {
name: '@fin.cx/xinvoice', name: '@fin.cx/xinvoice',
version: '1.3.3', version: '2.0.0',
description: 'A TypeScript module for creating, manipulating, and embedding XML data within PDF files specifically tailored for xinvoice packages.' description: 'A TypeScript module for creating, manipulating, and embedding XML data within PDF files specifically tailored for xinvoice packages.'
} }

View File

@ -90,7 +90,7 @@ export class XInvoice implements plugins.tsclass.business.ILetter {
} }
/** /**
* Creates an empty IContact object * Creates an empty TContact object
*/ */
private createEmptyContact(): plugins.tsclass.business.TContact { private createEmptyContact(): plugins.tsclass.business.TContact {
return { return {
@ -103,7 +103,23 @@ export class XInvoice implements plugins.tsclass.business.ILetter {
city: '', city: '',
country: '', country: '',
postalCode: '' postalCode: ''
} },
registrationDetails: {
vatId: '',
registrationId: '',
registrationName: ''
},
foundedDate: {
year: 2000,
month: 1,
day: 1
},
closedDate: {
year: 9999,
month: 12,
day: 31
},
status: 'active'
}; };
} }
@ -198,7 +214,15 @@ export class XInvoice implements plugins.tsclass.business.ILetter {
* @param pdfBuffer PDF buffer * @param pdfBuffer PDF buffer
*/ */
public async loadPdf(pdfBuffer: Uint8Array | Buffer): Promise<void> { public async loadPdf(pdfBuffer: Uint8Array | Buffer): Promise<void> {
this.pdf = Uint8Array.from(pdfBuffer); // Create a valid IPdf object
this.pdf = {
name: 'invoice.pdf',
id: `invoice-${Date.now()}`,
metadata: {
textExtraction: ''
},
buffer: Uint8Array.from(pdfBuffer)
};
try { try {
// Try to extract embedded XML // Try to extract embedded XML
@ -224,7 +248,7 @@ export class XInvoice implements plugins.tsclass.business.ILetter {
} }
try { try {
const pdfDoc = await PDFDocument.load(this.pdf); const pdfDoc = await PDFDocument.load(this.pdf.buffer);
// Get the document's metadata dictionary // Get the document's metadata dictionary
const namesDictObj = pdfDoc.catalog.lookup(PDFName.of('Names')); const namesDictObj = pdfDoc.catalog.lookup(PDFName.of('Names'));
@ -313,9 +337,9 @@ export class XInvoice implements plugins.tsclass.business.ILetter {
this.to = { ...letter.to }; this.to = { ...letter.to };
this.content = { this.content = {
invoiceData: letter.content.invoiceData ? { ...letter.content.invoiceData } : this.createEmptyInvoice(), invoiceData: letter.content.invoiceData ? { ...letter.content.invoiceData } : this.createEmptyInvoice(),
textData: letter.content.textData, textData: null,
timesheetData: letter.content.timesheetData, timesheetData: null,
contractData: letter.content.contractData contractData: null
}; };
this.needsCoverSheet = letter.needsCoverSheet; this.needsCoverSheet = letter.needsCoverSheet;
this.objectActions = [...letter.objectActions]; this.objectActions = [...letter.objectActions];
@ -412,7 +436,7 @@ export class XInvoice implements plugins.tsclass.business.ILetter {
const xmlContent = await this.exportXml(format); const xmlContent = await this.exportXml(format);
// Load the PDF // Load the PDF
const pdfDoc = await PDFDocument.load(this.pdf); const pdfDoc = await PDFDocument.load(this.pdf.buffer);
// Convert the XML string to a Uint8Array // Convert the XML string to a Uint8Array
const xmlBuffer = new TextEncoder().encode(xmlContent); const xmlBuffer = new TextEncoder().encode(xmlContent);
@ -452,8 +476,13 @@ export class XInvoice implements plugins.tsclass.business.ILetter {
// Save the modified PDF // Save the modified PDF
const modifiedPdfBytes = await pdfDoc.save(); const modifiedPdfBytes = await pdfDoc.save();
// Update the pdf property // Update the pdf property with a proper IPdf object
this.pdf = modifiedPdfBytes; this.pdf = {
name: this.pdf.name,
id: this.pdf.id,
metadata: this.pdf.metadata,
buffer: modifiedPdfBytes
};
return modifiedPdfBytes; return modifiedPdfBytes;
} catch (error) { } catch (error) {

View File

@ -27,31 +27,63 @@ export abstract class BaseDecoder {
*/ */
protected createDefaultLetter(): plugins.tsclass.business.ILetter { protected createDefaultLetter(): plugins.tsclass.business.ILetter {
// Create a default seller // Create a default seller
const seller: plugins.tsclass.business.IContact = { const seller: plugins.tsclass.business.TContact = {
name: 'Unknown Seller', name: 'Unknown Seller',
type: 'company', type: 'company',
description: 'Unknown Seller', // Required by IContact interface description: 'Unknown Seller',
address: { address: {
streetName: 'Unknown', streetName: 'Unknown',
houseNumber: '0', // Required by IAddress interface houseNumber: '0',
city: 'Unknown', city: 'Unknown',
country: 'Unknown', country: 'Unknown',
postalCode: 'Unknown', postalCode: 'Unknown',
}, },
registrationDetails: {
vatId: 'Unknown',
registrationId: 'Unknown',
registrationName: 'Unknown'
},
foundedDate: {
year: 2000,
month: 1,
day: 1
},
closedDate: {
year: 9999,
month: 12,
day: 31
},
status: 'active'
}; };
// Create a default buyer // Create a default buyer
const buyer: plugins.tsclass.business.IContact = { const buyer: plugins.tsclass.business.TContact = {
name: 'Unknown Buyer', name: 'Unknown Buyer',
type: 'company', type: 'company',
description: 'Unknown Buyer', // Required by IContact interface description: 'Unknown Buyer',
address: { address: {
streetName: 'Unknown', streetName: 'Unknown',
houseNumber: '0', // Required by IAddress interface houseNumber: '0',
city: 'Unknown', city: 'Unknown',
country: 'Unknown', country: 'Unknown',
postalCode: 'Unknown', postalCode: 'Unknown',
}, },
registrationDetails: {
vatId: 'Unknown',
registrationId: 'Unknown',
registrationName: 'Unknown'
},
foundedDate: {
year: 2000,
month: 1,
day: 1
},
closedDate: {
year: 9999,
month: 12,
day: 31
},
status: 'active'
}; };
// Create default invoice data // Create default invoice data

View File

@ -97,21 +97,37 @@ export class FacturXDecoder extends BaseDecoder {
} }
// Create seller // Create seller
const seller: plugins.tsclass.business.IContact = { const seller: plugins.tsclass.business.TContact = {
name: sellerName, name: sellerName,
type: 'company', type: 'company',
description: sellerName, description: sellerName,
address: { address: {
streetName: this.getElementText('ram:LineOne') || 'Unknown', streetName: this.getElementText('ram:LineOne') || 'Unknown',
houseNumber: '0', // Required by IAddress interface houseNumber: '0',
city: this.getElementText('ram:CityName') || 'Unknown', city: this.getElementText('ram:CityName') || 'Unknown',
country: this.getElementText('ram:CountryID') || 'Unknown', country: this.getElementText('ram:CountryID') || 'Unknown',
postalCode: this.getElementText('ram:PostcodeCode') || 'Unknown', postalCode: this.getElementText('ram:PostcodeCode') || 'Unknown',
}, },
registrationDetails: {
vatId: this.getElementText('ram:ID') || 'Unknown',
registrationId: this.getElementText('ram:ID') || 'Unknown',
registrationName: sellerName
},
foundedDate: {
year: 2000,
month: 1,
day: 1
},
closedDate: {
year: 9999,
month: 12,
day: 31
},
status: 'active'
}; };
// Create buyer // Create buyer
const buyer: plugins.tsclass.business.IContact = { const buyer: plugins.tsclass.business.TContact = {
name: buyerName, name: buyerName,
type: 'company', type: 'company',
description: buyerName, description: buyerName,
@ -122,6 +138,22 @@ export class FacturXDecoder extends BaseDecoder {
country: 'Unknown', country: 'Unknown',
postalCode: 'Unknown', postalCode: 'Unknown',
}, },
registrationDetails: {
vatId: 'Unknown',
registrationId: 'Unknown',
registrationName: buyerName
},
foundedDate: {
year: 2000,
month: 1,
day: 1
},
closedDate: {
year: 9999,
month: 12,
day: 31
},
status: 'active'
}; };
// Extract invoice type // Extract invoice type

View File

@ -32,8 +32,8 @@ export class FacturXEncoder {
} }
const invoice: plugins.tsclass.finance.IInvoice = letterArg.content.invoiceData; const invoice: plugins.tsclass.finance.IInvoice = letterArg.content.invoiceData;
const billedBy: plugins.tsclass.business.IContact = invoice.billedBy; const billedBy: plugins.tsclass.business.TContact = invoice.billedBy;
const billedTo: plugins.tsclass.business.IContact = invoice.billedTo; const billedTo: plugins.tsclass.business.TContact = invoice.billedTo;
// 2) Start building the document // 2) Start building the document
const doc = smartxmlInstance const doc = smartxmlInstance

View File

@ -118,21 +118,37 @@ export class XInvoiceDecoder extends BaseDecoder {
'Unknown Buyer'; 'Unknown Buyer';
// Create seller contact // Create seller contact
const seller: plugins.tsclass.business.IContact = { const seller: plugins.tsclass.business.TContact = {
name: sellerName, name: sellerName,
type: 'company', type: 'company',
description: sellerName, description: sellerName,
address: { address: {
streetName: sellerStreet, streetName: sellerStreet,
houseNumber: '0', // Required by IAddress interface houseNumber: '0',
city: sellerCity, city: sellerCity,
country: sellerCountry, country: sellerCountry,
postalCode: sellerPostcode, postalCode: sellerPostcode,
}, },
registrationDetails: {
vatId: this.getElementText('cac:AccountingSupplierParty/cac:Party/cac:PartyTaxScheme/cbc:CompanyID') || 'Unknown',
registrationId: this.getElementText('cac:AccountingSupplierParty/cac:Party/cac:PartyLegalEntity/cbc:CompanyID') || 'Unknown',
registrationName: sellerName
},
foundedDate: {
year: 2000,
month: 1,
day: 1
},
closedDate: {
year: 9999,
month: 12,
day: 31
},
status: 'active'
}; };
// Create buyer contact // Create buyer contact
const buyer: plugins.tsclass.business.IContact = { const buyer: plugins.tsclass.business.TContact = {
name: buyerName, name: buyerName,
type: 'company', type: 'company',
description: buyerName, description: buyerName,
@ -143,6 +159,22 @@ export class XInvoiceDecoder extends BaseDecoder {
country: 'Unknown', country: 'Unknown',
postalCode: 'Unknown', postalCode: 'Unknown',
}, },
registrationDetails: {
vatId: this.getElementText('cac:AccountingCustomerParty/cac:Party/cac:PartyTaxScheme/cbc:CompanyID') || 'Unknown',
registrationId: this.getElementText('cac:AccountingCustomerParty/cac:Party/cac:PartyLegalEntity/cbc:CompanyID') || 'Unknown',
registrationName: buyerName
},
foundedDate: {
year: 2000,
month: 1,
day: 1
},
closedDate: {
year: 9999,
month: 12,
day: 31
},
status: 'active'
}; };
// Extract invoice type // Extract invoice type

View File

@ -23,8 +23,8 @@ export class XInvoiceEncoder {
} }
const invoice: plugins.tsclass.finance.IInvoice = letterArg.content.invoiceData; const invoice: plugins.tsclass.finance.IInvoice = letterArg.content.invoiceData;
const billedBy: plugins.tsclass.business.IContact = invoice.billedBy; const billedBy: plugins.tsclass.business.TContact = invoice.billedBy;
const billedTo: plugins.tsclass.business.IContact = invoice.billedTo; const billedTo: plugins.tsclass.business.TContact = invoice.billedTo;
// Create the XML document // Create the XML document
const doc = smartxmlInstance const doc = smartxmlInstance
@ -76,9 +76,9 @@ export class XInvoiceEncoder {
const supplierPartyDetails = supplierParty.ele('cac:Party'); const supplierPartyDetails = supplierParty.ele('cac:Party');
// Seller VAT ID // Seller VAT ID
if (billedBy.vatId) { if (billedBy.type === 'company' && billedBy.registrationDetails?.vatId) {
const partyTaxScheme = supplierPartyDetails.ele('cac:PartyTaxScheme'); const partyTaxScheme = supplierPartyDetails.ele('cac:PartyTaxScheme');
partyTaxScheme.ele('cbc:CompanyID').txt(billedBy.vatId).up(); partyTaxScheme.ele('cbc:CompanyID').txt(billedBy.registrationDetails.vatId).up();
partyTaxScheme.ele('cac:TaxScheme') partyTaxScheme.ele('cac:TaxScheme')
.ele('cbc:ID').txt('VAT').up() .ele('cbc:ID').txt('VAT').up()
.up(); .up();
@ -117,9 +117,9 @@ export class XInvoiceEncoder {
const customerPartyDetails = customerParty.ele('cac:Party'); const customerPartyDetails = customerParty.ele('cac:Party');
// Buyer VAT ID // Buyer VAT ID
if (billedTo.vatId) { if (billedTo.type === 'company' && billedTo.registrationDetails?.vatId) {
const partyTaxScheme = customerPartyDetails.ele('cac:PartyTaxScheme'); const partyTaxScheme = customerPartyDetails.ele('cac:PartyTaxScheme');
partyTaxScheme.ele('cbc:CompanyID').txt(billedTo.vatId).up(); partyTaxScheme.ele('cbc:CompanyID').txt(billedTo.registrationDetails.vatId).up();
partyTaxScheme.ele('cac:TaxScheme') partyTaxScheme.ele('cac:TaxScheme')
.ele('cbc:ID').txt('VAT').up() .ele('cbc:ID').txt('VAT').up()
.up(); .up();