<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>DEV Community: Bohdan Shtelmakh</title>
    <description>The latest articles on DEV Community by Bohdan Shtelmakh (@bohard).</description>
    <link>https://dev.to/bohard</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F3858093%2Fe31afc6a-3466-4f23-b06d-66b984bdd2f8.png</url>
      <title>DEV Community: Bohdan Shtelmakh</title>
      <link>https://dev.to/bohard</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/bohard"/>
    <language>en</language>
    <item>
      <title>I built an HTML-to-PDF engine 10x faster than Puppeteer</title>
      <dc:creator>Bohdan Shtelmakh</dc:creator>
      <pubDate>Thu, 02 Apr 2026 17:37:49 +0000</pubDate>
      <link>https://dev.to/bohard/i-built-an-html-to-pdf-engine-10x-faster-than-puppeteer-43pa</link>
      <guid>https://dev.to/bohard/i-built-an-html-to-pdf-engine-10x-faster-than-puppeteer-43pa</guid>
      <description>&lt;p&gt;If you've ever generated PDFs on a backend — invoices, reports, receipts — you've probably used Puppeteer or Playwright. They work, but they're slow.&lt;/p&gt;

&lt;p&gt;Spinning up a headless Chrome instance to render a simple invoice takes &lt;strong&gt;2–4 seconds&lt;/strong&gt;, eats &lt;strong&gt;300+ MB of RAM&lt;/strong&gt;, and cold starts in serverless environments are brutal.&lt;/p&gt;

&lt;p&gt;I needed something faster. So I built &lt;strong&gt;html-pdf-lite&lt;/strong&gt; — a lightweight HTML-to-PDF converter for Node.js that skips Chrome entirely.&lt;/p&gt;

&lt;h2&gt;
  
  
  How fast?
&lt;/h2&gt;

&lt;div class="table-wrapper-paragraph"&gt;&lt;table&gt;
&lt;thead&gt;
&lt;tr&gt;
&lt;th&gt;Engine&lt;/th&gt;
&lt;th&gt;Time per PDF&lt;/th&gt;
&lt;th&gt;Memory&lt;/th&gt;
&lt;/tr&gt;
&lt;/thead&gt;
&lt;tbody&gt;
&lt;tr&gt;
&lt;td&gt;Puppeteer&lt;/td&gt;
&lt;td&gt;2–4s&lt;/td&gt;
&lt;td&gt;~300 MB&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;strong&gt;html-pdf-lite&lt;/strong&gt;&lt;/td&gt;
&lt;td&gt;&lt;strong&gt;200–400ms&lt;/strong&gt;&lt;/td&gt;
&lt;td&gt;~50 MB&lt;/td&gt;
&lt;/tr&gt;
&lt;/tbody&gt;
&lt;/table&gt;&lt;/div&gt;

&lt;p&gt;That's &lt;strong&gt;10x faster&lt;/strong&gt; with &lt;strong&gt;6x less memory&lt;/strong&gt;.&lt;/p&gt;

&lt;h2&gt;
  
  
  How it works
&lt;/h2&gt;

&lt;p&gt;Instead of launching a browser, html-pdf-lite:&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;Parses HTML with JSDOM&lt;/li&gt;
&lt;li&gt;Applies CSS cascade (specificity, inheritance, variables, shorthands)&lt;/li&gt;
&lt;li&gt;Renders directly to PDF using PDFKit&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;No Chrome. No Playwright. No browser binary to install.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npm &lt;span class="nb"&gt;install &lt;/span&gt;html-pdf-lite
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;





&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight javascript"&gt;&lt;code&gt;&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;renderPdfFromHtml&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nf"&gt;require&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;html-pdf-lite&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;fs&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nf"&gt;require&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;fs&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;html&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="s2"&gt;`
  &amp;lt;h1&amp;gt;Invoice #1042&amp;lt;/h1&amp;gt;
  &amp;lt;table&amp;gt;
    &amp;lt;tr&amp;gt;&amp;lt;td&amp;gt;Widget&amp;lt;/td&amp;gt;&amp;lt;td&amp;gt;$10.00&amp;lt;/td&amp;gt;&amp;lt;/tr&amp;gt;
    &amp;lt;tr&amp;gt;&amp;lt;td&amp;gt;Gadget&amp;lt;/td&amp;gt;&amp;lt;td&amp;gt;$25.00&amp;lt;/td&amp;gt;&amp;lt;/tr&amp;gt;
  &amp;lt;/table&amp;gt;
  &amp;lt;p&amp;gt;&amp;lt;strong&amp;gt;Total: $35.00&amp;lt;/strong&amp;gt;&amp;lt;/p&amp;gt;
`&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;pdf&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nf"&gt;renderPdfFromHtml&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;html&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;span class="nx"&gt;fs&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;writeFileSync&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;invoice.pdf&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;pdf&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;That's it. One function, returns a Buffer.&lt;/p&gt;

&lt;h2&gt;
  
  
  What it supports
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;Headings, paragraphs, spans, links&lt;/li&gt;
&lt;li&gt;Tables with colspan and styled cells&lt;/li&gt;
&lt;li&gt;Flexbox and CSS Grid layout&lt;/li&gt;
&lt;li&gt;Ordered and unordered lists&lt;/li&gt;
&lt;li&gt;Images (PNG, JPEG, SVG, data URIs)&lt;/li&gt;
&lt;li&gt;Inline SVG rendering&lt;/li&gt;
&lt;li&gt;Page breaks (&lt;code&gt;break-before&lt;/code&gt;, &lt;code&gt;break-inside: avoid&lt;/code&gt;)&lt;/li&gt;
&lt;li&gt;Internal PDF links (&lt;code&gt;href="#section"&lt;/code&gt;)&lt;/li&gt;
&lt;li&gt;CSS variables, shorthands, specificity cascade&lt;/li&gt;
&lt;li&gt;Custom fonts with system font auto-detection&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;@page&lt;/code&gt; margin rules&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  What it doesn't support
&lt;/h2&gt;

&lt;p&gt;This is &lt;strong&gt;not&lt;/strong&gt; a full browser engine. If you need:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Perfect flexbox/grid parity with Chrome&lt;/li&gt;
&lt;li&gt;JavaScript-rendered content&lt;/li&gt;
&lt;li&gt;Advanced CSS (animations, transforms, filters)&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Use Puppeteer. That's the right tool for those cases.&lt;/p&gt;

&lt;p&gt;html-pdf-lite is for when you have a &lt;strong&gt;known HTML template&lt;/strong&gt; and need a PDF &lt;strong&gt;fast&lt;/strong&gt; — APIs, invoices, reports, serverless functions.&lt;/p&gt;

&lt;h2&gt;
  
  
  Perfect for serverless
&lt;/h2&gt;

&lt;p&gt;No Chromium binary means:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Deploys to &lt;strong&gt;AWS Lambda&lt;/strong&gt; without layers or custom runtimes&lt;/li&gt;
&lt;li&gt;Works on &lt;strong&gt;Vercel&lt;/strong&gt;, &lt;strong&gt;Cloudflare Workers&lt;/strong&gt;, &lt;strong&gt;Railway&lt;/strong&gt;
&lt;/li&gt;
&lt;li&gt;Cold starts in milliseconds, not seconds&lt;/li&gt;
&lt;li&gt;Fits in small container images&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Open source
&lt;/h2&gt;

&lt;p&gt;MIT licensed. 11 test suites. CI on every push.&lt;/p&gt;

&lt;p&gt;GitHub: &lt;a href="https://github.com/BohdanShtelmakh/html-to-pdf" rel="noopener noreferrer"&gt;github.com/BohdanShtelmakh/html-to-pdf&lt;/a&gt;&lt;br&gt;
npm: &lt;a href="https://www.npmjs.com/package/html-pdf-lite" rel="noopener noreferrer"&gt;npmjs.com/package/html-pdf-lite&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;If it helped you, star the repo. If something doesn't render right, open an issue with a minimal HTML example and I'll fix it.&lt;/p&gt;

</description>
      <category>node</category>
      <category>javascript</category>
      <category>pdf</category>
      <category>webdev</category>
    </item>
  </channel>
</rss>
