1. Introduction

Fugerit Venus Doc Mind Map (fj-doc)
Failed to generate image: mmdc failed: Generating single mermaid chart

mindmap
  root((Venus<br/>Doc))
    Fugerti Doc Source Format
      XML Source - default
      JSON Source
      YAML Source
      Kotlin Source
    Output Format
      PDF - fop
      HTML - freemarker
      Markdown - freemarker
      Asciidoc - freemarker
      Excel - poi
      CSV - OpenCSV
    Utilities
      Freemarker Configuration
      Auto documentation library
      Configuration conversion tool
      Maven Plugin
      Table rendering
      Format validation
      Playground
Fugerit Venus Doc Typical Flow
Failed to generate image: mmdc failed: Generating single mermaid chart

flowchart TB
    D -->|picks| H
    subgraph source
    A[Fugerit Doc Flow] -->|starts| C{Mode}
    C -->|Static data| D[Fugerit Doc Format]
    C -->|Dynamic data| E[Freemarker pipeline + Data Model] --> |Generates| D(Fugerit Doc Format)
    C -->|Scripting| G[Kotlin DSL + Data Binding] --> |Generates| D(Fugerit Doc Format *)
    end
    subgraph output
    H{Doc Handler}
    H -->|mod-fop| L[PDF] -->|Renders| F(Output Format)
    H -->|mod-poi| M[XLSX] -->|Renders| F(Output Format)
    H -->|...| N[...] -->|Renders| F(Output Format)
    end

(*) Fugerit Doc Format is the core of the Venus Doc Framework and comes into a few flavours :

Copyright

@2024 Matteo Franci - CC BY 4.0 - ATTRIBUTION 4.0 INTERNATIONAL - https://creativecommons.org/licenses/by-nc-sa/4.0/deed.en

All trademarks, logos and brand names are the property of their respective owners. All company, product and service names used in this website are for identification purposes only. Use of these names,trademarks and brands does not imply endorsement.

1.1. What is Fugerit Venus Doc?

Fugerit Venus Doc is a document generation framework, largely based on Apache FreeMarker template engine and on a custom document format (available in XML (default), JSON, YAML and Kotlin).

The source format can be rendered to different output formats through Doc Handlers. (the actual rendering often depends on other open source libraries like Apache FOP or OpenCSV).

Here is a diagram showing the usage of the framework on a typical scenario.

  1. Using XML as document source format

  2. Picking two output format (HTML and MarkDown)

Fugerit Venus Doc engine
Failed to generate image: mmdc failed: Generating single mermaid chart

flowchart TD
    T --> F
    J --> F
    I1 --> D1 --> R1
    I2 --> D2 --> R2
    F --> O --> D1
    O --> D2
    subgraph FreeMarker Template
            T["#lt;doc#gt;
            #lt;metadata#gt;...#lt;/metadata#gt;
            #lt;body#gt;
            ...
            #lt;h head-level='2'#gt;Hello ?{name}!#lt;/para#gt;
            ...
            #lt;/body#gt;
            #lt;/doc#gt;"]
            style T text-align:left
    end
    subgraph Doc Process Context
        J["...
            model.setName(#quot;Venus ♀#quot;);
            ..."]
            style J text-align:left
    end
    subgraph Output Document Selection
        I1(HTML)
        I2(MarkDown)
        I3(PDF)
    end
    subgraph Venus FreeMarker Doc Process
            F(("Apache
            FreeMarker"))
            O["#lt;doc#gt;
            #lt;metadata#gt;...#lt;/metadata#gt;
            #lt;body#gt;
            ...
            #lt;h head-level=#quot;2#quot;#gt;Hello Venus ♀!#lt;/h#gt;
            ...
            #lt;/body#gt;
            #lt;/doc#gt;"]
            style O text-align:left
            D1(("HTML
            Doc Handler"))
            D2(("MarkDown
            Doc Handler"))
    end
    subgraph HTML Document
        R1["#lt;html#gt;
        ...
        #lt;h2#gt;Hello Venus ♀!#lt;/h2#gt;
        ...
        #lt;/html#gt;"]
        style R1 text-align:left
    end
    subgraph MarkDown Document
        R2["...
        ## Hello Venus ♀!
        ..."]
        style R2 text-align:left
    end

For who is familiar with Apache FreeMarker this is basically an evolution of its approach :

Apache FreeMarker template engine
Failed to generate image: mmdc failed: Generating single mermaid chart

flowchart TD
    T --> F
    J --> F
    F --> O
    subgraph Template
            T["#lt;html#gt;
            ...
            Hello ?{name}!
            ...
            #lt;/html#gt;"]
            style T text-align:left
    end
    subgraph Java Objects
        J["...
            model.setName(#quot;World#quot;);
            ..."]
            style J text-align:left
    end
    F(("Apache
    FreeMarker"))
    subgraph Output
        O["#lt;html#gt;
        ...
        Hello World!
        ...
        #lt;/html#gt;"]
        style O text-align:left
    end
This diagram is a reproduction of the one in What is Apache FreeMarker? page.

1.2. Release Notes

Whereas the CHANGELOG is a detailed list of modification to project, the release notes just refers to most important changes.

1.2.1. Version 8.11.2 [2024-11-21]

1.2.2. Version 8.11.1 [2024-11-19]

  • Fixed ImageValidator exception handling #262

1.2.3. Version 8.11.0 [2024-11-19]

  • DocValidatorTypeCheck facade to check file type #260

  • Check the inner type on P7MContentValidator type #260

1.2.4. Version 8.10.9 [2024-11-03]

  • [fj-doc-maven-plugin] fix version check for AsciiDoc example

  • Native support minor fixes

1.2.5. Version 8.10.8 [2024-11-02]

  • Native metadata for module fj-doc-mode-opencsv (and relevant tests)

1.2.6. Version 8.10.7 [2024-11-02]

  • Native metadata for fj-doc-base and fj-doc-freemarker are sorted in a stable way.

  • Added test of native modules against built executable.

1.2.7. Version 8.10.6 [2024-11-01]

1.2.8. Version 8.10.5 [2024-11-01]

1.3. GraalVM Ahead Of Time (AOT) Support

The full GraalVM support is currently under development.

Doc handlers with native support are marked as native ready.

Currently native ready modules can be found here.

There is a test project fj-doc-native-quarkus
and a GitHub workflow.

Most metadata have been generated from configuration using native-helper-maven-plugin.

2. Quickstart

This section cover the creation of a new maven project, configured for Venus Doc.

2.1. New maven project

Just run the org.fugerit.java:fj-doc-maven-plugin:init plugin :

mvn org.fugerit.java:fj-doc-maven-plugin:init \
-DgroupId=org.example.doc \
-DartifactId=fugerit-demo \
-Dflavour=quarkus-3 \
-Dextensions=base,freemarker,mod-fop

And you will get a project folder named after the artifactId.
See the README.md in the project folder for further infos.

In case of 'quarkus-3' flavour, for instance, simply :

cd fugerit-demo
mvn quarkus:dev

Open the Open swagger UI to test available services.

See the Maven Plugin Goal init for further options.

2.2. Existing maven project

Just run the org.fugerit.java:fj-doc-maven-plugin:add plugin inside the folder containing your pom :

mvn org.fugerit.java:fj-doc-maven-plugin:add \
-Dextensions=base,freemarker,mod-fop

Dependencies, project configuration and an example main will be added to the project.

See the Maven Plugin Goal Add for further options.

3. Maven Plugin

Here is the Fugerit Doc Maven Plugin Documentation.

Currently, three goals are available :

goal since description

add

8.6.0

add Venus Doc Configuration to an existing project.

init

8.7.2

create a new project already configured (actually an extension to 'add' goal); a UI is available on the playground too.

verify

8.7.0

verify the templates in a FreeMarker configuration (folder), note: it can be used on any Apache FreeMarker configuration, not only Fugerit Venus Doc.

3.1. Goal 'add'

This goal add Fugerit Venus Doc configuration to an existing maven project.

Simply run the plugin in the folder containing your project POM :

mvn org.fugerit.java:fj-doc-maven-plugin:add \
-Dextensions=base,freemarker,mod-fop

3.1.1. Goal 'add' available parameters

parameter required default description

version

true

latest stable

fj-doc version to add to the project (i.e. '8.7.5'), advice: keep the default unless a specific version is strictly needed.

extensions

true

base,freemarker

List of fj-doc core modules to add (*)

projectFolder

true

.

Maven project base folder

addDocFacade

true

true

If true, a stub doc configuration helper will be created

force

false

false

Will force project setup even if fj-doc already configured (warning: can overwrite configuration)

excludeXmlApis

false

false

It will exclude dependency xml-apis:xml-apis

addExclusions

false

Add comma separated exclusion, for instance : xml-apis:xml-apis,${groupId}:${artificatId}

addVerifyPlugin

true

true

If set to true, it will configure the 'verify' goal on the project

addJunit5

true

true

If set to true, it will add junit5 (test scope) and basic test

addLombok

true

true

If set to true, it will add lombok (provided scope) and slf4j-simple (test scope)

addDependencyOnTop

true

false

If set to true, added dependencies will be added before existing ones

freemarkerVersion

true

2.3.32

Freemarker compatibility version (max 2.3.33)

3.1.2. Available extensions.

short name full name type handler description notes

base

fj-doc-base

md

library base, xml as format for document template

native ready (1)

base-json

fj-doc-base-json

add support to use json documents as format for document template

base-yaml

fj-doc-base-yaml

add support to use yaml documents as format for document template

base-kotlin

fj-doc-base-kotlin

add support to use Kotlin DSL script (KTS) as format for document template

freemarker

fj-doc-freemarker

html

Template and configuration functionalities based on [Apache FreeMarker](https://freemarker.apache.org/)

native ready (1)

mod-fop

fj-doc-mod-fop

fo, pdf

Type handler based on [Apache FOP](https://xmlgraphics.apache.org/fop/)

mod-poi

fj-doc-mod-poi

xls, xlsx

Type handler based on [Apache POI](https://poi.apache.org/)

mod-opencsv

fj-doc-mod-opencsv

csv

Type handler based on [OpenCSV](https://opencsv.sourceforge.net/)

native ready (1)

mod-openpdf-ext

fj-doc-mod-openpdf-ext

pdf

Type handler based on [OpenPDF](https://github.com/LibrePDF/OpenPDF)

mod-openrtf-ext

fj-doc-mod-openrtf-ext

rtf

Type handler based on [OpenRTF](https://github.com/LibrePDF/OpenRTF)

(1) Native Ready means that the GraalVM metadata are included in the release.

3.2. Goal 'init'

Create a new project and add Venus Doc Configuration to it.

mvn org.fugerit.java:fj-doc-maven-plugin:init \
-DgroupId=org.example.doc \
-DartifactId=fugerit-demo \
-Dextensions=base,freemarker,mod-fop

Project folder will be ./${artifactId}/.

3.2.1. Goal 'init' available parameters

parameter required default description

groupId

true

new project group id

artifactId

true

new project artifact id

projectVersion

true

1.0.0-SNAPSHOT

new project version

javaRelease

true

21

java release version

flavour

true

vanilla

the flavour for the new project (see below for options)

flavourVersion

false

see below

override default framework version if supported (recommended : leave default or blank)

it is possible to set any property from 'add' goal, except 'projectFolder' which is set to ./${artifactId}.

3.3. Goal 'verify'

verify the templates in a FreeMarker configuration (folder), note: it can be used on any Apache FreeMarker configuration, not only Fugerit Venus Doc.

3.3.1. Verify at command line

mvn org.fugerit.java:fj-doc-maven-plugin:verify -DtemplateBasePath=./src/test/resources/fj_doc_test/template-fail

3.3.2. Verify at maven build time

  <plugin>
    <groupId>org.fugerit.java</groupId>
    <artifactId>fj-doc-maven-plugin</artifactId>
    <version>${fj-doc-version}</version>
    <executions>
      <execution>
        <id>freemarker-verify</id>
        <phase>compile</phase>
        <goals>
          <goal>verify</goal>
        </goals>
      </execution>
    </executions>
    <configuration>
      <!-- Where the FreeMarker templates are located -->
      <templateBasePath>${project.basedir}/src/main/resources/fugerit-blank/template</templateBasePath>
      <!-- WARNING: if set to 'true', build will fail when at least one syntax error is found -->
      <failOnErrors>true</failOnErrors>
      <!-- If 'true' a report will be generated (when 'true', param reportOutputFolder is required) -->
      <generateReport>true</generateReport>
      <!-- Template syntax verify report output folder -->
      <reportOutputFolder>${project.build.directory}/freemarker-syntax-verify-report</reportOutputFolder>
    </configuration>
  </plugin>

3.3.3. Goal 'verify' available parameters

parameter required default description

templateBasePath

true

Path to base folder containing FreeMarker templates

freemarkerVersion

false

latest stable

FreeMarker configuration ( i.e. 2.3.33)

templateFilePattern

false

Filter on templates to be checked, regex on filename, i.e. ".{0,}[.]ftl"

failOnErrors

true

true

If set to true the build will fail when template syntax errors will be found, otherwise errors will be only logged

generateReport

false

false

If set to true a report will be generated (and property 'reportOutputFolder' will be olso required).

reportOutputFolder

false

Output folder for the generated report.

reportOutputFormat

false

'html'

Output format for the generated report, supported : html (default), pdf, csv, xlsx, md.

4. The Doc Source Format

The Doc Source Format is the basic of Fugerit Venus Doc library.

4.1. XML Source Format

When writing a sample Venus Document, it is possible to draw from some online resources :

  1. Venus DOC XML Schema Definition - Current version of the Venus DOC XSD, the main source for writing valid Venus document meta model.

  2. Venus DOC XML Reference - the informations contained in the previous XSD in HTML format for convenience.

  3. Venus DOC meta informations reference - documentations for existing 'info' properties for 'metadata' section.

  4. Online Playground - To test how the XML elements are rendered to documents by DocHandlers.

Supported is provided by the base module dependency :

<dependency>
    <groupId>org.fugerit.java</groupId>
    <artifactId>fj-doc-base</artifactId>
    <version>${fj-doc-version}</version>
</dependency>

Here is an example xml :

<?xml version="1.0" encoding="utf-8"?>
<doc
	xmlns="http://javacoredoc.fugerit.org"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://javacoredoc.fugerit.org https://www.fugerit.org/data/java/doc/xsd/doc-2-1.xsd" >

  <!--
  	This is a Venus Fugerit Doc (https://github.com/fugerit-org/fj-doc) XML Source Document.
  	For documentation on how to write a valid Venus Doc XML Meta Model refer to :
  	https://venusdocs.fugerit.org/guide/#doc-format-entry-point
  -->

  <metadata>
	<!-- Margin for document : left;right;top;bottom -->
	<info name="margins">10;10;10;30</info>
	<!-- documenta meta information -->
	<info name="doc-title">Hello World</info>
	<info name="doc-author">fugerit79</info>
	<info name="doc-language">en</info>
  </metadata>
  <body>
	<para>Hello World!</para>
  </body>

</doc>

4.2. JSON and YAML Source Format

It is possible to use JSON or YAML as source. It is just needed to follow some conversion rules.

The conversion rules from xml to json/yaml are :

  1. xml meta information are translated as top level property (xmlns etc.)

  2. Every JSON/YAML object contains the information of a XML tag

  3. "_t" special property contains the tag name (for example "_t" : "metadata")

  4. "_v" special property contains the text content of an element

  5. "_e" special property contains a list of child elements

  6. any other xml attribute is mapped as a JSON/YAML property (for example "name" : "margins")

As it is possible to directly convert JSON and YAML, rules for YAML are the same as for JSON format.
All XML comments are ignored
For XML/JSON/YAML conversion a Online Playground is available.

Support for JSON format needs the following dependency :

<dependency>
    <groupId>org.fugerit.java</groupId>
    <artifactId>fj-doc-base-json</artifactId>
    <version>${fj-doc-version}</version>
</dependency>

Here is a simple JSON source document :

{
	"xmlns" : "http://javacoredoc.fugerit.org",
	"xsi:schemaLocation" : "http://javacoredoc.fugerit.org https://www.fugerit.org/data/java/doc/xsd/doc-2-0.xsd",
	"xmlns:xsi" : "http://www.w3.org/2001/XMLSchema-instance",
	"_t" : "doc",
	"_e" : [ {
		"_t" : "metadata",
		"_e" : [ {
			"name" : "margins",
			"_t" : "info",
			"_v" : "10;10;10;30"
		}, {
			"name" : "doc-title",
			"_t" : "info",
			"_v" : "Hello World"
		}, {
			"name" : "doc-author",
			"_t" : "info",
			"_v" : "fugerit79"
		}, {
			"name" : "doc-language",
			"_t" : "info",
			"_v" : "en"
		} ]
	}, {
		"_t" : "body",
		"_e" : [ {
			"_t" : "para",
			"_v" : "Hello World!"
		} ]
	} ]
}

Support for YAML format needs the following dependency :

<dependency>
    <groupId>org.fugerit.java</groupId>
    <artifactId>fj-doc-base-yaml</artifactId>
    <version>${fj-doc-version}</version>
</dependency>

Here is the equivalent YAML :

---
xmlns: "http://javacoredoc.fugerit.org"
xsi:schemaLocation: "http://javacoredoc.fugerit.org https://www.fugerit.org/data/java/doc/xsd/doc-2-1.xsd"
xmlns:xsi: "http://www.w3.org/2001/XMLSchema-instance"
_t: "doc"
_e:
- _t: "metadata"
  _e:
  - name: "margins"
    _t: "info"
    _v: "10;10;10;30"
  - name: "doc-title"
    _t: "info"
    _v: "Basic example"
  - name: "doc-subject"
    _t: "info"
    _v: "fj doc venus sample source xml"
  - name: "doc-author"
    _t: "info"
    _v: "fugerit79"
  - name: "doc-language"
    _t: "info"
    _v: "en"
  - name: "default-font-name"
    _t: "info"
    _v: "TitilliumWeb"
  - name: "excel-table-id"
    _t: "info"
    _v: "data-table=print"
  - name: "csv-table-id"
    _t: "info"
    _v: "data-table"
  - _t: "footer-ext"
    _e:
    - align: "right"
      _t: "para"
      _v: "${currentPage} / ${pageCount}"
- _t: "body"
  _e:
  - _t: "para"
    _v: "My sample title"
  - padding: "2"
    columns: "3"
    width: "100"
    id: "data-table"
    colwidths: "30;30;40"
    _t: "table"
    _e:
    - _t: "row"
      _e:
      - border-color: "#000000"
        border-width: "1"
        align: "center"
        _t: "cell"
        _e:
        - style: "bold"
          _t: "para"
          _v: "Name"
      - align: "center"
        _t: "cell"
        _e:
        - style: "bold"
          _t: "para"
          _v: "Surname"
      - align: "center"
        _t: "cell"
        _e:
        - style: "bold"
          _t: "para"
          _v: "Title"
    - _t: "row"
      _e:
      - _t: "cell"
        _e:
        - _t: "para"
          _v: "Luthien"
      - _t: "cell"
        _e:
        - _t: "para"
          _v: "Tinuviel"
      - _t: "cell"
        _e:
        - _t: "para"
          _v: "Queen"
    - _t: "row"
      _e:
      - _t: "cell"
        _e:
        - _t: "para"
          _v: "Thorin"
      - _t: "cell"
        _e:
        - _t: "para"
          _v: "Oakshield"
      - _t: "cell"
        _e:
        - _t: "para"
          _v: "King"

4.3. Kotlin Source Format (Experimental)

Supported is provided by the base module dependency.

<dependency>
    <groupId>org.fugerit.java</groupId>
    <artifactId>fj-doc-base-kotlin</artifactId>
    <version>${fj-doc-version}</version>
</dependency>
At the moment kotlin (KTS) support is still experimental.
When adding this extension, all the kotlin runtime must be included too.

Kotlin source is based on a custom DSL (Domain Specific Language).
The Fugerit Doc Kotlin DSL is basically a mapping of the xsd.

Here is a sample kotlin source document :

import org.fugerit.java.doc.base.kotlin.dsl.dslDoc

dslDoc {
    val docTitle = attStr( data, "docTitle" )
    meta {
        info( ( docTitle ) ).name( "doc-title" )
        info( ( "10;10;10;30" ) ).name( "margins" )
        info( ( "fj doc venus sample source Kotlin Template - kts" ) ).name( "doc-subject" )
        info( ( "fugerit79" ) ).name( "dock-author" )
        info( ( "en" ) ).name( "doc-language" )
        info( ( "TitilliumWeb" ) ).name( "default-font-name" )
        info( ( "data-table=print" ) ).name( "excel-table-id" )
        info( ( "data-table" ) ).name( "csv-table-id" )
        footerExt {
            para( '$'+"{currentPage} / "+'$'+"{pageCount}" ).align( "right" )
        }
    }
    body {
        h( docTitle ).headLevel( 1 )
        table {
            row {
                cell { para( "Name" ) }.align( "center" )
                cell { para( "Surname" ) }.align( "center" )
                cell { para( "Title" ) }.align( "center" )
            }.header( true )
            attListMap( data, "listPeople" ).forEach( { e -> row {
                cell { para( attStr( e, "name" ) ) }
                cell { para( attStr( e, "surname" ) ) }
                cell { para( attStr( e, "title" ) ) }
            } } )
        }.columns( 3 ).colwidths( "30;30;40" ).width( 100 ).id( "data-table" ).padding( 2 )
    }
}

4.3.1. DocKotlinParser

The DocKotlinParser provides a utility to eval the kotlin script with data model :

try ( Reader ktsReader = ... ) {
    Map<String,Object> dataModel = ...
    String xml = DocKotlinParser.dslDocToXml( ktsReader, dataModel );
}

And when using DocFacadeSource it is available with the source type SOURCE_TYPE_KOTLIN (value:9).

4.3.2. FreeMarkerKotlinStep

A new FreeMarkerKotlinStep is also available inside the FreemarkerDocProcessConfig

4.4. Doc Format Info Element

The info element is available to setup some specific properties of the document.

For instance in this document we set following info elements :

  • margin

  • doc-title

  • doc-author

  • doc-language

Some DocHandler will ignore some info elements.

<?xml version="1.0" encoding="utf-8"?>
<doc
	xmlns="http://javacoredoc.fugerit.org"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://javacoredoc.fugerit.org https://www.fugerit.org/data/java/doc/xsd/doc-2-1.xsd" >

  <metadata>
	<!-- Margin for document : left;right;top;bottom -->
	<info name="margins">10;10;10;30</info>
	<!-- documenta meta information -->
	<info name="doc-title">Hello World</info>
	<info name="doc-author">fugerit79</info>
	<info name="doc-language">en</info>
  </metadata>
  <body>
	<para>Hello World!</para>
  </body>

</doc>

5. Dynamic Data : Freemarker

In a typical workflow, freemarker will be used to generate the Venus Doc Format.

Fugerit Venus Doc - Dynamic Document Generation
Failed to generate image: mmdc failed: Generating single mermaid chart

stateDiagram-v2
    state "FreeMarker template" as s1
    state "DataModel" as s2
    state "FreemarkerDocProcessConfig" as s3
    state "Venus Doc Format" as s4
    state "Output Format" as s5
    s1 --> s3: Input
    s2 --> s3: Input
    s3 --> s4: Creates
    s4 --> s5: Renders

This is a simple tutorial for Fugerit Venus Doc full usage.

This tutorial is based on a quarkus project available at :
tutorial repository git.

The project has been created with the command :

mvn org.fugerit.java:fj-doc-maven-plugin:init \
-DgroupId=org.fugerit.java-tutorial \
-DartifactId=fj-doc-quarkus-tutorial \
-Dflavour=quarkus-3 \
-Dextensions=base,freemarker,mod-fop,mod-poi

5.1. FreemarkerDocProcessConfig configuration

The main configuration file is fm-doc-process-config.xml.

Here you can find :

5.1.1. Root element freemarker-doc-process-config

This is the root configuration element :

<freemarker-doc-process-config
    xmlns="https://freemarkerdocprocess.fugerit.org"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="https://freemarkerdocprocess.fugerit.org https://www.fugerit.org/data/java/doc/xsd/freemarker-doc-process-1-0.xsd"
    validating="true"
    failOnValidate="true"
    cleanSource="true">

Aside from namespace declaration, it is possible to set some general attribute for the configuration :

name type default description

validating

boolean

false

If set to 'true' the FreemarkerDocProcessConfig will try to validate the source. (since 8.9.1), NOTE: if active, source reader will be buffered, potentially resulting in higher memory usage. See also FAQ.

failOnValidate

boolean

false

If set to 'true' the FreemarkerDocProcessConfig will fail in case of validation errors, if 'false' will just print the result as warning. (since 8.9.1), NOTE: 'validating' is set to true, this attribute is ignored. See also FAQ

cleanSource

boolean

false

If set to 'true' the FreemarkerDocProcessConfig will try to clean the source (i.e. DocXmlUtils.cleanXml()). (since 8.9.1), NOTE: if active, source reader will be buffered, potentially resulting in higher memory usage. See also FAQ

5.1.2. Element docHandlerConfig

The main elements are the <docHandlerConfig/> containing the available output format doc handlers.

    <docHandlerConfig registerById="true">
        <!-- Type handler for markdown format -->
        <docHandler id="md-ext" info="md" type="org.fugerit.java.doc.base.typehandler.markdown.SimpleMarkdownExtTypeHandler" />
        <!-- Type henalder for xml format, generates the source xml:doc -->
        <docHandler id="xml-doc" info="xml" type="org.fugerit.java.doc.base.config.DocTypeHandlerXMLUTF8" />
        <!-- Type handler for html using freemarker -->
        <docHandler id="html-fm" info="html" type="org.fugerit.java.doc.freemarker.html.FreeMarkerHtmlTypeHandlerEscapeUTF8" />
        <!-- Type handler for html using freemarker (fragment version, only generates body content no html or head part -->
        <docHandler id="html-fragment-fm" info="fhtml" type="org.fugerit.java.doc.freemarker.html.FreeMarkerHtmlFragmentTypeHandlerEscapeUTF8" />
        <!-- Type handler generating xls:fo style sheet -->
        <docHandler id="fo-fop" info="fo" type="org.fugerit.java.doc.mod.fop.FreeMarkerFopTypeHandlerUTF8" />
        <!-- Type handler generating pdf -->
        <docHandler id="pdf-fop" info="pdf" type="org.fugerit.java.doc.mod.fop.PdfFopTypeHandler">
            <docHandlerCustomConfig charset="UTF-8" fop-config-mode="classloader" fop-config-classloader-path="fj-doc-quarkus-tutorial/fop-config.xml" fop-suppress-events="1"/>
        </docHandler>
        <!-- Type handler generating xls -->
        <docHandler id="xls-poi" info="xls" type="org.fugerit.java.doc.mod.poi.XlsPoiTypeHandler" />
        <!-- Type handler generating xlsx -->
        <docHandler id="xlsx-poi" info="xlsx" type="org.fugerit.java.doc.mod.poi.XlsxPoiTypeHandler" />
    </docHandlerConfig>

5.1.3. Element docChain

And the <docChain/> elements containing the configuration for the freemarker template and data model.

Usually you will need at least one configuration step :

    <docChain id="shared">
        <chainStep stepType="config">
            <config
                    id="fj_doc_config_fm_fjdocquarkustutorial"
                    class="org.fugerit.java.tutorial.fjdocquarkustutorial.DocHelper"
                    exception-handler="RETHROW_HANDLER"
                    fallback-on-null-loop-variable="false"
                    log-exception="false"
                    mode="class"
                    path="/fj-doc-quarkus-tutorial/template/"
                    version="2.3.32"
                    wrap-unchecked-exceptions="true"
                    load-bundled-functions="true"
            />
        </chainStep>
    </docChain>

And one or more document process step :

    <!-- example document chain -->
    <docChain id="document" parent="shared">
        <chainStep stepType="complex" map-atts="listPeople" template-path="${chainId}.ftl"/>
    </docChain>

5.1.4. Built-in step types

name type description

config

org.fugerit.java.doc.freemarker.config.​FreeMarkerConfigStep

responsable for freemarker and venus configuration

function

org.fugerit.java.doc.freemarker.config.​FreeMarkerFunctionStep

add freemarker functions to the data model (some built-in functions are available)

complex

org.fugerit.java.doc.freemarker.config.​FreeMarkerComplexProcessStep

it apply the freemarker template and render the output

map

org.fugerit.java.doc.freemarker.config.​FreeMarkerMapStep

Venus data model to Freemarker data model mapping

skipfm

org.fugerit.java.doc.freemarker.config.​FreeMarkerSkipProcessStep

When using this step, freemarker apply template will be skipped. (Since 8.9.7)

kotlin

org.fugerit.java.doc.freemarker.config.​FreeMarkerKotlinStep

When using this step, freemarker apply template will be skipped. It uses kotlin script, kts-path must be set, dataModel map can still be used and will be bound to the script (Since 8.10.0)

Additional step can be added setting the fully qualified class name in the type attribute.
When using skipfm no FreeMarker template syntax should be used in the template.

5.1.5. Built-in functions

name type description parameters

imageBase64CLFun

org.fugerit.java.doc.freemarker.fun.​ImageBase64CLFun

tries to load from classpath an image and converts it to base64

(1) - classloader path of the image to convert

textWrap

org.fugerit.java.doc.freemarker.fun.​TextWrapFun

convert a text with &#8203; characters (useful for long words on pdf)

(1) - the text to wrap with &#8203;

messageFormat

org.fugerit.java.doc.freemarker.fun.​SimpleMessageFun

formats text with stardard MessageFormat

(1) - the pattern, (2+) - parameters for message format

sumLong

org.fugerit.java.doc.freemarker.fun.​SimpleSumLongFun

sums numbers

(1+) - numbers to sum

cleanXml

org.fugerit.java.doc.freemarker.fun.​CleanXmlFun

cleans invalid xml characters with the regex [^\u0009\u000A\u000D\u0020-\uD7FF\uE000-\uFFFD\u10000-\u10FFF]+

(1) - the text to clean

cleanText

org.fugerit.java.doc.freemarker.fun.​CleanTextFun

cleans text with given regex

(1) - the text to clean, (2) - regex for the pattern to remove

formatDateTime

org.fugerit.java.doc.freemarker.fun.​FormatLocalDateTimeFun

formats a LocalDate, LocalTime or LocalDateTime

(1) - the date/time to format, (2) - the format pattern

These functions can all be loaded at once with the config step attribute load-bundled-functions="true".

5.2. Venus Freemarker Chain

With a venus freemarker chain, we use an Apache FreeMarker template (ftl) :

<?xml version="1.0" encoding="utf-8"?>
<doc
xmlns="http://javacoredoc.fugerit.org"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://javacoredoc.fugerit.org https://www.fugerit.org/data/java/doc/xsd/doc-2-1.xsd" >

    <#--
        This is a Venus Fugerit Doc (https://github.com/fugerit-org/fj-doc) FreeMarker Template XML (ftl[x]).
        For consideration of Venus Fugerit Doc and Apache FreeMarker integration see :
        https://venusdocs.fugerit.org/guide/#doc-freemarker-entry-point
        The result will be a :
    -->
    <!--
        This is a Venus Fugerit Doc (https://github.com/fugerit-org/fj-doc) XML Source Document.
        For documentation on how to write a valid Venus Doc XML Meta Model refer to :
        https://venusdocs.fugerit.org/guide/#doc-format-entry-point
    -->

    <#assign defaultTitle="My sample title">

    <metadata>
        <!-- Margin for document : left;right;top;bottom -->
        <info name="margins">10;10;10;30</info>
        <!-- documenta meta information -->
        <info name="doc-title">${docTitle!defaultTitle}</info>
        <info name="doc-subject">fj doc venus sample source FreeMarker Template XML - ftlx</info>
        <info name="doc-author">fugerit79</info>
        <info name="doc-language">en</info>
        <!-- property specific for xls/xlsx -->
        <info name="excel-table-id">data-table=print</info>
        <!-- property specific for csv -->
        <info name="csv-table-id">data-table</info>
        <footer-ext>
            <para align="right">${r"${currentPage}"} / ${r"${pageCount}"}</para>
        </footer-ext>
    </metadata>
    <body>
    <para>${docTitle!defaultTitle}</para>
    <table columns="3" colwidths="30;30;40"  width="100" id="data-table" padding="2">
        <row header="true">
            <cell align="center"><para>Name</para></cell>
            <cell align="center"><para>Surname</para></cell>
            <cell align="center"><para>Title</para></cell>
        </row>
        <#if listPeople??>
            <#list listPeople as current>
                <row>
                    <cell><para>${current.name}</para></cell>
                    <cell><para>${current.surname}</para></cell>
                    <cell><para>${current.title}</para></cell>
                </row>
            </#list>
        </#if>
    </table>
    </body>

</doc>

Adding to it the data model, and applying the template :

DocHelper docHelper = new DocHelper();
// create custom data for the fremarker template 'document.ftl'
List<People> listPeople = Arrays.asList(new People("Luthien", "Tinuviel", "Queen"), new People("Thorin", "Oakshield", "King"));
// output generation
docHelper.getDocProcessConfig().fullProcess("document", DocProcessContext.newContext("listPeople", listPeople), handlerId, baos);

The Venus Freemarker Template and the data model will produce a Venus Doc Format document source :

<?xml version="1.0" encoding="utf-8"?>
<doc
xmlns="http://javacoredoc.fugerit.org"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://javacoredoc.fugerit.org https://www.fugerit.org/data/java/doc/xsd/doc-2-1.xsd" >

        <!--
        This is a Venus Fugerit Doc (https://github.com/fugerit-org/fj-doc) XML Source Document.
        For documentation on how to write a valid Venus Doc XML Meta Model refer to :
        https://venusdocs.fugerit.org/guide/#doc-format-entry-point
    -->


    <metadata>
        <!-- Margin for document : left;right;top;bottom -->
        <info name="margins">10;10;10;30</info>
        <!-- documenta meta information -->
        <info name="doc-title">My sample title</info>
        <info name="doc-subject">fj doc venus sample source FreeMarker Template XML - ftlx</info>
        <info name="doc-author">fugerit79</info>
        <info name="doc-language">en</info>
        <!-- property specific for xls/xlsx -->
        <info name="excel-table-id">data-table=print</info>
        <!-- property specific for csv -->
        <info name="csv-table-id">data-table</info>
        <footer-ext>
            <para align="right">${currentPage} / ${pageCount}</para>
        </footer-ext>
    </metadata>
    <body>
    <para>My sample title</para>
    <table columns="3" colwidths="30;30;40"  width="100" id="data-table" padding="2">
        <row header="true">
            <cell align="center"><para>Name</para></cell>
            <cell align="center"><para>Surname</para></cell>
            <cell align="center"><para>Title</para></cell>
        </row>
                <row>
                    <cell><para>Luthien</para></cell>
                    <cell><para>Tinuviel</para></cell>
                    <cell><para>Queen</para></cell>
                </row>
                <row>
                    <cell><para>Thorin</para></cell>
                    <cell><para>Oakshield</para></cell>
                    <cell><para>King</para></cell>
                </row>
    </table>
    </body>

</doc>

And pass it to the chosen DocHandler in order to have the desired output format, for instance the markdown version :

My sample title

| Name | Surname | Title  |
|---------------|---------------|---------------|
| Luthien | Tinuviel | Queen  |
| Thorin | Oakshield | King  |

5.3. Venus Freemarker Usage

Adding everything together, we need a facade to read the configuration and process the Freemarker template chains :

package org.fugerit.java.tutorial.fjdocquarkustutorial;

import org.fugerit.java.doc.freemarker.process.FreemarkerDocProcessConfig;
import org.fugerit.java.doc.freemarker.process.FreemarkerDocProcessConfigFacade;

/**
* DocHelper, version : auto generated on 2024-09-25 23:53:33.687
*/
public class DocHelper {

     private FreemarkerDocProcessConfig docProcessConfig = FreemarkerDocProcessConfigFacade.loadConfigSafe( "cl://fj-doc-quarkus-tutorial/fm-doc-process-config.xml" );

     public FreemarkerDocProcessConfig getDocProcessConfig() { return this.docProcessConfig; }

}

And for instance a rest serive to expose the api :

package org.fugerit.java.tutorial.fjdocquarkustutorial;

import jakarta.ws.rs.GET;
import jakarta.ws.rs.Path;
import jakarta.ws.rs.Produces;
import jakarta.ws.rs.WebApplicationException;
import jakarta.ws.rs.core.MediaType;
import lombok.extern.slf4j.Slf4j;
import org.fugerit.java.doc.base.config.DocConfig;
import org.fugerit.java.doc.base.process.DocProcessContext;

import java.io.ByteArrayOutputStream;
import java.util.Arrays;
import java.util.List;

import org.eclipse.microprofile.openapi.annotations.Operation;
import org.eclipse.microprofile.openapi.annotations.responses.APIResponse;
import org.eclipse.microprofile.openapi.annotations.tags.Tag;
import org.eclipse.microprofile.openapi.annotations.tags.Tags;

@Slf4j
@Path("/doc")
public class DocResource {

    byte[] processDocument(String handlerId) {
        try (ByteArrayOutputStream baos = new ByteArrayOutputStream()) {
            // creates the doc helper
            DocHelper docHelper = new DocHelper();
            // create custom data for the fremarker template 'document.ftl'
            List<People> listPeople = Arrays.asList(new People("Luthien", "Tinuviel", "Queen"), new People("Thorin", "Oakshield", "King"));
            // output generation
            docHelper.getDocProcessConfig().fullProcess("document", DocProcessContext.newContext("listPeople", listPeople), handlerId, baos);
            // return the output
            return baos.toByteArray();
        } catch (Exception e) {
            String message = String.format("Error processing %s, error:%s", handlerId, e);
            log.error(message, e);
            throw new WebApplicationException(message, e);
        }
    }

    @APIResponse(responseCode = "200", description = "The Markdown document content" )
    @APIResponse(responseCode = "500", description = "In case of an unexpected error" )
    @Tags( { @Tag( name = "document" ), @Tag( name = "markdown" ) } )
    @Operation( operationId = "MarkdownExample", summary = "Example Markdown generation",
        description =  "Generates an example Markdown document using Fugerit Venus Doc handler" )
    @GET
    @Produces("text/markdown")
    @Path("/example.md")
    public byte[] markdownExample() {
        return processDocument(DocConfig.TYPE_MD);
    }

    @APIResponse(responseCode = "200", description = "The HTML document content" )
    @APIResponse(responseCode = "500", description = "In case of an unexpected error" )
    @Tags( { @Tag( name = "document" ), @Tag( name = "html" ) } )
    @Operation( operationId = "HTMLExample", summary = "Example HTML generation",
        description =  "Generates an example HTML document using Fugerit Venus Doc handler" )
    @GET
    @Produces("text/html")
    @Path("/example.html")
    public byte[] htmlExample() {
        return processDocument(DocConfig.TYPE_HTML);
    }

    @APIResponse(responseCode = "200", description = "The PDF document content" )
    @APIResponse(responseCode = "500", description = "In case of an unexpected error" )
    @Tags( { @Tag( name = "document" ), @Tag( name = "pdf" ) } )
    @Operation( operationId = "PDFExample", summary = "Example PDF generation",
        description =  "Generates an example PDF document using Fugerit Venus Doc handler" )
    @GET
    @Produces("application/pdf")
    @Path("/example.pdf")
    public byte[] pdfExample() {
        return processDocument(DocConfig.TYPE_PDF);
    }

    @APIResponse(responseCode = "200", description = "The Excel document content" )
    @APIResponse(responseCode = "500", description = "In case of an unexpected error" )
    @Tags( { @Tag( name = "document" ), @Tag( name = "excel" ) } )
    @Operation( operationId = "ExcelExample", summary = "Example Excel generation",
        description =  "Generates an example Excel document using Fugerit Venus Doc handler" )
    @GET
    @Produces("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet")
    @Path("/example.xlsx")
    public byte[] excelExample() {
        return processDocument(DocConfig.TYPE_XLSX);
    }

}

6. Doc Handlers

Doc Handler are modules responsible for rendering the output format.

Fugerit Venus Doc - Doc Handlers
Failed to generate image: mmdc failed: Generating single mermaid chart

flowchart TB
    D -->|picks| H
    D(Fugerit Doc Format)
    subgraph output
    H{Doc Handler}
    H -->|mod-fop| L[PDF] -->|Renders| F(Output Format)
    H -->|mod-poi| M[XLSX] -->|Renders| F(Output Format)
    H -->|...| N[...] -->|Renders| F(Output Format)
    end

6.1. DocHandler Module Index

doc-handler module type description notes

org.fugerit.java.doc.base.typehandler.markdown.​SimpleMarkdownExtTypeHandlerNoCommentsUTF8

fj-doc-base

MD

Renders a MarkDown (Extended) document.

native ready (1)

org.fugerit.java.doc.base.typehandler.markdown.​SimpleMarkdownBasicTypeHandlerNoCommentsUTF8

fj-doc-base

MD

Renders a MarkDown (Basic) document (tables are rendered as HTML).

native ready (1)

org.fugerit.java.doc.freemarker.html.​FreeMarkerHtmlTypeHandlerEscapeUTF8

fj-doc-freemarker

HTML

Renders HTML documents using Apache FreeMarker template engine.

native ready (1)

org.fugerit.java.doc.freemarker.html.​FreeMarkerHtmlFragmentTypeHandlerEscapeUTF8

fj-doc-freemarker

HTML Fragment

Same as above, but an HTML fragment is just the body of the HTML document.

native ready (1)

org.fugerit.java.doc.freemarker.asciidoc.​FreeMarkerAsciidocTypeHandlerUTF8

fj-doc-freemarker

ADOC

Renders AsciiDoc documents using Apache FreeMarker template engine.

native ready (1)

org.fugerit.java.doc.mod.fop.​PdfFopTypeHandler

fj-doc-mod-fop

PDF (PDF/A, PDF/UA)

A PDF doc handler based on Apache FOP Project. It offers options for PDF/A and PDF/UA formats.

org.fugerit.java.doc.mod.fop.​FreeMarkerFopTypeHandlerUTF8

fj-doc-mod-fop

FO (XLS-FO)

A XLS-FO doc handler based on Apache FOP Project.

org.fugerit.java.doc.mod.poi.​XlsxPoiTypeHandler

fj-doc-mod-poi

XLSX

Generates an XLSX document using Apache POI Project.

org.fugerit.java.doc.mod.poi.​XlsPoiTypeHandler

fj-doc-mod-poi

XLS

Generates an XLS document using Apache POI Project.

org.fugerit.java.doc.mod.opencsv.​OpenCSVTypeHandler

fj-doc-mod-opencsv

CSV

Generates a CSV document using OpenCSV.

native ready (1)

org.fugerit.java.doc.mod.openpdf.ext.​PdfTypeHandler

fj-doc-mod-openpdf-ext

PDF

Generates a PDF document using OpenPDF.

org.fugerit.java.doc.mod.openpdf.ext.​HtmlTypeHandler

fj-doc-mod-openpdf-ext

HTML

Generates a HTML document using OpenPDF.

org.fugerit.java.doc.mod.openrtf.ext.​RtfTypeHandler

fj-doc-mod-openpdf-ext

RTF

Generates a RTF document using OpenRTF.

(1) Native Ready means that the GraalVM metadata are included in the release.

Each section describing a specific doc handler will contain this quick reference :

In brief :

  • compliance level : COMPLETE, HIGH, MEDIUM, LOW, The level of support to Venus Doc Format.

  • compliance detail : The limitations to the support of the Venus Doc Format (for instance : unsupported elements or attributes).

  • native ready : YES, NO (If the doc handler is ready for native compilation with GraalVM).
    <<<

    === [fj-doc-base]

Here are described the doc handlers included in the basic module, which usually is always included :

<dependency>
    <groupId>org.fugerit.java</groupId>
    <artifactId>fj-doc-base</artifactId>
    <version>${fj-doc-version}</version>
</dependency>

6.1.1. MarkDown (Extended)

This doc handler would render a MarkDown (Extended) document. The Basic and Extended version only differ for table management. Basic version will render tables in html.

In brief :

  • compliance level : HIGH

  • compliance detail : All generic Venus Doc Format elements are supported.

  • native ready : YES

Add this element to <docHandlerConfig/> :

<docHandler id="md-ext" info="md" type="org.fugerit.java.doc.base.typehandler.markdown.SimpleMarkdownExtTypeHandlerNoCommentsUTF8" />

For a demo of this doc handler usage you can refer to the fj-doc-quarkus-tutorial full example or fj-doc-quarkus-tutorial base example.

6.1.2. MarkDown (Basic)

This doc handler would render a MarkDown (Basic) document. The Basic and Extended version only differ for table management. Basic version will render tables in html.

In brief :

  • compliance level : HIGH

  • compliance detail : All generic Venus Doc Format elements are supported.

  • native ready : YES

Add this element to <docHandlerConfig/> :

<docHandler id="md-basic" info="md" type="org.fugerit.java.doc.base.typehandler.markdown.SimpleMarkdownBasicTypeHandlerNoCommentsUTF8" />

For a demo of this doc handler usage you can refer to the fj-doc-quarkus-tutorial full example or fj-doc-quarkus-tutorial base example.
<<<

6.2. [fj-doc-freemarker]

To use this doc handler, you will need to add the following dependency :

<dependency>
    <groupId>org.fugerit.java</groupId>
    <artifactId>fj-doc-freemarker</artifactId>
    <version>${fj-doc-version}</version>
</dependency>

6.2.1. HTML

This doc handler would render a full html document.

In brief :

  • compliance level : HIGH

  • compliance detail : All generic Venus Doc Format elements are supported.

  • native ready : YES

Add this element to <docHandlerConfig/> :

<docHandler id="html-fm" info="html" type="org.fugerit.java.doc.freemarker.html.FreeMarkerHtmlTypeHandlerEscapeUTF8" />

For a demo of this doc handler usage you can refer to the fj-doc-quarkus-tutorial full example or fj-doc-quarkus-tutorial base example.

6.2.2. Fragment HTML

This doc handler would render only the body of the html document.
(useful to inject the body as a fragment).

In brief :

  • compliance level : HIGH

  • compliance detail : All generic Venus Doc Format elements are supported.

  • native ready : YES

Add this element to <docHandlerConfig/> :

<docHandler id="html-fragment-fm" info="fhtml" type="org.fugerit.java.doc.freemarker.html.FreeMarkerHtmlFragmentTypeHandlerEscapeUTF8" />

For a demo of this doc handler usage you can refer to the fj-doc-quarkus-tutorial full example or fj-doc-quarkus-tutorial base example.

6.2.3. Asciidoc (ADOC)

This doc handler would be rendered as an asciidoc document.

In brief :

  • compliance level : HIGH

  • compliance detail : All generic Venus Doc Format elements are supported.

  • native ready : YES

Add this element to <docHandlerConfig/> :

<docHandler id="asciidoc-fm" info="adoc" type="org.fugerit.java.doc.freemarker.asciidoc.FreeMarkerAsciidocTypeHandlerUTF8" />

For a demo of this doc handler usage you can refer to the fj-doc-quarkus-tutorial full example or fj-doc-quarkus-tutorial base example.
<<<

6.3. [fj-doc-mod-fop] A PDF/FO DocHandler

To use this doc handler, you will need to add the following dependency :

<dependency>
    <groupId>org.fugerit.java</groupId>
    <artifactId>fj-doc-mod-fop</artifactId>
    <version>${fj-doc-version}</version>
</dependency>

This module is based on Apache FOP Project for rendering and Apache FreeMarker for composing the source document model.

Usually Apache FOP uses XSL Transformations as a mean of customizing the content. But here we follow the templating approach using FreeMarker.

6.3.1. Basic PDF DocHandler

This doc handler would render a full PDF document with default FOP configuration.

In brief :

  • compliance level : HIGH

  • compliance detail : All generic Venus Doc Format elements are supported.

  • native ready : NO (As Apache FOP is not ready).

Add this element to <docHandlerConfig/> :

        <!-- Type handler generating pdf -->
        <docHandler id="pdf-fop" info="pdf" type="org.fugerit.java.doc.mod.fop.PdfFopTypeHandler"/>

For a demo of this doc handler usage you can refer to the fj-doc-quarkus-tutorial full example or fj-doc-quarkus-tutorial fop example.

6.3.2. Custom configured PDF DocHandler

For this DocHandler it is possible to customize some attributes, for instance :

        <!-- Type handler generating pdf -->
        <docHandler id="pdf-fop-config" info="pdf" type="org.fugerit.java.doc.mod.fop.PdfFopTypeHandler">
            <docHandlerCustomConfig charset="UTF-8" fop-config-mode="classloader" fop-config-classloader-path="fj-doc-quarkus-tutorial/fop-config.xml"
             fop-pool-min="20" fop-pool-max="40" fop-suppress-events="1"/>
        </docHandler>

docHandlerCustomConfig reference for org.fugerit.java.doc.mod.fop.PdfFopTypeHandler

name type default description

charset

string

UTF-8

This will set the charset to use.

fop-config-mode

string

Custom fop configuration mode, possible values are : classloader (path set fop-config-classloader-path) or inline (child element).

fop-config-classloader-path

string

Path to Apache FOP Configuration file.

fop-suppress-events

boolean

false

If set to true (or 1), will try to suppress event logging (for example : new page).

pdf-a-mode

string

If present will set pdf-a-mode, possible values are : PDF/A-1a, PDF/A-1b, PDF/A-2a, PDF/A-3a

pdf-ua-mode

string

If present, will set pdf-ua-mode, possible values are : PDF/UA-1. Partially compatible with pdf-a-mode.

fop-pool-min

int

0

If present, it will create a fo user agent pool, this is the minimum size of the pool.

fop-pool-max

int

0

If present, it will create a fo user agent pool, this is the maximum size fo the pool.

If pdf-a-mode is set, there will be a strict validation of the PDF (i.e. it will be checked if the font are all embedded and the images should comply to PDF/A standard).

6.3.3. PDF/A DocHandler

See also Apache FOP PDF/A.

Here is an example of PDF/A DocHandler with custom fop configuration :

		<docHandler id="pdf_a-fop" info="pdf" type="org.fugerit.java.doc.mod.fop.PdfFopTypeHandler">
			<docHandlerCustomConfig charset="UTF-8" fop-config-mode="classloader" fop-config-classloader-path="fop-config-pdfa.xml" pdf-a-mode="PDF/A-1b"/>
		</docHandler>

For a demo of this doc handler usage you can refer to the fj-doc-quarkus-tutorial full example or fj-doc-quarkus-tutorial fop pdf/a example.

the PDF/A configuration is not automatically generated by the Maven Plugin.

6.3.4. PDF/UA DocHandler (inline configuration)

	<docHandlerConfig registerById="true">
		<docHandler id="pdf_ua-fop" info="pdf" type="org.fugerit.java.doc.mod.fop.PdfFopTypeHandler">
			<docHandlerCustomConfig charset="UTF-8" fop-config-mode="inline" pdf-ua-mode="PDF/UA-1">
				<fop version="1.0">

				  <strict-configuration>true</strict-configuration>
				  <strict-validation>true</strict-validation>
				  <base>.</base>
				  <font-base>.</font-base>
					<renderers>
					   <renderer mime="application/pdf">
					   	  <pdf-ua-mode>PDF/UA-1</pdf-ua-mode>
					   	  <pdf-a-mode>PDF/A-1b</pdf-a-mode>
				      	  <version>1.4</version>
					   </renderer>
					</renderers>
				  <!-- Source resolution in dpi (dots/pixels per inch) for determining the size of pixels in SVG and bitmap images, default: 72dpi -->
				  <source-resolution>72</source-resolution>
				  <!-- Target resolution in dpi (dots/pixels per inch) for specifying the target resolution for generated bitmaps, default: 72dpi -->
				  <target-resolution>72</target-resolution>
				  <default-page-settings height="11in" width="8.26in"/>
				</fop>
			</docHandlerCustomConfig>
		</docHandler>

6.3.5. FO DocHandler

This doc handler would render a full FO intermediate document.

In brief :

  • compliance level : HIGH

  • compliance detail : All generic Venus Doc Format elements are supported.

  • native ready : YES

Add this element to <docHandlerConfig/> :

        <!-- Type handler generating xls:fo style sheet -->
        <docHandler id="fo-fop" info="fo" type="org.fugerit.java.doc.mod.fop.FreeMarkerFopTypeHandlerUTF8" />

For a demo of this doc handler usage you can refer to the fj-doc-quarkus-tutorial full example or fj-doc-quarkus-tutorial fop example.
<<<

6.4. [fj-doc-mod-poi] : a XLS/XLSX DocHandler

To use this doc handler, you will need to add the following dependency :

<dependency>
    <groupId>org.fugerit.java</groupId>
    <artifactId>fj-doc-mod-poi</artifactId>
    <version>${fj-doc-version}</version>
</dependency>

This module is based on Apache POI Project.

6.4.1. POI Handler Basics

The following information apply to both xlsx and xls DocHandler.

You will need at least a table, with an id set. (in this example the id is data-table).

And the excel-table-id element with the comma separated enumeration of the table to render as sheet. :

<info name="excel-table-id">data-table=print</info>

The given table (data-table) will be rendered as the named sheet (print) in the couple : $tableId=$sheetName.

Here is a full example.

<?xml version="1.0" encoding="utf-8"?>
<doc
xmlns="http://javacoredoc.fugerit.org"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://javacoredoc.fugerit.org https://www.fugerit.org/data/java/doc/xsd/doc-2-1.xsd" >
  <metadata>
  	<!-- property specific for xls/xlsx -->
  	<info name="excel-table-id">data-table=print</info>
  </metadata>
  <body>
    	<table columns="3" colwidths="30;30;40"  width="100" id="data-table" padding="2">
    		<row>
    			<cell align="center" border-color="#000000" border-width="1"><para style="bold">Name</para></cell>
    			<cell align="center"><para style="bold">Surname</para></cell>
    			<cell align="center"><para style="bold">Title</para></cell>
    		</row>
       		<row>
    			<cell><para><![CDATA[Luthien]]></para></cell>
    			<cell><para><![CDATA[Tinuviel]]></para></cell>
    			<cell><para><![CDATA[Queen]]></para></cell>
    		</row>
       		<row>
    			<cell><para><![CDATA[Thorin]]></para></cell>
    			<cell><para><![CDATA[Oakshield]]></para></cell>
    			<cell><para><![CDATA[King]]></para></cell>
    		</row>
    	</table>
  </body>
</doc>

More elements specific to excel format are available here.

6.4.2. Xlsx DocHandler

This doc handler would a XLSX document.

In brief :

  • compliance level : MEDIUM

  • compliance detail : Only single table elements of the Venus Doc Format are rendered.

  • native ready : NO (As Apache POI is not ready).

Add this element to <docHandlerConfig/> :

<!-- XLSX type hanlder  -->
<docHandler id="xlsx-poi" info="xlsx" type="org.fugerit.java.doc.mod.poi.XlsxPoiTypeHandler" />

For a demo of this doc handler usage you can refer to the fj-doc-quarkus-tutorial full example or fj-doc-quarkus-tutorial poi example.

6.4.3. Xls DocHandler

This doc handler would a XLS document.

In brief :

  • compliance level : MEDIUM

  • compliance detail : Only single table elements of the Venus Doc Format are rendered.

  • native ready : NO (As Apache POI is not ready).

Add this element to <docHandlerConfig/> :

<!-- XLSX type hanlder  -->
<docHandler id="xls-poi" info="xlsx" type="org.fugerit.java.doc.mod.poi.XlsPoiTypeHandler" />

6.5. [fj-doc-mod-opencsv] : a CSV DocHandler

To use this doc handler, you will need to add the following dependency :

In brief :

  • compliance level : MEDIUM

  • compliance detail : Only a single table element of the Venus Doc Format is rendered as a CSV document.

  • native ready : YES

<dependency>
    <groupId>org.fugerit.java</groupId>
    <artifactId>fj-doc-mod-opencsv</artifactId>
    <version>${fj-doc-version}</version>
</dependency>

This module is based on OpenCSV.

This doc handler would a CSV document.

Add this element to <docHandlerConfig/> :

<!-- CSV type hanlder  -->
<docHandler id="csv-opencsv" info="csv" type="org.fugerit.java.doc.mod.opencsv.OpenCSVTypeHandler"/>

For a demo of this doc handler usage you can refer to the fj-doc-quarkus-tutorial full example or fj-doc-quarkus-tutorial opencsv example.

You will need at least a table, with an id set. (in this example the id is data-table).

And the csv-table-id element with the comma separated enumeration of the table to render as sheet. :

<info name="csv-table-id">data-table</info>

Here is a full example.

<?xml version="1.0" encoding="utf-8"?>
<doc
xmlns="http://javacoredoc.fugerit.org"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://javacoredoc.fugerit.org https://www.fugerit.org/data/java/doc/xsd/doc-2-1.xsd" >
  <metadata>
  	<info name="csv-table-id">data-table</info>
  </metadata>
  <body>
    	<table columns="3" colwidths="30;30;40"  width="100" id="data-table" padding="2">
    		<row>
    			<cell align="center" border-color="#000000" border-width="1"><para style="bold">Name</para></cell>
    			<cell align="center"><para style="bold">Surname</para></cell>
    			<cell align="center"><para style="bold">Title</para></cell>
    		</row>
       		<row>
    			<cell><para><![CDATA[Luthien]]></para></cell>
    			<cell><para><![CDATA[Tinuviel]]></para></cell>
    			<cell><para><![CDATA[Queen]]></para></cell>
    		</row>
       		<row>
    			<cell><para><![CDATA[Thorin]]></para></cell>
    			<cell><para><![CDATA[Oakshield]]></para></cell>
    			<cell><para><![CDATA[King]]></para></cell>
    		</row>
    	</table>
  </body>
</doc>

More elements specific to CSV format are available here.

In comparison to Xlsx Doc Handler the OpenCSV one is able to render only one table at once.
<<<

6.6. [fj-doc-mod-openpdf-ext] : a PDF and HTML DocHandler

To use this doc handler, you will need to add the following dependency :

In brief :

  • compliance level : MEDIUM

  • compliance detail : Most elements of the Venus Doc Format are supported. lists have not been implemented yet.

  • native ready : NO (As OpenPDF is not ready).

<dependency>
    <groupId>org.fugerit.java</groupId>
    <artifactId>fj-doc-mod-openpdf-ext</artifactId>
    <version>${fj-doc-version}</version>
</dependency>

This module is based on OpenPDF (based on a fork of iText).

6.6.1. [fj-doc-mod-openpdf-ext-pdf] : PDF DocHandler

This doc handler would a PDF document.

Add this element to <docHandlerConfig/> :

<!-- OpenPDF type hanlder  -->
<docHandler id="openpdf" info="openpdf" type="org.fugerit.java.doc.mod.openpdf.ext.PdfTypeHandler"/>

For a demo of this doc handler usage you can refer to the fj-doc-quarkus-tutorial full example or fj-doc-quarkus-tutorial openpdf ext example.

More elements specific to fixed size formats, like PDF, are available here.

docHandlerCustomConfig reference for org.fugerit.java.doc.mod.openpdf.ext.PdfTypeHandler"

name type default description

charset

string

UTF-8

This will set the charset to use.

Additionaly fonts can be configured as child elements of docHandlerCustomConfig

name type default description

name

string

Name of the font

path

string

Path of the font (in classloader of file)

Here is a custom configuration example :

<!-- OpenPDF type hanlder  -->
<docHandler id="openpdf" info="openpdf" type="org.fugerit.java.doc.mod.openpdf.ext.PdfTypeHandler">
    <docHandlerCustomConfig charset="UTF-8">
        <font name="TitilliumWeb" path="font/TitilliumWeb-Regular.ttf"/>
    </docHandlerCustomConfig>
</docHandler>

6.6.2. [fj-doc-mod-openpdf-ext-html] : HTML DocHandler

This doc handler would a HTML document.

In brief :

  • compliance level : MEDIUM

  • compliance detail : Most elements of the Venus Doc Format are supported. lists have not been implemented yet.

  • native ready : NO (As OpenPDF is not ready).

Add this element to <docHandlerConfig/> :

<!-- OpenHTML type hanlder  -->
<docHandler id="openpdf-html" info="openpdf-html" type="org.fugerit.java.doc.mod.openpdf.ext.HtmlTypeHandler"/>

For a demo of this doc handler usage you can refer to the fj-doc-quarkus-tutorial full example or fj-doc-quarkus-tutorial openpdf ext example.
<<<

6.7. [fj-doc-mod-openrtf-ext] : a RTF DocHandler

To use this doc handler, you will need to add the following dependency :

In brief :

  • compliance level : MEDIUM

  • compliance detail : Most elements of the Venus Doc Format are supported. lists have not been implemented yet.

  • native ready : NO (As OpenRTF is not ready).

<dependency>
    <groupId>org.fugerit.java</groupId>
    <artifactId>fj-doc-mod-openrtf-ext</artifactId>
    <version>${fj-doc-version}</version>
</dependency>

This module is based on OpenRTF.

This doc handler would a RTF document.

Add this element to <docHandlerConfig/> :

<!-- OpenRTF type hanlder  -->
<docHandler id="openrtf" info="openrtf" type="org.fugerit.java.doc.mod.openrtf.ext.RtfTypeHandler"/>

For a demo of this doc handler usage you can refer to the fj-doc-quarkus-tutorial full example or fj-doc-quarkus-tutorial openrtf ext example.

More elements specific to fixed size formats, like RTF, are available here.

7. Frequently asked questions (FAQ)

This sections contains some answers to frequently asked questions.

7.1. How do I create a new project using Fugerit Venus Doc?

It is possible to simply run the plugin :

mvn org.fugerit.java:fj-doc-maven-plugin:init \
-DgroupId=org.example.doc \
-DartifactId=fugerit-demo \
-Dextensions=base,freemarker,mod-fop

For more information see : Maven Plugin Goal Init.

7.2. How do I validate the doc format after freemarker processing?

If using the FreemarkerDocProcessConfig configuration
you can simply set to true the attributes :

  • validating="true"

  • failOnValidate="true" (if you want the generation to fail in case of validation errors)

For instance :

<freemarker-doc-process-config
    xmlns="https://freemarkerdocprocess.fugerit.org"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="https://freemarkerdocprocess.fugerit.org https://www.fugerit.org/data/java/doc/xsd/freemarker-doc-process-1-0.xsd"
    validating="true"
    failOnValidate="true">

On log you will find something like :

  • DocValidationResult failed!, errors : 2

  • Validation error 0, org.xml.sax.SAXParseException; lineNumber: 42; columnNumber: 22; cvc-complex-type.2.4.a: Invalid content was found starting with element '{"http://javacoredoc.fugerit.org":h}'. One of '{"http://javacoredoc.fugerit.org":phrase, "http://javacoredoc.fugerit.org":para}' is expected.

  • Validation error 1, org.xml.sax.SAXParseException; lineNumber: 49; columnNumber: 49; cvc-complex-type.3.2.2: Attribute 'attribute-not-allowed' is not allowed to appear in element 'para'.

7.3. How do I clean source document before parsing to Doc Model?

If using the FreemarkerDocProcessConfig configuration
you can simply set to true the attributes :

  • cleanSource="true"

For instance :

<freemarker-doc-process-config
    xmlns="https://freemarkerdocprocess.fugerit.org"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="https://freemarkerdocprocess.fugerit.org https://www.fugerit.org/data/java/doc/xsd/freemarker-doc-process-1-0.xsd"
    cleanSource="true">

In case of XML source, invalid characters will be stripped :

\u0009\u000A\u000D\u0020-\uD7FF\uE000-\uFFFD\u10000-\u10FFF

For instance the following XML :

<para id="to-clean">test clean \u0002 end test.</para>

Would be cleaned to :

<para id="to-clean">test clean end test.</para>

7.4. How do I have a Excel document cells resize to fit the content.

It is possible to use the elements :

For instance :

<info name="excel-try-autoresize">true</info>
<info name="excel-fail-on-autoresize-error">false</info>

The DocHandler will try to automatically resize the cell content of the Excel Sheet.

7.5. How do I configure kotlin step in FreemarkerDocProcessConfig?

As stated in the Kotlin Source Format documentation

First of all we need to add the kotlin source support to our project :

<dependency>
    <groupId>org.fugerit.java</groupId>
    <artifactId>fj-doc-base-kotlin</artifactId>
    <version>${fj-doc-version}</version>
</dependency>

Then let’s create this very simple script in the resource folder of the project kts/kotlin-01.kts :

import org.fugerit.java.doc.base.kotlin.dsl.dslDoc

dslDoc {
    meta {
        info( "DSL Kotlin Sample" ).name( "doc-title" )
        info( "10;10;10;30" ).name( "margins" )
        info( "fugerit79" ).name( "dock-author" )
        info( "en" ).name( "doc-language" )
        info( "data-table=print" ).name( "excel-table-id" )
    }
    body {
        table {
            row {
                cell { para( "Name" ) }.align( "center" )
                cell { para( "Surname" ) }.align( "center" )
                cell { para( "Title" ) }.align( "center" )
            }.header( true )
            attListMap( data, "userList" ).forEach( { e -> row {
                cell { para( attStr( e, "name" ) ) }
                cell { para( attStr( e, "surname" ) ) }
                cell { para( attStr( e, "title" ) ) }
            } } )
        }.width( 100 ).columns( 3 ).colwidths( "30;30;40" ).id( "data-table" ).padding( 2 )
    }
}

Then we should create the FreemarkerDocProcessConfig configuration (or modify existing one) inside our resources folder config/freemarker-doc-process.xml :

<?xml version="1.0" encoding="utf-8"?>
<freemarker-doc-process-config
	xmlns="https://freemarkerdocprocess.fugerit.org"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="https://freemarkerdocprocess.fugerit.org https://www.fugerit.org/data/java/doc/xsd/freemarker-doc-process-1-0.xsd" >

	<docHandlerConfig registerById="true">
		<docHandler id="md-ext" info="md" type="org.fugerit.java.doc.base.typehandler.markdown.SimpleMarkdownExtTypeHandler" />
	</docHandlerConfig>

	<docChain id="kotlin-01">
		<chainStep stepType="kotlin" kts-path="kts/kotlin-01.kts"/>
	</docChain>

</freemarker-doc-process-config>
stepType kotlin is standalone and it does not freemarker configuration step.
stepType kotlin uses attribute kts-path, which resolve the path in the class loader.

Then we can create and use our configuration :

FreemarkerDocProcessConfig config = FreemarkerDocProcessConfigFacade.loadConfigSafe( "cl://config/freemarker-doc-process.xml" );
List<UserModel> userList = new ArrayList<>();
userList.add( new UserModel( "Queen" , "Luthien", "Tinuviel" ) );
userList.add( new UserModel( "King" , "Thorin", "Oakshield" ) );
userList.add( new UserModel( "Strider" , "Aragorn II", null ) );
DocProcessContext context = DocProcessContext.newContext( "list", userList );
try ( FileOutputStream os = FileOutputStream( ... ) ) {
    config.fullProcess( "kotlin-01", context, DocConfig.TYPE_MD, os );
}

The result should be something like :

| Name | Surname | Title  |
|---------------|---------------|---------------|
| Luthien | Tinuviel | Queen  |
| Thorin | Oakshield | King  |
| Aragorn II | null | Strider  |

7.6. PDF FOP doc handler ignores endline?

Usually PDF FOP doc handler will ignore endline.

To have endline actually be processed, it is possible to use the https://venusdocs.fugerit.org/fj-doc-base/src/main/docs/doc_xsd_config_ref.html#para attribute (when applicable).

Here is an example :

    <para>This will be rendered
as a single line.</para>
    <para white-space-collapse="preserve">This will be rendered
as two lines.</para>
when getting data from outside the template, endline should be property encoded (for instance with \n in properties file).

8. Optimizations

In this chapter the main Venus optimizations ara described.
<<<

8.1. Eager vs Lazy initialization

By default, Fugerit Venus Doc has a lazy initialization approach.

In some situation it could be a good idea to have an eager (Ahead Of Time) initialization.

This can be achieved using a built-in utility :

org.fugerit.java.doc.base.config.InitHandler

Here is an example of eager initialization based on Jakarta EE event API.

package org.fugerit.java.doc.demo;

import io.quarkus.runtime.StartupEvent;
import jakarta.enterprise.context.ApplicationScoped;
import jakarta.enterprise.event.Observes;
import jakarta.inject.Inject;
import org.fugerit.java.doc.base.config.InitHandler;

@ApplicationScoped
public class AppInit {

    @Inject
    DocHelper docHelper;

    void onStart(@Observes StartupEvent ev) {
        /*
         * This will initialize all the doc handlers using async mode.
         * (use method InitHandler.initDocAll() for synced startup)
         */
        InitHandler.initDocAllAsync(
                docHelper.getDocProcessConfig().getFacade().handlers() );
    }

}
This will initialize all the doc handlers specified as arguments by trying to create a simple document.

9. Fugerit Venus Doc Quarkus Playground

The quarkus playground is a standalone stateless application showing the capabilities of Fugerit Venus Doc.

Three version are available :

Quarkus Playground Menu
Failed to generate image: mmdc failed: Generating single mermaid chart

mindmap
  root((Home))
    Doc Conversion
      From
        XML
        JSON
        YAML
      To
        XML
        JSON
        YAML
    Doc Editor and Generator
      Source type
        FreeMarker Template
        XML
        JSON
        YAML
      Doc Handler
        HTML
        PDF FOP
        PDF/A FOP
        XLSX
        MarkDown
        Venus XML Doc
        XSL-FO
        CDV
        PDF OpenPDF
        RTF OpenRTF
        AsciiDoc
    Other Utilities
      Doc Project Init
      Doc Type Validator
      Doc Config Convert