io.alapierre.ksef.fop.PdfGenerator Maven / Gradle / Ivy
package io.alapierre.ksef.fop;
import lombok.val;
import org.apache.fop.apps.*;
import org.apache.fop.configuration.Configuration;
import org.apache.fop.configuration.ConfigurationException;
import org.apache.fop.configuration.DefaultConfigurationBuilder;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import javax.xml.transform.*;
import javax.xml.transform.sax.SAXResult;
import javax.xml.transform.stream.StreamSource;
import java.io.*;
import java.time.LocalDate;
import java.time.format.DateTimeFormatter;
import java.util.Base64;
import java.util.Optional;
/**
* @author Adrian Lapierre {@literal [email protected]}
* Copyrights by original author 2023.11.11
*/
public class PdfGenerator {
private final FopFactory fopFactory;
private InvoicePdfConfig invoicePdfConfig = new InvoicePdfConfig();
/**
* Create generator with Apache FOP config file from classpath
*
* @param fopConfig config file name
* @throws IOException throws when cant load given config file from classpath
* @throws ConfigurationException throws when config file has errors
*/
public PdfGenerator(String fopConfig, InvoicePdfConfig invoicePdfConfig) throws IOException, ConfigurationException {
this(loadResource(fopConfig));
this.invoicePdfConfig = invoicePdfConfig;
}
/**
* Create generator with Apache FOP config file from classpath
*
* @param fopConfig config file name
* @throws IOException throws when cant load given config file from classpath
* @throws ConfigurationException throws when config file has errors
*/
public PdfGenerator(String fopConfig) throws IOException, ConfigurationException {
this(loadResource(fopConfig));
}
/**
* Create generator with Apache FOP config
*
* @param fopConfig InputStream to read FOP config file
*
* @throws ConfigurationException throws when config file has errors
*/
public PdfGenerator(InputStream fopConfig) throws ConfigurationException {
val builder = new FopFactoryBuilder(new File(".").toURI());
DefaultConfigurationBuilder cfgBuilder = new DefaultConfigurationBuilder();
Configuration cfg = cfgBuilder.build(fopConfig);
builder.setConfiguration(cfg);
fopFactory = builder.build();
}
/**
* Generates UPO PDF from given XML and OutputStream
* @param upoXML UPO XML
* @param out destination OutputStream
* @throws IOException throws when IO error occurs
* @throws TransformerException throws when XSLT transformer error occurs
* @throws FOPException throws when FOP error occurs
*/
public void generateUpo(Source upoXML, OutputStream out) throws IOException, TransformerException, FOPException {
FOUserAgent foUserAgent = fopFactory.newFOUserAgent();
Fop fop = fopFactory.newFop("application/pdf", foUserAgent, out);
TransformerFactory factory = TransformerFactory.newInstance();
Transformer transformer = factory.newTransformer(new StreamSource(loadResource("ksef_upo.fop")));
Result res = new SAXResult(fop.getDefaultHandler());
transformer.transform(upoXML, res);
}
/**
* Generates a regular invoice PDF based on the input XML, using InvoiceGenerationParams for configurable options.
*
* This method is responsible for creating a new invoice and does not handle duplicate-related logic.
*
* @param invoiceXml e-invoice FA(2) XML
* @param params An instance of InvoiceGenerationParams, containing settings like KSeF number, verification link,
* QR code, and logo.
* @param out The OutputStream where the generated PDF will be written.
* @throws IOException throws when IO error occurs
* @throws TransformerException throws when XSLT transformer error occurs
* @throws FOPException throws when FOP error occurs
*/
public void generateInvoice(Source invoiceXml, InvoiceGenerationParams params, OutputStream out)
throws IOException, TransformerException, FOPException {
generatePdfInvoice(invoiceXml, params, null, out);
}
/**
* Generates a duplicate invoice PDF based on the input XML, using InvoiceGenerationParams for configurable options.
*
* This method specifically handles the generation of a duplicate invoice, inserting the duplicate date.
*
* @param invoiceXml The source XML file representing the e-invoice FA(2) format.
* @param params An instance of InvoiceGenerationParams, containing settings like KSeF number, verification link,
* QR code, logo, and duplicate date.
* @param duplicateDate The date when the duplicate invoice was issued.
* @param out The OutputStream where the generated PDF will be written.
* @throws IOException If an I/O error occurs during the generation process.
* @throws TransformerException If an error occurs during the XSLT transformation.
* @throws FOPException If an error occurs during the PDF rendering process.
*/
public void generateDuplicateInvoice(Source invoiceXml, InvoiceGenerationParams params, LocalDate duplicateDate, OutputStream out)
throws IOException, TransformerException, FOPException {
generatePdfInvoice(invoiceXml, params, duplicateDate, out);
}
/**
* Internal method that performs the actual generation of the PDF, applying any additional data such as
* KSeF number, QR codes, and handling duplicate invoices if applicable.
*
* @param invoiceXml The source XML representing the invoice.
* @param params An instance of InvoiceGenerationParams containing additional data for the invoice.
* @param duplicateDate The date the duplicate invoice was issued (can be null for regular invoices).
* @param out The OutputStream where the generated PDF will be written.
* @throws FOPException If an error occurs during the FOP processing.
* @throws IOException If an I/O error occurs.
* @throws TransformerException If an error occurs during the XSLT transformation.
*/
private void generatePdfInvoice(@NotNull Source invoiceXml, InvoiceGenerationParams params, @Nullable LocalDate duplicateDate, OutputStream out)
throws FOPException, IOException, TransformerException {
FOUserAgent foUserAgent = fopFactory.newFOUserAgent();
Fop fop = fopFactory.newFop("application/pdf", foUserAgent, out);
TransformerFactory factory = TransformerFactory.newInstance();
Transformer transformer = factory.newTransformer(new StreamSource(loadResource("ksef_invoice.xsl")));
insertAdditionalInvoiceData(params, duplicateDate, transformer);
Result res = new SAXResult(fop.getDefaultHandler());
transformer.transform(invoiceXml, res);
}
private void insertAdditionalInvoiceData(InvoiceGenerationParams params, @Nullable LocalDate duplicateDate, @NotNull Transformer transformer) {
Optional.ofNullable(params.getQrCode())
.map(Base64.getEncoder()::encodeToString)
.ifPresent(encodedQrCode -> setParameterIfNotNull("qrCode", encodedQrCode, transformer));
Optional.ofNullable(params.getLogo())
.map(Base64.getEncoder()::encodeToString)
.ifPresent(encodedLogo -> setParameterIfNotNull("logo", encodedLogo, transformer));
Optional.ofNullable(duplicateDate)
.map(localDate -> localDate.format(DateTimeFormatter.ISO_LOCAL_DATE))
.ifPresent(formattedDate -> setParameterIfNotNull("duplicateDate", formattedDate, transformer));
Optional.ofNullable(params.getCurrencyDate())
.map(localDate -> localDate.format(DateTimeFormatter.ISO_LOCAL_DATE))
.ifPresent(formattedDate -> setParameterIfNotNull("currencyDate", formattedDate, transformer));
setParameterIfNotNull("nrKsef", params.getKsefNumber(), transformer);
setParameterIfNotNull("verificationLink", params.getVerificationLink(), transformer);
setParameterIfNotNull("showFooter", invoicePdfConfig.isShowFooter(), transformer);
}
private void setParameterIfNotNull(@NotNull String name,
@Nullable Object value,
@NotNull Transformer transformer) {
if (value != null) {
transformer.setParameter(name, value);
}
}
private static InputStream loadResource(String resource) throws IOException {
val res = PdfGenerator.class.getClassLoader().getResourceAsStream(resource);
if (res == null) throw new IOException("Can't load classpath resource " + resource);
return res;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy