Michael Altfield's gravatar

WordPress Profiling with XHProf (Debugging & Optimizing Speed)

This guide will show you how to generate and view XHProf reports of your WordPress Site.

This is useful so you can drill-down and see exactly how many microseconds each of your scripts and functions (themes & plugins) are running when generating a page — slowing down your website visitors’ page load speed.

Debugging & Optimizing WordPress Speed with XHProf

Intro

Profiling is a critical tool for debugging the performance of your web application in production.

XHProf was originally developed by Facebook to profile (and optimize) their PHP web application (facebook.com). It was open-sourced in 2009.

Facebook no longer maintains XHProf, and as development continued past PHP7, the XHProf code rot lead to a number of XHProf forks, including:

In this guide, we will be using the XHProf fork maintained by Tideways — as it is available in the Debian repos and therefore can be securely downloaded & installed without downloading & compiling code from the Internet.

Assumptions

Versions

This guide was written in June 2022, and it uses the following software and versions:

  1. Debian 11 (bullseye)
  2. Apache 2.4
  3. WordPress 5.9.3
  4. php-tideways v5.0.4-2
  5. longxinH/xhprof v2.3.5

If you’re using a different OS, web server, wordpress, or xhprof versions, then adaptations to the below commands & files may be necessary.

WordPress Document Root

This guide assumes that your wordpress document root is located at the following directory

/var/www/html/wordpress/htdocs/

If your wordpress install is located in a distinct directory, then adaptations to the below commands will be necessary.

xhprof-html Document Root

This guide assumes that you will install a new `xhprof-html` vhost with a document root at the following directory

/var/www/html/xhprof/htdocs/

If you’d like to keep your web server’s document roots organized in a distinct path, then adaptations to the below commands will be necessary.

XHProf File Storage Dir

This guide assumes that you will be writing your `.xhprof` files to the following directory:

/var/www/html/wordpress/xhprof/

If you’d like to store your `.xhprof` files in a distinct path, then adaptations to the below commands will be necessary.

Installing XHProf

Most XHProf guides on the ‘net tell you to clone some random git repo and then (without doing any cryptographic authenticity nor integrity checks), proceed to compile binaries on your machine, then run `make install` as root.

There are numerous supply chain risks in blindly trusting public publishing infrastructure like that.

As such, the fork of XHProf that we choose to use for this guide is the only one that can be securely downloaded from the Debian repos using apt: `php-tideways`

Install Debian Packages

Of course, you should already have a functioning apache2 web server installed with php modules installed & enabled

sudo apt-get install apache2 libapache2-mod-php php-common

And to install the `php-tideways` extension, run

sudo apt-get install php-tideways

The above php-tideways package should have:

  1. installed the tideways_xhprof.so module on your system and
  2. added the tideways.ini file to be dynamically added to your main PHP php.ini config file

For a list of files added to your system by installing the `php-tideways` module, run

sudo dpkg-query -L php-tideways

Finally, restart apache to apply the config changes

sudo systemctl restart apache2

You can also confirm in the CLI that the tideways module is enabled in PHP with the following command:

root@host:~# php -m | grep -i tideways
tideways_xhprof
root@host:~#

Configuring WordPress to use XHProf

XHProf is a tool for profiling PHP Applications generally. As such, the official PHP documentation describing how to use XHProf usually says something like this:

<?php
tideways_xhprof_enable();
 
my_app()
 
$data = tideways_xhprof_disable();
file_put_contents( 'my_app.xhprof', serialize($data) );

So, basically:

  1. You need to enable XHProf with `tideways_xhprof_enable()` at the start of your page load,
  2. You need to disable XHProf with `tideways_xhprof_disable()` (and collect its output) at the end of your page load, and
  3. You need to write the output from the `tideways_xhprof_disable()` function above to a file somewhere (for later analysis)

…but what does that mean for a wordpress site?

Well, one solution is to integrate it into your theme’s `header.php` and `footer.php` files.

But better than `header.php` is to put it at the end of your wp-config.php file — just before the `require()` for `wp-settings.php`.

And better than `footer.php` is to use PHP’s register_shutdown_function() to disable XHProf and save our results to the `.xhprof` file.

Profile only on-demand

Modify your `wp-config.php` file so that it looks like this at the bottom:

/* That's all, stop editing! Happy publishing. */
 
/** Absolute path to the WordPress directory. */
if ( ! defined( 'ABSPATH' ) ) {
   define( 'ABSPATH', __DIR__ . '/' );
}
 
# PHP profiling with tideways-xhprof
if ( isset( $_GET['profile'] ) && $_GET['profile'] === 'secret-string' ) {
   tideways_xhprof_enable( TIDEWAYS_FLAGS_MEMORY + TIDEWAYS_FLAGS_CPU );
   register_shutdown_function( function() {
      $results = tideways_xhprof_disable();
      include_once( '/var/www/html/xhprof/htdocs/xhprof_lib/utils/xhprof_runs.php' );
      $XHProfRuns = new XHProfRuns_Default( '/var/www/html/wordpress/xhprof/' );
      $XHProfRuns->save_run( $results, date('Y-m-d\TH:i:s\Z', time() - date('Z')) );
   });
}
 
/** Sets up WordPress vars and included files. */
require_once ABSPATH . 'wp-settings.php';

The above code block will check to see if a GET variable named `profile` was defined and if it was set to `secret-string`. For example, if your wordpress website is available at https://wordpress.example.com/, then loading the following page will trigger XHProf to run and generate a timestamped `.xhprof` file in `/var/www/html/wordpress/xhprof/`

Note that running XHProf does add some performance penalty. Depending on your options, it could slow your site down 4% – 200%.

⚠ WARNING: To prevent Mallory from launching a DDoS attack on your website, you should probably change `secret-string` above to some unique passphrase that only you know 🙂

Profile every 1,000 requests

Alternatively, you can capture a random 1 out if every 1,000 requests to your wordpress site by adding something like this to the bottom of your `wp-config.php` file

/* That's all, stop editing! Happy publishing. */
 
/** Absolute path to the WordPress directory. */
if ( ! defined( 'ABSPATH' ) ) {
   define( 'ABSPATH', __DIR__ . '/' );
}
 
# PHP profiling with tideways-xhprof
if ( rand( 1, 1000 ) === 1 ) {
   tideways_xhprof_enable( TIDEWAYS_FLAGS_MEMORY + TIDEWAYS_FLAGS_CPU );
   register_shutdown_function( function() {
      $results = tideways_xhprof_disable();
      include_once( '/var/www/html/xhprof/htdocs/xhprof_lib/utils/xhprof_runs.php' );
      $XHProfRuns = new XHProfRuns_Default( '/var/www/html/wordpress/xhprof/' );
      $XHProfRuns->save_run( $results, date('Y-m-d\TH:i:s\Z', time() - date('Z')) );
   });
}
 
/** Sets up WordPress vars and included files. */
require_once ABSPATH . 'wp-settings.php';

Just make sure to periodically clean-out your XHProf File Storage Dir, as it will grow over time 🙂

XHProf File Storage Dir

Of course, we need to create the XHProf Storage Directory.

By default, XHProf generally likes to store your `.xhprof` files to the PHP `sys_temp_dir`, which is often system-wide readable & writable. That’s not very secure, so we should specify a more locked-down path.

We create the XHProf File Storage dir inside the wordpress vhost dir (but outside the document root). Make sure your web server has read/write access to this directory.

sudo mkdir -p /var/www/html/wordpress/xhprof/
sudo chown root:www-data /var/www/html/wordpress/xhprof/
sudo chmod 0770 /var/www/html/wordpress/xhprof/

xhprof_lib & xhprof_html

Simply installing `php-tideways` and updating your `wp-config.php` file with the code block above is not enough. We still need:

  1. To install the `xhprof_lib` dir (note the `include_once()` above)
  2. To install the xhprof_html` dir to actually visualize and understand the `.xhprof` files (else you’ll just have to read an enormous array dumped to a file in plaintext)

xhprof_html

There’s a lot of fancy SaaS tools for capturing and visualizing web application profiling data at scale in the cloud. This guide doesn’t cover such solutions. What we use is `xhprof_html` — which is sometimes referred to as the “poor man’s XHProf GUI”

The original (no longer maintained) Facebook xhprof repo included a simple tool to drill-down and visualize the xhprof results saved to `.xhprof` files. It’s located in a directory called `xhprof_html`

Curiously, The tideways repo itself (forked from the original repo) does not include the `xhprof_html` directory (possibly because tideways is one of these businesses that sells xhprof visualization SaaS). Rather, the tideways website actually links you to the original Facebook repo to obtain the `xhprof_lib` & `xhprof_html` directories.

[tideways] stores the trace in your temporary directory which the default UI uses to look for data. Install the xhprof_lib and xhprof_html directories from this repository into your webfolder and navigate to xhprof_html/index.php to see a list of trace.

(source)

xhprof_lib

I’ve read that the `.xhprof` file format is basically just the serialized output of the `xhprof_disable()`, but I had issues getting `xhprof_html/index.php` to read my `.xhprof` files written this way.

# this won't work
$results = tideways_xhprof_disable();
file_put_contents( '/var/www/html/wordpress/xhprof/' .date('Y-m-d\TH:i:s\Z', time() - date('Z')). '.xhprof' , serialize( $results ) );

Instead, I found that I needed to write the files using the `XHProfRuns_Default()`’s `save_run()` method.

# this will work
$results = tideways_xhprof_disable();
include_once( '/var/www/html/xhprof/htdocs/xhprof_lib/utils/xhprof_runs.php' );
$XHProfRuns = new XHProfRuns_Default( '/var/www/html/wordpress/xhprof/' );
$XHProfRuns->save_run( $results, date('Y-m-d\TH:i:s\Z', time() - date('Z')) );

As the `XHProfRuns_Default()` object is defined in `xhprof_lib/utils/xhprof_runs.php' file, this is why we need the `xhprof_lib/` dir.

vhost

Both `xhprof_lib` & xhprof_html` are available together in the same repo. Here we use the `longxinH` fork because it seems to be the most maintained fork of xhprof at the time of writing.

First, we create a new web server document root for the vhost.

sudo mkdir -p /var/www/html/xhprof/htdocs/

Now we checkout the xhprof git repo into this directory

cd /var/www/html/xhprof/htdocs/
sudo git clone https://github.com/longxinH/xhprof.git .

By default, `xhprof_lib` will write your `.xhprof` files to the PHP `sys_temp_dir`, which is often system-wide readable & writable. That’s not very secure, so we should specify a more locked-down path.

You can specify the directory location for XHProf runs in at least two places:

  1. As `xhprof.output_dir` in your `php.ini` file or
  2. As the first argument to the `XHProfRuns_Default()` constructor

We choose the latter, so make a backup of your `xhprof_html/index.php` file, and edit it so that `new XHProfRuns_Default()` becomes `new XHProfRuns_Default('/var/www/html/wordpress/xhprof/')`

root@host:/var/www/html/xhprof/htdocs/# sudo cp xhprof_html/index.php xhprof_html/index.orig.php
root@host:/var/www/html/xhprof/htdocs/# sudo vim xhprof_html/index.php
root@host:/var/www/html/xhprof/htdocs/#
 
root@host:/var/www/html/xhprof/htdocs/# diff index.orig.php index.php
83c83
< $xhprof_runs_impl = new XHProfRuns_Default();
---
> $xhprof_runs_impl = new XHProfRuns_Default( '/var/www/html/wordpress/xhprof/' );
root@host:/var/www/html/xhprof/xhprof-html#

Before we create the vhost (so apache actually serves the `/var/www/html/xhprof/htdocs` docroot), let’s create an `htpasswd` file (in the xhprof vhost dir but outside the document root) so that we can use it to password-protect the xhprof vhost (so it’s not available to everyone on the internet unless they have the username & password)

Run this command to add a user `xhprof-admin`. Type your passphrase at the prompt.

sudo htpasswd -Bc /var/www/html/xhprof/.htpasswd xhprof-admin

Now, let’s harden the permissions of these web server files

sudo chown -R root:www-data "/var/www/html/xhprof/"
sudo find "/var/www/html/xhprof/" -type d -exec chmod 0050 {} \;
sudo find "/var/www/html/xhprof/" -type f -exec chmod 0040 {} \;

Finally, we add the apache vhost file.

sudo bash -c 'cat << EOF > /etc/apache2/sites-available/xhprof.conf
<VirtualHost 127.0.0.1:443>
 
   ServerName xhprof.example.com
   DocumentRoot "/var/www/html/xhprof/htdocs"
 
   # block dot (hidden) files
   <LocationMatch "/\.(?!well\-known)">
      Require all denied
   </LocationMatch>
 
   # block config files
   <LocationMatch "config.php">
      Require all denied
   </LocationMatch>
 
   <Directory "/var/www/html/xhprof/htdocs">
      AuthType Basic
      AuthName "Protected"
      AuthUserFile /var/www/html/xhprof/.htpasswd
      Require valid-user
 
      # Harden vhost docroot by blocking all request types except the 3 essentials
      # Note: The debian wiki actually says not to use this since it can result in
      #       disabling auth modules, so be careful
      #        * https://wiki.debian.org/Apache/Hardening
      <LimitExcept GET POST HEAD>
         Require all denied
      </LimitExcept>
 
      Options -Indexes -Includes
      AllowOverride None
   </Directory>
 
</VirtualHost>
EOF'
 
sudo ln -s /etc/apache2/sites-available/xhprof.conf /etc/apache2/sites-enabled/xhprof.conf
 
sudo vim /etc/apache2/sites-available/xhprof.conf

Be sure to change:

  1. `example.com` to your domain
  2. `127.0.0.1` to your server’s IP address (if you want to serve it publicly)
  3. Add lines for your TLS cert for https, which is outside the scope of this article

Test the sanity of your apache config and restart the `apache2` service

sudo apachectl configtest && sudo systemctl restart apache2

Generating XHProf Reports

You should now be able to generate `.xhprof` files!

If you added the first code block above to your `wp-config.php` file, then just access your site and set the `profile` GET variable to your `secret-string`.

Your wordpress site will load like normal, but it will spit-out a timestamped `.xhprof` file to your XHProf Storage Directory.

root@host:/var/www/html# du -sh /var/www/html/wordpress/xhprof/*
1,8M    wordpress/xhprof/62906fb2c49b4.2022-05-27T06:29:06Z.xhprof
root@host:/var/www/html#

Viewing XHProf Reports

You should now be able to view your `.xhprof` files using your xhprof vhost (password protected username = `xhprof-admin` and the password you typed into `htpasswd` above).

After authenticating in the above page, you should be prompted with a list of `.xhprof` files.

If you click on a given report, then it will show you the list of all the functions that were called to generate the page. By default, they’re sorted by the wall time that the function call ran in microseconds (1,000,000 microseconds = 1 second)

In this example, we see that the wordpress `main()` function’s wall clock runtime was 1,498,090 microseconds = 1.5 seconds.

Screenshot shows a list of XHProf runs

The XHProf vhost shows the list of XHProf runs. Clicking on a run will show you the report.

Screenshot shows the top of an XHProf report

List of functions and their run time (in microseconds)


Once you know which functions are slow, you can drill-down into them and begin debugging to see what are your webpage generation bottlenecks.

Screenshot shows the whole first page of the xhprof report

Further Reading

Struggling to follow this guide?

If your organization has a wordpress site that’s running slow, and you can’t figure out why, please contact me about your project.

I can provide support for an hourly fee.

Related Posts

Leave a Reply

You can use these HTML tags

<a href="" title=""> <abbr title=""> <acronym title=""> <b> <blockquote cite=""> <cite> <code> <del datetime=""> <em> <i> <q cite=""> <s> <strike> <strong>