The Home for Magento 2 Excellence

Quality-tested Magento 2 modules. Explore. Evaluate. Elevate. #magento2

748 Modules
496 Ready
245 Need Help
🏆 Leaderboard
Actively Maintained v1.0.2

Varnish ESI Theme Context

hryvinskyi/magento2-esi-page-layout

Ensures correct theme context and cache segmentation for Varnish ESI block requests in multi-theme Magento setups. Prevents ESI blocks from being rendered with the wrong theme and skips inline rendering of TTL blocks that will be served via ESI includes.

5
Downloads
Below average
0
GitHub Stars
1mo ago
Last Release
0
Open Issues
Build Passing Strict Compliant
Ready to install

Build Tests

Composer Install
DI Compile
Templates

Code Quality

Coding Standard
PHPStan Level 8

Tested on Magento 2.4.8-p4

Recent Test History

Each release is tested against the latest Magento version at that time.

v1.0.2 on Magento 2.4.8-p4
Mar 22, 2026
v1.0.0 on Magento 2.4.8-p4
Mar 21, 2026

Share This Module's Status

Varnish ESI Theme Context Magento compatibility status badge

README

Loaded from GitHub

Varnish ESI Theme Context for Magento 2

Ensures correct theme context and cache segmentation for Varnish ESI block requests in multi-theme Magento setups.

Problem

When Varnish fetches ESI blocks via /page_cache/block/esi, the request does not carry any theme context. In a multi-theme store (e.g. Vendor/base and Venor/balckfriday), ESI blocks may be rendered and cached using the wrong theme, causing incorrect markup or broken styles.

Additionally, blocks with ttl are rendered inline during full-page generation even though they will be replaced by <esi:include> tags — wasting server resources.

Solution

This module solves both issues with three components:

1. Observer: ProcessLayoutRenderElement

Replaces the core Magento\PageCache\Observer\ProcessLayoutRenderElement (disabled in frontend scope). Adds an esi_theme parameter directly to the ESI URL during <esi:include> tag generation, so the theme path is carried through to the ESI request.

2. Plugin: RestoreEsiContextPlugin

Before plugin on Magento\PageCache\Controller\Block\Esi::execute(). Reads esi_theme from the request, sets the design theme via DesignInterface::setDesignTheme(), and adds a layout cache key (esi_theme_{path}) so layout cache is segmented per theme.

3. Plugin: SkipRenderLayoutElementPlugin

Around plugin on Magento\Framework\View\Layout::renderNonCachedElement(). When Varnish full-page cache is active and the page is cacheable, blocks with a TTL are skipped (return empty string) since they will be served via ESI includes instead.

Known Issues with Third-Party Modules

Amasty: Incorrect Entity-Specific Handle Registration

Warning: Avoid using Amasty modules if you want your site to be solid and have good performance.

Amasty modules (amasty/module-shop-by-brand, amasty/shopby) have a bug in their category controllers where they call addPageLayoutHandles() with both type and id parameters in a single call without setting $entitySpecific = false:

// Amasty (BROKEN) — marks ALL handles as entity-specific, including page-type handles
$page->addPageLayoutHandles(['type' => $type, 'id' => $category->getId()], 'catalog_category_view');

Magento core does this correctly with separate calls:

// Magento core (CORRECT) — type handles are NOT entity-specific, only id handles are
$page->addPageLayoutHandles(['type' => $pageType], null, false);
$page->addPageLayoutHandles(['id' => $category->getId()]);

This causes catalog_category_view_type_layered to be registered as an entity-specific handle. Magento's ProcessLayoutRenderElement observer then strips it from ESI URLs via array_diff(), so when Varnish fetches the ESI block, the layout is loaded without the layered navigation handle, resulting in broken block rendering on category and brand pages.

Installation

composer require hryvinskyi/magento2-esi-page-layout
bin/magento module:enable Hryvinskyi_EsiPageLayout
bin/magento setup:di:compile
bin/magento cache:flush

Configuration

No configuration required. The module activates automatically when Varnish full-page cache is enabled.

Requirements

  • Magento 2.4.x
  • PHP 8.1+

This content is fetched directly from the module's GitHub repository. We are not the authors of this content and take no responsibility for its accuracy, completeness, or any consequences arising from its use.