WordPress is dynamic, meaning that every time a visitor accesses your site it executes many server-side processes to generate a page. It queries the database, executes PHP scripts, and builds the final page for the user. This makes it flexible and powerful but also slow, especially as traffic increases.
Static page caching solves this by serving stored copies of your fully rendered dynamic pages to visitors. This avoids all the server-side processing, but still loads static pages from a single server. If that server is in Virginia, USA and a visitor is in Germany, it may be slower than expected. CloudFlare’s edge cache solves this problem.
They operate one of the largest networks of data centers worldwide, with servers located in over 200 cities. When a visitor accesses your site, Cloudflare serves cached content from the data center closest to them, minimizing the distance data has to travel and reducing latency. For our previous use case, the visitor in Germany would load your statically cached website from a server in Germany, as close to that user as possible.
To serve a WordPress site statically through edge cache, you’ll need to set up specific caching rules in Cloudflare that will cache all public pages served by WordPress, while ignoring sensitive internal resources like /wp-admin/
and /wp-login.php
. You also need to avoid having a logged-in user store new cached versions of pages.
To do this, we’ll need two Cloudflare cache rules:
To create both of these rules, you’ll need to do the following.
This rule aggressively caches all other content that does not match the criteria set in our bypass rule.
/wp
wordpress_logged_in
The full expression syntax can be copied directly into CloudFlare if you’d like to save some time:
(not starts_with(http.request.uri.path, "/wp-")) or (not http.cookie contains "wordpress_logged_in")
This rule must be first in the order, BEFORE the bypass rule.
This rule prevents caching for any requests to WordPress admin pages or by users who are logged in.
/wp
wordpress_logged_in
The full expression syntax can be copied directly into CloudFlare if you’d like to save some time:
(starts_with(http.request.uri.path, "/wp-")) or (http.cookie contains "wordpress_logged_in")
This rule must be last in the order, AFTER the bypass rule.
Now that you have our CloudFlare cache rules in place, you need to set HTTP headers like Cache-Control
(and a few others) to tell CloudFlare when the cached rule should be active. It’s a strange step that threw me off for a while, but it seems to be needed to make things actively cache the way we want (up to a 1 year).
Add the following function to your theme’s functions.php
to set the appropriate HTTP headers:
function set_custom_cache_headers() {
if (is_user_logged_in()) {
header('Cache-Control: no-cache, no-store, must-revalidate');
header('Pragma: no-cache');
header('Expires: 0');
} else {
header('Cache-Control: public, max-age=31536000, stale-while-revalidate=3600');
}
}
add_action('send_headers', 'set_custom_cache_headers');
This will avoid storing any cached values for a logged-in user, protecting you from unwanted results such as having pages with an admin bar at the top cached for every visitor to see.
The final step is to automatically clear cache when posts, pages or other changes occur in WordPress. The simplest way to do this is to leverage the logic already provided by the Cloudflare plugin, with a set of specific configuration options.
That’s it, you now have a very good static caching system in place for your WordPress website.
By implementing these caching strategies, you’re essentially turning your WordPress site into a lightning-fast static site for logged-out users, while dynamically serving logged-in users. This setup is simple yet effective, but it may require additional fine-tuning in production environments to meet specific needs. When properly optimized, this approach can drastically enhance site performance, providing a solid foundation for building a fast and reliable WordPress site.