@jsreport/jsreport-core
The minimalist jsreport rendering core.
The full distribution can be found in the jsreport package.
jsreport is a platform providing dynamic documents assembling and printing. It supports various document types or printing techniques.
@jsreport/jsreport-core
contains the jsreport rendering core that is useless alone. It is up to you which jsreport extensions you combine
Quick example
To generate a document using jsreport, you always need a javascript templating engine. The engine is used to dynamically assemble the document based on the input values. For start lets pick @jsreport/jsreport-handlebars engine and install it using npm.
Next to the engine, you need something we call recipe. A recipe represents the technique used to print the document. It can be an HTML to pdf conversion, DOCX rendering, and others. In this example lets pick @jsreport/jsreport-chrome-pdf. This recipe implements HTML to pdf conversion using chrome. So in this example, we use handlebars to assemble HTML based on the input data and then print the output into the final pdf.
npm install @jsreport/jsreport-core
npm install @jsreport/jsreport-handlebars
npm install puppeteer @jsreport/jsreport-chrome-pdf
const fs = require('fs').promises
const jsreport = require('@jsreport/jsreport-core')()
jsreport.use(require('@jsreport/jsreport-chrome-pdf')())
jsreport.use(require('@jsreport/jsreport-handlebars')())
await jsreport.init()
const result = await jsreport.render({
template: {
content: '<h1>Hello {{foo}}</h1>',
engine: 'handlebars',
recipe: 'chrome-pdf'
},
data: {
foo: "world"
}
})
await fs.writeFile('out.pdf', result.content)
Render
render
is the main method that invokes report generation. A single parameter is an object representing describing what to render. It has the following structure:
{
template: {
engine: "handlebars",
recipe: "chrome-pdf",
content: "<h1>{{foo}}</h1>",
helpers: "function foo() { ...} " +
"function foo2() { ... }"
...
},
data: { foo: "hello world"}
...
}
In case you have the template stored in the jsreport templates store, you can reference the template using a name or path.
{
template: {
name: '/myfolder/mytemplate'
},
data: { foo: "hello world"}
...
}
The render returns a promise with the single response value
{
content: ...
stream: ...
meta: { ... }
}
The convention is that jsreport repository extension starts with jsreport-xxx
, but the extension real name and also the recipes or engines it registers excludes the jsreport-
prefix. This means if you install extension @jsreport/jsreport-handlebars
the engine's name you specify in the render should be handlebars
.
Require in the helpers
jsreport by default runs helpers in the sandbox where is the require
function blocked. To unblock particular modules or local scripts you need to configure sandbox.allowedModules
option.
const jsreport = require('@jsreport/jsreport-core')({
sandbox: { allowedModules: ['moment'] }
})
const jsreport = require('@jsreport/jsreport-core')({
sandbox: { allowedModules: '*' }
})
Additionally, jsreport provides global variables which can be used to build the local script path and read it.
const jsreport = require('@jsreport/jsreport-core')({
sandbox: { allowedModules: '*' }
})
await jsreport.init()
await jsreport.render({
template: {
content: '<script>{{jquery}}</script>',
helpers: `
function jquery() {
const fs = require('fs')
const path = require('path')
return fs.readFileSync(path.join(__rootDirectory, 'jquery.js'))
}
`,
engine: 'handlebars',
recipe: 'chrome-pdf'
}
})
The following variables are available in the global scope:
__rootDirectory
- two directories up from jsreport-core__appDirectory
- directory of the script which is used when starting node__parentModuleDirectory
- directory of script which was initializing jsreport-core
Extensions
You need to install additional packages (extensions) even for the simplest pdf printing. This is the philosophy of jsreport, and you will need to install additional extensions very often. There are many extensions adding support for persisting templates, dynamic script evaluation, adding browser based reports studio or exposing API. To get the idea of the whole platform you can install the full jsreport distribution and pick what you like. Then you can go back to jsreport-core
and install extensions you need.
You are also welcome to write your own extension or even publish it to the community. See the following articles how to get started.
The best place to find availible extensions is in the jsreport documentation or you can search in this monorepo's packages - every package with jsreport-
prefix is an extension.
Extensions auto discovery
jsreport by default auto-discovers extensions in the application's directory tree. This means, there is no need to explicitely require
and call jsreport.use
for all extensions you want to use. However it is prefered for the clarity.
Configuration
jsreport accepts options as the first parameter. The core options are the following:
require('@jsreport/jsreport-core')({
rootDirectory: path.join(__dirname, '../../'),
tempDirectory: path.join(dataDirectory, 'temp'),
logger: {
silent: false
},
sandbox: {
cache: {
max: 100,
enabled: true
}
},
loadConfig: false,
autoTempCleanup: true,
useExtensionsLocationCache: true
})
jsreport-core
is also able to load configuration from other sources including configuration file, environment variables and command line parameters. This can be opted in through option loadConfig:true
. If this option is set to true the configuration is merged from the following sources in the particular order:
- configuration file jsreport.config.json or the one specified in
configFile
environment variable - command-line arguments
- process environment variables
- options passed directly to
require('@jsreport/jsreport-core')({})
Each extension (recipe, store...) usually provides some options you can apply and adapt its behavior. These options can be typically set through standard configuration under the top-level extensions
property, options in there with the name corresponding to the extension's name are forwarded to the particular extension. This is the common way how to globally configure all extensions at one place.
require('@jsreport/jsreport-core')({
...
"extensions": {
"scripts": {
"allowedModules": ["url"]
}
}
})
You can find configuration notes for the full jsreport distribution here.
Logging
jsreport leverages winston logging abstraction together with debug utility. To output logs in the console just simply set the DEBUG
environment variable
DEBUG=jsreport node app.js
on windows do
set DEBUG=jsreport & node app.js
To declarative configure logging levels and outputs you can see this page which contains all the details for that.
jsreport also exposes logger
property which can be used to adapt the logging as you like. You can for example just add winston console transport and filter in only important log messages into console.
const winston = require('winston')
const jsreport = require('@jsreport/jsreport-core')()
jsreport.logger.add(winston.transports.Console, { level: 'info' })
Typescript
jsreport types are in the DefinitelyTyped repository.
You can install @types/jsreport-core and invidual types for extensions, or get all types at once from @types/jsreport.
You can also find jsreport typescript examples here.
Listeners
jsreport extensions are mainly using the system of event listeners to adapt the rendering process. Extensions, can for example listen to event, which is called before the rendering process starts and adapt the input values.
jsreport.beforeRenderListeners.add('name-of-listener', (req, res) => {
req.template.content = 'Changing the template in listener!'
})
To start the listening, you must first add the listener function to the right listener. In the example is used beforeRenderListeners
which is called before the rendering starts. jsreport then in the right time sequentially fires all the listener functions and let them do the required work. If the function returns a promise, jsreport awaits until it is fulfilled.
Note this technique can be used in extensions, but also outside in nodejs application using jsreport.
jsreport currently support these main listeners
initializeListeners()
- called when all extensions are initialized
beforeRenderListeners(req, res)
- very first in the rendering pipeline, used to load templates and parse input data
validateRenderListeners(req, res)
- possible to reject rendering before it starts, jsut return failed promise or exception
afterTemplatingEnginesExecutedListeners(req, res)
- engine like handlebars or jsrender extracted the content, the res.content
contains Buffer with extracted content
afterRenderListeners(req, res)
- recipes are executed, res.content
contains final buffer which will be returned as a stream back, the last change to modify the output or send it elsewhere
renderErrorListeners(req, res, err)
- fired when there is error somewhere in the rendering pipelinecloseListeners()
called when jsreport is about to be closed, you will usually put here some code that clean up some resource
Studio
jsreport includes also visual html studio and rest API. This is provided through two extensions, @jsreport/jsreport-express extension to have a web server available and @jsreport/jsreport-studio for the web UI, both extensions should be installed in order to have the studio ready. See the documentation of each extension for the details.
Template store
jsreport-core
includes API for persisting and accessing report templates. This API is then used by extensions mainly in combination with jsreport studio. jsreport-core
implements just in-memory persistence, but you can add other persistence methods through extensions, see the template stores docummentation
The persistence API is almost compatible to the mongodb API:
jsreport.documentStore.collection('templates')
.find({name: 'test'})
.then((res) => {})
jsreport.documentStore.collection('templates')
.update({name: 'test'}, { $set: { attr: 'value' })
.then((res) => {})
jsreport.documentStore.collection('templates')
.insert({name: 'test'})
.then((res) => {})
jsreport.documentStore.collection('templates')
.remove({name: 'test'})
.then((res) => {})
Changelog
4.2.0
- the response object has new structure and api, the
response.output
is new addition and contains different methods to work with the response output. the response.content
, response.stream
are now soft deprecated (will be removed in future versions) - the internal architecture has been updated to support working with template content and report output as streams, this means that the render will be able to handle and produce bigger reports without getting hit by out of memory errors
4.1.0
- update deps to fix npm audit
- fix memory being held when timeouts are large
- fix node.js 21 compatibility when sandbox is used (
trustUserCode: false
- support using different cache key per template engine if it supports it
4.0.1
- fix parameter mutations passed to store methods producing unexpected changes in store
4.0.0
- minimum node.js version is now
18.15.0
- remove old migration options
migrateXlsxTemplatesToAssets
, migrateResourcesToAssets
- sandbox now uses SES instead of vm2 for evaluating user code
- internal changes to support multi admin users
3.12.0
- update vm2 to fix security issues
- render requests are now rotated across not busy workers but considering its last usage too
- require in sandbox now uses a custom require implementation that takes care of isolate module resolving and that uses our own cache (different to built in require.cache) to avoid using a lot of memory when there are a lot of requests. if module isolation is not needed (because user can trust the templates) then it can be disabled by using
sandbox.isolateModules: false
3.11.4
- update unset-value to fix security issue
3.11.3
- update vm2 to fix security issue
- automatically disable full profiling after some time to avoid performance degradation
- improvements to full profile serialization (prevent blocking)
- fix profiles cleaning and calculate timeout in beforeRender
3.11.2
- add
options.onReqReady
to be able to receive the parsed req values
3.11.1
- fix error when trying to read
req.options
on reporter main code when enableRequestReportTimeout
is enabled
3.11.0
- log when worker returns bad res.content
- fix profiler leaks
- remove settings sync API and avoid loading all items to memory
- throw weak error when validating duplicated entity
- ensure we end with profiles with error state when there is server or req timeout
3.10.0
mainReporter.executeWorkerAction
now supports cancellation with AbortController.signal
- add support for specifying what are the main document properties of templates entitySet
3.9.0
- add more store methods
collection.findAdmin
, collection.findOneAdmin
, reporter.adminRequest
to easily allow execure store queries without taking into account permissions - improve logging for child requests and user level logs
- differentiate between template not found errors and permissions related errors (it is now more clean what is the cause of specific error)
- normalize to error when non-errors are throw (like throw "string")
- improve errors in helpers (it now includes the helper name)
- improve error message when template was not found in child request
- improve error handling in sandbox
3.8.1
- update vm2 for fix security issue
3.8.0
- make "config.json" a reserved name for entities
3.7.0
- add support for multiple source entities when copying and moving
- fix some issues with blobs and profiles
- format user logs differently on stdout
- fix logging of req as http.IncomingRequest
- fix profile compatibility with jsreport container based execution
- fix memory leak in profiling
- add support for logs of main reporter to show in profile
3.6.1
- update @jsreport/advanced-workers to fix bug with cloning req.data when
trustUserCode
is true
3.6.0
- improve handling of worker exit
- add a way to disable the safe execution and prevent the performance penalty
- cache system helpers function
profiler.maxDiffSize
applies to both req and res- improved the support for running jsreport as serverless function
- added new apis in
jsreport-proxy
for better working with async helpers
3.5.0
- fix parsing issue of code with comment in the sandbox (helpers, scripts)
- improve profiling when there is big data
- make transactions support in store configurable
- improve timeout for the whole request
- fix applying req.options.timeout when enableRequestReportTimeout is true
- optimization regarding profile persistence
3.4.2
- update dep
vm2
to fix security vulnerability in sandbox
3.4.1
- fix passing data to async report
- fix blob appends
3.4.0
- fix for reports execution
- fix for render profiling
- fix for blob storage remove
- update deps to fix npm audit
3.1.0
- fix blob storage append to not existing blob (mongo)
- use relative path to the currently evaluated entity
- fix performance issue in sandbox with long buffer (don't use restore() of sandbox through a method attached to the sandbox)
- update migration
xlsxTemplatesToAssets
, resourcesToAssets
to inherit permissions and change name for resource script to ${template.name}_resources
- refactor ListenerCollection for better stack traces
- fix startup extensions logs not recognized as npm source
- fix extensions cache entry root path
- don't crash process when monitoring persist fails
- fix compilation (updates to vm2)
License
LGPL