Revisions: Improve performance of `WP_Text_Diff_Renderer_Table`.
`WP_Text_Diff_Renderer_Table` is used to generate the diff view in revisions, but there were some cases that could cause it to take excessive amounts of time to run. Some noteable cases include: - When a large number of new lines were inserted in the middle of the post from one revision to the next. - When both revisions contain >100 lines. - When either revision contains a lot of long lines. In one extreme test case, the diff view took over a minute to generate. With this change, it now takes less than a second. See #35667. Built from https://develop.svn.wordpress.org/branches/5.0@43775 git-svn-id: http://core.svn.wordpress.org/branches/5.0@43604 1a063a9b-81f0-0310-95a4-ce76da25c4cd
This commit is contained in:
parent
bae53caa70
commit
648850ea73
|
@ -55,6 +55,22 @@ class WP_Text_Diff_Renderer_Table extends Text_Diff_Renderer {
|
||||||
|
|
||||||
protected $compat_fields = array( '_show_split_view', 'inline_diff_renderer', '_diff_threshold' );
|
protected $compat_fields = array( '_show_split_view', 'inline_diff_renderer', '_diff_threshold' );
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Caches the output of count_chars() in compute_string_distance()
|
||||||
|
*
|
||||||
|
* @var array
|
||||||
|
* @since 5.0.0
|
||||||
|
*/
|
||||||
|
protected $count_cache = array();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Caches the difference calculation in compute_string_distance()
|
||||||
|
*
|
||||||
|
* @var array
|
||||||
|
* @since 5.0.0
|
||||||
|
*/
|
||||||
|
protected $difference_cache = array();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor - Call parent constructor with params array.
|
* Constructor - Call parent constructor with params array.
|
||||||
*
|
*
|
||||||
|
@ -391,13 +407,11 @@ class WP_Text_Diff_Renderer_Table extends Text_Diff_Renderer {
|
||||||
if ( false === $final_pos ) { // This orig is paired with a blank final.
|
if ( false === $final_pos ) { // This orig is paired with a blank final.
|
||||||
array_splice( $final_rows, $orig_pos, 0, -1 );
|
array_splice( $final_rows, $orig_pos, 0, -1 );
|
||||||
} elseif ( $final_pos < $orig_pos ) { // This orig's match is up a ways. Pad final with blank rows.
|
} elseif ( $final_pos < $orig_pos ) { // This orig's match is up a ways. Pad final with blank rows.
|
||||||
$diff_pos = $final_pos - $orig_pos;
|
$diff_array = range( -1, $final_pos - $orig_pos );
|
||||||
while ( $diff_pos < 0 )
|
array_splice( $final_rows, $orig_pos, 0, $diff_array );
|
||||||
array_splice( $final_rows, $orig_pos, 0, $diff_pos++ );
|
|
||||||
} elseif ( $final_pos > $orig_pos ) { // This orig's match is down a ways. Pad orig with blank rows.
|
} elseif ( $final_pos > $orig_pos ) { // This orig's match is down a ways. Pad orig with blank rows.
|
||||||
$diff_pos = $orig_pos - $final_pos;
|
$diff_array = range( -1, $orig_pos - $final_pos );
|
||||||
while ( $diff_pos < 0 )
|
array_splice( $orig_rows, $orig_pos, 0, $diff_array );
|
||||||
array_splice( $orig_rows, $orig_pos, 0, $diff_pos++ );
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -425,12 +439,28 @@ class WP_Text_Diff_Renderer_Table extends Text_Diff_Renderer {
|
||||||
* @return int
|
* @return int
|
||||||
*/
|
*/
|
||||||
public function compute_string_distance( $string1, $string2 ) {
|
public function compute_string_distance( $string1, $string2 ) {
|
||||||
// Vectors containing character frequency for all chars in each string
|
// Use an md5 hash of the strings for a count cache, as it's fast to generate, and collisions aren't a concern.
|
||||||
$chars1 = count_chars($string1);
|
$count_key1 = md5( $string1 );
|
||||||
$chars2 = count_chars($string2);
|
$count_key2 = md5( $string2 );
|
||||||
|
|
||||||
// L1-norm of difference vector.
|
// Cache vectors containing character frequency for all chars in each string.
|
||||||
$difference = array_sum( array_map( array($this, 'difference'), $chars1, $chars2 ) );
|
if ( ! isset( $this->count_cache[ $count_key1 ] ) ) {
|
||||||
|
$this->count_cache[ $count_key1 ] = count_chars( $string1 );
|
||||||
|
}
|
||||||
|
if ( ! isset( $this->count_cache[ $count_key2 ] ) ) {
|
||||||
|
$this->count_cache[ $count_key2 ] = count_chars( $string2 );
|
||||||
|
}
|
||||||
|
|
||||||
|
$chars1 = $this->count_cache[ $count_key1 ];
|
||||||
|
$chars2 = $this->count_cache[ $count_key2 ];
|
||||||
|
|
||||||
|
$difference_key = md5( implode( ',', $chars1 ) . ':' . implode( ',', $chars2 ) );
|
||||||
|
if ( ! isset( $this->difference_cache[ $difference_key ] ) ) {
|
||||||
|
// L1-norm of difference vector.
|
||||||
|
$this->difference_cache[ $difference_key ] = array_sum( array_map( array( $this, 'difference' ), $chars1, $chars2 ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
$difference = $this->difference_cache[ $difference_key ];
|
||||||
|
|
||||||
// $string1 has zero length? Odd. Give huge penalty by not dividing.
|
// $string1 has zero length? Odd. Give huge penalty by not dividing.
|
||||||
if ( !$string1 )
|
if ( !$string1 )
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
*
|
*
|
||||||
* @global string $wp_version
|
* @global string $wp_version
|
||||||
*/
|
*/
|
||||||
$wp_version = '5.0-alpha-43774';
|
$wp_version = '5.0-alpha-43775';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Holds the WordPress DB revision, increments when changes are made to the WordPress DB schema.
|
* Holds the WordPress DB revision, increments when changes are made to the WordPress DB schema.
|
||||||
|
|
Loading…
Reference in New Issue