Make WordPress Core

Opened 13 years ago

Closed 8 years ago

Last modified 7 years ago

#17817 closed defect (bug) (fixed)

do_action/apply_filters/etc. recursion on same filter kills underlying call

Reported by: kernfel's profile kernfel Owned by: pento's profile pento
Milestone: 4.7 Priority: normal
Severity: normal Version: 2.2
Component: Plugins Keywords: has-patch needs-testing has-unit-tests
Focuses: Cc:

Description

Affects @wp-includes/plugin.php: do_action, do_action_ref_array, apply_filters, apply_filters_ref_array, _wp_call_all_hook

When calling a specific hook from a function that was called through that same hook, the remaining hooked functions from the first iteration will be discarded.
This is due to the handling of the array of registered functions using internal array pointers instead of a more robust approach.

In my example, this problem arose when I tried to programmatically delete a menu in reaction to the removal of a category. I hooked into the delete_term action to do so, upon which another function hooked into delete_term would no longer fire.
The obvious workaround is to adjust the priorities accordingly, but it shouldn't be necessary.

The current implementation as in apply_filters is:

reset( $wp_filter[ $tag ] );

if ( empty($args) )
	$args = func_get_args();

do {
	foreach( (array) current($wp_filter[$tag]) as $the_ )
		if ( !is_null($the_['function']) ){
			$args[1] = $value;
			$value = call_user_func_array($the_['function'], array_slice($args, 1, (int) $the_['accepted_args']));
		}

} while ( next($wp_filter[$tag]) !== false );

Using the following method instead, both iterations would develop properly:

if ( empty($args) )
	$args = func_get_args();

foreach ( $wp_filter[$tag] as $filters )
	foreach( (array) $filters as $the_ )
		if ( !is_null($the_['function']) ){
			$args[1] = $value;
			$value = call_user_func_array($the_['function'], array_slice($args, 1, (int) $the_['accepted_args']));
		}

Attachments (45)

test.php (5.2 KB) - added by cheeserolls 12 years ago.
Comparison of 3 different methods of looping through actions
17817.patch (1.6 KB) - added by SergeyBiryukov 11 years ago.
17817.2.patch (15.0 KB) - added by jbrinley 11 years ago.
Implement the WP_Hook_Iterator for action/filter loops
17817.3.patch (27.5 KB) - added by jbrinley 11 years ago.
17817.4.patch (22.4 KB) - added by jbrinley 11 years ago.
17817.5.patch (23.9 KB) - added by jbrinley 10 years ago.
17817.6.patch (25.3 KB) - added by jbrinley 10 years ago.
17817.6.2.patch (25.2 KB) - added by jbrinley 10 years ago.
17817.7.patch (17.2 KB) - added by leewillis77 10 years ago.
17817.8.patch (25.7 KB) - added by leewillis77 10 years ago.
17817.9.patch (26.8 KB) - added by jbrinley 10 years ago.
17817.10.patch (28.7 KB) - added by jbrinley 10 years ago.
17817.11.patch (28.3 KB) - added by jbrinley 10 years ago.
17817.diff (31.6 KB) - added by wonderboymusic 10 years ago.
17817.12.patch (29.8 KB) - added by jbrinley 10 years ago.
17817.13.patch (30.0 KB) - added by jbrinley 10 years ago.
17817.14.patch (50.3 KB) - added by jbrinley 9 years ago.
17817.2.diff (50.1 KB) - added by wonderboymusic 9 years ago.
17817.3.diff (50.1 KB) - added by jorbin 9 years ago.
17817.4.diff (50.1 KB) - added by jorbin 9 years ago.
17817.5.diff (31.0 KB) - added by DrewAPicture 9 years ago.
Docs audit
17817.6.diff (63.8 KB) - added by DrewAPicture 9 years ago.
.5 + missing tests
17817.7.diff (51.5 KB) - added by jorbin 9 years ago.
17817.8.diff (53.7 KB) - added by jbrinley 9 years ago.
17817.9.diff (53.4 KB) - added by SergeyBiryukov 9 years ago.
17817.10.diff (18.7 KB) - added by dougwollison 9 years ago.
Updated doc block for apply_filters; shouldn't claim $priority and $accepted_args are optional. Also removed random extra semicolon after an if block.
17817.10.2.diff (85.0 KB) - added by dougwollison 9 years ago.
Correct version of previous diff; a number of the new files were missing.
17817.10.3.diff (72.2 KB) - added by khag7 9 years ago.
Fixes 17817.10.2 - class-wp-hook.php has entire contents duplicated
17817.11.diff (56.7 KB) - added by pento 8 years ago.
Apply cleanly against trunk.
17817.12.diff (57.9 KB) - added by pento 8 years ago.
17817.13.diff (58.0 KB) - added by pento 8 years ago.
17817.14.diff (59.1 KB) - added by pento 8 years ago.
17817.15.diff (59.3 KB) - added by pento 8 years ago.
17817.16.diff (59.3 KB) - added by pento 8 years ago.
17817.17.diff (61.9 KB) - added by pento 8 years ago.
17817.18.diff (64.8 KB) - added by pento 8 years ago.
17817.19.diff (21.2 KB) - added by pento 8 years ago.
17817.20.diff (21.2 KB) - added by pento 8 years ago.
17817.21.diff (21.2 KB) - added by pento 8 years ago.
17817.22.diff (65.2 KB) - added by pento 8 years ago.
17817.23.diff (65.7 KB) - added by pento 8 years ago.
17817.24.diff (64.8 KB) - added by aaroncampbell 8 years ago.
17817.25.diff (1.1 KB) - added by boonebgorges 8 years ago.
17817.26.diff (868 bytes) - added by jorbin 8 years ago.
17817.27.diff (2.1 KB) - added by dd32 8 years ago.
Upgrade filters added manually in advanced-cache.php

Download all attachments as: .zip

Change History (283)

#1 @scribu
13 years ago

Array pointers are used for performance reasons.

Related: #9968

#3 @scribu
12 years ago

Another dup: #20998

#4 @SergeyBiryukov
12 years ago

  • Component changed from General to Plugins
  • Milestone Awaiting Review deleted
  • Resolution set to duplicate
  • Status changed from new to closed

#21169 has a patch.

#5 @mrubiolvn
12 years ago

  • Resolution duplicate deleted
  • Status changed from closed to reopened

#21169 does not resolve this issue. The problem here is that the if we've got recursive calls on apply_filters, when we return one level up, the array pointer might not be in the expected position(because of the reset and next/each over the same array). That'd require to make a copy of $wp_filter[ $tag ], and iterate over the copy, to be fixed.

#6 @scribu
12 years ago

@mrubiolvn: Please provide some sample code that demonstrates the bug.

#7 @SergeyBiryukov
12 years ago

  • Milestone set to Awaiting Review

#8 @mrubiolvn
12 years ago

function dummyfunc($postId) {
	$post = get_post($postId);
	$childPosts = get_posts( 'post_parent='.$postId );
	foreach($childPosts as $childPost) {
		$childPost->post_title = 'whatever';
		wp_update_post($childPost);
	}
	error_log(__FUNCTION__.' triggered for post id:'.$postId);
}
function dummyfunc2($postId) {
	error_log(__FUNCTION__.' triggered for post id:'.$postId);
}

add_action('save_post','dummyfunc',10);
add_action('save_post','dummyfunc2',15);

Now the flow would be like this:

save_post is triggered, and eventually dummyfunc is executed.
It saves a child post, so save_post is triggered again, only for the child post this time.
At this point, $wp_filter [ 'save_post' ] pointer is reset. When we're done with all the filters for the child post, and any other child posts, the execution will continue right after where we left it with the initial post. Now, we do execute next(or each with #21169) over $wp_filter [ 'save_post' ]. But the pointer will likely be on the end of the array, so dummyfunc2 will not be executed for the initial post.

#9 @foo123
12 years ago

  • Keywords 2nd-opinion added
  • Type changed from defect (bug) to enhancement
  • Version set to 3.4.1

another solution while retaining internal pointers is to use a dummy variable (since this would duplicate the intrnal pointer , according to PHP docs)

$foo=$wp_filter[ $tag ];
reset( $foo );

if ( empty($args) )
	$args = func_get_args();

do {
	foreach( (array) current($foo) as $the_ )
		if ( !is_null($the_['function']) ){
			$args[1] = $value;
			$value = call_user_func_array($the_['function'], array_slice($args, 1, (int) $the_['accepted_args']));
		}

} while ( next($foo) !== false );

#10 @foo123
12 years ago

  • Type changed from enhancement to defect (bug)

changing the type back

#11 @SergeyBiryukov
12 years ago

#23035 was marked as a duplicate.

@cheeserolls
12 years ago

Comparison of 3 different methods of looping through actions

#12 @cheeserolls
12 years ago

  • Cc dh@… added
  • Keywords has-patch added
  • Severity changed from minor to normal

foo123's solution works, but it's slow because when we execute $foo=wp_filter[$tag] the whole (often very large) array gets copied to a new variable $foo.

Instead, I suggest this version:

foreach ($wp_filter[$tag] as &$action_to_execute) {
	foreach ((array)$action_to_execute as $the_)
		if ( !is_null($the_['function']) )
			call_user_func_array($the_['function'], array_slice($args, 0, (int) $the_['accepted_args']));
}

As with foo123's solution, a new dummy variable is used - $action_to_execute - but it's created by reference, so the array isn't copied. All nested actions get executed. Furthermore, I did some benchmarking tests, and found that this is actually quicker than the current implementation. (see above attached file).

I think this really should be fixed ASAP.

It's a serious bug, because a plugin author cannot safely call wordpress functions like wp_update_post() from inside an action. (Unless he/she first reads the entire source code for wp_update_post() and all other functions that are used, to make sure that the same action isn't called).

Also, it specifically affects blogs with both WPML and ACF plugins installed (2 of the most common plugins in WP). ACF and WPML both use the 'save_posts' action, and this bug prevents ACF's 'save_post' action being called.

#13 @scribu
12 years ago

  • Keywords 2nd-opinion removed
  • Milestone changed from Awaiting Review to 3.6

If there's no performance degradation, we should give this a shot.

#14 @scribu
12 years ago

  • Keywords needs-unit-tests added

But unit tests are essential.

#15 @toscho
12 years ago

  • Cc info@… added

#16 @cyberhobo
12 years ago

  • Cc cyberhobo@… added

#17 follow-up: @jbrinley
12 years ago

  • Cc jonathanbrinley@… added

I don't agree with this change. If you move to a foreach loop, you'll lose the ability to add another callback to the same hook with a later priority.

function my_first_callback( $post_id, $post ) {
  if ( $post->post_type == 'pistachio' ) {
    add_action('save_post', 'my_second_callback', 15, 2);
  }
}
function my_second_callback( $post_id, $post ) {
  // do something else
}
add_action('save_post', 'my_first_callback', 10, 2);

I had always assumed that having this ability was a conscious design decision, not a bug. Removing it would definitely break some plugins.

I wrote a blog post a couple of years back about this issue (http://xplus3.net/2011/08/18/wordpress-action-nesting/), and the workaround is fairly simple. In my opinion, it should be up to a plugin author to take care of the housekeeping if he decides to nest action calls.

#18 @cheeserolls
12 years ago

Jbrinley, I don't agree with your comment.

You said:

In my opinion, it should be up to a plugin author to take care of the housekeeping if he decides to nest action calls.

My whole point is that it usually won't be a conscious decision by a plugin author to nest action calls. They won't be directly calling do_action('something'). But they might well use another wordpress function, which they read about in the wordpress API, which inadvertently sets off some other actions.

For example, I discovered this 'bug' through some strange behaviour in the Wordpress Multilingual plugin. They need to automatically update post B when post A is saved. So of course they hooked into the 'save_post' action when post A is saved, and called wp_update_post() on post B. This seems like a completely logical and sensible approach, but it 'accidentally' caused a nested 'save_post' action, and the inner action broke the outer action.

You're saying, the plugin author should be left to clean up his own mess. But how on earth would the plugin author be expected to know he'd made a mess in the first place? You cannot expect a standard wordpress function called in an action, to prevent the rest of the actions executing. The only way a plugin author could know, would be to have read the source code of do_action, wp_update_post, and everything else that stemmed from it.

I appreciate your point that you won't be able from inside a hook, to add another callback on the same hook. I must admit that it did not occur to me. But there's an easy workaround for that too: A plugin author just has to put the conditional logic inside the second callback instead of the first. In your example:

function my_first_callback( $post_id, $post ) {
  // do something
}
function my_second_callback( $post_id, $post ) {
  if ( $post->post_type == 'pistachio' ) {
    // do something else
  }
}
add_action('save_post', 'my_first_callback', 10, 2);
add_action('save_post', 'my_second_callback', 15, 2);

In contrast the workaround you suggest for plugins in your blog post (Remembering the position of the array pointer, then manually finishing the do_action loop after you break it.) is torturous and error prone, and depends on plugin authors having huge amount of knowledge about WP's internal code. It's against the whole spirit of plugins.

It violates the 'principle of least astonishment'.

A plugin author who sets up new actions after a sequence of actions has already started would surely not be too surprised to find it didn't work. And would easily be able to fix it.

A plugin author who calls a wp function inside his action, would be astonished to find that he had inadvertently prevented all other plugins that used the same action from working.

Furthermore, if a plugin author has a problem because he can't add new callbacks to the same hook, at least he will break his own plugin, and no-one else's. And he'll therefore know that it's not working and have a chance to fix it. The current behaviour is much nastier, because a plugin author is likely to instead break other plugins, and won't even know that he's done it.

#19 @jbrinley
12 years ago

If it were a matter of creating the WP plugin API from scratch, I would agree with you. But you're talking about a backwards-incompatible change; plugins will break. We need to approach that much more cautiously.

The only way a plugin author could know, would be to have read the source code of do_action, wp_update_post, and everything else that stemmed from it.

Or we could update the codex so that users of the function know that they'll be triggering certain hooks.

If you want to go forward with this, the first step is to figure out a way to deprecate the old functionality without breaking it. E.g., throw a _deprecated_argument warning or some such when calling add_action/add_filter on the currently running hook. Keep that in there for a version or two before breaking things. Set a target version a bit in the future so plugin developers know when their code will stop working. This is a big change; people deserve the notice.

#20 @MZAWeb
12 years ago

  • Cc wordpress@… added

#21 @cheeserolls
12 years ago

Well, again, some good points. A backwards-incompatible change which breaks plugins isn't very nice. But I find the idea of waiting a few versions to fix something which must surely be considered a bug is unappealing too.

I've just had a really good look at the plugin.php code, and have come up with the following alternative approach:

Still use current() and next() to loop through the actions/filters, thus keeping the efficiency of not copying the array, and the ability to add actions to a hook from inside the execution of that same hook. But, give the do_acttion() / apply_filter() functions responsibility for keeping track of nestings.

There is already a global var called $wp_current_filter which maintains a stack of currently executing filters. At the start of execution, the name of the filter is pushed onto it, and then popped off at the end. This variable is only used by the current_filter() function, which just returns the name of the currently executing filter.

That stack could also be used to store the POSITION of the array pointer at the currently executing filter. Then, if the same filter is nested, when we return to the outer filter, the saved position could be used to continue executing the other filters from the point that it left off.

I think this approach would not change any previous behaviour, and therefore not break existing plugins, but it would stop an inner action from breaking the outer one. And so would probably fix a lot of plugins which didn't even know they were broken.

Any thoughts?

#22 @Chouby
11 years ago

  • Cc frederic.demarle@… added

#23 in reply to: ↑ 17 @Denis-de-Bernardy
11 years ago

Replying to jbrinley:

I don't agree with this change. If you move to a foreach loop, you'll lose the ability to add another callback to the same hook with a later priority.
(...)
I had always assumed that having this ability was a conscious design decision, not a bug. Removing it would definitely break some plugins.

Hold... Are you seriously arguing that we should continue to live with breaking much about every plugin that hooks after akismet in the name of backwards compatibility? :-) (see #9968, which is the exact same issue.)

#24 @iandunn
11 years ago

  • Cc ian.dunn@… added

#25 @SergeyBiryukov
11 years ago

#24691 was marked as a duplicate.

#26 follow-up: @markjaquith
11 years ago

  • Keywords has-patch removed
  • Milestone changed from 3.6 to Future Release

No patch, punting out of 3.6.

#27 in reply to: ↑ 26 @Denis-de-Bernardy
11 years ago

Replying to markjaquith:

No patch, punting out of 3.6.

Can you explain how the patch in the ticket's description isn't good?

If it's a mere matter of copy/pasting it and creating a diff, I'm sure someone who is following this ticket will be happy to do it so it gets in 3.6.

If it's something else, please explain what and why (and probably close this and other tickets, since the suggested patch is the only rational fix.)

#28 @SergeyBiryukov
11 years ago

  • Keywords has-patch added

#29 @jbrinley
11 years ago

  • Keywords needs-testing added; needs-unit-tests removed

I think the correct answer to this and related tickets (#9968, #21169, and other duplicates) is to create a iterator that more accurately encapsulates the array iteration while tracking newly added/removed callbacks. The patch I'm submitting (https://gist.github.com/jbrinley/7024059, I'll attach in just a moment) should do so in a backwards-compatible way. I had three key objectives with this:

  1. Maintain backwards compatibility
  2. Support recursive calls to functions that trigger hooks
  3. Support manipulation of hook callbacks while the hook is running

In this patch you'll find:

  • A new class, WP_Hook_Iterator, that can be used in place of a foreach loop to get each callback for an action/filter hook
  • Updates to apply_filters, apply_filters_ref_array, do_action, do_action_ref_array, and _wp_call_all_hook to use the new iterator
  • Unit tests asserting that you can call a hook recursively
  • Unit tests asserting that you can manipulate the callbacks of a hook while that hook is running

This also passes all current unit tests, including the test added for #21169 (in tests/phpunit/tests/filters.php).

Obviously, this is a key area in WordPress, so this patch deserves careful scrutiny. Are there any edge cases we're missing in the tests? How does this affect performance? Can we accomplish the same goals more efficiently?

@jbrinley
11 years ago

Implement the WP_Hook_Iterator for action/filter loops

#30 @jadpm
11 years ago

  • Cc jadpm added

#31 @lkraav
11 years ago

  • Cc leho@… added

#32 @jbrinley
11 years ago

I've run some performance benchmarking on this patch, using the code at https://gist.github.com/jbrinley/7518483.

Running 10000 times with 3 callbacks
Total Time (Old): 0.887964
Average Time (Old): 0.000042
Total Time (New): 1.961968
Average Time (New): 0.000149

Running 10000 times with 15 callbacks
Total Time (Old): 3.754449
Average Time (Old): 0.000160
Total Time (New): 8.169822
Average Time (New): 0.000606

Running 10000 times with 150 callbacks
Total Time (Old): 38.625702
Average Time (Old): 0.001561
Total Time (New): 184.439555
Average Time (New): 0.016029

Summary: calling do_action/apply_filters with the new iterator is less performant. With callbacks on three random priorities, it runs in about 354% of the time. With 15 random callbacks, that increases to 404%. Crank it up to 150 different priorities, and that increases to 1026%.

I would love for those numbers to be smaller, but a path thither is not clear to me. Using a global $merged_filters or similar doesn't work with nested hooks. If anyone has any suggestions, I'd appreciate a comment here.

How do those numbers translate to real-world performance impact? On a stock WP install on my test server, apply_filters increases from 66ms to 88ms. I also see a negligible gain of a couple of ms in add_filter by removing the unset on $merged_filters;

#33 @jbrinley
11 years ago

I've attached a new version, 17817.3.patch. This one is a bit more invasive, changing the way the $wp_filter global works (it's now an object containing an array of WP_Hook objects). This allows the iterators to watch for changes to filters, enabling a more efficient algorithm for iterating over the priorities.

Running the same benchmarking script (slightly modified to account for the new data structure):

Running 10000 times with 3 callbacks
Total Time (Old): 0.957014
Average Time (Old): 0.000042
Total Time (New): 2.570118
Average Time (New): 0.000155

Difference: 369%

Running 10000 times with 15 callbacks
Total Time (Old): 3.978540
Average Time (Old): 0.000155
Total Time (New): 9.040840
Average Time (New): 0.000469

Difference: 302%

Running 1000 times with 300 callbacks
Total Time (Old): 8.850893
Average Time (Old): 0.003229
Total Time (New): 17.300127
Average Time (New): 0.007968

Difference: 246%

Running 1000 times with 3000 callbacks
Total Time (Old): 88.171389
Average Time (Old): 0.027288
Total Time (New): 157.092921
Average Time (New): 0.059824

Difference: 219%

Summary: still less performant, but the difference actually decreases now as the number of callbacks increases.

On real page loads, I'm seeing about a 3% increase in total load time (increasing from ~660ms to ~680ms on my server) when profiling with xdebug. Using ab, I see an increase of about 5% (from ~190ms to ~200ms).

This still passes all unit tests, including when run with the flag "--group hooks,21169,17817,9968".

@jbrinley
11 years ago

#34 @jbrinley
11 years ago

Just realized that I left in some inappropriate usages of spl_object_hash. I've updated 17817.3.patch to remove them.

#35 @SergeyBiryukov
11 years ago

#26239 was marked as a duplicate.

#36 @jchristopher
11 years ago

  • Cc jonathan@… added

#37 follow-up: @rzen
11 years ago

The patch attached to #26239 is a clean and lightweight solution to the issue at hand. I'm certainly not opposed to a more robust rewrite — there are many benefits to be gained no doubt — but it would be great if we could fix the issue first, then iterate and improve from there.

#38 @sc0ttkclark
11 years ago

  • Cc lol@… added
  • Keywords dev-feedback added

#39 @TJNowell
11 years ago

  • Cc contact@… added

#40 in reply to: ↑ 37 ; follow-up: @jbrinley
11 years ago

Replying to rzen:

The patch attached to #26239 is a clean and lightweight solution to the issue at hand. I'm certainly not opposed to a more robust rewrite — there are many benefits to be gained no doubt — but it would be great if we could fix the issue first, then iterate and improve from there.

Just like all the other solutions that make a copy of the array, http://core.trac.wordpress.org/attachment/ticket/26239/26239.patch is not backwards compatible. You can't modify callbacks for a filter while it is running.

#41 in reply to: ↑ 40 @Denis-de-Bernardy
11 years ago

Replying to jbrinley:

Just like all the other solutions that make a copy of the array, http://core.trac.wordpress.org/attachment/ticket/26239/26239.patch is not backwards compatible. You can't modify callbacks for a filter while it is running.

Expanding on my earlier point in this regard, maintaining this backward compatible behavior is of dubious merit at the very best. Particularly if maintaining it involves a plugin API that is an order of magnitude slower.

Look… Whoever writes the following code expecting 'bar' to come out deserves to see their code blow up in their face:

add_action('foo', function() {
  add_action('foo', function() {
    echo 'bar';
  }
});
do_action('foo');

Yes, it currently works in WP. But it shouldn't: you don't catch a plane that is already departed.

On the contrary, the following *should* output 'bar' and doesn't work:

function foo() {
  remove_action('foo', 'foo');
}
add_action('foo', 'foo');

function bar() {
  echo('bar');
}
add_action('foo', 'bar');

do_action('foo');

Our current approach is to crash the entire plane if anyone wants to get off.

Trying the same examples in a framework with a sane observer pattern implementation will yield the expected results. Whereas here we are haggling over keeping some manifestly broken behavior around for the sake of maintaining backwards compatibility that nobody cares about.

Can we please stop this madness?

/rant

#42 follow-up: @sc0ttkclark
11 years ago

@Denis-de-Bernardy: What is your suggestion? Just wanted to be sure it was clear what you were championing, could you clarify or suggest something different?

@SergeyBiryukov: The patch http://core.trac.wordpress.org/attachment/ticket/17817/17817.3.patch -- is this is something that can move forward since it passes unit tests, or does this require further feedback/direction from core devs? I know it's too big of a change to make 3.8, but what are the next steps to get this ticket resolved in *some* way?

#43 @wonderboymusic
11 years ago

The code looks great, but it can't be that slow

#44 @jbrinley
11 years ago

@sc0ttkclark - I'm not quite comfortable saying that 17817.3.patch is ready for core, yet. As @wonderboymusic just said, it needs to be more performant. If core devs, or anyone else, has suggestions along those lines, they would be welcome. It's already improved from 17817.2.patch, based on feedback @nacin and @markjaquith gave at WordCamp Orlando last week.

We definitely need to get this bug fixed, and it needs to be done in a backwards-compatible way. A minor performance impact is probably worth it. I'm just not sure what the threshold is for "minor". 5% is probably too big. 1-2%, maybe even 3%, would probably be acceptable, in my opinion.

#45 in reply to: ↑ 42 @Denis-de-Bernardy
11 years ago

Replying to sc0ttkclark:

@Denis-de-Bernardy: What is your suggestion? Just wanted to be sure it was clear what you were championing, could you clarify or suggest something different?

My own opinion on this hasn't moved by one iota in over 4 years (#9968): we should replace our current mess with foreach() loops where needed, and shed a crocodile tear or two if we inconveniently break an already half-broken plugin in the process.

@SergeyBiryukov: The patch http://core.trac.wordpress.org/attachment/ticket/17817/17817.3.patch -- is this is something that can move forward since it passes unit tests, or does this require further feedback/direction from core devs? I know it's too big of a change to make 3.8, but what are the next steps to get this ticket resolved in *some* way?

I'll volunteer an opinion on that patch: it's nice to see some object-oriented code, but if we introduce changes of that magnitude, we might as well backport Symfony's EventDispatcher to make it php 5.2 compatible, and wrap it around the legacy API functions.

http://symfony.com/doc/current/components/event_dispatcher/introduction.html

That way, we'd be building on top of something sound, and eventually begin paving the way towards potentially interesting behaviors such as stopping event propagation — which, incidentally, could be used to "break" the legacy API functions while keeping a perfectly functional API at its side.

#46 @SergeyBiryukov
11 years ago

  • Version changed from 3.4.1 to 2.2

Related: [4955] (for #3875).

#47 @KoenRijpstra
11 years ago

  • Cc k.rijpstra@… added

#48 follow-up: @jbrinley
11 years ago

Here's my next attempt. Executive summary: Still slower (but more reasonably so) in some cases, slightly faster in the most common case, backwards compatible, passes all unit tests. Let's ship it.

Details:

Results of benchmarking using https://gist.github.com/jbrinley/9458582:

Running 10000 times with 0 callbacks
Total Time (Old): 0.093368
Average Time (Old): 0.000004
Total Time (New): 0.078037
Average Time (New): 0.000004
New runs in 83.58% of the time of old.

Running 10000 times with 3 callbacks
Total Time (Old): 1.299951
Average Time (Old): 0.000064
Total Time (New): 1.822003
Average Time (New): 0.000117
New runs in 140.16% of the time of old.

Running 10000 times with 15 callbacks
Total Time (Old): 4.006840
Average Time (Old): 0.000184
Total Time (New): 6.443348
Average Time (New): 0.000390
New runs in 160.81% of the time of old.

Running 1000 times with 300 callbacks
Total Time (Old): 8.345846
Average Time (Old): 0.003690
Total Time (New): 13.177368
Average Time (New): 0.007553
New runs in 157.89% of the time of old.

Running 100 times with 3000 callbacks
Total Time (Old): 8.087688
Average Time (Old): 0.031045
Total Time (New): 10.641660
Average Time (New): 0.052878
New runs in 131.58% of the time of old.

You'll notice that with 0 callbacks, the new code is actually a bit more performant than the existing code. For the vast majority of hooks, this will be the case. In all other cases, the new code for running do_action() takes 30-60% longer than the old code.

Translating this into actual page loads using ab on a stock WP install (svn trunk@27490, twentyfourteen theme):

Existing code:

Connection Times (ms)
              min  mean[+/-sd] median   max
Connect:        0    0   0.1      0       0
Processing:   177  190  11.0    191     209
Waiting:      177  188   8.8    191     205
Total:        177  190  11.0    192     209

17817.4.patch:

Connection Times (ms)
              min  mean[+/-sd] median   max
Connect:        0    0   0.1      0       0
Processing:   175  186   7.3    189     196
Waiting:      175  185   7.3    189     196
Total:        175  186   7.3    189     197

A statistically insignificant 2% faster with the new code, largely due to the fact that most hooks don't have any callbacks most of the time.

@jbrinley
11 years ago

#49 in reply to: ↑ 48 @rzen
11 years ago

Replying to jbrinley:

Here's my next attempt. Executive summary: Still slower (but more reasonably so) in some cases, slightly faster in the most common case, backwards compatible, passes all unit tests. Let's ship it.

I like everything about this summary. Faster than the other proposed solutions, passes all tests, backwards compatible (unlike the other solutions), AND it fixes the issue. The only way it could be better is if it were faster across the board.

Really hope we can get a core dev to bless this one :)

#50 @mbijon
11 years ago

+1 for attachment:17817.4.patch

At the very least we should include the tests from '17817.4.patch' in the suite & mark them as skipped. No reason to not include those tests, as they'll help steer any suggested updates/enhancements

#51 @mbijon
11 years ago

  • Keywords has-unit-tests added

#52 follow-up: @Denis-de-Bernardy
11 years ago

  • Version changed from 2.2 to trunk

Likely -1 here. Won't 17814.4 break plugins that look into the $wp_filter global in an effort to remove observers added by plugins that are anonymous? Plugins and such that change $wp_filter directly, such as what occurs in some of the unit tests?

#53 in reply to: ↑ 52 ; follow-up: @jbrinley
11 years ago

Replying to Denis-de-Bernardy:

Likely -1 here. Won't 17814.4 break plugins that look into the $wp_filter global in an effort to remove observers added by plugins that are anonymous? Plugins and such that change $wp_filter directly, such as what occurs in some of the unit tests?

@Denis-de-Bernardy, would your concerns be assuaged if WP_Hook implemented ArrayAccess, IteratorAggregate, and Countable (basically extending ArrayObject)?

In my opinion, it's unnecessary bloat for the benefit of a few edge cases where plugins take advantage of undocumented WP internals. But, my opinion aside, the solution is relatively straight forward. I'd appreciate your feedback, and maybe even a unit test or two specifying exactly what you expect from it.

#54 @toscho
11 years ago

If we are are talking about removing anonymous objects – this must continue to work. This solution is active on many sites, we would break too much.

WP_Hook_Iterator::__construct() should be SOLID, i.e. depend on an abstraction, not on a concrete class. There should be an interface for WP_Hook.

#55 @SergeyBiryukov
11 years ago

#27488 was marked as a duplicate.

#56 in reply to: ↑ 53 @Denis-de-Bernardy
11 years ago

Replying to jbrinley:

Replying to Denis-de-Bernardy:

Likely -1 here. Won't 17814.4 break plugins that look into the $wp_filter global in an effort to remove observers added by plugins that are anonymous? Plugins and such that change $wp_filter directly, such as what occurs in some of the unit tests?

@Denis-de-Bernardy, would your concerns be assuaged if WP_Hook implemented ArrayAccess, IteratorAggregate, and Countable (basically extending ArrayObject)?

Yeah. Look at doctrine/common, or is that collection... At any rate, there's a collection class in there that implements all three, complete with unit tests.

In my opinion, it's unnecessary bloat for the benefit of a few edge cases

It's not a few edge cases. See toscho's reply: he links to one of the highest voted questions in WP Answers.

Nor is it plugins taking advantage of undocumented internals: people make do with what they have -- including, for that matter, WP itself when it messes around with wp_filters in its own test suite.

My own use, or the WP unit test suite itself, is to pre-load wp filters prior to loading WP. So it should ideally convert any pre-existing wp_filter global as appropriate at the top of the file if any are present.

#57 @jbrinley
11 years ago

  • Version changed from trunk to 2.2

Would appreciate some feedback from @SergeyBiryukov or other core devs. Is patch 17817.4 ready to go, or do Denis-de-Bernardy's concerns merit additional consideration?

This bug has been plaguing us for years; I just need to know what it will take to get this fixed in core.

This ticket was mentioned in IRC in #wordpress-dev by jbrinley. View the logs.


10 years ago

#59 @edward_plainview
10 years ago

I've recently encountered this same ... "bug" with my Broadcast plugin that hooked into the same save_post hook that another plugin at an earlier priority used. Took me hours of debugging to find the problem.

Long story short: the earlier plugin is called, calls save_post itself, broadcast is run as per save_post, and then the hook finishes, without calling the "outer" save_post loop.

This needs to be fixed since I wasn't expecting the hook looping to be broken. My patch uses foreach loops, but I see that others have suggested foreaches also.

This ticket was mentioned in IRC in #wordpress-dev by jbrinley. View the logs.


10 years ago

@jbrinley
10 years ago

#61 @jbrinley
10 years ago

I've attached 17817.5.patch. That should address the concern from comment 52; existing code for removing anonymous callbacks should continue to work. Also added another unit test to verify (all other unit tests still passing).

Any objections? Can we move forward with this?

This ticket was mentioned in IRC in #wordpress-dev by jbrinley. View the logs.


10 years ago

#63 @wonderboymusic
10 years ago

  • Milestone changed from Future Release to 4.0

Let's give this a good look for 4.0

#64 @wonderboymusic
10 years ago

The first time I ran unit tests, it took 1.5 minutes longer. The second time, 45 seconds. I think we need to squeeze some more performance for this to be viable.

#65 follow-up: @wonderboymusic
10 years ago

  • Keywords needs-patch added; has-patch needs-testing removed
  • Milestone changed from 4.0 to Future Release

See above - needs performance gains to be considered.

This ticket was mentioned in IRC in #wordpress-dev by jbrinley. View the logs.


10 years ago

#67 in reply to: ↑ 65 @Denis-de-Bernardy
10 years ago

  • Keywords 2nd-opinion added

Replying to wonderboymusic:

See above - needs performance gains to be considered.

What was the objection to switching a simple foreach loop again? Isn't it time we take a rational approach to this and acknowledge that a very slight backwards compat issue in 4.0 might be fine in the interest of adding some sanity to the plugin API?

Last edited 10 years ago by Denis-de-Bernardy (previous) (diff)

@jbrinley
10 years ago

#68 @jbrinley
10 years ago

  • Keywords has-patch added; needs-patch 2nd-opinion removed

Went through a lot of iterations before arriving at the version just attached in 17817.6.patch.

Let's start with performance: this is as good as or better than what currently exists. Better with 0 callbacks or 3+ callbacks, about the same at 1-2 callbacks. Results of the test at https://gist.github.com/jbrinley/7518483

Running 10000 times with 0 callbacks
Total Time (Old): 0.042208
Average Time (Old): 0.000004
Total Time (New): 0.037750
Average Time (New): 0.000004
New runs in 89.44% of the time of old.

Running 10000 times with 1 callbacks
Total Time (Old): 0.212133
Average Time (Old): 0.000021
Total Time (New): 0.239030
Average Time (New): 0.000024
New runs in 112.68% of the time of old.

Running 10000 times with 2 callbacks
Total Time (Old): 0.312370
Average Time (Old): 0.000031
Total Time (New): 0.314523
Average Time (New): 0.000031
New runs in 100.69% of the time of old.

Running 10000 times with 3 callbacks
Total Time (Old): 0.400735
Average Time (Old): 0.000040
Total Time (New): 0.384914
Average Time (New): 0.000038
New runs in 96.05% of the time of old.

Running 10000 times with 5 callbacks
Total Time (Old): 0.604012
Average Time (Old): 0.000060
Total Time (New): 0.515111
Average Time (New): 0.000052
New runs in 85.28% of the time of old.

Running 10000 times with 10 callbacks
Total Time (Old): 1.133043
Average Time (Old): 0.000113
Total Time (New): 0.888408
Average Time (New): 0.000089
New runs in 78.41% of the time of old.

Running 10000 times with 15 callbacks
Total Time (Old): 1.535434
Average Time (Old): 0.000154
Total Time (New): 1.266079
Average Time (New): 0.000127
New runs in 82.46% of the time of old.

Running 1000 times with 300 callbacks
Total Time (Old): 2.447496
Average Time (Old): 0.002447
Total Time (New): 2.088821
Average Time (New): 0.002089
New runs in 85.35% of the time of old.

Running 100 times with 3000 callbacks
Total Time (Old): 0.999711
Average Time (Old): 0.009997
Total Time (New): 0.810893
Average Time (New): 0.008109
New runs in 81.11% of the time of old.

@wonderboymusic was concerned about the extra time spent running unit tests. On my system, running tests against current WP trunk takes about 3:23 (average of 4 runs). Running with this patch: 3:28 (average of 4 runs).

I think the performance concerns have been addressed. Many thanks to those of you who kept pushing for more; eventually we got it.

On to functionality: Anonymous callbacks can be removed. Plugins that setup callbacks before WordPress initializes (e.g., batcache, the WP unit test suite) will continue to work. Hooks can be called recursively with no unexpected side effects. Callbacks can be added and removed mid-run with no unexpected side effects.

I removed a unit test added in 4.0 for ticket #29070. It tested that the array pointer for the global $wp_filter didn't change when calling has_action(). Since this new implementation no longer depends on the global array pointer, that test was rendered superfluous. The array pointer does change, but it doesn't matter.

I abandoned the custom iterator that characterized my previous patches. No matter how much I optimized, it just was not performing as well as I would have liked. I was able to squeeze a lot more out of it by keeping the management of the iteration within the WP_Hook class, at the slight cost of a cleaner separation of responsibilities.

Any questions? Any concerns? Anything holding this back?

#69 @rzen
10 years ago

John, you magnificent and glorious man, these performance stats are excellent! I'm going to put it through some paces on the code where @sc0ttkclark and I first noticed the issue.

#70 @sc0ttkclark
10 years ago

Saw this come through my e-mail and was so happy!

@jbrinley
10 years ago

#71 @jbrinley
10 years ago

Just realized that the patch left in a bit of debugging code. Ignore 17817.6.patch, use 17817.6.2.patch.

#72 follow-ups: @leewillis77
10 years ago

Great to see this being worked on - I just wasted a good two hours trying to work out why a hook wasn't being fired with this as the root cause (Not in my code thankfully).

Good news - the patch here resolves the issue I was having. However - it causes issues where there are multiple callbacks registered against a hook, and an earlier hook accepts fewer arguments than a subsequent hook, e.g.

add_action($tag, array(&$a1, 'action'), 10, 2);
add_action($tag, array(&$a2, 'action'));
add_action($tag, array(&$a3, 'action'), 10, 2);

In this situation, the third callback will only receive one argument, not two. The revised patch attached resolves this, complete with a test to prove it. I cannot for the life of me work out why though - it's as though the array_slice is affecting $args (Which it shouldn't).

In my defense, it's late in the day though ...

So - this probably needs more work (At least we'd need to verify that creating a copy of the args rather than reusing a reference doesn't impact on performance (CPU or memory)), but throwing this out for discussion.

#73 @leewillis77
10 years ago

Apparently that diff didn't include the new WP_Hook class. Updated patch attached. For info, the only real change to the code is the change from:

$func_args =& $args;

to

$func_args = $args;

in WP_Hook::do_action()

#74 in reply to: ↑ 72 @jbrinley
10 years ago

Replying to leewillis77:

In this situation, the third callback will only receive one argument, not two. The revised patch attached resolves this, complete with a test to prove it. I cannot for the life of me work out why though - it's as though the array_slice is affecting $args (Which it shouldn't).

Great catch, @leewillis77. After the assign-by-ref happens on one iteration, the reference is maintained in the next iteration when the array is sliced.

I'll run some benchmarking, but I expect the change to be inconsequential. Thanks for testing!

#75 follow-ups: @leewillis77
10 years ago

Question: Should do_action() and apply_filters() check whether the callback is callable before attempting to call it - thus ending the age old "PHP Warning: call_user_func_array() expects parameter 1 to be a valid callback" errors when invalid callbacks are used - or is that warning considered a "feature"?

See also the discussion on https://core.trac.wordpress.org/ticket/29803

Does anyone know from experience whether is_callable() is suitably performant - or do we need to test if we want to add it?

#76 in reply to: ↑ 75 @jbrinley
10 years ago

Replying to leewillis77:

Does anyone know from experience whether is_callable() is suitably performant - or do we need to test if we want to add it?

I did some rough testing of it at one point. I can't recall the exact numbers, but it slowed down the loop considerably.

@jbrinley
10 years ago

#77 in reply to: ↑ 72 @jbrinley
10 years ago

Replying to leewillis77:

At least we'd need to verify that creating a copy of the args rather than reusing a reference doesn't impact on performance (CPU or memory)

In most cases, the difference was inconsequential. I had to bump the number of arguments up to around 100 before I started seeing a noticeable (i.e., > 5%) impact on performance. 17817.9.patch negates that impact, by avoiding the copy entirely (there will still be a copy when the callback is called, but there won't be two copies made).

#78 in reply to: ↑ 75 @jbrinley
10 years ago

Replying to leewillis77:

Does anyone know from experience whether is_callable() is suitably performant - or do we need to test if we want to add it?

Just ran some tests. is_callable() adds about 10% to the do_action execution time.

#79 @leewillis77
10 years ago

OK, so sounds like that's not worth adding - 17817.9.patch looking good to me - cheers Jonathan.

This ticket was mentioned in IRC in #wordpress-dev by jbrinley. View the logs.


10 years ago

#81 @jbrinley
10 years ago

@johnbillion requested some memory statistics in IRC. Had to break the tests out into multiple files so we could isolate processes and report peak memory independently. https://gist.github.com/jbrinley/7518483 is updated to reflect the change. Looks like the new code uses about 3% more memory for the hook system (i.e., about 50-100 KB more memory).

$ php tests.local/test_17817.php 
Running 10000 times with 0 callbacks
Total Time (Old): 0.078679
Average Time (Old): 0.000008
Peak Memory after Old: 1755592

Total Time (New): 0.057374
Average Time (New): 0.000006
Peak Memory after New: 1800080

New runs in 72.92% of the time of old.
New runs in 102.53% of the memory of old.

$ php tests.local/test_17817.php 
Running 10000 times with 1 callbacks
Total Time (Old): 0.264266
Average Time (Old): 0.000026
Peak Memory after Old: 1758256

Total Time (New): 0.260950
Average Time (New): 0.000026
Peak Memory after New: 1804336

New runs in 98.75% of the time of old.
New runs in 102.62% of the memory of old.

$ php tests.local/test_17817.php 
Running 10000 times with 2 callbacks
Total Time (Old): 0.339878
Average Time (Old): 0.000034
Peak Memory after Old: 1759384

Total Time (New): 0.340070
Average Time (New): 0.000034
Peak Memory after New: 1805408

New runs in 100.06% of the time of old.
New runs in 102.62% of the memory of old.

$ php tests.local/test_17817.php 
Running 10000 times with 3 callbacks
Total Time (Old): 0.434463
Average Time (Old): 0.000043
Peak Memory after Old: 1760080

Total Time (New): 0.422050
Average Time (New): 0.000042
Peak Memory after New: 1806296

New runs in 97.14% of the time of old.
New runs in 102.63% of the memory of old.

$ php tests.local/test_17817.php 
Running 10000 times with 5 callbacks
Total Time (Old): 0.630719
Average Time (Old): 0.000063
Peak Memory after Old: 1761904

Total Time (New): 0.557231
Average Time (New): 0.000056
Peak Memory after New: 1808352

New runs in 88.35% of the time of old.
New runs in 102.64% of the memory of old.

$ php tests.local/test_17817.php 
Running 10000 times with 10 callbacks
Total Time (Old): 1.133310
Average Time (Old): 0.000113
Peak Memory after Old: 1766352

Total Time (New): 0.917521
Average Time (New): 0.000092
Peak Memory after New: 1813496

New runs in 80.96% of the time of old.
New runs in 102.67% of the memory of old.

$ php tests.local/test_17817.php 
Running 1000 times with 100 callbacks
Total Time (Old): 0.913040
Average Time (Old): 0.000913
Peak Memory after Old: 497704

Total Time (New): 0.748710
Average Time (New): 0.000749
Peak Memory after New: 574104

New runs in 82.00% of the time of old.
New runs in 115.35% of the memory of old.

$ php tests.local/test_17817.php 
Running 1000 times with 1000 callbacks
Total Time (Old): 6.859408
Average Time (Old): 0.006859
Peak Memory after Old: 1273632

Total Time (New): 5.400719
Average Time (New): 0.005401
Peak Memory after New: 1400144

New runs in 78.73% of the time of old.
New runs in 109.93% of the memory of old.

This ticket was mentioned in Slack in #core by jbrinley. View the logs.


10 years ago

This ticket was mentioned in Slack in #core by jbrinley. View the logs.


10 years ago

This ticket was mentioned in Slack in #core by jbrinley. View the logs.


10 years ago

This ticket was mentioned in Slack in #core by dancameron. View the logs.


10 years ago

This ticket was mentioned in Slack in #core by jbrinley. View the logs.


10 years ago

@jbrinley
10 years ago

#87 @jbrinley
10 years ago

Per discussion with @johnbillion in Slack, 17817.10.patch implements the ArrayAccess interface for the WP_Hook class, adding backwards compatibility with code directly accessing the third level of the global $wp_filter array. Also adds a unit test for the new methods.

@jbrinley
10 years ago

#88 @jbrinley
10 years ago

@boonebgorges helpfully points out that we still need to avoid SPL interfaces. Fortunately, Countable is the only SPL interface implemented here, and it's actually not used for anything and can be safely removed. Therefore, 17817.11.patch.

This ticket was mentioned in Slack in #core by travisnorthcutt. View the logs.


10 years ago

This ticket was mentioned in Slack in #core by drew. View the logs.


10 years ago

#91 @helgatheviking
10 years ago

Now that there is a patch with good performance stats, what are the odds of this making it into a release soon?

#92 @johnbillion
10 years ago

  • Keywords 4.2-early added

This ticket was mentioned in Slack in #core by jbrinley. View the logs.


10 years ago

#94 @iseulde
10 years ago

  • Milestone changed from Future Release to 4.2

has-patch 4.2-early so moving to 4.2.

This ticket was mentioned in Slack in #core by jbrinley. View the logs.


10 years ago

This ticket was mentioned in Slack in #core by jbrinley. View the logs.


10 years ago

This ticket was mentioned in Slack in #core by jbrinley. View the logs.


10 years ago

This ticket was mentioned in Slack in #core by drew. View the logs.


10 years ago

#99 @jbrinley
10 years ago

Followup to discussion in today's dev chat:

I've reviewed every instance of $wp_filter that can be found at https://github.com/search?q=%24wp_filter+user%3Awp-plugins&type=Code (it's a little finicky, so if no results load refresh and you may get some). Most instances are not a cause for concern; they're already compatible thanks to IteratorAggregate or ArrayAccess. I did find a handful of cases that would have compatibility issues:

Group 1 - Plugins that are messing with the $wp_filter global to work around this very issue. I only found one example here.

Group 2 - Plugins that do an is_array() check on a member of the $wp_filter global. Again, I found one example (and it's within debug tools for a plugin).

Group 3 - Plugins that directly unset individual callbacks instead of using remove_action() / remove_filter() (this doesn't work because ArrayAccess doesn't allow indirect modification of an offset).

Group 4 - Plugins that directly set individual callbacks, instead of using add_action() / add_filter() (same technical limitation as Group 3).

Nine examples out of 40K+ plugins, and all of them could be easily fixed. As much as I wish there were zero compatibility issues, I think this is more than reasonable.

#100 @sc0ttkclark
10 years ago

I remember having to do that BadgeOS hack :)

That's great news, in terms of your findings, eager to see this put to bed.

This ticket was mentioned in Slack in #core by jbrinley. View the logs.


10 years ago

#102 follow-up: @DrewAPicture
10 years ago

  • Keywords needs-docs added

Some notes:

  1. Worth making a decision about how/if we plan to support making this extensible, i.e. marking as final or not
  1. Decide if there's merit in not introducing methods that directly mirror their legacy counterparts, e.g. new methods that take four parameters right out of the gate.
  1. We need to do a full inline-docs pass on this before the initial commit (if it happens).
  • Some missing parameter and return descriptions
  • Missing @since tags
  • Missing @access tags
  • Inline documentation in these new methods explaining what's going on
  1. I enjoy a good whitespace refactoring as much as the next developer, but I'd like to see a cleaner a patch with fewer vanilla whitespacing changes.

This ticket was mentioned in Slack in #core by drew. View the logs.


10 years ago

#104 @nacin
10 years ago

This should not land in core without lead developer approval.

@jbrinley
10 years ago

#105 @jbrinley
10 years ago

17817.12.patch attached. It incorporates @wonderboymusic's formatting updates (but removes those from lines that would otherwise be untouched).

There were a few failing unit tests. Some came from #20920. That fix is now incorporated into this patch. Some came from #30936, but were really a consequence of #28535. Hook objects were backed up by reference, so new hooks added weren't removed. This updates WP_UnitTestCase to use clone when backing up and restoring $wp_filter.

Patch should now apply cleanly and all unit tests should pass.

Thanks for the feedback, @wonderboymusic and @DrewAPicture.

#106 in reply to: ↑ 102 @toscho
10 years ago

Replying to DrewAPicture:

  1. Worth making a decision about how/if we plan to support making this extensible, i.e. marking as final or not

Only make it final if we have an interface. Abstract dependency injection or mocking is a nightmare already with WP_Post, let’s not repeat this mistake.

#107 @jbrinley
10 years ago

Some more statistics regarding memory usage, that may be more helpful than those above:

Peak memory loading the homepage with the 2014 theme.
Before Patch: 16.56 MB
After Patch: 16.62 MB

Peak memory loading a single post with the 2014 theme.
Before Patch: 16.74 MB
After Patch: 16.81 MB

Peak memory loading the dashboard.
Before Patch: 20.30 MB
After Patch: 21.38 MB

Peak memory loading /wp-admin/edit.php
Before Patch: 20.31 MB
After Patch: 21.05 MB

#108 @jbrinley
10 years ago

Per @boone's suggestion, I've put together draft release notes to go with this patch. The draft is available at https://docs.google.com/document/d/1TWd3i0GX7WsfosyCHG-EnnhANErVBsPnofvXp2qzX58/edit?usp=sharing

I would appreciate feedback.

#109 @jbrinley
10 years ago

On second though, a gist is probably better than google drive. Here it is: https://gist.github.com/jbrinley/eaaad00b52e1316c6904

This ticket was mentioned in Slack in #core by jbrinley. View the logs.


10 years ago

#111 @jbrinley
10 years ago

More statistics, from today's Slack conversation:

nacin [18:16]

@jbrinley anything we can do to specifically improve upon any areas where it is slower? e.g. 12% slower for one callback.

jbrinley [22:04]

@nacin: I tried to find a way to optimize the one callback case, but failed. Anything I tried was either less performant overall, or failed to correctly handle edge cases (like a hook with a single callback, but that callback adds other callbacks while it's running).

The actual difference we're talking about is fairly minute. In a worst case scenario, where every hook that WP fires has a single callback assigned to it, 12% extra time running do_action() or apply_filters() adds approximately 0.01s to WP's total load time. But surely most of the hooks with one callback will be offset by other hooks with zero callbacks. When I load the home page of a clean install (2014 theme), apply_filters() or do_action is called 2520 times with zero callbacks vs 164 times with 1 callback.

summary: .00044 seconds lost is more than offset by .00112 seconds gained

This ticket was mentioned in Slack in #core by jbrinley. View the logs.


10 years ago

This ticket was mentioned in Slack in #core by jbrinley. View the logs.


10 years ago

This ticket was mentioned in Slack in #core by jbrinley. View the logs.


10 years ago

This ticket was mentioned in Slack in #core by jbrinley. View the logs.


10 years ago

This ticket was mentioned in Slack in #core by jbrinley. View the logs.


10 years ago

#117 follow-up: @nacin
10 years ago

A day later than promised, but here is my review.

Minor style:

  • Please go over uses of empty() and ! empty() Only use it in conditionals when trying to avoid a notice. Otherwise use if ( $a ) or if ( ! $a ).
  • false and true, not FALSE or TRUE.
  • $the_[ 'function' ] should be $the_['function'], as 'function' is a literal. Same for $the_['accepted_args'].
  • There is a space before !.

Performance:

  • $this->callbacks[ $priority ][ $idx ] should store a numeric array. If appropriate, use constants such as self::FUNCTION and self::ACCEPTED_ARGS for 0 and 1 indexes. This will reduce memory usage. I have wanted to do this in core, but we couldn't for back compat (direct access of the array, of course). Can we do it now? Would this be manipulation in getIterator() (or our own iterator)? Lightweight object with ArrayAccess? Would any of these be worth it?
  • Does the ksort() need to occur on add_filter()? This appears wasteful. I recognize this is a very key part of the nesting aspects of this change, but it feels heavy. (I wish PHP were more amenable to linked list manipulation, as in other languages we could do this without resorting to a sort. It's possible slicing would be faster than sorting, but I have no data to back this up.)
  • $callbacks appears to unnecessarily be used by reference when $this->callbacks is iterated over in a foreach in a few places.
  • I don't see why has_filters() isn't just return ! empty( $this->callbacks ).
  • Should we try to avoid array_slice() in apply_filters() the same way we do in do_action()?
  • The apply_filters() function now calls array_shift() in an unconditional fashion, even if there are no callbacks to apply. I recognize why it is done, but it feels like it is bringing back the same kind of thing we optimized in [17638].
  • _wp_call_all_hook() appears like it is just an unnecessary function call at this point and should probably stop being used.

Other:

  • What would be the use case for nesting with the all hook?
  • It's possible that someone could be overwriting $wp_filter[ $tag ] with their own array. This would trigger a fatal error when a method is called. I don't know what to do about this, and I don't know how serious it is.
  • It feels weird that _wp_filter_build_unique_id() was moved from the add_filter() function to the add_filter() method, but it remains in has_filter(), remove_filter(), etc. It appears that some functionality was deliberately left over in the functions -- like has_filter() when called without a function to check. Is this by design? (Is there a use case for WP_Hook used outside the context of these internal functions?)
  • class-wp-hook.php should probably be included from plugin.php for compatibility, the way functions.php includes option.php.

Tests:

  • This feels like it has a lot of good test coverage for what has been added, but I think we need a lot more test coverage for the existing functions and the new methods. That there's no new WP_Hook in the test suite seems to be a smell. We should also split up individual (sets of) assertions into separate tests.

This has taken a long time to progress to this state. I want to mention a few notable factors:

  • Project history: This API is a pillar of WordPress. It is simple yet powerful. It's been tweaked over time, but the core of its underlying architecture has not significantly changed since it was first written more than a decade ago. The developers who wrote it are no longer active in the project's development. It is particularly challenging to consider a rewrite.
  • Personal history: Last time I tried to touch something in plugin.php, I broke things really badly. See #21321. My comment there is innocuous and merely suggests there were "some side effects here". Let me be clear: This broke wordpress.org harder than I had ever broken it before, and ever broken it since. If the original developers can no longer be counted on to review this, and the last time (three years ago) I tried to go anywhere near this I blew things sky high, you can perhaps understand the reluctancy.

Conclusions and next steps:

  • I want to recognize, and I very much appreciate, @jbrinley's long march shepherding this. Great work so far, and great improvements over time. Thank you for continuing to stay on our case about this.
  • I am not comfortable committing this without ample lead time. It is too late for 4.2. While I should have found the time to review this earlier in the cycle, day two of a cycle is too late. (See supra the personal and project factors.) And this review notes a number of things to evaluate/study/tweak.
  • It could definitely use some more tests, not just coverage for the new stuff, but to prevent regressions.
  • We need a blog post written for make/core that explains to plugin authors exactly how they will benefit from this, and how code might break because of this.
  • We probably do need one final cost-benefit analysis, where "benefits" are what this provides to plugin developers, and "costs" are what might break and how hard. The blog post could double as a request for comments.
  • I am willing to deploy this to .org for a few hours on a weekend and see if it breaks anything.
  • I am willing to commit this to trunk after 4.2 is released.
Last edited 10 years ago by nacin (previous) (diff)

#118 @DrewAPicture
10 years ago

  • Keywords 4.3-early added; 4.2-early removed
  • Milestone changed from 4.2 to Future Release

This is 4.3-early material. See comment:117.

@jbrinley
10 years ago

#119 in reply to: ↑ 117 @jbrinley
10 years ago

Replying to nacin:

Minor style:

All should be fixed in 17817.13.patch.

Performance:

  • $this->callbacks[ $priority ][ $idx ] should store a numeric array. If appropriate, use constants such as self::FUNCTION and self::ACCEPTED_ARGS for 0 and 1 indexes. This will reduce memory usage. I have wanted to do this in core, but we couldn't for back compat (direct access of the array, of course). Can we do it now? Would this be manipulation in getIterator() (or our own iterator)? Lightweight object with ArrayAccess? Would any of these be worth it?

This would probably cause some minor additional back-compat issues beyond what we've already identified. I'm not sure of the memory difference, but can run some tests.

  • Does the ksort() need to occur on add_filter()? This appears wasteful. I recognize this is a very key part of the nesting aspects of this change, but it feels heavy.

As you say, it's key to supporting recursion. The sort runs when callbacks are added at a new priority to a hook that already has callbacks at other priorities. So it will generally run maybe a few dozen times (55 on the homepage of a fresh install, maybe a few dozen more depending on your plugins). The alternative is to go back to the $merged_filters approach (albeit encapsulated within each hook), and check if it's sorted every time apply_filters / do_action runs, which can come out to several hundreds of checks, and then also doing the sorting on many of those occasions (about 40 times instead of 55, on the page load above). I don't yet have much data to say if one approach is heavier than the other. My intuition is that the difference is inconsequential.

  • $callbacks appears to unnecessarily be used by reference when $this->callbacks is iterated over in a foreach in a few places.

Fixed in 17817.13.patch.

  • I don't see why has_filters() isn't just return ! empty( $this->callbacks ).

For backwards compatibility, if a plugin takes advantage of ArrayAccess to remove a callback but doesn't remove the priority index. Looping through the priorities ensures that we don't return a false positive. Assuming $this->callbacks is empty as it should be, the foreach won't have anything to iterate over and false will be returned. Wrapping the foreach in if ( $this->callbacks ) didn't show any performance improvement.

  • Should we try to avoid array_slice() in apply_filters() the same way we do in do_action()?

Fixed in 17817.13.patch.

  • The apply_filters() function now calls array_shift() in an unconditional fashion, even if there are no callbacks to apply. I recognize why it is done, but it feels like it is bringing back the same kind of thing we optimized in [17638].

Fixed in 17817.13.patch. array_shift() will only be called if there's a WP_Hook for the hook.

  • _wp_call_all_hook() appears like it is just an unnecessary function call at this point and should probably stop being used.

Are you suggesting getting rid of the 'all' hook, or just inlining $wp_filter['all']->do_all_hook( $args );? I'll assume the latter. Are there any back-compat concerns for getting rid of the function?

Other:

  • What would be the use case for nesting with the all hook?

I don't particularly see a use case. But if any callback on the all hook triggers another hook, all _will_ be called recursively, so it seems we should make it predictable.

  • It's possible that someone could be overwriting $wp_filter[ $tag ] with their own array. This would trigger a fatal error when a method is called. I don't know what to do about this, and I don't know how serious it is.

Comment #99 above addresses everything that I found (not to say that there aren't any plugins I didn't find). Every case I found where someone was directly setting $wp_filter[ $tag ], it was to restore what was previously there. So a WP_Hook would be correctly restored. If someone does try to directly set it as an array, then yes, it would trigger a fatal, and there's nothing I can think of that we can do to stop them.

  • It feels weird that _wp_filter_build_unique_id() was moved from the add_filter() function to the add_filter() method, but it remains in has_filter(), remove_filter(), etc. It appears that some functionality was deliberately left over in the functions -- like has_filter() when called without a function to check. Is this by design? (Is there a use case for WP_Hook used outside the context of these internal functions?)

Fixed in 17817.13.patch. I've moved all calls to _wp_filter_build_unique_id() inside of WP_Hook. Also changed the order of args for some functions to put $tag at the end (in anticipation of deprecating the argument in 40 years when we drop 5.2 support).

  • class-wp-hook.php should probably be included from plugin.php for compatibility, the way functions.php includes option.php.

Heh. I had it there, and then Scott moved it to wp-settings.php. Now back to plugin.php.

  • This feels like it has a lot of good test coverage for what has been added, but I think we need a lot more test coverage for the existing functions and the new methods. That there's no new WP_Hook in the test suite seems to be a smell. We should also split up individual (sets of) assertions into separate tests.

I'll work on this shortly.

  • We need a blog post written for make/core that explains to plugin authors exactly how they will benefit from this, and how code might break because of this.

I'll work on expanding https://gist.github.com/jbrinley/eaaad00b52e1316c6904 to include a "benefits" section.

Thanks for the review, @nacin. I'm looking forward to seeing this in core.

This ticket was mentioned in Slack in #core by jbrinley. View the logs.


10 years ago

#121 follow-up: @helen
9 years ago

  • Milestone changed from Future Release to 4.3
  • Owner set to nacin
  • Status changed from reopened to assigned

I am moving this as the first ticket into 4.3 also as recognition of the hard work and shepherding that's gone on here. It's on @nacin, though. He promised. :)

@jbrinley
9 years ago

#122 @jbrinley
9 years ago

Attached 17817.14.patch. Refreshed onto 4.2, and adds a series of new unit tests for methods of the WP_Hook class (probably not yet all that @nacin wants).

#123 @obenland
9 years ago

  • Keywords 4.3-early removed

This ticket was mentioned in Slack in #core by johnbillion. View the logs.


9 years ago

#125 @chriscct7
9 years ago

#22127 was marked as a duplicate.

#126 @jorbin
9 years ago

  • Keywords changed from dev-feedback, has-unit-tests, has-patch, needs-docs to dev-feedback has-unit-tests has-patch needs-docs

@nacin - Still want to give this one a shot or are you going to punt again?

This ticket was mentioned in Slack in #core by nacin. View the logs.


9 years ago

#128 @wonderboymusic
9 years ago

  • Milestone changed from 4.3 to Future Release

With a heavy heart

#129 in reply to: ↑ 121 @samuelsidler
9 years ago

Replying to helen:

He promised. :)

:(

#130 @johnbillion
9 years ago

  • Keywords 4.4-early like-totally-4.4-early added

#131 @wonderboymusic
9 years ago

  • Milestone changed from Future Release to 4.4

Let's focus on this part:
Owned by: @nacin

#132 follow-up: @nacin
9 years ago

Last I checked this ticket, I saw "(probably not yet all that @nacin wants)" regarding unit tests. Has anything advanced on that front?

If people feel comfortable with it, I'm game for dropping it into trunk now that 4.3 is out. Just don't be surprised if there are major explosions and we have to yank it quickly and try again after more tweaks and unit tests.

#133 in reply to: ↑ 132 ; follow-up: @pento
9 years ago

  • Keywords commit added; needs-docs 4.4-early like-totally-4.4-early removed

Replying to nacin:

If people feel comfortable with it, I'm game for dropping it into trunk now that 4.3 is out.

+1, let's do that and see what explodes.

#134 in reply to: ↑ 133 @wonderboymusic
9 years ago

Replying to pento:

Replying to nacin:

If people feel comfortable with it, I'm game for dropping it into trunk now that 4.3 is out.

+1, let's do that and see what explodes.

+1 for explosion, we all know how to revert

#135 follow-up: @pento
9 years ago

  • Keywords needs-refresh added; dev-feedback commit removed
  • Owner changed from nacin to pento

Needs some fixes before commit:

  • 17817.14.patch doesn't apply cleanly against trunk.
  • @since should be updated to 4.4.0.
  • @DrewAPicture - should each method in WP_Hook have an @since doc, or is the one on the class enough?

#136 in reply to: ↑ 135 @DrewAPicture
9 years ago

  • Keywords needs-docs added

Replying to pento:

  • DrewAPicture - should each method in WP_Hook have an @since doc, or is the one on the class enough?

The class + every method, every property.

If we can get a refreshed patch in place, I can help with the docs.

#137 @wonderboymusic
9 years ago

  • Keywords needs-refresh removed

For God and Country, I have refreshed this patch 17817.2.diff

This ticket was mentioned in Slack in #core by wonderboymusic. View the logs.


9 years ago

#139 @DrewAPicture
9 years ago

  • Owner changed from pento to DrewAPicture
  • Status changed from assigned to reviewing

@wonderboymusic: Your community thanks you for your service. Next up: Docs review.

@jorbin
9 years ago

#140 follow-up: @jorbin
9 years ago

I modified @wonderboymusic's patch to add a few more @ticket references in the unit tests.

Looking at this more, I think it might make sense to mark everything as final, at least initially. It's easy to remove final, but damn near impossible to add it back later on without breaking things. As hooks are the foundation of the WordPress API, I think it's important that they work consistently across WordPress installs.

#141 in reply to: ↑ 140 @wonderboymusic
9 years ago

Replying to jorbin:

Looking at this more, I think it might make sense to mark everything as final, at least initially.

methods in a class marked final inherit that sentiment, so marking the class makes sense and is a good idea

@jorbin
9 years ago

#142 @jorbin
9 years ago

Updated to include final on the class.

@DrewAPicture
9 years ago

Docs audit

#143 @DrewAPicture
9 years ago

  • Keywords needs-docs removed
  • Owner changed from DrewAPicture to wonderboymusic

Did a docs pass in 17817.5.diff off of 17817.4.diff.

Added:

  • A file header
  • Complete property docs
  • Complete method docs
  • Some inline-spacing and a few comments

Changed:

  • I changed the argument order for ->add_filter() to have the two optional parameters be last (and to better follow the standalone pattern of 'hook', 'callback', 'priority', 'args'. Besides, it's always a little weird when optional parameters are in between required ones.

@wonderboymusic: Tagging you back into the ring.

#144 @wonderboymusic
9 years ago

I like these numbers

 PHPUnit #1
 2.04 mins/179.75MB
 
 PHPUnit #2
 1.9 mins/179.75MB
 
 Patch adds 618 assertions
 
 PHPUnit #1 w/ patch
 2.21min/180MB
 
 PHPUnit #2 w/ patch
 1.96min/180MB

#145 follow-up: @kitchin
9 years ago

17817.5.diff is missing the /tests/phpunit/tests/hooks/* from 17817.4.diff
Also I wonder if the function/method signatures in 17817.5 are so similar that it's confusing:

function add_filter( $tag, $function_to_add, $priority = 10, $accepted_args = 1 )
public function add_filter( $function_to_add, $tag, $priority, $accepted_args )

#146 @DrewAPicture
9 years ago

Looks like I missed re-adding a couple of the tests files back on the last patch. Regenerating ...

#147 @DrewAPicture
9 years ago

Regenerated .5 with the missing tests in 17817.6.diff

@DrewAPicture
9 years ago

.5 + missing tests

@jorbin
9 years ago

#148 @jorbin
9 years ago

  • Keywords commit added

attachment:17817.7.diff updates the unit tests based on the signiture change @drewapicture proposed (and I agree with). Unit tests run clean and quick on the latest nightly of PHP 7.

I think we are ready to play ball.

#149 follow-up: @wonderboymusic
9 years ago

chatted with @nacin earlier, he is doing final review

#150 in reply to: ↑ 149 @wonderboymusic
9 years ago

Replying to wonderboymusic:

chatted with @nacin earlier, he is doing final review

still in @nacin's hands, I imagine he'll update us soon

This ticket was mentioned in Slack in #core by jorbin. View the logs.


9 years ago

#152 @DrewAPicture
9 years ago

#21172 was marked as a duplicate.

#153 @wonderboymusic
9 years ago

  • Owner changed from wonderboymusic to nacin

organizing tickets

#154 @johnbillion
9 years ago

As per Slack chat, deadline for this to go in is Monday otherwise it's getting too late in the cycle to get a good soaking.

#155 @chriscct7
9 years ago

#21169 was marked as a duplicate.

#156 @ocean90
9 years ago

  • Keywords needs-testing added; commit removed

I have applied 17817.7.diff on my dotorg sandbox and it broke our forums which are running bbPress trunk.

I can reproduce the same with the current stable version 2.5.8 of bbPress on my local install. These are the warnings I get:

( ! ) Warning: ksort() expects parameter 1 to be array, object given in /srv/www/wp-shared/develop-plugins/bbpress/includes/core/template-functions.php on line 316
Call Stack
#	Time	Memory	Function	Location
1	0.0018	244024	{main}( )	.../index.php:0
2	0.0025	244472	require( '/srv/www/wp-develop/svn/src/wp-blog-header.php' )	.../index.php:17
3	0.0028	245368	require_once( '/srv/www/wp-develop/svn/src/wp-load.php' )	.../wp-blog-header.php:12
4	0.0038	246400	require_once( '/srv/www/wp-develop/svn/wp-config.php' )	.../wp-load.php:42
5	0.0041	249208	require_once( '/srv/www/wp-develop/svn/src/wp-settings.php' )	.../wp-config.php:102
6	0.1794	4354752	do_action( string(17), ??? )	.../wp-settings.php:336
7	0.1794	4355424	WP_Hook->do_action( array(1) )	.../plugin.php:454
8	0.1794	4356320	call_user_func_array:{/srv/www/wp-develop/svn/src/wp-includes/class-wp-hook.php:282} ( string(21), array(1) )	.../class-wp-hook.php:282
9	0.1795	4356688	bbp_after_setup_theme( string(0) )	.../class-wp-hook.php:282
10	0.1795	4356736	do_action( string(21), ??? )	.../sub-actions.php:345
11	0.1795	4357408	WP_Hook->do_action( array(1) )	.../plugin.php:454
12	0.1796	4358560	call_user_func_array:{/srv/www/wp-develop/svn/src/wp-includes/class-wp-hook.php:282} ( string(24), array(1) )	.../class-wp-hook.php:282
13	0.1796	4358936	bbp_load_theme_functions( string(0) )	.../class-wp-hook.php:282
14	0.1796	4359104	bbp_locate_template( string(21), bool, ??? )	.../template-loader.php:162
15	0.1796	4359304	bbp_get_template_stack( )	.../template-functions.php:65
16	0.1796	4359760	ksort ( object(WP_Hook)[750] )	.../template-functions.php:316

// Nine occurrences of this one:
( ! ) Notice: Undefined index: function in /srv/www/wp-shared/develop-plugins/bbpress/includes/core/template-functions.php on line 326
Call Stack
#	Time	Memory	Function	Location
1	0.0018	244024	{main}( )	.../index.php:0
2	0.0025	244472	require( '/srv/www/wp-develop/svn/src/wp-blog-header.php' )	.../index.php:17
3	0.0028	245368	require_once( '/srv/www/wp-develop/svn/src/wp-load.php' )	.../wp-blog-header.php:12
4	0.0038	246400	require_once( '/srv/www/wp-develop/svn/wp-config.php' )	.../wp-load.php:42
5	0.0041	249208	require_once( '/srv/www/wp-develop/svn/src/wp-settings.php' )	.../wp-config.php:102
6	0.1794	4354752	do_action( string(17), ??? )	.../wp-settings.php:336
7	0.1794	4355424	WP_Hook->do_action( array(1) )	.../plugin.php:454
8	0.1794	4356320	call_user_func_array:{/srv/www/wp-develop/svn/src/wp-includes/class-wp-hook.php:282} ( string(21), array(1) )	.../class-wp-hook.php:282
9	0.1795	4356688	bbp_after_setup_theme( string(0) )	.../class-wp-hook.php:282
10	0.1795	4356736	do_action( string(21), ??? )	.../sub-actions.php:345
11	0.1795	4357408	WP_Hook->do_action( array(1) )	.../plugin.php:454
12	0.1796	4358560	call_user_func_array:{/srv/www/wp-develop/svn/src/wp-includes/class-wp-hook.php:282} ( string(24), array(1) )	.../class-wp-hook.php:282
13	0.1796	4358936	bbp_load_theme_functions( string(0) )	.../class-wp-hook.php:282
14	0.1796	4359104	bbp_locate_template( string(21), bool, ??? )	.../template-loader.php:162
15	0.1796	4359304	bbp_get_template_stack( )	.../template-functions.php:65

#157 @DrewAPicture
9 years ago

Note, this would also fix the problem with priorities demonstrated in #27428.

@jbrinley
9 years ago

#158 @jbrinley
9 years ago

Thanks for testing, @ocean90. Those errors are disheartening. There's really no way to be 100% backwards compatible with something that touches so deeply into the internals of $wp_filter. bbPress could be revised to work with both the old and the new $wp_filter with code such as:

function bbp_get_template_stack() {
	global $wp_filter, $merged_filters, $wp_current_filter;

	// Setup some default variables
	$tag  = 'bbp_template_stack';
	$args = $stack = array();

	// Add 'bbp_template_stack' to the current filter array
	$wp_current_filter[] = $tag;

	if ( class_exists( 'WP_Hook' ) ) {
		$filter = $wp_filter[ $tag ]->callbacks;
	} else {
		$filter = &$wp_filter[ $tag ];

		// Sort
		if ( ! isset( $merged_filters[ $tag ] ) ) {
			ksort( $filter );
			$merged_filters[ $tag ] = true;
		}
	}

	// Ensure we're always at the beginning of the filter array
	reset( $filter );

	// Loop through 'bbp_template_stack' filters, and call callback functions
	do {
		foreach ( (array) current( $filter ) as $the_ ) {
			if ( ! is_null( $the_['function'] ) ) {
				$args[1] = $stack;
				$stack[] = call_user_func_array( $the_['function'], array_slice( $args, 1, (int) $the_['accepted_args'] ) );
			}
		}
	} while ( next( $filter ) !== false );

	// Remove 'bbp_template_stack' from the current filter array
	array_pop( $wp_current_filter );

	// Remove empties and duplicates
	$stack = array_unique( array_filter( $stack ) );

	return (array) apply_filters( 'bbp_get_template_stack', $stack ) ;
}

#159 @jbrinley
9 years ago

In looking at @ocean90's issue, I realized that while IteratorAggregate is 5.2 compatible, the ArrayIterator it was returning is SPL. 17817.8.diff reworks WP_Hook to implement Iterator instead of IteratorAggregate, handling the array traversal itself rather than delegating to ArrayIterator.

This ticket was mentioned in Slack in #bbpress by ocean90. View the logs.


9 years ago

#161 @helen
9 years ago

cc @jjj for the bbPress errors above.

#162 follow-up: @helen
9 years ago

Sigh, didn't catch that age-old mistake fast enough. That would be @johnjamesjacoby.

#163 in reply to: ↑ 162 ; follow-up: @johnjamesjacoby
9 years ago

Replying to helen:

Sigh, didn't catch that age-old mistake fast enough. That would be @johnjamesjacoby.

Thanks for the ping. Two thoughts:

  • Both BuddyPress and bbPress will need an update to match the new approach in WordPress core
  • Happy to take over the @jjj account if someone wants to bless that task for all of our conveniences

#164 in reply to: ↑ 163 ; follow-up: @helen
9 years ago

Replying to johnjamesjacoby:

  • Both BuddyPress and bbPress will need an update to match the new approach in WordPress core

More concerned about back-compat here (not taking down sites that update core before the plugin or fail to update the plugin for whatever reason) and your thoughts on the approach, since you're doing deeper things with hooks.

#165 in reply to: ↑ 164 @johnjamesjacoby
9 years ago

Replying to helen:

Replying to johnjamesjacoby:

  • Both BuddyPress and bbPress will need an update to match the new approach in WordPress core

More concerned about back-compat here (not taking down sites that update core before the plugin or fail to update the plugin for whatever reason) and your thoughts on the approach, since you're doing deeper things with hooks.

IIRC, the leads in this ticket predicted a high likelihood of breakage, and deemed it necessary for the sake of progress & performance. See this comment from nacin among others here.

The bbs are a good litmus test for this type of breakage because they hug tightly to internal API's, sometimes in creative ways. Each bb'breakage is a scar on their surface; a point in time where WordPress got better because the bb's absorbed an impact.

Because WooCommerce and others have taken inspiration from this template-stack code, I'm weary of slipping it in quickly without more exposure: a post on Make, a clear description of why this break is necessary, and maybe even suggestions for alternative approaches.

All of that said, changing this code at all scares the daylights out of me. Many subsystems rely on this foundation, and the bb's may have only helped bring attention to one specific implementation.

Last edited 9 years ago by johnjamesjacoby (previous) (diff)

#166 @wonderboymusic
9 years ago

  • Milestone changed from 4.4 to Future Release
  • Owner nacin deleted

#167 @johnjamesjacoby
9 years ago

Also, both BuddyPress & bbPress have considerations for juggling WordPress version nuances. If either project needs to move away from direct touches to the $wp_filter global (which is a known fragility) it's possible to achieve the same functionality with with a variable in their respective singletons, or a number of other ways. All about the greater good.

#168 @boonebgorges
9 years ago

The lameness of not doing this for 4.4 is extensive and profound, but I guess it's the right choice.

If we are confident that we're going to go with this in 4.5 (or whenever), we should start the education campaign today. One, developers cannot always drop everything and respond to stuff like this in the course of a few weeks or even a few months. Two, and more importantly: the sooner affected plugins can push out a fix, the more of a gap we put between the fix being available and the breaking change going into WordPress. The critical situation to avoid is one where an out-of-date version of BuddyPress (or whatever) is on a site that updates to WP_Hook; putting more time between a fixed BuddyPress and WP_Hook is going to make this less likely to happen.

#169 follow-up: @nacin
9 years ago

Some of you may know that I have been playing with this patch for a few weeks with the intention of committing it. I also played with it on my WordPress.org sandbox (a self-imposed prerequisite prior to committing), but I hadn't tested all scenarios and missed that it broke the bb's. Good catch, @ocean90. I didn't notice anything else, which is great.

I want to try something here, which we've done with success before -- putting something in trunk that has a very slim chance of making it to the final release, in order to get a much better idea of what it could break. Thus, I would be comfortable committing this to trunk in the next 7 days under the following conditions:

  1. We need some kind of fix or workaround (somewhere -- .org, core, or bb's) to avoid .org breakage. [ACTION NEEDED]
  2. We publish https://gist.github.com/jbrinley/eaaad00b52e1316c6904 on make/core, with some adjustments...
  3. The post should say it'll ship in 4.5, but that it is in trunk now to assess the extent of the damage. We should leave it in trunk for up to 14-21 days. This would imply its removal by beta 2 or beta 3. I will manage the fallout. Note "up to 14-21 days" -- if the breakage is particularly severe, it will have to come out quickly.

The time it spends in core will further inform how much further it has to go, whether it needs a significant shift in strategy, and whether it's possible at all.

It's up to all of you to find success on point 1. But if that happens, I will handle points 2 and 3, and the commit itself.

#170 in reply to: ↑ 169 ; follow-up: @netweb
9 years ago

Replying to nacin:

  1. We need some kind of fix or workaround (somewhere -- .org, core, or bb's) to avoid .org breakage. [ACTION NEEDED]

ACTION STARTED: https://bbpress.trac.wordpress.org/ticket/2871 WP_Hook

I'm yet to wrap my head around all of this or even read/test the patch fully; anyways, we'll get a start on for bbPress

#171 @boonebgorges
9 years ago

BuddyPress has been updated so that it no longer manipulates $wp_filter sub-arrays directly. https://buddypress.trac.wordpress.org/changeset/10279

#172 @rotozeev
9 years ago

Support of NextGen Gallery plugin have sad that this bug is a cause of my problem. There is a problem of crossposting to another blogs: shortcodes of NGG doesn't transform to html due to the apply_filters("the_content", ...) and as result I obtain the text like [singlepic id=831] instead picture. So, may be somebody knows how I can easy improve this problem?

#174 in reply to: ↑ 145 @SergeyBiryukov
9 years ago

Replying to kitchin:

Also I wonder if the function/method signatures in 17817.5 are so similar that it's confusing:

function add_filter( $tag, $function_to_add, $priority = 10, $accepted_args = 1 )
public function add_filter( $function_to_add, $tag, $priority, $accepted_args )

Good point, 17817.9.diff makes the order of arguments for WP_Hook::add_filter(), WP_Hook::remove_filter(), and WP_Hook::has_filter() consistent with the respective functions.

#175 in reply to: ↑ 170 @netweb
9 years ago

Replying to netweb:

ACTION STARTED: https://bbpress.trac.wordpress.org/ticket/2871 WP_Hook

Done, see [bbpress5944]

#176 @DrewAPicture
9 years ago

#36199 was marked as a duplicate.

@dougwollison
9 years ago

Updated doc block for apply_filters; shouldn't claim $priority and $accepted_args are optional. Also removed random extra semicolon after an if block.

@dougwollison
9 years ago

Correct version of previous diff; a number of the new files were missing.

#177 @boonebgorges
9 years ago

#36097 was marked as a duplicate.

@khag7
9 years ago

Fixes 17817.10.2 - class-wp-hook.php has entire contents duplicated

#178 follow-up: @khag7
9 years ago

17817.10.2.diff had an issue, 17817.10.3.diff fixes it.

It looks like @dougwollison copy and pasted the code for that file so it was in there twice. First 480 lines were identical to next 480 lines.

Last edited 9 years ago by khag7 (previous) (diff)

#179 in reply to: ↑ 178 @dougwollison
9 years ago

Replying to khag7:

17817.10.2.diff had an issue, 17817.10.3.diff fixes it.

It looks like @dougwollison copy and pasted the code for that file so it was in there twice. First 480 lines were identical to next 480 lines.

Thanks, legitimately not sure how I managed that.

By the way, would it be worth adding coding-standards cleanup to plugins.php while we're at this? Fixing brace-less if/foreach/while blocks mainly.

This ticket was mentioned in Slack in #core by casiepa. View the logs.


9 years ago

This ticket was mentioned in Slack in #core by netweb. View the logs.


8 years ago

#182 @parisholley
8 years ago

any idea when this is going to make it into a release? about pulled my hair out today trying to figure out why things weren't working and this was the root cause.

#183 @barry.hughes
8 years ago

This has been a frequent source of problems for us, it would be great to see it move forward into a release!

Last edited 8 years ago by barry.hughes (previous) (diff)

#184 @Compute
8 years ago

Something for 4.7 early maybe?

#185 @pento
8 years ago

  • Milestone changed from Future Release to 4.7
  • Owner set to pento

@pento
8 years ago

Apply cleanly against trunk.

This ticket was mentioned in Slack in #core by ocean90. View the logs.


8 years ago

This ticket was mentioned in Slack in #core by jorbin. View the logs.


8 years ago

#188 @noplanman
8 years ago

#37679 was marked as a duplicate.

This ticket was mentioned in Slack in #core by helen. View the logs.


8 years ago

This ticket was mentioned in Slack in #core by pento. View the logs.


8 years ago

This ticket was mentioned in Slack in #core by helen. View the logs.


8 years ago

@pento
8 years ago

This ticket was mentioned in Slack in #core by pento. View the logs.


8 years ago

@pento
8 years ago

#193 @pento
8 years ago

  • Keywords needs-unit-tests added; has-unit-tests removed

17817.12.diff fixes some unit tests, and updates the @since docs.

There was a bug I've been running into, triggered by this code. It's kind of ugly, but not insane.

<?php
/*
Plugin Name: lol
 */
add_filter( 'the_content', 'foo', 999999 );

function foo( $content ) {
        return $content . bar();
}

function bar() {
        remove_filter( 'the_content', 'foo', 999999 );
        $bar = apply_filters( 'the_content', 'bar' );
        add_filter( 'the_content', 'foo', 999999 );
        return $bar;
}

When remove_filter() is called, it eventually gets to WP_Hook::resort_active_iterations(). While the logic in this method was sound, the foreach() was resetting the array pointer for each item in $this->iterations.

In the above example, the result was that, when foo() returned to WP_Hook::apply_filters(), the internal pointer of $this->iterations[ $nesting_level ] had been reset to the first element, instead of staying where it was supposed to.

17817.13.diff fixes this bug by operating on a reference to each element in the WP_Hook::resort_active_iterations() loop.

It also bails on the loop early when the current() element is false, there's no need to loop through an array that we're already at the end of.

17817.13.diff needs unit tests to confirm that this behaviour is fixed.

This ticket was mentioned in Slack in #core by pento. View the logs.


8 years ago

#195 @noplanman
8 years ago

@pento Latest diff isn't working for me. It seems to be the issue of the pointer losing the correct position. (I think the loop on line class-wp-hook.php:293)

Interestingly though, if I remove all references of $nesting_level (or rather, keep it at 0 always), it works perfectly for my initial issue #37679.

@pento
8 years ago

@pento
8 years ago

@pento
8 years ago

#196 @pento
8 years ago

  • Keywords has-unit-tests added; needs-unit-tests removed

17817.16.diff adds a unit test for the filters being added and removed from the currently running hook.

@noplanman: The new test added (test_remove_and_add in tests/hooks/add_filter.php) seems to cover the remove/add behaviour you described in #37679, but I was unable to reproduce the output you described.

Could I get you to have a look at the test, and say how it differs from your code?

#197 @noplanman
8 years ago

@pento, I see your test is adding the filters with the same priority. The problem only happens if the priorities are different.

Please check #37679 again to see what I mean.

Unfortunately I can't modify your test, as PHP/PHPUnit dies on me with a Process finished with exit code 139 (interrupted by signal 11: SIGSEGV) when I change the priorities, which is very odd.
I'm trying to find a solution for this, without luck so far.
Maybe you have an idea why PHP is failing like this?

@pento
8 years ago

#198 @pento
8 years ago

Ah, I see what's happening.

  • p11 removes itself, and so ::resort_active_iterations() causes the current $this->iterations to skip forward to priority 12.
  • p11 re-adds itself, but now the current priority in $this->iterations is 12, so ::resort_active_iterations() doesn't go backwards.
  • When ::apply_filters() finishes priority 11, it assumes that the current $this->iterations hasn't changed, so it grabs the next priority (after 12).

17817.17.diff alters ::resort_active_iterations(), to detect when a new priority level has been created, that matches the current priority according to ::apply_filters(), and moves the current $this->iterations back to that priority level.

@noplanman: Could you test 17817.17.diff?

#199 @noplanman
8 years ago

Hi @pento

Happy to test and get this into core =)

With filters the tests work nicely. But unfortunately my problem persists using your latest code changes.

Using do_action doesn't seem to work. Here my tests for you to play with:

public function test_remove_and_add_last_action_1() {
  $this->hook = new Wp_Hook();

  $this->hook->add_filter( 'remove_and_add', '__return_empty_string', 10, 0 );

  $this->hook->add_filter( 'remove_and_add', array( $this, '_action_remove_and_add1' ), 11, 0 );
  $this->hook->add_filter( 'remove_and_add', array( $this, '_action_remove_and_add2' ), 12, 0 );

  $this->action_output = '';
  $this->hook->do_action( [] );

  // Works! First add1, then add2. ($this->action_output = '12')
  $this->assertSame( '12', $this->action_output );
}

public function test_remove_and_add_last_action_2() {
  $this->hook = new Wp_Hook();

  $this->hook->add_filter( 'remove_and_add', '__return_empty_string', 10, 0 );

  $this->hook->add_filter( 'remove_and_add', array( $this, '_action_remove_and_add1' ), 13, 0 );
  $this->hook->add_filter( 'remove_and_add', array( $this, '_action_remove_and_add2' ), 12, 0 );

  $this->action_output = '';
  $this->hook->do_action( [] );

  // Fails! First add2, but no add1. ($this->action_output = '2')
  $this->assertSame( '21', $this->action_output );
}

public function _action_remove_and_add1() {
  $this->action_output .= '1';
}

public function _action_remove_and_add2() {
  $this->hook->remove_filter( 'remove_and_add', array( $this, '_action_remove_and_add2' ), 12 );
  $this->hook->add_filter( 'remove_and_add', array( $this, '_action_remove_and_add2' ), 12, 0 );

  $this->action_output .= '2';
}

Hope this helps.

@pento
8 years ago

#200 @pento
8 years ago

Oh, good catch @noplanman. I totally forgot to update WP_Hook::do_action() to keep track of the current priority. :-)

17817.18.diff fixes this, and adds unit tests.

#201 @noplanman
8 years ago

Ok, can confirm that it's working now, great job! :-D

If you need any other testing, just let me know.

#202 @pento
8 years ago

Great news!

I've been doing more testing on W.org and WP.com, everything appears to be working smoothly. As such, the plan is to commit in the coming week. Please don't hold back on trying to break it over the next few days, I'd prefer to fix bugs now, rather than after it rolls out in the nightly. :-)

As @nacin suggested a lifetime ago, the draft post that @jbrinley wrote will be published on make/core, with some comments around timing for the final keep/revert decision.

@pento
8 years ago

@pento
8 years ago

#203 @pento
8 years ago

The last few patches reminded why it's super annoying that apply_filters() and do_action() are very-similar-but-different bits of code.

17817.20.diff merges the two, so ::do_action() now calls ::apply_filters(). I've added a unit test to check that multiple hooks against an action will still get the same value passed.

This was surprisingly easy to do, which was nice, but makes me very suspicious. Please read the code and think about ways to make it explode. :-)

@pento
8 years ago

#204 @pento
8 years ago

17817.21.diff expands the test added in 17817.20.diff to test across multiple priorities.

#205 @ocean90
8 years ago

@pento Patches .19 to .21 are missing the WP_Hook class. Can you please add a patch which includes everything? 😉

@pento
8 years ago

#206 @pento
8 years ago

Well, at least someone around here is paying attention.

17817.22.diff exists, for all of your WP_Hook needs.

#207 @ocean90
8 years ago

Thanks @pento.

I run the tests with 17817.22.diff on PHP 7.1 RC1 and it seems like it also fixes #37295 and #37772.

But there is one failure when running phpunit -c tests/phpunit/multisite.xml:

1) Test_Nav_Menus::test_wp_get_associated_nav_menu_items
Trying to get property of non-object

/src/wp-includes/ms-blogs.php:1221
/src/wp-includes/class-wp-hook.php:294
/src/wp-includes/class-wp-hook.php:319
/src/wp-includes/plugin.php:453
/src/wp-includes/post.php:2414
/tests/phpunit/tests/post/nav-menu.php:101

I created a PR on @jorbin's GitHub mirror to run the tests on Travis CI: https://github.com/aaronjorbin/develop.wordpress/pull/16 / https://travis-ci.org/aaronjorbin/develop.wordpress/builds/157840503

@pento
8 years ago

#208 follow-up: @pento
8 years ago

Noice. ::do_action() didn't like it when an action was called recursively. :-)

17817.23.diff fixes this by adding a recursion check to ::do_action(). It also updates test_do_action_doesnt_change_value to test for recursion, too.

https://travis-ci.org/aaronjorbin/develop.wordpress/builds/157912226

This ticket was mentioned in Slack in #core-multisite by jeremyfelt. View the logs.


8 years ago

#210 in reply to: ↑ 208 @ocean90
8 years ago

Replying to pento:

Noice. ::do_action() didn't like it when an action was called recursively. :-)

The Travis CI build was successful for 17817.23.diff. I'm also running this patch on my w.org sandbox without any issues so far.

The lost five methods of WP_Hook could need some docs love, missing periods and incomplete @return tags.

#211 @aaroncampbell
8 years ago

17817.24.diff adds the @return for the methods that @ocean90 pointed out were missing it. It also removes the return from the rewind() method since it's ignored anyway. All tests still appear to pass, and I pounded on the iterator a decent bit and the rewinding seems to still work as expected.

This ticket was mentioned in Slack in #core by ocean90. View the logs.


8 years ago

This ticket was mentioned in Slack in #core by pento. View the logs.


8 years ago

#214 @pento
8 years ago

  • Resolution set to fixed
  • Status changed from reviewing to closed

In 38571:

Hooks: Add the new class WP_Hook, and modify hook handling to make use of it.

Filters and actions have been the basis of WordPress' plugin functionality since time immemorial, they've always been a reliable method for acting upon the current state of WordPress, and will continue to be so.

Over the years, however, edge cases have cropped up. Particularly when it comes to recursively executing hooks, or a hook adding and removing itself, the existing implementation struggled to keep up with more complex use cases.

And so, we introduce WP_Hook. By changing $wp_filter from an array of arrays, to an array of objects, we reduce the complexity of the hook handling code, as the processing code (see ::apply_filters()) only needs to be aware of itself, rather than the state of all hooks. At the same time, we're able te handle more complex use cases, as the object can more easily keep track of its own state than an array ever could.

Props jbrinley for the original architecture and design of this patch.
Props SergeyBiryukov, cheeserolls, Denis-de-Bernardy, leewillis77, wonderboymusic, nacin, jorbin, DrewAPicture, ocean90, dougwollison, khag7, pento, noplanman and aaroncampbell for their testing, suggestions, contributions, patch maintenance, cajoling and patience as we got through this.
Fixes #17817.

#215 @DrewAPicture
8 years ago

In 38573:

Docs: Fix minor formatting for inline docs in WP_Hook following its introduction in [38571].

See #17817.

#216 @jacobsantos
8 years ago

Oh. Oh dear god. Oh you poor poor people. I am so, so very sorry.

I guess it will be okay. It should be okay. As long as everything works. Looking at the code that was committed...

I'm sorry. The design decisions that were made years and years ago were mostly compromises for PHP4. Things had to work on PHP4 you see, PHP4 had priority.

Also, the decision was that we didn't care about plugins affected by this bug, because at the time it was an edge case. I feel that it still is by the way. Looks like I'm going to maintain a separate build of WordPress that reverts this mess of a commit.

I don't care. You will either regret this commit and revert it or it won't matter and my fears will be unfounded. Either way, I think it would be a good time for a fork. Or not. It should be fine. I mean, it isn't like this patch resembles a patch that was submitted years ago back when we had a similar problem and rejected that shit faster than the fastest thing you can think that can go fast.

It is funny to me. How history repeats itself. I hope evil didn't prevail this time. Who knows? I forget why a similar solution was rejected so long ago. Performance reasons? Must have been performance reasons and PHP4. WordPress sucked back when PHP4 had to be supported. Imagine being stuck supporting a older version of PHP. What a nightmare.

I still think it is funny that this whole thing could be solved with one line and fuck it, let's just add class! I mean, you can't really solve this problem in a good way. Well, you can, you can do what you did in the commit.

You know. I'm sure it will be fine. It'll be fine. PHP5 and PHP7 are way faster than they were back in 2007. Servers are way faster than they were in 2007.

Well. Good luck. I'll make sure the door doesn't hit my ass on the way out again.

This ticket was mentioned in Slack in #core by jacobsantos. View the logs.


8 years ago

#218 follow-up: @noplanman
8 years ago

@jacobsantos

I still think it is funny that this whole thing could be solved with one line

For learning sake, what fix are you referring to?
One line really isn't a big change and I'd really like to understand how it would fix this issue properly.

#219 in reply to: ↑ 218 @dougwollison
8 years ago

Replying to noplanman:

@jacobsantos

I still think it is funny that this whole thing could be solved with one line

For learning sake, what fix are you referring to?
One line really isn't a big change and I'd really like to understand how it would fix this issue properly.

If I had to guess, something like what me and others have proposed over the years in our duplicate tickets. Namely, copying the hooks list for a tag to a new variable and iterating over that. But it's been pointed out there are apparently performance issues with that solution.

#220 follow-up: @noplanman
8 years ago

Ok, I see what you mean.

I'm just curious to know if recursive adding/removing of hooks within a hook would work properly then, as the array would be changing during the loop.

At the moment I'm just happy that @pento has put in the effort to get this fixed.

#221 in reply to: ↑ 220 @jacobsantos
8 years ago

Replying to noplanman:

Ok, I see what you mean.

I'm just curious to know if recursive adding/removing of hooks within a hook would work properly then, as the array would be changing during the loop.

At the moment I'm just happy that @pento has put in the effort to get this fixed.

Yeah, that was it exactly. The problem is what do you want to happen? If you add a callback to a hook that is already running, then what do you want to happen?

  1. The callback is called at the next time the hook is called.
  2. The callback is called later within the same hook execution. So if you add a callback to hook 'something' running with priority 10, while running at priority 10, that same callback is called at some point soon after.

The solution can't have both. Or realistically, it could have both, but solutions for both are different and counter each other. I am sure this was mentioned at some point above. With the patch, great, I am guessing the patch is the option 2.

The problem is what do people mean when they add to the same hook they are running on? So if the programmer starts adding callbacks to the 'wp_init' hook, do they mean they want it to run soon after? I understand there is a chicken and the egg problem. That is why there is an early process startup execution and late process startup execution hook. So you can manage what goes into the latter in the former.

The point is more so that if you want to remove a callback from a hook, you may do so. What logical reason does a person have for adding more hooks to 'wp_init'? Do you mean you want to execute the callbacks? Why not just execute the callbacks?


For me, the problem is not that the problem should or should not have been fixed. Incorrect behavior is incorrect behavior and should be fixed. The problem is that the solution required is hideous. Not the code provided and the code required to accurately and completely fix the problem. The problem is fundamental with how PHP handles arrays and references and pointers. Requiring code that breaks this behavior in some way to provide consistency.

This is code that when it breaks, it will be damn near impossible to fix. It will be damn near impossible to optimize. The original patch like the one committed was beautiful in that only one person knew how it worked. Only one or a few people would ever know how it worked. You do not want code like that in any project. You may think you understand how this code works, but you don't. Or at least, if it was like the original patch. The code that was committed appears to be a bit more straightforward, so I am not sure if that just means I know more or that the author is a better code artist.

I only got to the point where what appears to be an incredibly expensive method is called, each time a callback is added to a hook. Sure, optimization is sometimes counter-intuitive, but when has sorting ever been a fast process? For each addition? Insanity. I would buy a hat and eat it, if sorting each time didn't have a huge performance hit.

It is said that copying the array would have performance issues. How does what was committed not have huge performance issues?

The effort is awesome, sure, but you just used C4 to clean a window. I guess technically, the window is clean, albeit scattered to the wind.

#222 @noplanman
8 years ago

Incorrect behavior is incorrect behavior and should be fixed.

Absolutely, though utopian. Especially in a large project, it's extremely difficult to "force" developers/users to use it properly. Because what does "properly" even mean. This is where flexibility comes in and allowing developers to express and create amazing things. Of course there are boundaries, but isn't software here to try to break those?

I totally agree with you, that in the scenario that you described, there are different way of interpreting what the developer wanted to achieve. That's a tough boundary to crack, as it's something a machine can't (maybe yet) know. Edge cases like that are clearly an exception, but should still have their place, motivating more developers to come up with unique and gorgeous solutions.
The same will be the case with this fix. Some time down the line, there will probably be new changes, optimisations, maybe a whole rewrite, who knows!

But hey, we're all here to learn together, right? Striving to become better and write smarter code :-)

I think we can put this discussion on pause here, as it isn't really contributing to this current issue any more, but thanks a lot for elaborating on your point!

#223 @flixos90
8 years ago

A small improvement idea: I think it would make sense to separate the WP_Hook::has_filter() and WP_Hook::has_filters() methods completely so that both have a single responsibility: The $function_to_check parameter would become required for WP_Hook::has_filter() and we would no longer call WP_Hook::has_filters() from WP_Hook::has_filter(). The logic to be backwards compatible could then happen in the regular has_filter() function - this way at least the new class methods would only have one specific purpose.

Another thing I observed: This might have a reason I'm overlooking right now, but why aren't we storing the tag (hook name) in the class as property (and pass it to the constructor)? This would allow us to get rid of passing the $tag parameter on all the methods. Sorry to come up with this now, I haven't really paid attention to this movement before :(

#224 follow-up: @danielbachhuber
8 years ago

FYI - Because WP-CLI directly modifies the $wp_filter global, this WP_Hook thing breaks the implementation we adopted in #34936. See https://travis-ci.org/wp-cli/wp-cli/jobs/158420911

If r38282 stays in, I can get around this by requiring plugins.php early in WP-CLI (#37707) and using add_action() and add_filter().

#225 @jorbin
8 years ago

I don't see any reason that r38282 won't be staying in.

#226 follow-ups: @nerrad
8 years ago

So we were getting failing unit tests in our builds after this was deployed and so far it looks like it traces to our usage of the tests_add_filter() function found in /tests/phpunit/functions.php. That function overrides $wp_filter and doesn't implement the new changes.

#227 in reply to: ↑ 226 @jorbin
8 years ago

Replying to nerrad:

So we were getting failing unit tests in our builds after this was deployed and so far it looks like it traces to our usage of the tests_add_filter() function found in /tests/phpunit/functions.php. That function overrides $wp_filter and doesn't implement the new changes.

Can you provide a sample test that demonstrates this failure please.

#228 @nerrad
8 years ago

Here's the code I see for tests_add_filter (to be clear, this is pulled directly from the WordPress Core UnitTest files):

// For adding hooks before loading WP
function tests_add_filter($tag, $function_to_add, $priority = 10, $accepted_args = 1) {
	global $wp_filter;

	$idx = _test_filter_build_unique_id($tag, $function_to_add, $priority);
	$wp_filter[$tag][$priority][$idx] = array('function' => $function_to_add, 'accepted_args' => $accepted_args);
	return true;
}

All we have in our tests is implementation of that filter in our plugin unit tests bootstrap. So basically we do something like:

tests_add_filter( 'some_filter_we_want_to_hook_into', '__return_false' )

Last edited 8 years ago by nerrad (previous) (diff)

#229 in reply to: ↑ 224 ; follow-up: @danielbachhuber
8 years ago

Replying to danielbachhuber:

FYI - Because WP-CLI directly modifies the $wp_filter global, this WP_Hook thing breaks the implementation we adopted in #34936.

As it turns out, WP-CLI broke because of poor coding on my part. My bad. See https://github.com/wp-cli/wp-cli/pull/3384 for the fix.

However, I discovered this breaks the intent of #37707 because it depends on WPINC, which is only available after wp-settings.php has begun loading.

This ticket was mentioned in Slack in #buddypress by boone. View the logs.


8 years ago

#231 in reply to: ↑ 226 ; follow-up: @boonebgorges
8 years ago

  • Resolution fixed deleted
  • Status changed from closed to reopened

Replying to nerrad:

So we were getting failing unit tests in our builds after this was deployed and so far it looks like it traces to our usage of the tests_add_filter() function found in /tests/phpunit/functions.php. That function overrides $wp_filter and doesn't implement the new changes.

I can confirm that there's an issue here. tests_add_filter() still works OK before WP has loaded - WP_Hook::build_preinitialized_hooks() takes care of the manually-set $wp_filter keys. But if you run tests_add_filter() after these pre-bootstrap filters are normalized, they never get converted to WP_Hook objects. 17817.25.diff is a simple fix, though the class_exists() check might break once WP_Hook is autoloaded; we may want a global flag or a constant or somesuch to keep track of whether preexisting hooks have been converted.

@jorbin
8 years ago

#232 in reply to: ↑ 229 ; follow-up: @jorbin
8 years ago

Replying to danielbachhuber:

Replying to danielbachhuber:

FYI - Because WP-CLI directly modifies the $wp_filter global, this WP_Hook thing breaks the implementation we adopted in #34936.

As it turns out, WP-CLI broke because of poor coding on my part. My bad. See https://github.com/wp-cli/wp-cli/pull/3384 for the fix.

However, I discovered this breaks the intent of #37707 because it depends on WPINC, which is only available after wp-settings.php has begun loading.

The above patch fixes this issue, though the more I think about it, the more I wonder if WP_Hook should even be in a separate file. Is there ever a time you would want one and not the other?

#233 in reply to: ↑ 231 ; follow-up: @dd32
8 years ago

Replying to boonebgorges:

I can confirm that there's an issue here. tests_add_filter() still works OK before WP has loaded - WP_Hook::build_preinitialized_hooks() takes care of the manually-set $wp_filter keys. But if you run tests_add_filter() after these pre-bootstrap filters are normalized, they never get converted to WP_Hook objects. 17817.25.diff is a simple fix, though the class_exists() check might break once WP_Hook is autoloaded; we may want a global flag or a constant or somesuch to keep track of whether preexisting hooks have been converted.

Once @jorbin's 17817.26.diff is in, can't 17817.25.diff just be a check on function_exists( 'add_filter' ) and use the API when available, falling back to directly modifying when it's called before initialisation?

#234 in reply to: ↑ 232 @pento
8 years ago

Replying to jorbin:

The above patch fixes this issue, though the more I think about it, the more I wonder if WP_Hook should even be in a separate file. Is there ever a time you would want one and not the other?

17817.26.diff will break things in weird ways - if something loads plugin.php without loading class-wp-hook.php, it still won't be able to use add_filter(), etc. And once plugin.php is loaded, it can't be relied on to process $wp_filter as it does at the top of the file.

Discussion of the correct fix for that can continue in #37707.

#235 in reply to: ↑ 233 @pento
8 years ago

Replying to dd32:

Once @jorbin's 17817.26.diff is in, can't 17817.25.diff just be a check on function_exists( 'add_filter' ) and use the API when available, falling back to directly modifying when it's called before initialisation?

It doesn't need to wait for 17817.26.diff. In the test suite, phpunit/includes/functions.php is loaded, it plays some funky music, then wp-settings.php is loaded - WP_Hook will be loaded the same before or after whatever the fix for #37707 is.

#236 @pento
8 years ago

  • Resolution set to fixed
  • Status changed from reopened to closed

In 38582:

Tests: Use add_filter() when it's available.

The tests_add_filter() helper function directly manipulates the $wp_filter global, instead of using add_filter(). We can use add_filter() when it's available, and fall back to manipulating $wp_filter when it isn't, relying on the $wp_filter bootstrap code at the top of plugin.php to handle conversion.

Props boonebgorges, dd32 and pento: WordPress Thought Leadership Triumvirate.
Fixes #17817.

#237 @kraftbj
8 years ago

This change introduced #38011 which changes the behavior when a function called by a hook adds another function onto the same hook with an earlier priority. The new behavior is logical, but different than previously.

@dd32
8 years ago

Upgrade filters added manually in advanced-cache.php

This ticket was mentioned in Slack in #core-js by adamsilverstein. View the logs.


7 years ago

Note: See TracTickets for help on using tickets.