Dynamic Document Generator

Overview

This version is for Appian 23.2 and above. Cloud customers on a lower version of Appian that want to install this plugin should open a support case. Self-managed customers can find the latest jar for Appian 23.1 and lower version in the folder pre-23.2

Convert, merge and generate DOCX, PDF and XML files automatically in an Appian process

Key Features & Functionality

  • PDF from single DOCX with Fonts
    • Convert multiple DOCX files into a single PDF. Please note that the translation from DOCX to PDF is not always 1:1. The template may need to be tweaked or simplified to achieve the desired results.
  • PDF from DOCX without Fonts
    • Convert multiple DOCX files into a single PDF. Please note that the translation from DOCX to PDF is not always 1:1. The template may need to be tweaked or simplified to achieve the desired results. This will not support fonts.
  • PDF from XSL-FO Transformation
    • Uses XSLT to convert an XML to XSL-FO which is then converted to PDF
  • PDF from HTML
    • Converts an HTML document into a single page PDF.
  • PDF from HTML Transformation
    • Uses XSLT to convert an XML to HTML which is then converted to PDF
  • Text Doc from XSLT
    • Uses XSLT to convert an XML to a plain text document (such as HTML, another XML doc, etc).
  • DOCX from XHTML With Styling
    • Converts a valid XHTML file into a DOCX, allowing you to provide your own template file for style reference. Useful for creating dynamic documents that are editable
  • DOCX Merge
    • Merges one or more DOCX files together. If a header or footer exists in any document, they will be removed during the merge process.
    • A configurable separator can be specified: none, line break or page break

Function: xsltransform - Transform source XML using XSL Transform

Anonymous
  • After utilizing OWASP dependency-check, we have identified the following vulnerabilities (outlined below). Are there any plans in place to mitigate these CVEs?

    • CVE-2020-8908
    • CVE-2023-2976
  • v2.1.0 Release Notes
    • This is a migration of function that used to exist in the "XSL Transformation" , which is no longer listed on the Appian Appmarket**
    • Both plugins cannot exist on the same environment so first delete XSL Transformation before updating to the newest version of DDG
    • New Features:
    • New function: xsltransform

  • I can confirm at least in version 2.0.4 Arabic Letters can be displayed, if you provide a proper Font Document as an input

  • v2.0.4 Release Notes
    • Bug Fixes: PDF from Single DOCX with Font
    • Issue with mapping fonts

  • Hello,

    I am looking for a recommendation with a solution we are bulding using the smart service PDF from XSL FO Transformation With Fonts, to get text justified when there is no space on it, but keeping words together correctly when text has proper spaces.

    We have found two different solutions that help us with part of the incident, but we are looking for another one to join both of them:

    1st Soultion: correct text (spaces or urls) are wrapping correctly but texts without spaces are going out of the border:

    <fo:block margin-left="0mm" margin-top="3mm" margin-bottom="0mm" linefeed-treatment='preserve' >
    <xsl:value-of select="descripcion_txt" />
    </fo:block>

    Result: 

    2nd Solutioncorrect text (spaces or urls) are being cut when "x" number of characters are reaching the border, but texts without spaces are shown correct.

    For this solution we are using 2 templates:

    <xsl:template name="zero_width_space_1">
    <xsl:param name="data"/>
    <xsl:param name="counter" select="0"/>
    <xsl:choose>
    <xsl:when test="$counter &lt; string-length($data)">
    <xsl:value-of select='concat(substring($data,$counter,1),"&#8203;")'/>
    <xsl:call-template name="zero_width_space_2">
    <xsl:with-param name="data" select="$data"/>
    <xsl:with-param name="counter" select="$counter+1"/>
    </xsl:call-template>
    </xsl:when>
    <xsl:otherwise>
    </xsl:otherwise>
    </xsl:choose>
    </xsl:template>

    <xsl:template name="zero_width_space_2">
    <xsl:param name="data"/>
    <xsl:param name="counter"/>
    <xsl:value-of select='concat(substring($data,$counter,1),"&#8203;")'/>
    <xsl:call-template name="zero_width_space_1">
    <xsl:with-param name="data" select="$data"/>
    <xsl:with-param name="counter" select="$counter+1"/>
    </xsl:call-template>
    </xsl:template>

    Block:

    <fo:block margin-left="0mm" margin-top="3mm" margin-bottom="0mm" linefeed-treatment='preserve' >
    <xsl:call-template name="zero_width_space_1">
    <xsl:with-param name="data" select="descripcion_txt"/>
    </xsl:call-template>
    </fo:block>

    Here is an example of the xml and xsl we are using:

    XML

    <doc>
    	<prefijo>
    		<nombre>Example</nombre>
    		<fecha>2023-03-16z</fecha>
    	</prefijo>
    	<seccion_campos>
    		<titulo_txt>DESCRIPCIÓN DEL ACTO O MOTIVO:</titulo_txt>
    		<descripcion_txt>Peropueden surgir empresas tecnológicas en países con legislaciones que hacen difícil cualquier emprendimiento? No hay duda de que un mal clima de negocios, burocracias infernales y la corrupción son grandes trabas. Es difícil crear una empresa innovadora en Venezuela, donde según los estudios del Banco Mundial de los que hablábamos en el prólogo hacen falta 17 trámites legales para registrar una nueva empresa, o en Argentina, donde se requieren 14, o en Brasil y Colombia, donde hacen falta 13, y que por lo general tardan varios meses en completarse.
    		
    I cannot find that it is an error from an XSLFO standpoint this surprises me, so maybe I just missed it, but from an implementation standpoint in other words, from within FOP, there really is no way to process hyphenation without knowing which language (and therefore which hyphenation dictionary to use. Probably what is happening is that FOP jumps to a default dictionary no doubt English or U.S. English. If you are using that default language your result is ok, but it is only by accident the FOP developers could just as easily have chosen Turkish as the default.	
    	
    		LOS INNOVADORES QUIEREN VIVIR EN LUGARES VIBRANTES </descripcion_txt>
    	</seccion_campos>
    	<seccion_campos>
    		<titulo_txt>EFECTO SOBRE LOS TENEDORES:</titulo_txt>
    		<descripcion_txt>efaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaecto</descripcion_txt>
    	</seccion_campos>
    	<seccion_campos>
    		<titulo_txt>NOTA COMPARATIVA RESPECTO A LAS DIFERENCIAS MÁS RELEVANTES:</titulo_txt>
    		<descripcion_txt>find that it is an error from an XSLFO standpoint this surprises me, so maybe I just missed it, but from an implementation standpoint in other words, from within FOP, there really is no way to process hyphenation without knowing which language (and therefore which hyphenation dictionary to use. Probably what is happening is that FOP jumps to a default dictionary no doubt English or U.S. English. If you are using that default language your result is ok, but it is https://bmv-test.appiancloud.com/suite/design/lMBfe49_FfR7AnMpoon-ShAtibjpv4wDQFXudCPCREuKKkRzDRcUEpwDURnyVjx6t5Id2ExlrieqaqxXUWH2p-p8UnK2B1i6V-dwFn5rL-hfDlzgQ
    https://bmv-dev.appiancloud.com/suite/design/ksBCPzGojqGmiD_JHPtxO7RCAQ-3vL2745szbJZQQ8SUhuzlQk7uOeJCGKZaLcUdUNjeq78ZzD3wC_HeaHmkRH4V2TqtAoPkovRsTk
    
    
    </descripcion_txt>
    	</seccion_campos>
    	<seccion_campos>
    		<titulo_txt>URL:</titulo_txt>
    		<descripcion_txt>find that it is an error from an XSLFO standpoint this surprises me, so maybe I just missed it, but from an implementation standpoint in other words, from within FOP, there really is no way to process hyphenation without knowing which language (and therefore which hyphenation dictionary to use. Probably what is happening is that FOP jumps to a default dictionary no doubt English or U.S. English. If you are using that default language your result is ok, but it is https://grupovass.sharepoint.com/:x:/r/teams/BMV-Emisnet/_layouts/15/Doc.aspx?sourcedoc=%7B491E1B70-8EE3-4C32-B1B2-8EFAAC1E1713%7D&file=Incidentes%20reportados%20por%20BMV.xlsx&action=default&mobileredirect=true&clickparams=eyJBcHBOYW1lIjoiVGVhbXMtRGVza3RvcCIsIkFwcFZlcnNpb24iOiIyNy8yMzAyMDUwMTQyMSIsIkhhc0ZlZGVyYXRlZFVzZXIiOmZhbHNlfQ%3D%3D&cid=b21dabe8-a8cf-432e-a4bd-844b12946b5e</descripcion_txt>
    	</seccion_campos>
    </doc>

    XSL

    <?xml version="1.0" encoding="UTF-8"?>
    <xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:fo="http://www.w3.org/1999/XSL/Format" >
    
     <xsl:template name="zero_width_space_1">
    	<xsl:param name="data"/>
    	<xsl:param name="counter" select="0"/>
    	<xsl:choose>
    		<xsl:when test="$counter < string-length($data)">
    			<xsl:value-of select='concat(substring($data,$counter,1),"​")'/>
    			<xsl:call-template name="zero_width_space_2">
    				<xsl:with-param name="data" select="$data"/>
    				<xsl:with-param name="counter" select="$counter+1"/>
    			</xsl:call-template>
    		</xsl:when>
    		<xsl:otherwise>
    		</xsl:otherwise>
    	</xsl:choose>
     </xsl:template>
     
     <xsl:template name="zero_width_space_2">
    	<xsl:param name="data"/>
    	<xsl:param name="counter"/>
    	<xsl:value-of select='concat(substring($data,$counter,1),"​")'/>
    	<xsl:call-template name="zero_width_space_1">
    		<xsl:with-param name="data" select="$data"/>
    		<xsl:with-param name="counter" select="$counter+1"/>
    	</xsl:call-template>
     </xsl:template>
     
      <xsl:template name="getDateBy">
       <xsl:param name="date_dt"/>
       <xsl:value-of select="concat('',substring($date_dt,9,2),'/',substring($date_dt,6,2),'/',substring($date_dt,1,4))" />
      </xsl:template>
      
    	<xsl:template match="doc">
    		<fo:root  xml:lang="es" xmlns:fo="http://www.w3.org/1999/XSL/Format">
    			<fo:layout-master-set>
    				<fo:simple-page-master master-name="A4" page-width="220mm" page-height="280mm" margin-top="1cm" margin-bottom="1cm" margin-left="1cm" margin-right="1cm">
    					<fo:region-body margin-top="6cm" margin-bottom="1cm" margin-left="0mm" margin-right="5mm" />
    					<fo:region-before margin-top="0cm" margin-bottom="20cm" margin-left="0cm" margin-right="0cm" />
    					<fo:region-after margin-top="0cm" margin-bottom="1cm" margin-left="0cm" margin-right="3cm" />
    				</fo:simple-page-master>
    			</fo:layout-master-set>
    	
    			<fo:page-sequence master-reference="A4">
    			
    				<!-- HEADER -->
    				<fo:static-content flow-name="xsl-region-before" >
    					<fo:block-container height="0cm" background-color="white" margin-top="-20mm" margin-left="-10mm" margin-right="43mm" >
    					
    					<!-- NOMBRE DEL PREFIJO -->
    					<fo:block text-align="left" font-size="13pt" margin-top="25mm" margin-left="20mm" color="black" font-weight="bold" >
    						<xsl:value-of select="prefijo/nombre" />                 
    					</fo:block> 
    					<!-- TERMINA EL NOMBRE DEL PREFIJO -->
    					 
    					 <!-- LOGO -->
    					 <fo:block margin-left="135mm" margin-top="-15mm">
    					  <fo:external-graphic 
    						src = "url('')"
    						height="10.5cm" 
    						content-width="8.5cm" />     
    					 </fo:block>
    					<!-- TERMINA EL LOGO -->
    							
    					<!-- FECHA -->
    					<fo:block margin-left="15mm" margin-bottom="-30mm" margin-top="-63mm" >
    						<fo:table table-layout="fixed" width="100%" >
    							<fo:table-column column-width="249mm" />
    								<fo:table-body>
    									<fo:table-row>
    									<fo:table-cell>
    										<fo:block-container font-size="11pt" border-bottom-style="solid" border-color="black" >
    											<fo:block margin-left="-5mm" color="black" >Fecha:  
    												 <xsl:call-template name="getDateBy">
    													 <xsl:with-param name="date_dt" select="prefijo/fecha"/>
    												 </xsl:call-template>
    											</fo:block>
    										</fo:block-container>
    									</fo:table-cell>
    								</fo:table-row>
    							</fo:table-body>
    						</fo:table>
    					</fo:block>
    					<!-- TERMINA LA FECHA -->
    					
    				 </fo:block-container>
    				</fo:static-content>
    				<!-- TERMINA EL HEADER -->
    				
    				<!-- FOOTER -->
    				<fo:static-content flow-name="xsl-region-after" height="10cm">
    					<fo:block text-align="left" font-size="9pt" border-bottom-style="solid" border-top-style="solid" border-color="black" >
    						<fo:table table-layout="fixed" width="100%" >
    							<fo:table-column column-width="198mm" />
    							<fo:table-column column-width="2mm" />
    							<fo:table-body>
    								<fo:table-row>
    									<fo:table-cell>
    										<fo:block-container >
    											<fo:block margin-left="0mm" margin-top="1mm" margin-bottom="0.5mm" >
    												 Customer
    											</fo:block>
    										</fo:block-container>
    									</fo:table-cell>
    									<fo:table-cell>
    										<fo:block-container >
    											<fo:block margin-left="0mm" margin-top="1mm" margin-bottom="0.5mm" >
    												 <fo:page-number/>
    											</fo:block>
    										</fo:block-container>
    									</fo:table-cell>
    								</fo:table-row>
    							</fo:table-body>
    						</fo:table>
    					</fo:block>
    				</fo:static-content>
    				<!-- TERMINA EL FOOTER -->
    				
    				<!-- BODY -->
    				<fo:flow flow-name="xsl-region-body" >
    					<fo:block text-align="justify" font-size="9pt" >
    						<fo:table table-layout="fixed" width="100%" >
    							<fo:table-column column-width="200mm" />
    							<fo:table-body>
    								
    								<!-- SECCION DE CAMPOS DINAMICOS -->
    								<fo:table-row>
    									<fo:table-cell>
    										<fo:block font-size="11pt" >
    											<fo:table table-layout="fixed" width="100%" >
    												<fo:table-column column-width="200mm" />
    												<fo:table-column column-width="200mm" />
    												<fo:table-body>
    													<xsl:for-each select="seccion_campos">
    														<fo:table-row>
    															<fo:table-cell>
    																<fo:block-container >
    																	<fo:block margin-left="0mm" margin-top="6mm" margin-bottom="0mm" font-weight="bold" >
    																		 <xsl:value-of select="titulo_txt" />    
    																	</fo:block>
    																</fo:block-container>
    															</fo:table-cell>
    														</fo:table-row>
    														<fo:table-row>
    															<fo:table-cell>
    																<fo:block-container >
    																	<fo:block margin-left="0mm" margin-top="3mm" margin-bottom="0mm" linefeed-treatment='preserve' >
    																		<!-- 	 <xsl:value-of select="descripcion_txt" />			 -->
    																		<xsl:call-template name="zero_width_space_1">
    																				<xsl:with-param name="data" select="descripcion_txt"/>
    																			</xsl:call-template>
    																	</fo:block>
    																</fo:block-container>
    															</fo:table-cell>
    														</fo:table-row>
    													</xsl:for-each>
    												</fo:table-body>
    											</fo:table>
    										</fo:block>
    									</fo:table-cell>
    								</fo:table-row>
    								<!-- TERMINA SECCION DE CAMPOS DINAMICOS -->
    								
    							</fo:table-body>
    						</fo:table>
    					</fo:block>
    				<fo:block id="TheVeryLastPage"> </fo:block>
    				</fo:flow>
    				<!-- TERMINA EL BODY -->
    				
    			</fo:page-sequence>
    		</fo:root>
    	</xsl:template>
    </xsl:stylesheet>

  • Please upload an example document with problems and the issues you are seeing.

  • Hello! 

    I tested it but still same issue.

    Regards,

    Dimitrios

  • Hi

    Do you know if this has been fixed after the new version.

    I appreciate your comments.

    Best regards

  • v2.0.3 Release Notes
    • Bug Fixes:
    • DOCX from XHTML fixes
    • Issue with base64 images
    • Issue with null template
  • Hello, ! I have tried your suggestion but it does not work. Still the same issue. 

    By the way, I want to report that the HTML to PDF conversion seems fragile. I am trying to convert a DOCX to PDF and I am getting the below issues as reported also by ,

    In general, it seems that the conversions are a little bit fragile.