Chrome-pdf recipe is using headless chrome to print html content into pdf files.
The settings reflects the headless chrome API where you can also find detail information.
These basic settings are typically stored with the template, but you can also send them through API calls  inside the template.chrome property.
The options can be also dynamically set from within the page javascript using:
<script>
    ...
    window.JSREPORT_CHROME_PDF_OPTIONS = {
        landscape:  true
    }
</script>
Use chrome-pdf node in the standard config file.
"extensions": {
  "chrome-pdf": {
    "timeout": 30000,
    "launchOptions": {...}
  }
}
to find more information about what is available in launchOptions configuration object you can check the docs here.
you can also use top level chrome property in configuration, the difference is that this configuration will be shared with any other extension that uses chrome and the configuration snippet above is specifically for options in chrome-pdf extension.
"chrome": {
  "timeout": 30000
}
The fonts can be easily embedded into PDF reports using the assets extension. You can find the tutorial on how to do it here.
CSS contains styles like page-break-before that you can use to specify html page breaks. This can be used as well with chrome-pdf in order to specify page breaks inside pdf files.
<h1>Hello from Page 1</h1>
<div style='page-break-before: always;'></div>
<h1>Hello from Page 2</h1>
You can also use css property page-break-inside to for example avoid breaking an element to multiple pages.
You may need to postpone pdf printing until some javascript async tasks are processed. If this is your case set the chrome.waitForJS=true in the API or Wait for printing trigger in the studio menu. Then the printing won't start until you set window.JSREPORT_READY_TO_START=true inside your template's javascript.
...
<script>
    // do some calculations or something async
    setTimeout(function() {
        window.JSREPORT_READY_TO_START = true; //this will start the pdf printing
    }, 500);
    ...
</script>
See an example of using printing triggers when rendering charts here.
The header and footer are evaluated as if they were a full jsreport template. This means you can add, for example, a child template reference into a header and it will be extracted. You can also use main template helpers or data in the header/footer. Remember that in order to show the header/footer you need to activate the displayHeaderFooter option first and add some top, bottom margin to the template in order to give the page some space to show the header/footer.
Inside the header/footer template you can use some special css classes to make chrome inject some content for you. the special css classes supported by chrome are the following:
date -> injects formatted print datetitle -> injects the content of the document titleurl -> injects the document locationpageNumber -> injects current page numbertotalPages -> insject the total pagesThere are some issues with native header/footer you should be aware of:
-webkit-print-color-adjust: exact as workaroundIn the most of the cases it is better to use pdf-utils instead which is less limiting and without these issues.
The chrome by default displays the current date in the header and file name in the footer when displayHeaderFooterSelected. You can place an empty tag <span/> to the header or footer to avoid this behavior.
The header/footer has some extra padding you can remove using:
<style>#header, #footer { padding: 0 !important; }</style>
Example showing how to use the special css classes and the workaround for the scaling issues.
<!--header template content-->
<html>
  <head>
    <style>
      /* defining explicit font-size solves the scaling issue */
      html, body {
        font-size: 12px;
      }
    </style>
  </head>
  <body>
    <!--
      defining some elements with the special css classes makes chrome
      inject content in runtime
    -->
    Page <span class="pageNumber"></span> of <span class="totalPages"></span>
  </body>
</html>
The pdf-utils extension provides advanced and more rich features to merge dynamic content into the chrome pdf output, like rich header/footer, print page numbers, watermarks, merge pages with different orientation, etc. make sure to check the docs for some examples.
Chrome by default uses print CSS media query when printing pdf. This impacts CSS frameworks like Bootstrap which usually produces different results for print media type. The pdf in this case applies different styles then html. You can adapt/unite this by changing media type settings from print to screen in the template's chrome settings.
By default, chrome will wait for the content on your page to load, if some of the content takes a long time to load then it will hang the whole rendering. this may not be the ideal scenario for certain cases. to have better control for these kind of resources you can use the chromeResourceWithTimeout(resourceURL, timeoutMs) helper.
For example to load an image that it is nice to have but not critical for the content (we just give it a maximum of 2s to load), then you can define it as:
<img src="{{chromeResourceWithTimeout 'http://some-domain/image.png' 2000}}" />
First, make sure the chrome-pdf recipe is the bottleneck by checking the studio profile tab.
ARIA
Then you can try to disable chrome accessibility tags generation which affects the rendering time and the pdf size for reports with many elements. This can be done by adding aria-hidden="true" attribute to the HTML body or wrapping element.
Image sizes
The images printed to the pdf keep the original size despite the width and height attributes set. Visually the images are properly sized, but the stored size is the same as the original and can dramatically increase the pdf output. The solution is to resize the images to the desired size before starting the chrome pdf printing. One of the approaches using templating engines helper is mentioned here.
Also the css style object-fit: cover may cause pdf size increase and you may need to avoid it.
Scale the HW Trying to increase CPU or removing container limits is always a good idea when troubleshooting performance.
Isolate the problem Chrome is a big black box you won't be able to analyze deeply. The only option is to isolate the problem by removing parts of the template content. You can isolate the problem to a chart that needs to disable animations or a long table that needs to optimize some CSS. This is always specific to a particular report.
Search forum for community tips Try searching jsreport forum. You may have luck and find a tip that will be the solution for you https://forum.jsreport.net/search
Split long reports Sometimes you can use jsreport script to split a long report into multiple smaller parts which you combine after rendering. This is demonstrated in the following demo.
The request data or template definition can be reached inside inline script tag using function await window.jsreport.getRequest()
recipe: <span id='recipe' />
<script>
  (async () => {
    const req = await window.jsreport.getRequest()
    document.getElementById('recipe').innerText = req.template.recipe
  })()
</script>
To improve performance you can also select just specific data you need.
some prop: <span id='someProp' />
<script>
  (async () => {
    const someProp = await window.jsreport.getRequest('data.someProp')
    document.getElementById('someProp').innerText = someProp
  })()
</script>
Another option is to serailize data into the html during the templating engine evaluation using toJS system helper.
<script>
  const reportData = {{{toJS this}}}
  ...
</script>
You can also print an existing webpage through chrome-pdf recipe without a need to define your templates in jsreport studio. Just send a request like this:
{
  "template": {
    "recipe": "chrome-pdf",
    "engine": "none",
    "chrome": {
      "url": "https://jsreport.net"
    }
  }
}
Or you can create an empty template and define the url using jsreport script.
function beforeRender(req, res) {
  req.template.chrome = {
     "url": "https://jsreport.net"
  }
}
The recipe by default allocates a single instance of chrome per worker thread and reuses it. This means that for the configuration "workers": { "numberOfWorkers": 5 } there will be 5 chrome instances allocated.
This is reasonable for most of the cases, but in case your report is initiating many nested reports, you may want to increase the parallelization by increasing the number of chrome instances allocated per thread.
{
  "chrome": {
    "numberOfWorkers": 3
  }
}
You can change the allocation strategy and let the recipe always create a new instance of chrome if that is what you prefer. This increases the parallelization of the nested reports to the maximum. However, note that starting a new chrome process costs about 100ms.
{
  "chrome": {
    "strategy": "dedicated-process"
  }
}
Finally there is a third option, you can connect to an existing instance chrome that it is running somewhere else. This is useful if you want to use external services that host chrome.
{
  "chrome": {
    "strategy": "connect",
    "connectOptions": {
      "browserWSEndpoint": "<endpoint to the remote instance of chrome>"
    }
  }
}
In many cases, you can switch to the html recipe and debug the output using the F12 browser's tool just like any other page.
To troubleshoot the javascript evaluated in the chrome-pdf recipe, you can write to the console.log and inspect the outputs in the studio profile tab when clicking on the chrome-pdf operation.
For a full chrome-pdf debugging experience, find the video tutorial here. Full credits go to the author.
Note the same debugging techniques applies also to the chrome-image recipe.
Protocol error (Page.printToPDF): Printing failed can becaused by various things, but the most common solution is
<div />) are heavily slowing down chrome pdf rendering, don't use them
<div style='height:500'>
  <img src='foo' />
</div>
border-collapse: collapse which does not work properly when content is split across pages. So different approach should be done in order to replicate borders that work properly across pages. Example for this solution available here
<table> in a loop, we create new <tbody> elements. Example for this solution available here
<style>
  * {
    text-rendering: geometricprecision !important;
  }
</style>
--no-sandbox argument. This can be achieved using the following config. See also puppeteer troubleshooting.
 "extensions": {
  "chrome-pdf": {
    "launchOptions": {
      "args": ["--no-sandbox"]
    }
  }
}