Gotenberg JS/TS client
A simple JS/TS client for interacting with a Gotenberg API.
Gotenberg is a Docker-powered stateless API for converting HTML, Markdown and Office documents to PDF.
- HTML and Markdown conversions using Google Chrome headless
- Office conversions (.txt, .rtf, .docx, .doc, .odt, .pptx, .ppt, .odp and so on) using unoconv
- Assets: send your header, footer, images, fonts, stylesheets and so on for converting your HTML and Markdown to beautiful PDFs!
- Easily interact with the API using Go and PHP libraries (and now - JavaScript too ;)
Install
$ yarn add gotenberg-js-client
Or using npm
$ npm install --save gotenberg-js-client
NB ⚠️
This library is not yet fully compatible with Gotenberg 7.
You can find more info in this comment.
There are three main issues:
- Gotenberg 7 has introduced new concept of conversion modules, thus, changing conversion URLs, so now they are different, than ones, this library creates. This can be sidestepped using custom connection string or adjusting URL manually (see this comment).
- New modules has some new possibilities/parameters, which are impossible to pass, using this library. This can be sidestepped using
adjust
, and casting to any
, if you use TypeScript (see this issue for reference). - Gotenberg 7 can potentially has many many different custom conversion modules, you can even write your own one. You can combine p.1 and p.2 to use any module with any path with any parameters, but I guess it will look not good. But it should work nonetheless.
So, nothing you can live without, but there are some inconveniences. For a while ;)
Usage
import { pipe, gotenberg, convert, html, please } from 'gotenberg-js-client'
const toPDF = pipe(
gotenberg('http://localhost:3000'),
convert,
html,
please
)
const pdf = await toPDF('file://index.html')
const pdf = await toPDF(fs.createReadStream('index.html'))
const pdf = await toPDF('<html>...</html>')
pdf.pipe(fs.createWriteStream('index.pdf'))
app.get('/pdf', function (req, res) {
pdf.pipe(res)
})
You can define any source like string
, Buffer
, file link, stream.Readable
, or URL
(for url conversions).
Detailed sources format you can find here.
Header, footer and assets
You can define sources as array or object, for example:
const pdf = await toPDF({
'index.html': 'file://index.html',
'header.html': 'file://header.html',
'footer.html': 'file://footer.html',
'style.css': 'file://style.css',
'img.png': 'file://img.png',
'font.wof': 'file://font.wof',
})
const pdf = await toPDF([
['index.html', 'file://index.html'],
['header.html', 'file://header.html'],
['footer.html', 'file://footer.html'],
])
const pdf = await toPDF([
'file://index.html',
'file://header.html',
'file://footer.html',
])
Instead of array you can use any iterable, like Map
, Set
, arguments
, iterator from generator function, or any object with [Symbol.iterator]
defined.
Detailed sources format you can find here.
Paper size, margins, orientation
When converting HTML or Markdown, you can use to
helper, to set paper size, margins and orientation:
import {
pipe,
gotenberg,
convert,
html,
please,
to,
a4,
landscape,
} from 'gotenberg-js-client'
const toPDF = pipe(
gotenberg('http://localhost:3000'),
convert,
html,
to(a4, landscape),
please
)
You can use simple object(s) for to
argument(s) as well:
to({
paperWidth: 8.27,
paperHeight: 11.69,
marginTop: 0,
marginBottom: 0,
marginLeft: 0,
marginRight: 0,
landscape: true,
})
to([8.27, 11.69], [0, 0, 0, 0], { landscape: true })
to({ paper: [8.27, 11.69], margins: [0, 0, 0, 0], landscape: true })
to({ width: 8.27, height: 11.69 }, { landscape: true })
to({ top: 0, bottom: 0 })
When using array for paper size, order should be [width, height]
When using array for margins, order should be [top, right, bottom, left]
(just like in CSS)
Common options
You can set common options, like resultFilename, or waitTimeout, or, actually, you can override any option, using set
helper:
set({
resultFilename: 'foo.pdf',
waitTimeout: 2.5,
})
There are some modifiers functions as well, like filename
, timeout
, delay
, webhook
and googleChromeRpccBufferSize
:
set(filename('foo.pdf'), timeout(2.5))
Also you can specify page ranges using set(range)
(will not work with merge
):
set(range('1-1'))
or scale, using set(scale)
(works with HTML, Markdown and URL conversions):
set(scale(0.75))
import { pipe, gotenberg, convert, markdown, please } from 'gotenberg-js-client'
const toPDF = pipe(
gotenberg('http://localhost:3000'),
convert,
markdown,
please
)
const pdf = await toPDF({
'index.html': `
<!doctype html>
<html lang="en">
<head>
<meta charset="utf-8">
<title>My PDF</title>
</head>
<body>
{{ toHTML .DirPath "content.md" }}
</body>
</html>`,
'content.md': `
# My awesome markdown
...
`,
})
Note: I use strings here as an example, remind that you can use other supported source type.
import {
pipe,
gotenberg,
convert,
office,
to,
landscape,
set,
filename,
please,
} from 'gotenberg-js-client'
const toPDF = pipe(
gotenberg('http://localhost:3000'),
convert,
office,
to(landscape),
set(filename('result.pdf')),
please
)
const pdf = await toPDF('file://document.docx')
Note: I use file link here as an example, remind that you can use other supported source type, say, Buffer
, or stream.Readable
:
https.get(
'https://file-examples.com/wp-content/uploads/2017/02/file-sample_100kB.docx',
async (document) => {
const pdf = await toPDF({ 'document.docx': document })
}
)
import { pipe, gotenberg, convert, url, please } from 'gotenberg-js-client'
const toPDF = pipe(
gotenberg('http://localhost:3000'),
convert,
url,
please
)
const pdf = await toPDF('https://google.com')
const pdf = await toPDF(new URL('https://google.com'))
Note: The only supported source for Url conversion is text url or instance of URL
class.
You can set remote url header (for example, for authentication or host specifying) with helper add(header)
(or add(headers)
, or both):
const toPDF = pipe(
gotenberg('http://localhost:3000'),
convert,
url,
add(
header('Foo-Header', 'Foo'),
header('Bar-Header', 'Bar'),
headers({ 'Baz1-Header': 'Baz1', 'Baz2-Header': 'Baz2' })
),
please
)
(This also applies for Webhook headers, just use webhookHeader
instead of header
and webhookHeaders
instead of headers
).
Like you would think:
import { pipe, gotenberg, merge, please } from 'gotenberg-js-client'
const toMergedPDF = pipe(
gotenberg('http://localhost:3000'),
merge,
please
)
Advanced fine adjustment
There is special function adjust
, which you can use to modify any field in prepared internal Request
object. You can check internal Request
object structure in types. Any object, passed to adjust
, will be merged with prepared Request
.
For example, you can modify url
, if your Gotenberg instance is working behind reverse proxy with some weird url replacement rules:
import { pipe, gotenberg, convert, html, adjust, please } from 'gotenberg-js-client'
const toPDF = pipe(
gotenberg('http://localhost:3000'),
convert,
html,
adjust({ url: '/hidden/html/conversion' }),
please
)
But, using that function, remember about Peter Parker principle:
"With great power comes great responsibility"
Bonus
If you happen to use this package from JavaScript, you will, obviously, lost type safety, but in return, you can use proposed pipe operator (with Babel plugin), to get beauty like this:
const toPDF = source =>
source
|> gotenberg('http://localhost:3000')
|> convert
|> html
|> to(a4, noMargins)
|> set(filename('out.pdf'))
|> please
Names clashes
If you don't like to have simple imported names in your namespace, you can use import *
syntax:
import * as got from 'gotenberg-js-client'
const toPDF = got.pipe(
got.gotenberg('http://localhost:3000'),
got.convert,
got.html,
got.please
)
Setplex OTT Platform