Настройка Varnish для кэширования Magento 2

Наша компания занимается разработкой, поддержкой и обслуживанием сайтов любой сложности. От простых одностраничных сайтов до масштабных кластерных систем построенных на микро сервисах. Опыт разработчиков подтвержден сертификатами от вендоров.

Разработка и обслуживание любых видов сайтов:

Информационные сайты или веб-приложения
Сайты визитки, landing page, корпоративные сайты, онлайн каталоги, квиз, промо-сайты, блоги, новостные ресурсы, информационные порталы, форумы, агрегаторы
Сайты или веб-приложения электронной коммерции
Интернет-магазины, B2B-порталы, маркетплейсы, онлайн-обменники, кэшбэк-сайты, биржи, дропшиппинг-платформы, парсеры товаров
Веб-приложения для управления бизнес-процессами
CRM-системы, ERP-системы, корпоративные порталы, системы управления производством, парсеры информации
Сайты или веб-приложения электронных услуг
Доски объявлений, онлайн-школы, онлайн-кинотеатры, конструкторы сайтов, порталы предоставления электронных услуг, видеохостинги, тематические порталы

Это лишь некоторые из технических типов сайтов, с которыми мы работаем, и каждый из них может иметь свои специфические особенности и функциональность, а также быть адаптированным под конкретные потребности и цели клиента

Предлагаемые услуги
Показано 1 из 1 услугВсе 2065 услуг
Настройка Varnish для кэширования Magento 2
Сложная
~2-3 рабочих дня
Часто задаваемые вопросы

Наши компетенции:

Этапы разработки

Последние работы

  • image_website-b2b-advance_0.png
    Разработка сайта компании B2B ADVANCE
    1262
  • image_web-applications_feedme_466_0.webp
    Разработка веб-приложения для компании FEEDME
    1171
  • image_websites_belfingroup_462_0.webp
    Разработка веб-сайта для компании БЕЛФИНГРУПП
    874
  • image_ecommerce_furnoro_435_0.webp
    Разработка интернет магазина для компании FURNORO
    1094
  • image_crm_enviok_479_0.webp
    Разработка веб-приложения для компании Enviok
    831
  • image_bitrix-bitrix-24-1c_fixper_448_0.png
    Разработка веб-сайта для компании ФИКСПЕР
    851

Настройка Varnish для кэширования Magento 2

Varnish — HTTP-акселератор, работающий перед веб-сервером. Для Magento 2 это стандартный production-стек: без него сайт с 500+ одновременными посетителями уходит в CPU-saturate на PHP-FPM. Magento 2 имеет встроенную поддержку Varnish через ESI и X-Magento-Tags для инвалидации кэша.

Архитектура стека

Браузер → Varnish :80/:443 → Nginx :8080 → PHP-FPM → MySQL/Redis
                         ↘ ESI requests (блоки корзины, авторизация)

Varnish слушает на 80/443, nginx перенесён на порт 8080 и принимает только трафик от Varnish. Для SSL termination перед Varnish ставится nginx или HAProxy на 443 с проксированием на Varnish 80.

Установка Varnish

# Ubuntu 22.04
curl -s https://packagecloud.io/install/repositories/varnishcache/varnish74/script.deb.sh | sudo bash
apt install varnish

# Файл сервиса
systemctl edit varnish
[Service]
ExecStart=
ExecStart=/usr/sbin/varnishd \
  -a :80 \
  -T localhost:6082 \
  -f /etc/varnish/default.vcl \
  -s malloc,2g \
  -p thread_pools=2 \
  -p thread_pool_max=1000 \
  -p thread_pool_timeout=300

Размер -s malloc,2g — под кэш Magento достаточно 2–4 GB. Меньше 1 GB нет смысла: Magento страницы весят 80–200 KB и при маленьком кэше hit rate будет низким.

VCL-конфигурация для Magento 2

Magento 2 предоставляет готовый VCL через admin: Stores > Configuration > Advanced > System > Full Page Cache > Varnish Configuration > Export VCL. Экспортируем и используем как основу, но стандартный VCL требует правок.

Ключевые части VCL:

vcl 4.1;

import std;

backend default {
    .host = "127.0.0.1";
    .port = "8080";
    .connect_timeout = 600s;
    .first_byte_timeout = 600s;
    .between_bytes_timeout = 600s;
}

acl purge {
    "localhost";
    "127.0.0.1";
}

sub vcl_recv {
    # Передаём реальный IP
    if (req.restarts == 0) {
        if (req.http.X-Forwarded-For) {
            set req.http.X-Forwarded-For = req.http.X-Forwarded-For + ", " + client.ip;
        } else {
            set req.http.X-Forwarded-For = client.ip;
        }
    }

    # PURGE-запросы от Magento
    if (req.method == "PURGE") {
        if (!client.ip ~ purge) {
            return (synth(405, "Not allowed"));
        }
        return (purge);
    }

    # BAN по X-Magento-Tags (для инвалидации блоков)
    if (req.method == "BAN") {
        if (!client.ip ~ purge) {
            return (synth(405, "Not allowed"));
        }
        if (req.http.X-Magento-Tags-Pattern) {
            ban("obj.http.X-Magento-Tags ~ " + req.http.X-Magento-Tags-Pattern);
        }
        return (synth(200, "Banned"));
    }

    # Не кэшируем HTTPS-индикатор
    if (req.http.X-Forwarded-Proto == "https") {
        set req.http.Ssl-Offloaded = "1";
    }

    # Не кэшируем cart, checkout, account
    if (req.url ~ "/(checkout|customer|account|cart|wishlist)") {
        return (pass);
    }

    # Удаляем куки на статических ресурсах
    if (req.url ~ "\.(css|js|png|jpg|jpeg|webp|gif|ico|woff2|svg)(\?.*)?$") {
        unset req.http.Cookie;
        return (hash);
    }

    # Нормализуем Google Analytics параметры
    set req.url = regsuball(req.url, "(^|&)(utm_[a-z]+|gclid|gclsrc|fbclid)=[^&]*", "");
    set req.url = regsub(req.url, "^(.*)\?&?(.*)?$", "\1?\2");
    set req.url = regsub(req.url, "^(.*)\?$", "\1");

    return (hash);
}

sub vcl_hash {
    hash_data(req.url);
    if (req.http.host) {
        hash_data(req.http.host);
    } else {
        hash_data(server.ip);
    }

    # Разные кэши для HTTP/HTTPS
    if (req.http.Ssl-Offloaded) {
        hash_data(req.http.Ssl-Offloaded);
    }

    return (lookup);
}

sub vcl_backend_response {
    # Не кэшируем 5xx
    if (beresp.status >= 500) {
        set beresp.uncacheable = true;
        set beresp.ttl = 1s;
        return (deliver);
    }

    # TTL по типу контента
    if (beresp.http.content-type ~ "text/html") {
        set beresp.ttl = 1d;
        set beresp.grace = 1h;
    }

    if (bereq.url ~ "\.(css|js|woff2)(\?.*)?$") {
        set beresp.ttl = 1y;
    }

    # Убираем куки из кэшируемых ответов
    if (beresp.ttl > 0s) {
        unset beresp.http.Set-Cookie;
    }

    return (deliver);
}

sub vcl_deliver {
    # Отладочные заголовки (отключить на prod)
    if (obj.hits > 0) {
        set resp.http.X-Cache = "HIT";
        set resp.http.X-Cache-Hits = obj.hits;
    } else {
        set resp.http.X-Cache = "MISS";
    }

    # Убираем внутренние заголовки Magento
    unset resp.http.X-Magento-Tags;
    unset resp.http.X-Powered-By;
    unset resp.http.Server;

    return (deliver);
}

Настройка Magento 2 на стороне приложения

Admin > Stores > Configuration > Advanced > System > Full Page Cache:

  • Caching Application: Varnish Cache
  • TTL for public content: 86400 (24 часа)

Admin > Stores > Configuration > Advanced > System > Full Page Cache > Varnish Configuration:

  • Access list: 127.0.0.1
  • Backend host: 127.0.0.1
  • Backend port: 8080

Включаем FPC:

bin/magento cache:enable full_page
bin/magento config:set system/full_page_cache/caching_application 2
# 2 = Varnish, 1 = Magento built-in

ESI — динамические блоки внутри кэшированных страниц

Magento 2 использует ESI (Edge Side Includes) для блоков, которые персонализированы: корзина, имя пользователя, wishlist. В VCL ESI уже включён через beresp.do_esi = true, который Magento устанавливает в заголовке X-Esi: 1.

Проверка что ESI работает:

curl -I https://myshop.ru/ | grep X-Cache
# Должен вернуть: X-Cache: HIT

curl -I https://myshop.ru/checkout/cart/ | grep X-Cache
# Должен вернуть: X-Cache: MISS (корзина не кэшируется)

Инвалидация кэша

Magento инвалидирует кэш автоматически при изменении товаров/категорий через BAN по тегам:

# Ручная инвалидация конкретной страницы
curl -X PURGE http://127.0.0.1:80/catalog/product/view/id/42

# Инвалидация всего кэша
varnishadm "ban req.url ~ /"

# Статистика кэша
varnishstat -1 -f MAIN.cache_hit -f MAIN.cache_miss

Целевой hit rate для Magento — 85–95%. Ниже 70% означает проблему: либо слишком много некэшируемых запросов, либо VCL некорректно обрабатывает куки.

Мониторинг

# Реальное время — запросы и их статус
varnishlog -q 'RespHeader:X-Cache' -i RespHeader,ReqURL,RespStatus

# Топ некэшируемых URL
varnishlog -q 'RespHeader:X-Cache eq MISS' -i ReqURL | \
  awk '{print $2}' | sort | uniq -c | sort -rn | head -20

Nginx как SSL terminator

server {
    listen 443 ssl http2;
    server_name myshop.ru;

    ssl_certificate /etc/letsencrypt/live/myshop.ru/fullchain.pem;
    ssl_certificate_key /etc/letsencrypt/live/myshop.ru/privkey.pem;

    location / {
        proxy_pass http://127.0.0.1:80;
        proxy_set_header Host $host;
        proxy_set_header X-Real-IP $remote_addr;
        proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
        proxy_set_header X-Forwarded-Proto https;
        proxy_set_header Ssl-Offloaded "1";
    }
}

Сроки работ

Установка Varnish, настройка VCL под Magento 2, перенос nginx на 8080, настройка SSL termination: 1–2 дня. Тестирование hit rate, инвалидации, ESI-блоков: 1 день. Нагрузочное тестирование и финальная настройка параметров (malloc, thread_pools): 0.5–1 день.