Skip to content

Commit

Permalink
initial support for rendering PDF files locally
Browse files Browse the repository at this point in the history
  • Loading branch information
toudi committed Dec 27, 2023
1 parent 9ad0cd7 commit 0bf951c
Show file tree
Hide file tree
Showing 28 changed files with 2,878 additions and 28 deletions.
1 change: 1 addition & 0 deletions .goreleaser.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,7 @@ archives:
files:
- przykladowe-pliki-wejsciowe
- klucze
- examples
- src: docs/ksef-dokumentacja-uzytkownika.pdf
dst: ksef-dokumentacja-uzytkownika.pdf

Expand Down
117 changes: 117 additions & 0 deletions cmd/ksef/commands/render-pdf.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,117 @@
package commands

import (
"crypto/sha256"
"encoding/base64"
"encoding/hex"
"flag"
"fmt"
"ksef/internal/logging"
"ksef/internal/pdf"
"ksef/internal/registry"
"ksef/internal/utils"
"os"
)

type renderPDFCommand struct {
Command
}
type renderPDFArgsType struct {
path string
resolverConfig utils.FilepathResolverConfig
invoice string
}

var RenderPDFCommand *renderPDFCommand
var renderPDFArgs renderPDFArgsType

func init() {
RenderPDFCommand = &renderPDFCommand{
Command: Command{
Name: "render-pdf",
FlagSet: flag.NewFlagSet("render-pdf", flag.ExitOnError),
Description: "drukuje PDF dla wskazanej faktury używając lokalnego szablonu",
Run: renderPDFRun,
Args: renderPDFArgs,
},
}

RenderPDFCommand.FlagSet.StringVar(
&renderPDFArgs.path,
"p",
"",
"ścieżka do pliku rejestru",
)
RenderPDFCommand.FlagSet.StringVar(
&renderPDFArgs.resolverConfig.Path,
"o",
"",
"ścieżka do zapisu PDF (domyślnie katalog pliku statusu + {nrRef}.pdf)",
)
RenderPDFCommand.FlagSet.BoolVar(
&renderPDFArgs.resolverConfig.Mkdir,
"m",
false,
"stwórz katalog, jeśli wskazany do zapisu nie istnieje",
)
RenderPDFCommand.FlagSet.StringVar(
&renderPDFArgs.invoice,
"i",
"",
"plik XML do wizualizacji",
)

registerCommand(&RenderPDFCommand.Command)
}

func renderPDFRun(c *Command) error {
if renderPDFArgs.path == "" || renderPDFArgs.invoice == "" {
RenderPDFCommand.FlagSet.Usage()
return nil
}

registry, err := registry.LoadRegistry(renderPDFArgs.path)
if err != nil {
return fmt.Errorf("unable to load registry from file: %v", err)
}

if registry.Environment == "" {
return fmt.Errorf("file deserialized correctly, but environment is empty")
}

fileContent, err := os.ReadFile(renderPDFArgs.invoice)
if err != nil {
return fmt.Errorf("nie udało się odczytać pliku źródłowego")
}

hasher := sha256.New()
hasher.Write(fileContent)
fileChecksum := hex.EncodeToString(hasher.Sum(nil))
fileBase64 := base64.StdEncoding.EncodeToString(fileContent)

logging.PDFRendererLogger.Debug("calculated checksum", "checksum", fileChecksum)

invoiceMeta := registry.GetInvoiceByChecksum(fileChecksum)
if invoiceMeta.Checksum != fileChecksum {
return fmt.Errorf("nie udało się znaleźć faktury na podstawie kryteriów wejściowych")
}

renderPDFArgs.resolverConfig.DefaultFilename = fmt.Sprintf(
"%s.pdf",
invoiceMeta.SEIReferenceNumber,
)

outputPath, err := utils.ResolveFilepath(renderPDFArgs.resolverConfig)
if err == utils.ErrDoesNotExistAndMkdirNotSpecified {
return fmt.Errorf("wskazany katalog nie istnieje a nie użyłeś opcji `-m`")
}
if err != nil {
return fmt.Errorf("błąd tworzenia katalogu wyjściowego: %v", err)
}

printingEngine, err := pdf.GetLocalPrintingEngine()
if err != nil {
return fmt.Errorf("nie udało się zainicjować silnika drukowania: %v", err)
}
return printingEngine.Print(fileBase64, invoiceMeta, outputPath)
}
57 changes: 32 additions & 25 deletions cmd/ksef/commands/status.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ import (
registryPkg "ksef/internal/registry"
"ksef/internal/sei/api/client"
"ksef/internal/sei/api/upo"
"os"
"ksef/internal/utils"
"path/filepath"
)

Expand Down Expand Up @@ -35,8 +35,18 @@ func init() {
}

StatusCommand.FlagSet.StringVar(&statusArgs.path, "p", "", "ścieżka do pliku rejestru")
StatusCommand.FlagSet.StringVar(&statusArgs.downloadUPOArgs.Output, "o", "", "ścieżka do zapisu UPO (domyślnie katalog pliku rejestru + {nrRef}.pdf)")
StatusCommand.FlagSet.BoolVar(&statusArgs.downloadUPOArgs.Mkdir, "m", false, "stwórz katalog, jeśli wskazany do zapisu nie istnieje")
StatusCommand.FlagSet.StringVar(
&statusArgs.downloadUPOArgs.Output,
"o",
"",
"ścieżka do zapisu UPO (domyślnie katalog pliku rejestru + {nrRef}.pdf)",
)
StatusCommand.FlagSet.BoolVar(
&statusArgs.downloadUPOArgs.Mkdir,
"m",
false,
"stwórz katalog, jeśli wskazany do zapisu nie istnieje",
)
StatusCommand.FlagSet.BoolVar(&statusArgs.xml, "xml", false, "zapis UPO jako plik XML")

registerCommand(&StatusCommand.Command)
Expand All @@ -54,7 +64,10 @@ func statusRun(c *Command) error {
}

if registry.Environment == "" || registry.SessionID == "" {
return fmt.Errorf("file deserialized correctly, but either environment or referenceNo are empty: %+v", registry)
return fmt.Errorf(
"file deserialized correctly, but either environment or referenceNo are empty: %+v",
registry,
)
}

statusArgs.downloadUPOArgs.OutputFormat = upo.UPOFormatPDF
Expand All @@ -67,29 +80,23 @@ func statusRun(c *Command) error {
statusArgs.downloadUPOArgs.Output = filepath.Dir(statusArgs.path)
}

// let's validate output.
// first, let's check if this is a file or a directory.
outputExt := filepath.Ext(statusArgs.downloadUPOArgs.Output)
outputPath := filepath.Dir(statusArgs.downloadUPOArgs.Output)
outputPath, err := utils.ResolveFilepath(
utils.FilepathResolverConfig{
Path: statusArgs.downloadUPOArgs.Output,
Mkdir: statusArgs.downloadUPOArgs.Mkdir,
DefaultFilename: fmt.Sprintf(
"%s.%s",
registry.SessionID,
statusArgs.downloadUPOArgs.OutputFormat,
),
},
)

if outputExt == "" {
// since there is no filename extension we have to treat the whole thing as a path.
outputPath = statusArgs.downloadUPOArgs.Output
statusArgs.downloadUPOArgs.Output = filepath.Join(outputPath, fmt.Sprintf("%s.%s", registry.SessionID, statusArgs.downloadUPOArgs.OutputFormat))
if err == utils.ErrDoesNotExistAndMkdirNotSpecified {
return fmt.Errorf("wskazany katalog nie istnieje a nie użyłeś opcji `-m`")
}

// let's validate output directory
_, err = os.Stat(outputPath)

if os.IsNotExist(err) {
// that's still fine at this point. let's check if we can create it.
if !statusArgs.downloadUPOArgs.Mkdir {
return fmt.Errorf("wskazany katalog nie istnieje a nie użyłeś opcji `-m`")
}
if err = os.MkdirAll(outputPath, 0755); err != nil {
return fmt.Errorf("błąd tworzenia katalogu wyjściowego: %v", err)
}

if err != nil {
return fmt.Errorf("błąd tworzenia katalogu wyjściowego: %v", err)
}

statusArgs.downloadUPOArgs.OutputPath = outputPath
Expand Down
5 changes: 4 additions & 1 deletion docs/.vitepress/config.mjs
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,10 @@ export default defineConfig({
{ text: 'Pobieranie faktur', link: '/content/komendy/download'},
{ text: 'Pobieranie UPO', link: '/content/komendy/upo'},
{ text: 'Identyfikator płatności', link: '/content/komendy/payment-id'},
{ text: 'Wizualizacja PDF', link: '/content/komendy/wizualizacja-pdf'},
{ text: 'Wizualizacja PDF', items: [
{text: "Wizualizacja KSeF", link: '/content/komendy/wizualizacja-pdf'},
{text: "Wizualizacja lokalna", link: '/content/komendy/lokalny-pdf'},
]},
]
}
],
Expand Down
154 changes: 154 additions & 0 deletions docs/content/komendy/lokalny-pdf.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,154 @@
# Wizualiacja PDF na podstawie własnego szablonu

::: warning
Ta funkcjonalność jest cały czas rozwijana - jestem otwarty na wszelkie sugestie
:::

Niestety, w związku z tym, że eFaktura jest plikiem XML mamy pod górkę już na starcie. Teoretycznie, aby skonwertować plik XML na HTML wystarczy użyć transformacji XSLT. Tyle tylko, że plik faktury zawiera przestrzenie nazw co powoduje, że zapytania XPath stają się niezbyt przyjemne i czytelne.

Zamiast tego wpadłem na pomysł aby do wizualizacji służyła mikroaplikacja javascriptowa która na wejściu przyjmie surowy plik XML, następnie przetworzy go na obiekt javascriptowy, wyświetli w HTML a następnie wydrukuje za pomocą puppetteer

::: warning
Zdaję sobie sprawę, że jest to wybór **niemerytoryczny** i że nie każdemu może przypaść do gustu. Dlatego zakładam, że silników drukujących mogłoby być kilka i konfigurowane byłyby w pliku konfiguracyjnym a ten który zaimplementowałem możnaby traktować jedynie jako wersję "referencyjną"
:::

::: danger
Przykładowa aplikacja uruchamia przeglądarkę chrome / chromium z parametrem `--disable-web-security` aby załadować lokalne pliki z dysku. Opcją numer dwa byłoby serwowanie plików przez program kliencki na jakimś losowym porcie - póki co po prostu nie chce mi się tego implementować. Inną opcją jest użycie programu gotenberg ale wymaga to działającego dockera
:::

## Rozumiem zagrożenie, co mam zrobić?

Zerknij na katalog `examples/local-pdf-printout`. Zawiera on dwa katalogi:

| katalog | znaczenie |
| ----------------- | --------------------------------------------------------------- |
| invoice-rendering | Zawiera **szablon aplikacji** wyświetlającej fakturę oraz QRKod |
| printing | Zawiera projekt oraz przykład skryptu drukującego |

### Instalacja bibliotek

```shell
cd invoice-rendering
npm install
cd ../printing
npm install
```

### Zmiana szablonu

Referencyjna aplikacja ma zaimplementowany szablon który na 99% nie będzie spełniać Twoich oczekiwań ale da Ci pogląd na to w jaki sposób można odnosić się do danych zawartych w fakturze, jak renderować QRKod etc etc.

Mikroaplikacja jest napisana w Vue.js. Ponownie - jest to wybór **niemerytoryczny** i przyczyna jest bardzo prosta - kod wyjściowy generowany przez Vue.js bardziej przypada mi do gustu niż inne silniki.

W mojej opinii nie ma to jednak **najmniejszego** znaczenia, ponieważ - ponownie - mój sposób jest jedynie przykładowy i równie dobrze możesz napisać aplikację w react / angularze albo svelte. Grunt, żeby działała w identyczny sposób.

Oto kilka najistotniejszych plików wraz z krótkim omówieniem

| plik | znaczenie |
| ----------------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| index.html | Podstawowy szablon. Zwróć uwagę na specyficzne wartości wpisane do sekcji `head > meta`. Na ich podstawie aplikacja będzie w stanie zwizualizować fakturę |
| src/annotate-invoice.js | Tworzenie anotacji (dodatkowych pomocniczych informacji do wyświetlenia). Zawiera funkcję która podmienia pola z XML na mnemoniki a także oblicza częściowe i całkowite kwoty netto / brutto / VAT |
| src/components/\*.vue | pliki komponentów do renderowania faktury. Jeśli kiedykolwiek robiłeś coś w vue to z pewnością dasz radę je ogarnąć |

### Sekcja `head > meta`

Jest to dość istotna kwestia związana z szablonem. Wygląda ona następująco:

```html
<meta name="invoice:qrcode" content="__qrcode_url__" />
<meta name="invoice:seiRefNo" content="__invoice_sei_ref_no__" />
<meta name="invoice" content="__invoice_base64__" />
```

Do sekcji `meta` mój program przekazuje następujące dane

| zmienna | znaczenie |
| ------------------------ | ----------------------------------------------------------- |
| `__qrcode_url__` | Link do weryfikacji faktury który znajduje się na kodzie QR |
| `__invoice_sei_ref_no__` | Numer faktury który widnieje w KSeF |
| `__invoice_base64__` | Treść pliku XML z fakturą zakodowana w base64 |

::: info
Jeśli testujesz szablon, użyj następującej komendy:

```shell
npm run dev
```

Wówczas uruchomi się serwer na porcie 5173. Oczywiście jeśli zostawisz domyślne wartości meta to szablon się wysypie. Pamiętaj więc, żeby do testów wstawić tam własne wartości. O ile link QR i numer faktury z KSeF nie mają większego znaczenia o tyle jeśli nie wpiszesz faktycznego base64 z XML'a z fakturą do `meta[name=invoice]` to .. niczego nie potestujesz :)
:::

### Budowanie wynikowego szablonu

Jeśli dostosowałeś szablon do swoich potrzeb to zbuduj jego finalną wersję:

```shell
npm run build -- --outDir=../template
```

::: warning
Podczas wizualizacji program tworzy plik `render.html` który wysyła do procesu drukującego. Teoretycznie więc jeśli będziesz mieć sporo wydruków, warto rozważyć trzymanie szablonu w ramdysku żeby nie niszczyć sobie dysku :-)
:::

## Jak powiązać to z klientem?

```text
./ksef render-pdf
Usage of render-pdf:
-i string
plik XML do wizualizacji
-m stwórz katalog, jeśli wskazany do zapisu nie istnieje
-o string
ścieżka do zapisu PDF (domyślnie katalog pliku statusu + {nrRef}.pdf)
-p string
ścieżka do pliku rejestru
```

Klient działa w następujący sposób:

1. Oblicza sumę sha256 pliku XML faktury
2. Znajduje metadane KSeF faktury w pliku rejestru
3. Koduje plik do base64
4. Bierze wskazany przez Ciebie szablon HTML a następnie wypełnia pola w `meta` odpowiednimi wartościami
5. Tworzy plik render.html (oryginalny index.html pozostaje bez zmian)
6. Renderuje PDF

Więcej o konfiguracji silników do renderowania PDF czytaj w rozdziale [Konfiguracja](/content/konfiguracja)

### Przykłady wywołań:

::: warning
Warunkiem **koniecznym** jest wskazanie lokalizacji pliku konfiguracyjnego poprzez przełącznik `-c`
:::

#### Zapisanie PDF'a w katalogu `katalog` z domyślną nazwą `{numerFakturyZKSeF}.pdf`

```shell
./ksef -c config.yaml render-pdf -i katalog/invoice.xml -o katalog -p katalog/registry.yaml
```

#### Zapisanie PDF'a w katalogu z własną nazwą

```shell
./ksef -c config.yaml render-pdf -i katalog/invoice.xml -o katalog/mojafaktura.pdf -p katalog/registry.yaml
```

#### Zapisanie PDF'a w innym katalogu z domyślną nazwą

```shell
./ksef -c config.yaml render-pdf -i katalog/invoice.xml -o inny-katalog -m -p katalog/registry.yaml
```

::: warning
zwróć uwagę na flagę `-m` - w przeciwnym wypadku jeśli `inny-katalog` nie istnieje, program zwróci błąd
:::

#### Zapisanie PDF'a w innym katalogu z własną nazwą

```shell
./ksef -c config.yaml render-pdf -i katalog/invoice.xml -o inny-katalog/mojafaktura.pdf -m -p katalog/registry.yaml
```

::: warning
zwróć uwagę na flagę `-m` - w przeciwnym wypadku jeśli `inny-katalog` nie istnieje, program zwróci błąd
:::
Loading

0 comments on commit 0bf951c

Please sign in to comment.