Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Move the notoptions lookup before the options cache lookup #8030

Open
wants to merge 5 commits into
base: trunk
Choose a base branch
from

Conversation

joemcgill
Copy link
Member

Trac ticket: https://core.trac.wordpress.org/ticket/62692

This is an alternate approach to #8004 which moves the check of 'notoptions' before checking the 'options' cache while leaving the 'alloptions' check first.


This Pull Request is for code review only. Please keep all other discussion in the Trac ticket. Do not merge this Pull Request. See GitHub Pull Requests for Code Review in the Core Handbook for more details.

Copy link

github-actions bot commented Dec 20, 2024

The following accounts have interacted with this PR and/or linked issues. I will continue to update these lists as activity occurs. You can also manually ask me to refresh this list by adding the props-bot label.

Unlinked Accounts

The following contributors have not linked their GitHub and WordPress.org accounts: @jonnynews.

Contributors, please read how to link your accounts to ensure your work is properly credited in WordPress releases.

Core Committers: Use this line as a base for the props when committing in SVN:

Props joemcgill, peterwilsoncc, siliconforks.

To understand the WordPress project's expectations around crediting contributors, please review the Contributor Attribution page in the Core Handbook.

Copy link

Test using WordPress Playground

The changes in this pull request can previewed and tested using a WordPress Playground instance.

WordPress Playground is an experimental project that creates a full WordPress instance entirely within the browser.

Some things to be aware of

  • The Plugin and Theme Directories cannot be accessed within Playground.
  • All changes will be lost when closing a tab with a Playground instance.
  • All changes will be lost when refreshing the page.
  • A fresh instance is created each time the link below is clicked.
  • Every time this pull request is updated, a new ZIP file containing all changes is created. If changes are not reflected in the Playground instance,
    it's possible that the most recent build failed, or has not completed. Check the list of workflow runs to be sure.

For more details about these limitations and more, check out the Limitations page in the WordPress Playground documentation.

Test this pull request with WordPress Playground.

if ( isset( $alloptions[ $option ] ) ) {
$value = $alloptions[ $option ];
} else {
// Check for non-existent options first to avoid unnecessary object cache lookups and DB hits.
$notoptions = wp_cache_get( 'notoptions', 'options' );
Copy link

@jonnynews jonnynews Dec 20, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Would it make sense to use wp_cache_get_multiple here?

$options = wp_cache_get_multiple( array( 'notoptions', $option ), 'options' );
$notoptions =  $options['notoptions'] ?? array();
$value =  $options[$option] ?? false;

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't think so, because the goal is to avoid the call to the object cache for the option at all if it doesn't exist. If we could get the alloptions and notoptions cache values in one request, that would be an improvement, but not a very meaningful one.

One thing to consider about the call to wp_cache_get( 'notoptions', 'options' ); is that the first time it is called, it will do check the external object cache if available, but then additional calls in the same request will already have that value in memory.

// Prevent non-existent `notoptions` key from triggering multiple key lookups.
if ( ! is_array( $notoptions ) ) {
$notoptions = array();
wp_cache_set( 'notoptions', $notoptions, 'options' );

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It appears that this line (initializing notoptions in the cache to an empty array) is removed entirely in this PR? Is that intentional? Is this line no longer needed?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good spot! I was thinking that there was no real purpose in caching an empty value here. However, as @peterwilsoncc's tests demonstrate, without that line each additional time we check for 'notoptions' a call will be made to the external object cache rather than returning the empty array from memory.

I've added that back in e3c900e and it looks like that fixes the problem:

  • exists, autoload: 1
  • exists, not autoloaded: 3
  • does not exist: 3

@peterwilsoncc
Copy link
Contributor

peterwilsoncc commented Dec 23, 2024

I wrote some rough tests for this and while it fixes the problem for a non-existent option, it appears to introduce the same problem to an option that exists but is not autoloaded.

These tests pass with this patch but as you can see, the cache hits for exists, not autoloaded goes up from three to twelve. The numbers for trunk are in comments in the data provider.

The test I was running follows. It will almost certainly break on some of the third party tests so is not ready for commit.

/**
 * Test that get_option() does not hit the external cache multiple times for the same option.
 *
 * @ticket 62692
 *
 * @dataProvider data_get_option_does_not_hit_the_external_cache_multiple_times_for_the_same_option
 *
 * @param int    $expected_connections Expected number of connections to the memcached server.
 * @param bool   $option_exists        Whether the option should be set. Default true.
 * @param string $autoload             Whether the option should be auto loaded. Default true.
 */
public function test_get_option_does_not_hit_the_external_cache_multiple_times_for_the_same_option( $expected_connections, $option_exists = true, $autoload = true ) {
	if ( ! wp_using_ext_object_cache() ) {
		$this->markTestSkipped( 'This test requires an external object cache.' );
	}

	if ( ! function_exists( 'wp_cache_get_stats' ) ) {
		$this->markTestSkipped( 'This test requires the Memcached PECL extension.' );
	}

	if ( $option_exists ) {
		add_option( 'ticket-62692', 'value', '', $autoload );
	}

	wp_cache_delete_multiple( array( 'ticket-62692', 'notoptions', 'alloptions' ), 'options' );

	$stats             = wp_cache_get_stats();
	$connections_start = array_shift( $stats )['cmd_get'];

	$call_getter = 10;
	while ( $call_getter-- ) {
		get_option( 'ticket-62692' );
	}

	$stats           = wp_cache_get_stats();
	$connections_end = array_shift( $stats )['cmd_get'];

	$this->assertSame( $expected_connections, $connections_end - $connections_start );
}

/**
 * Data provider.
 *
 * @return array[]
 */
public function data_get_option_does_not_hit_the_external_cache_multiple_times_for_the_same_option() {
	return array(
		'exists, autoload'       => array( 1, true, true ), // 1 on trunk.
		'exists, not autoloaded' => array( 12, true, false ), // 3 on trunk.
		'does not exist'         => array( 3, false ), // 12 on trunk.
	);
}

@joemcgill
Copy link
Member Author

As @siliconforks noticed, I had failed to cache the empty 'notoptions' array, which was causing extra hits to external cache when the option existed and was not autoloaded, rather than pulling the 'notoptions' value from memory. This has been addressed in e3c900e.

@joemcgill
Copy link
Member Author

@peterwilsoncc I went ahead and committed your tests to this PR but noticed you mention this in Trac:

...they need work so as not to break host tests using a different persistent cache implementation

Can you share more details about what needs to be updated?

@peterwilsoncc
Copy link
Contributor

peterwilsoncc commented Dec 24, 2024 via email

@peterwilsoncc
Copy link
Contributor

@joemcgill I've taken the liberty of pushing an update to my earlier tests to your branch. edae415 introduces a helper to get the number of get calls to the object cache.

Your changes in e3c900e cause Tests_Option_Option::test_get_option_notoptions_do_not_load_cache() to fail, as the notoptions cache is now expected to exist whenever get_option() is called, regardless of whether the option exists or not. The test was introduced in Core-58277 so it might be a simple case of removing it.

@joemcgill
Copy link
Member Author

Thanks @peterwilsoncc. I think the new unit test added in c3c6b56 is sufficient so that we can just remove the failing test that is no longer accurately representing the expected behavior. I've removed that test and added made a few small tweaks to the existing ones. I think this change is pretty much ready for final sign-off.

@joemcgill joemcgill mentioned this pull request Jan 3, 2025
@joemcgill joemcgill self-assigned this Jan 3, 2025
Copy link
Contributor

@peterwilsoncc peterwilsoncc left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This looks good to me and works as expected doing manual testing (see the gist used for local tests) with the memcached implementation used in the test suite.

Copy link
Contributor

@rmccue rmccue left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This looks good to me and restores the intention of notoptions without breaking the improvement aimed for previously.

For reference, I thought it'd be handy to spell out the change's impacts.

Before

The paths now are:

Option exists, is autoloaded:

  • Get alloptions from cache, once per request
  • Look up from array

That's best case 0 calls, average case 0 calls, worst case 1 call

Option exists, not autoloaded:

  • Get alloptions from cache, once per request
  • Look up from array
  • Get option from cache

That's best case 1 call, average case 1 call, worst case 2 calls

Option doesn't exist, in notoptions

  • Get alloptions from cache, once per request
  • Look up from array
  • Get option from cache
  • Get notoptions from cache, once per request
  • Look up from array

That's best case 1 call, average case 1 call, worst case 3 calls

(Assume stored in notoptions already and no DB fallbacks.)

After

The paths now are:

Option exists, is autoloaded:

  • Get alloptions from cache, once per request
  • Look up from array

That's best case 0 calls, average case 0 calls, worst case 1 call

Option exists, not autoloaded:

  • Get alloptions from cache, once per request
  • Look up from array
  • Get notoptions from cache, once per request
  • Look up from array
  • Get option from cache

That's best case 1 call, average case 1 call, worst case 3 calls

Option doesn't exist

  • Get alloptions from cache, once per request
  • Look up from array
  • Get notoptions from cache, once per request
  • Look up from array

That's best case 0 calls, average case 0 calls, worst case 2 calls


So, to some degree this improves the best case for not-exist at the expense of the worst case for exists-no-autoload, but that worst case is only hit a maximum of once per request, whereas the average case for not-exist is potentially per call.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
Status: Code Review 👀
Development

Successfully merging this pull request may close these issues.

5 participants