Encrypt pdf with password, sign pdf with a certificate, fill output pdf meta, dynamically merge or concatenate multiple pdf templates into one output pdf.
jsreport extension which can merge or concatenate multiple pdf templates into a single output. The merge functionality can be used to add a dynamic header based on the content of a particular page or even a table of contents. The join can be used to prepend a cover to the pdf or to change page orientation dynamically through the single report. These advanced functions are provided on the top of standard pdf recipes and fill the gap to reach fully dynamic and unlimited pdf outputs.
The pdf manipulations are configured using the studio, respectively individual template menu. The main part of this extension is the pdf merge and append operations, but the extension can also encrypt pdf, sign it with a certificate or fill basic pdf metadata. This document covers the pdf merge/append operations at the beginning and the rest at the very end.
Each template can contain an ordered list of pdf operations which will be invoked after the main template rendering is finished. The operation required attributes are template and type. The template specifies what to render and the type what to do with the result. There are three operation types:
The operations are executed sequentially in the specified order. This is an important attribute because the input of every operation is the output of the previous one. This behavior can be explained on the following example.
Example
It is required that the output pdf has at the beginning a single cover page and every other page includes a dynamic header. This can be solved with one prepend and one merge operation. However, if you move the merge operation after the prepend, it results in the undesired output with the header also merged into the first cover page. This is because the prepend operation in such an order runs the first and expands the final output with another page. This expanded pdf is then used as input of the second operation which invokes merge into every page which means also to the cover. The correct order, in this case, should be the merge operation the first and the prepend as the second.
The input data from the original template are forwarded to the rendering of the pdf operation. The operation input includes additional information about the current pdf.
{
...user data
$pdf: {
// array representing pdf before the current operation started
pages: [{
// explained bellow
items: [{}],
group: {}
}]
}
}
Note the $pdf
is just another prop on the templating engine data context. When you are inside a loop, the context is different and you need to reach it in the templating engine specific way.
handlebars
{{@root.$pdf}}
jsrender
{{:~root.$pdf}}
These two operations are straight forward. It is typically used to concatenate multiple heterogeneous reports into one. This fills the limitation of the pdf recipes which typically doesn't allow to have for example one page in portrait orientation and the second in the landscape. The solution is to create one append operation producing portrait pdf and one producing landscape.
The merge operation can be expressed as putting two transparent documents/papers on each other. This opens various use cases and possibilities. The merge can be used for example to add a watermark to every page. This requires an extra template with a positioned and transparent image in the middle. The merge operation with such a template puts the image on every page of the parent pdf.
The header scenario works the similar way. It is only required the header has already enough space in the parent template. This is typically done using margin settings provided by the recipe. This way the header can be positioned to the top, bottom, or even rotated to the side.
The merge operation includes three additional parameters.
With merge whole documents
enabled, the header template development requires more code but the resulting template has better performance. Such a header template typically needs to iterate over all pages and add a header with a page break. The output pdf then includes just headers which can be in one chunk merged into the current pdf. See this demonstrated on the following example.
In case the merge operation includes render for every page
attribute the input data includes additionally $pdf.pageNumber
and $pdf.pageIndex
.
{
...user data
$pdf: {
pages: [...],
pageNumber: 5,
pageIndex: 4
}
It is possible to add hidden information to a page of the main content which can be used in the pdf operations afterward. The page content can for example include a list of numbers that are summarized in the page header.
The extension provides templating engine helper pdfAddPageItem
which adds value or hash to the page.
{{#each prices}}
<span>{{value}}</span>
{{{pdfAddPageItem value}}}
{{/each}}
The value can be then reached inside the pdf operation on $pdf.pages[x].items[y]
. See the example of merge with items.
The passed item can be also an object with properties.
handlebars:
{{{pdfAddPageItem name="Jan" age=33}}}
jsrender:
{{pdfAddPageItem name="Jan" age=33 /}}
Note passing objects like this works only for handlebars and jsrender.
Another use case is when the report is represented by multiple groups of pages and it is required to make the header dynamic based on the particular group. This can be reached using helper pdfCreatePagesGroup
.
The main pdf report producing the list of students can look like this
{{#each students}}
<h1 style='page-break-before: always'>{{name}}</h1>
{{{pdfCreatePagesGroup name}}}
<div>lots of other content expanding to multiple pages</div>
....
{{/each}}
The difference to page items is that the group doesn't need to be specific to a single page. It should be created just once at the top of the group and all pages share it until there isn't another the one created.
The group information can be then retrieved from $pdf.pages[x].group
property. See the example of merge with groups.
The passed group can also be an object with properties
handlebars:
{{{pdfCreatePagesGroup name="Jan" age=33}}}
jsrender:
{{pdfAddPageItem name="Jan" age=33 /}}
The pdf utils can be used also to dynamically create pdf table of contents including the outlines. This is done mainly using pdf-utils fundamentals like merge operation and page items. The flow is the following.
The main template renders TOC at the top just like any other content using templating engines. To make the links clickable the html should use anchors (a
tags) with #
in href
pointing to particular headings. Like
<a href='#hello-world'>Hello world</a>
...
<h1 id='hello-world'>Hello world</a>
The problem is the main template doesn't know the page numbers for the links. This can be solved with pdf-utils fundamentals.
Every heading should be followed by page item which is later used to identify it in the merge operation.
<h1 id='hello-world'>Hello world</a>
{{{pdfAddPageItem id='hello-world'}}}
Then there needs to be a new template that renders the same content as the TOC at the top of the main and it gets merged using pdf utils operation to the main template. This extra template can then use the pdf utils properties to calculate the page numbers of the headings.
function getPage(root, id) {
for (let i = 0; i < root.$pdf.pages.length; i++) {
const item = root.$pdf.pages[i].items.find(item => item.id === id)
if (item) {
return i + 1
}
}
}
The last step is to add the pdf outlines for easier navigation. This is done using anchor's special attribute data-pdf-outline
.
<a href='#hello-world' data-pdf-outline'>
Hello world
</a>
This by default uses the anchors inner content for the outline title which can be overwritten using data-pdf-outline-title
attribute. To create hierarchic outlines structure the data-pdf-outline-parent
with id
of the parent outline can be used.
This documentation mainly highlights the idea of how the TOC can be implemented using pdf-utils. It can look a bit tedious to implement and maintain, but this can be improved a lot using jsreport child-templates which helps with reusing the TOC part.
You can find a complex example which does all of this together here: https://playground.jsreport.net/w/admin/akYBA4rS
For more advanced scenarios which require dynamic manipulation of the operations to be done on the PDF, there are methods (pdfUtils.parse
, pdfUtils.prepend
, pdfUtils.append
, pdfUtils.merge
, pdfUtils.outlines
) exposed in the special module jsreport-proxy
, which is available in scripts.
const jsreport = require('jsreport-proxy')
async function afterRender(req, res) {
// after rendering with some pdf recipe we can use .parse to read the text in the pdf
const $pdf = await jsreport.pdfUtils.parse(res.content, true)
// we can check some condition and apply some operation, in this case we append the result of
// another render into the current PDF
if ($pdf.pages[0] && $pdf.pages[0].text != null && $pdf.pages[0].text.includes('First')) {
const appendRes = await jsreport.render({
template: {
content: 'Second page',
engine: 'none',
recipe: 'chrome-pdf'
},
data: {
...req.data,
$pdf
}
})
res.content = await jsreport.pdfUtils.append(res.content, appendRes.content)
}
}
You can find a complete example using pdf-utils in script here
Methods available:
parse(sourcePdfBuf, includeText = false)
parameters:
sourcePdfBuf
-> a buffer which contains the pdf to parseincludeText
-> indicates if text should be parsed or notreturns:
{
pages: [{
items: 'in case there are items described previously ',
group: 'in case there is group mark described previously',
text: 'in case includeText=true'
}]
}
prepend(sourcePdfBuf, prependedPdfBuf)
parameters:
sourcePdfBuf
-> pdf buffer to which to prepend the second paramprependedPdfBuf
-> pdf buffer prepended before sourcereturns:
append(sourcePdfBuf, appendedPdfBuf)
parameters:
sourcePdfBuf
-> pdf buffer to which to append the second paramappendedPdfBuf
-> pdf buffer appended after sourcereturns:
merge(sourcePdfBuf, extraPdfBuf, mergeToFront = false)
parameters:
sourcePdfBuf
-> pdf buffer to which to mergeextraPdfBuf
-> it can be one buffer representing pdf document which gets merged into source or an array of pdf buffers representing individual pages which gets merged into source page by pagemergeToFront
-> boolean that indicates if the pages should be merged in front of the current pagesreturns:
outlines(sourcePdfBuf, outlines)
parameters:
sourcePdfBuf
-> source pdf bufferoutlines
-> Array of objects ({ id: <string>, title: <string>, parent: <string> }
) which are used to define outlines in the PDFreturns:
removePages(sourcePdfBuf, pageNumbers)
parameters:
sourcePdfBuf
-> source pdf bufferpageNumbers
-> Array of numbers or one number, the page numbers starts from 1returns:
The pdf utils extension can fill the basic pdf document metadata Title
, Author
, Subject
, Keywords
, Creator
, and Producer
. These can be filled using the pdf utils extension studio UI. Or you can do it also through the API call.
{
"template": {
"content": "Main Template",
"recipe": "chrome-pdf",
"engine": "handlebars",
"pdfMeta": {
"title": "title",
"author": "author",
"subject": "subject",
"keywords": "keywords",
"creator": "creator",
"producer": "producer"
}
}
}
The pdf utils extension supports encrypting output pdf and protecting it with a password. Note this is always disabled when you run from the studio. Use the download button from the run context menu to see the protected output.
The two passwords can be specified user password and owner password. One or both of them. Behavior differs according to passwords you provide:
Additionally, you can select specific permissions:
printing - notAllowed
| lowResolution
| highResolution
modifying - true
| false
copying - true
| false
annotating - true
| false
fillingForms - true
| false
contentAccessibility - true
| false
documentAssembly - true
| false
The complete details can be found in the pdf specification here.
The pdf encryption can be set through jsreport studio and also through API call.
{
"template": {
"content": "Main Template",
"recipe": "chrome-pdf",
"engine": "handlebars",
"pdfPassword": {
"password": "password",
"ownerPassword": "password",
"printing": "HighResolution",
"modifying": true,
"copying": true,
"annotating" true,
"fillingForms": true
"contentAccessibility": true
"documentAssembly": true
}
}
}
You can use the extension to sign the output pdf with the p12 certificates. Note this is always disabled when you run from the studio. Use the download button from the run context menu to see the signed output.
Open studio and upload p12 certificate as a new asset. For the testing purposes, you can create your p12 certificate using openssl.
openssl genrsa -out myKey.pem
openssl req -new -key myKey.pem -out cert.csr
openssl x509 -req -in cert.csr -signkey myKey.pem -out cert.crt
openssl pkcs12 -export -in cert.crt -inkey myKey.pem -out cert.p12
If your certificate has a password, fill it in through asset pdf sign settings properties inside the studio.
In this case, you need to configure or disable also jsreport encryption because this extension uses the encryption API to protect password before storing.
The next step is to select the certificate in the pdf utils template menu.
The last step is to verify the pdf is properly signed. Install the certificate on your client machine and make sure it is trusted. Then open the Adobe Acrobat and you should see the message "Signed and all signatures are valid".
You may want to avoid storing your certificates inside jsreport store. In this case, you can send the encoded certificate inline in your rendering request.
POST http://jsreportserver:5488/api/report
{
"template": {
"name": "my template",
"pdfSign": {
"certificateAsset": {
"content": "base64 encoded certificate"
"encoding": "base64",
"password": "certificate password"
},
"reason": "I am signing this pdf"
}
}
}
The pdf forms are an experimental jsreport feature and the API may change in the future versions. Please help us and report problems or submit feature requests.
The extension adds also helper pdfFormField
which can be used to create pdf forms. You can place the helper call anywhere in your html/css and it gets properly positioned.
Example:
<div>
<span>Name:</span>
<span>
{{{pdfFormField name='firstName' type='text' width='200px' height='20px'}}}
</span>
</div>
<div>
{{{pdfFormField
name='btnSubmit'
type='button'
action='submit'
label='submit'
exportFormat=true
url='http://mydomain.com'
backgroundColor='#AAAA00'
width='245px'
height='20px'
}}}
</div>
The helper call common arguments are the following:
name (required)
- the field identification
type (required)
- currently supported text, button, combo
width (required)
- the field width, needs to be in px
height (required)
- the field height, needs to be in px
color - the color string css style for text color
backgroundColor - the color string css style for background color
border - the pdf like border definition, defaults to "0,0,0". Please refer to the pdf spec
fontFamily - the font isn't inherited from the outer text at this moment and if you want to use a specific font, you need to set here one of the pdf standard fonts. The custom embedded fonts aren't supported in fields at this moment
fontSize - the text size, needs to be in px
. The default size is computed to fit the field height and width
textAlign - one of the left, center, right
{{{pdfFormField name='firstName' color='#FF0000' required=true type='text' width='200px' height='20px'}}}
value - the visible value
defaultValue - value stored in form, when nothing is filled, this value isn't visible
readOnly - bool if the field is read only
required - bool if the field is required
multiline - bool for multiline text fields
password - bool for password like text fields
The text fields can be quickly formatted using the following pdfFormField
arguments.
formatType - one of the date
, time
, percent
, number
, zip
, zipPlus4
, phone
or ssn
formatMask - specifies the value and display format and can include:
d
- single digit day of monthdd
- double digit day of monthm
- month digitmm
- month double digitmmm
- abbreviated month namemmmm
- full month nameyy
- two digit yearyyyy
- four digit yearhh
- hour for 12 hour clockHH
- hour for 24 hour clockMM
- two digit minutett
- am or pmformatMask - value is one of the following HH:mm
, hh:mm
, HH:mm:ss
, hh:mm:ss
formatFractionalDigits - the number of places after the decimal point
formatSepComma - bool if pdf should display a comma separator, otherwise do not display a separator.
formatNegStyle the value must be one of MinusBlack
, Red
, ParensBlack
, ParensRed
formatCurrency a currency symbol to display
formatCurrencyPrepend set to true to prepend the currency symbol
{{{pdfFormField name='gender' type='combo' value='b' items='man,woman' width='100px' height='20px'}}}
The combo field additionally requires the items
attribute. That can be a string where items are separated using comma ,
or an actual array. The value
, defaultValue
, required
, readOnly
attributes are the same as for the text fields.
{{{pdfFormField name='btn1' type='button' exportFormat=true url='http://myendpoint.com' action='submit' label='submit' width='200px' height='50px'}}}
{{{pdfFormField name='btn2' type='button' action='reset' label='reset' width='200px' height='50px'}}}
action - submit/reset
label - text on the button
url - url where to send the form after submit
The submit button can additionally include one or more submit flags. Please see the description in the pdf specification here.
{{{pdfFormField name='test' type='signature' width='100px' height='50px'}}}
The extension features can be used also directly through API without a need to use the studio or persist templates.
{
"template": {
"content": "Main Template",
"recipe": "chrome-pdf",
"engine": "handlebars",
"chrome": {
"marginTop": "50px"
},
"pdfOperations": [{
"template": {
"content": "Header",
"recipe": "chrome-pdf",
"engine": "handlebars"
},
"type": "merge"
}],
"pdfMeta": {
"title": "title",
"author": "author",
"subject": "subject",
"keywords": "keywords",
"creator": "creator",
"producer": "producer"
},
"pdfSign": {
"certificateAsset": {
"content": "base64 encoded certificate"
"encoding": "base64",
"password": "certificate password"
},
"reason": "I am signing this pdf"
},
"pdfPassword": {
"password": "password",
"ownerPassword": "password",
"printing": "HighResolution",
"modifying": true,
"copying": true,
"annotating" true,
"fillingForms": true
"contentAccessibility": true
"documentAssembly": true
}
},
"options": {
"pdfUtils": {
"removeHiddenMarks": true
}
}
}
The operation specification looks like this:
{
"type": "merge" | "append" | "prepend",
"mergeWholeDocument": true | false,
"renderForEveryPage": true | false,
"template": { "content": "Hello"... },
"templateShortid": "xxx"
}