Электронный счёт ZUGFeRD / Factur-X
Создайте электронный счёт ZUGFeRD / Factur-X -- гибридный PDF, который люди могут читать, а машины -- автоматически обрабатывать.
Что такое ZUGFeRD / Factur-X?
ZUGFeRD (Германия) и Factur-X (Франция / ЕС) -- это один и тот же стандарт под двумя названиями. Соответствующий стандарту счёт содержит:
- Визуальный PDF -- человекочитаемый счёт, отрендеренный как PDF/A-3 (или PDF/A-4f)
- Встроенный XML -- машиночитаемые данные счёта в формате UN/CEFACT Cross Industry Invoice (CII), прикреплённые внутри PDF
Это обеспечивает полностью автоматизированную обработку, сохраняя при этом привычный визуальный документ для бухгалтеров и аудиторов.
Профили
| Профиль | Типичное использование |
|---|---|
| Minimum | Номер счёта, дата, итоги, налог |
| Basic WL | + Позиции, условия оплаты |
| Basic | + Детализированные позиции |
| EN 16931 (Comfort) | Соответствие требованиям госзакупок ЕС |
| Extended | Сложные счета с несколькими налогами |
| XRechnung | Выставление счетов госсектору Германии |
Требование PDF/A
ZUGFeRD требует, чтобы PDF-хост был PDF/A-3 или PDF/A-4f (ISO 19005). Вариант "f" допускает встроенные файлы. TCPDF-Next автоматически обрабатывает метаданные соответствия, Output Intent и флаги вложений файлов.
Полный пример кода
php
<?php
declare(strict_types=1);
require __DIR__ . '/vendor/autoload.php';
use TcpdfNext\Document;
use TcpdfNext\Enums\PdfALevel;
use TcpdfNext\Archive\OutputIntent;
use TcpdfNext\Archive\EmbeddedFile;
use TcpdfNext\Archive\AFRelationship;
// ── 1. Create a PDF/A-4f document ──────────────────────────────────────
$pdf = Document::create()
->setPdfALevel(PdfALevel::A4F)
->setOutputIntent(OutputIntent::sRGB())
->setTitle('Invoice INV-2026-0042')
->setAuthor('Acme GmbH')
->setSubject('Invoice for Order PO-2026-0815');
// ── 2. Render the visual invoice ───────────────────────────────────────
$invoiceHtml = <<<'HTML'
<h1 style="color:#1a2634;">INVOICE</h1>
<table style="width:100%; margin-bottom:20px;">
<tr>
<td><strong>Acme GmbH</strong><br>Friedrichstr. 123<br>10117 Berlin, DE</td>
<td style="text-align:right;"><strong>INV-2026-0042</strong><br>Date: 2026-02-15<br>Due: 2026-03-17</td>
</tr>
</table>
<table border="1" cellpadding="6" style="width:100%; border-collapse:collapse;">
<thead>
<tr style="background:#1a2634; color:#fff;">
<th>#</th><th>Description</th><th style="text-align:right;">Amount</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>Professional Services -- February 2026</td>
<td style="text-align:right;">EUR 1,000.00</td>
</tr>
</tbody>
<tfoot>
<tr><td colspan="2" style="text-align:right;">Net</td><td style="text-align:right;">EUR 1,000.00</td></tr>
<tr><td colspan="2" style="text-align:right;">VAT 19%</td><td style="text-align:right;">EUR 190.00</td></tr>
<tr style="font-weight:bold;">
<td colspan="2" style="text-align:right;">Total</td>
<td style="text-align:right;">EUR 1,190.00</td>
</tr>
</tfoot>
</table>
HTML;
$pdf->addPage()
->setFont('helvetica', size: 10)
->writeHtml($invoiceHtml);
// ── 3. Build the Factur-X XML (EN 16931 profile) ───────────────────────
$xml = <<<'XML'
<?xml version="1.0" encoding="UTF-8"?>
<rsm:CrossIndustryInvoice
xmlns:rsm="urn:un:unece:uncefact:data:standard:CrossIndustryInvoice:100"
xmlns:ram="urn:un:unece:uncefact:data:standard:ReusableAggregateBusinessInformationEntity:100"
xmlns:udt="urn:un:unece:uncefact:data:standard:UnqualifiedDataType:100">
<rsm:ExchangedDocumentContext>
<ram:GuidelineSpecifiedDocumentContextParameter>
<ram:ID>urn:factur-x.eu:1p0:en16931</ram:ID>
</ram:GuidelineSpecifiedDocumentContextParameter>
</rsm:ExchangedDocumentContext>
<rsm:ExchangedDocument>
<ram:ID>INV-2026-0042</ram:ID>
<ram:TypeCode>380</ram:TypeCode>
<ram:IssueDateTime>
<udt:DateTimeString format="102">20260215</udt:DateTimeString>
</ram:IssueDateTime>
</rsm:ExchangedDocument>
<rsm:SupplyChainTradeTransaction>
<ram:ApplicableHeaderTradeAgreement>
<ram:SellerTradeParty>
<ram:Name>Acme GmbH</ram:Name>
<ram:SpecifiedTaxRegistration>
<ram:ID schemeID="VA">DE123456789</ram:ID>
</ram:SpecifiedTaxRegistration>
</ram:SellerTradeParty>
<ram:BuyerTradeParty>
<ram:Name>Customer Corp</ram:Name>
</ram:BuyerTradeParty>
</ram:ApplicableHeaderTradeAgreement>
<ram:ApplicableHeaderTradeSettlement>
<ram:InvoiceCurrencyCode>EUR</ram:InvoiceCurrencyCode>
<ram:SpecifiedTradeSettlementHeaderMonetarySummation>
<ram:LineTotalAmount>1000.00</ram:LineTotalAmount>
<ram:TaxBasisTotalAmount>1000.00</ram:TaxBasisTotalAmount>
<ram:TaxTotalAmount currencyID="EUR">190.00</ram:TaxTotalAmount>
<ram:GrandTotalAmount>1190.00</ram:GrandTotalAmount>
<ram:DuePayableAmount>1190.00</ram:DuePayableAmount>
</ram:SpecifiedTradeSettlementHeaderMonetarySummation>
</ram:ApplicableHeaderTradeSettlement>
</rsm:SupplyChainTradeTransaction>
</rsm:CrossIndustryInvoice>
XML;
// ── 4. Embed the XML as an associated file ─────────────────────────────
$pdf->addEmbeddedFile(
EmbeddedFile::create()
->setFilename('factur-x.xml')
->setMimeType('text/xml')
->setContent($xml)
->setRelationship(AFRelationship::Alternative)
->setDescription('Factur-X EN 16931 invoice data')
);
// ── 5. Set ZUGFeRD XMP metadata ────────────────────────────────────────
$pdf->getMetadata()
->setXmpProperty('fx:DocumentType', 'INVOICE')
->setXmpProperty('fx:DocumentFileName', 'factur-x.xml')
->setXmpProperty('fx:Version', '1.0')
->setXmpProperty('fx:ConformanceLevel', 'EN 16931');
// ── 6. Save ────────────────────────────────────────────────────────────
$pdf->save(__DIR__ . '/INV-2026-0042.pdf');
echo 'ZUGFeRD invoice created.' . PHP_EOL;Валидация
bash
# PDF/A conformance
verapdf --flavour 4f INV-2026-0042.pdf
# Factur-X XML schema
java -jar Mustang-CLI.jar --action validate --source INV-2026-0042.pdfTIP
Mustangproject -- это бесплатный валидатор с открытым исходным кодом для счетов ZUGFeRD/Factur-X.
Схема рабочего процесса
ERP / биллинговые данные
|
v
TCPDF-Next генерация
|
v
PDF/A-4f + встроенный factur-x.xml
|
v
(опционально) цифровая подпись PAdES
|
+-------+-------+-------+
| | | |
Email Портал PEPPOL АрхивДополнительные материалы
- Архивирование PDF/A-4 -- детали соответствия ISO 19005-4
- Процесс PAdES B-LTA -- цифровая подпись счетов