The jqassistant-asciidoctor-extension
is an asciidoctor extension that offers a convenient way of creating a jQAssistant report inside the asciidoctor pipeline.
The plugin is using the asciidoctor extension api to hook into the asciidoctor pipeline.
You can use it to convert your jqassistant-report.xml (jQAssistant generates this for you) directly to html, pdf, ePub and others, as long as there’s an Asciidoctor converter for it.
As an example of how to use the plugin, you can take a look at the following maven plugin configuration for the asciidoctor-maven-plugin
.
<!-- Define the following inside the build part of your pom.xml -->
<plugins>
<plugin>
<groupId>org.asciidoctor</groupId>
<artifactId>asciidoctor-maven-plugin</artifactId>
<version>3.1.1</version>
<configuration>
<attributes>
<jqassistant-report-path>
${project.build.directory}/jqassistant/jqassistant-report.xml <!--(1)-->
</jqassistant-report-path>
<jqassistant-templates-path>
your-custom-template-jar-or-folder-name <!--(2)-->
</jqassistant-templates-path>
<imagesDir>
your-desired-folder-to-store-pictures-inside-your-output-directory <!--(3)-->
</imagesDir>
</attributes>
</configuration>
<executions>
<execution>
<id>output-html</id>
<phase>verify</phase>
<goals>
<goal>process-asciidoc</goal>
</goals>
</execution>
</executions>
<dependencies>
<dependency>
<groupId>org.jqassistant.tooling.asciidoctorj</groupId>
<artifactId>jqassistant-asciidoctorj-extensions</artifactId>
<version>1.1.0</version> <!--(4)-->
</dependency>
</dependencies>
</plugin>
</plugins>
-
place the path to your jqassistant-report.xml here. Check Finding your jqassistant-report.xml for information.
-
OPTIONAL: If not declared, the extension will use the default templates. If you want to use your templates, place the name of your custom templates location here. For more information refer to section Using your own template.
-
OPTIONAL: If you add this option, the extension will copy all pictures (f.e. diagrams) generated by jQAssistant into the defined folder. The folder will be created by asciidoctor relative to the generated docs folder.
-
place your desired version of the extension here. Check maven repository for the newest version.
In general, there are two possible macros (called includes) you can use with the jqassistant-asciidoctor-extension
.
-
include::jQAssistant:Summary[]
will give you a summary table of all constraints and concepts with their id, description, severity and status. They’re sorted by Status and id name. -
include::jQAssistant:Rules[]
will give you a detailed list of all constraints and concepts with additional information like result tables or graphs
For both of them, you have the option to assign two additional attributes:
-
concepts
-
constraints
You can assign an id or id wildcard to either of them. In this case, you get back some specific jQAssisstant concepts and constraints. This could look like this:
-
include::jQAssistant:Summary[concepts="jQA:my:specific:concept-id"]
would result in a summary table that only contains the concept with the name jQA:my:specific:concept-id (if it exists). Constraints are not rendered in this case. -
include::jQAssistant:Summary[concepts="jQA:my:specific:concept-id", constraints="*"]
would result in a summary table containing the concept with the name jQA:my:specific:concept-id. Also, all constraints will be displayed. -
include::jQAssistant:Rules[constraints="jQA:a*", concepts="jQA:my:specific:concept-id"]
would result in a list containing all constraints with their details that start with "jQA:a". Also, it will contain exactly one concept (if it exists) with the id "jQA:my:specific:concept-id" (this concept will also be displayed with its details).
Note
|
The described behavior only applies if you use the default templates. If you assign custom templates, these includes may display different contents. |
Note
|
Normally include::… 's are used to reference and display other files inside the adoc file they’re written in. In our case, tho these include::… 's are replaced by the jqassistant-asciidoctor-extension .
|
Note
|
The standard behaviour of the default templates is to render Rule-Ids in the include::jQAssistant:Rules[] with level 0 headings (adoc notation: = ). If you want to change that, you may use the leveloffset attribute. F.e. include::jQAssistant:Rules[leveloffset=+1] : This would yield an output with all Rule-Ids having a level 1 heading instead of level 0 (adoc notation: = → == ).
|
Normally you can refer to your jqassistant-report.xml via ${project.build.directory}/jqassistant/jqassistant-report.xml
. You can check the exact location by searching for jqassistant-report.xml
in your jQA output directory (this xml is created by jQA at runtime).
This extension uses freemarker templates to personalize your report. The basic idea behind a freemarker template is, to fill a template with the data contained in a root element. Normally there are default templates already provided with this extension. If you want to declare your own templates, put the jar or folder with your freemarker templates inside your resource folder. All templates that are not defined will use the default templates. Add the name of the jar or folder to your configuration as described in Using the jqassistant-asciidoctor-extension.
For this part, you need to have a basic understanding of how to write a freemarker template and how to reference the data from your root element. For information on how to write your freemarker template please refer to the Freemarker template author’s guide.
There are five templates you can create for the extension to use.
-
IconEnabler
This template is used to assign your own document attributes for the asciidoctor processing. Its content is placed at the beginning of each adoc. This happens before the processing of your includes. See IconEnabler for the structure of the default IconEnabler. -
Summary
This template is used for theinclude::jQAssistant:Summary[]
call. -
RulesConcept
This template is used for the concepts part of theinclude::jQAssistant:Rules[]
call. -
RulesConstraint
This template is used for the constraints part of theinclude::jQAssistant:Rules[]
call. -
NoResult
This template is used in case of no matching concepts or constraints (both ininclude::jQAssistant:Summary[]
andinclude::jQAssistant:Rules[]
). It’s main purpose is to signalize a wrong id filter.
Note
|
For templates 2-5 the RuleRoot object is used to enrich the template with information. See RuleRoot usage for more information. |
:caution-caption: 🔥 ERROR
:tip-caption: ✅ SUCCESS
:warning-caption: ⚠ WARNING
:note-caption: ⚡ SKIPPED
One possibility here is to add the :icons:
attribute and set it to font
. This allows you to use icons in your other templates. Another possibility is to remove all the content in the IconEnabler. This allows you to use the standard asciidoctor icons for the admonition blocks.
The jqassistant-asciidoctor-extension
provides a root element, that is then combined with your freemarker template to create the finished report. You can take a look at the following example template to understand the data structure provided by the extension.
<#list concepts as rule> (1)
[#jqassistant_${rule.id}]
= ${rule.id} (2)
****
<#if rule.status == "SUCCESS"> (2)
TIP: ${rule.description} + (2)
Status: [green]#${rule.status}#, Severity: ${rule.severity} (2)
<#elseif rule.status == "WARNING">
WARNING: ${rule.description} +
Status: [yellow]#${rule.status}#, Severity: ${rule.severity}
<#elseif rule.status == "FAILURE">
CAUTION: ${rule.description} +
Status: [red]#${rule.status}#, Severity: ${rule.severity}
<#else>
NOTE: ${rule.description} +
Status: [grey]#${rule.status}#, Severity: ${rule.severity}
</#if>
<#if rule.hasReports> (3)
<#list rule.reports.links as labeledLink> (5)
link:${labeledLink.link}[${labeledLink.label}] (6)
</#list>
<#list rule.reports.images as labeledImage> (5)
[caption="", title=${labeledImage.label}] (6)
image::${labeledImage.link}[${labeledImage.label}]
</#list>
<#elseif rule.hasResult> (4)
|===
<#list rule.resultColumnKeys as key>|${key} </#list> (7)
<#list rule.resultRows as row> (8)
<#list row as cell> (9)
|${cell}
</#list>
</#list>
|===
</#if>
****
</#list>
-
You can refer to your returned constraints and concepts via the
constraints
orconcepts
identifiers. Specifically, each of them is internally handled as a sorted set (primarily sorted by status from failure to warning to success to skipped and secondarily by id in alphabetical order). In case theNoResult
template is used, the RuleRoot object holds an empty list of concepts and constraints. -
For each concept and constraint (here represented by
rule
) you can refer to their id, description, status or severity by usingrule.id
,rule.description
,rule.status
, andrule.severity
. -
Via
rule.hasReports
you can check whether the corresponding concept or constraint generated any pictures (e.g. graphs) or external files (e.g. CSV’s) to display. -
Via
rule.hasResult
you can check whether the corresponding concept or constraint generated a result table to display. This table may contain things like dependencies or other related information. -
rule.reports
gives you access to the list of external files (viarule.reports.links
) and to a list of generated images (viarule.reports.images
) -
Both external files and images use the same internal type (namely URLWithLabel) to give access to the location (
labeledLink.link
orlabeledImage.link
depending on the naming in the<#list>
block) and to the label (labeledLink.label
orlabeledImage.label
). (Consider usinghasReports
beforehand. See (5)) -
You can access the header of your table using the
resultColumnKeys
identifier. This gives you a list of the column names for the table. (Consider usinghasResult
beforehand. See (4)) -
You can access a list of all rows via
resultRows
-
You can access each individual cell (containing a string value) of one row via
<#list row as cell>
(each row itself is a list of Strings). In our example we listed each row via<#list rule.resultRows as row>
and for each listed their cells (strings) via<#list row as cell>
.
[opts="header"]
|===
|Id |Description |Severity |Status
<#list constraints as constraint>
|<<jqassistant_${constraint.id}>> |${constraint.description} |${constraint.severity}
|[<#if constraint.status == "SUCCESS">green<#elseif constraint.status == "FAILURE">red<#elseif constraint.status == "WARNING">yellow<#else>grey</#if>]#${constraint.status}#
</#list>
<#list concepts as concept>
|<<jqassistant_${concept.id}>> |${concept.description} |${concept.severity}
|[<#if concept.status == "SUCCESS">green<#elseif concept.status == "FAILURE">red<#elseif concept.status == "WARNING">yellow<#else>grey</#if>]#${concept.status}#
</#list>
|===