Mend.io Vulnerability Database
The largest open source vulnerability database
What is a Vulnerability ID?
New vulnerability? Tell us about it!
CVE-2026-41311
Published:April 26, 2026
Updated:April 26, 2026
Summary A circular block reference in "{% layout %}" / "{% block %}" causes an infinite recursive loop, consuming all available memory (~4GB) and crashing the Node.js process with "FATAL ERROR: JavaScript heap out of memory". This allows any user who can submit a Liquid template to perform a Denial of Service attack. Details In "src/tags/block.ts", during OUTPUT mode, each block looks up its render function from "ctx.getRegister('blocks')[this.block]". When a block with name "a" is nested inside another block also named "a" in a child template, the inner block finds the outer block's render function and calls it. The outer block's templates contain the inner block again, creating infinite recursion with no termination condition. Relevant code ("src/tags/block.ts", "getBlockRender" method): private getBlockRender (ctx: Context) { const { liquid, templates } = this const renderChild = ctx.getRegister('blocks')[this.block] const renderCurrent = function * (superBlock: BlockDrop, emitter: Emitter) { ctx.push({ block: superBlock }) yield liquid.renderer.renderTemplates(templates, ctx, emitter) ctx.pop() } return renderChild ? (superBlock: BlockDrop, emitter: Emitter) => renderChild( new BlockDrop( (emitter: Emitter) => renderCurrent(superBlock, emitter) ), emitter) : renderCurrent } When "renderChild" exists (same-name block found), it calls "renderChild" which re-renders templates containing the nested block, which again finds "renderChild", and so on — infinite loop. PoC 1. Create a layout file ("layout.html"): <header>{% block a %}default-a{% endblock %}</header> <main>{% block b %}default-b{% endblock %}</main> <footer>{% block c %}default-c{% endblock %}</footer> 2. Create a template that uses the layout: {% layout "layout" %} {% block a %}outer-a {% block a %}inner-a{% endblock %}{% endblock %} {% block b %}content-b{% endblock %} {% block c %}content-c{% endblock %} 3. Render: const { Liquid } = require('liquidjs') const liquid = new Liquid({ root: './', extname: '.html' }) liquid.renderFile('template').then(console.log) // Result: process hangs, memory grows to ~4GB, then crashes with OOM The anonymous block variant also triggers the same issue: {% layout "parent" %} {%block%}A{%block%}B{%endblock%}{%endblock%} Impact Denial of Service (DoS). Any application that accepts user-provided or user-influenced Liquid templates — such as CMS platforms, email template builders, multi-tenant SaaS products, or static site generators with untrusted input — can be crashed by a single malicious template. The attack requires no authentication beyond the ability to submit a template, and no special configuration. The Node.js process is killed by the OS due to memory exhaustion, causing complete service disruption.
Affected Packages
liquidjs (CDN_JS):
Affected version(s) >=10.8.0 <v10.25.6
Fix Suggestion:
Update to version v10.25.6
https://github.com/harttle/liquidjs.git (GITHUB):
Affected version(s) >=v1.2.0 <v10.25.6
Fix Suggestion:
Update to version v10.25.6
liquidjs (NPM):
Affected version(s) >=1.9.1 <10.25.7
Fix Suggestion:
Update to version 10.25.7
liquidjs (NPM):
Affected version(s) >=1.9.1 <v10.25.6
Fix Suggestion:
Update to version v10.25.6
Do you need more information?
Contact Us
CVSS v4
Base Score:
8.7
Attack Vector
NETWORK
Attack Complexity
LOW
Attack Requirements
NONE
Privileges Required
NONE
User Interaction
NONE
Vulnerable System Confidentiality
NONE
Vulnerable System Integrity
NONE
Vulnerable System Availability
HIGH
Subsequent System Confidentiality
NONE
Subsequent System Integrity
NONE
Subsequent System Availability
NONE
CVSS v3
Base Score:
7.5
Attack Vector
NETWORK
Attack Complexity
LOW
Privileges Required
NONE
User Interaction
NONE
Scope
UNCHANGED
Confidentiality
NONE
Integrity
NONE
Availability
HIGH
Weakness Type (CWE)
Uncontrolled Recursion