WordPress.org

Make WordPress Core

Ticket #21760: 21760.10.diff

File 21760.10.diff, 9.2 KB (added by tollmanz, 6 years ago)
  • src/wp-includes/functions.php

     
    35023502 * Determine if the scheme of the given URL is https.
    35033503 *
    35043504 * @since 4.0.0
    3505  * 
     3505 *
    35063506 * @param  string  $url The URL
    35073507 * @return boolean      True if the given URL uses https, false if not (or if the URL is not valid).
    35083508 */
     
    45974597
    45984598        return (bool) $var;
    45994599}
     4600
     4601/**
     4602 * Helper function to retrieve an incrementer identified by $group
     4603 *
     4604 * @since 4.0.0
     4605 *
     4606 * @param string $group The cache group for the incrementer.
     4607 * @param bool $force Whether or not to generate a new incrementor.
     4608 * @return int The timestamp representing 'last_changed'.
     4609 */
     4610function wp_get_last_changed( $group, $force = false ) {
     4611        $last_changed = wp_cache_get( 'last_changed', $group );
     4612        if ( ! $last_changed || true === $force ) {
     4613                $last_changed = microtime();
     4614                wp_cache_set( 'last_changed', $last_changed, $group );
     4615        }
     4616        return $last_changed;
     4617}
     4618 No newline at end of file
  • src/wp-includes/taxonomy.php

     
    962962                return $error;
    963963        }
    964964
     965        $group = $taxonomy . ':' . wp_get_last_changed( 'terms' );
    965966        if ( is_object($term) && empty($term->filter) ) {
    966                 wp_cache_add($term->term_id, $term, $taxonomy);
     967                wp_cache_add( $term->term_id, $term, $taxonomy );
     968                wp_cache_add( "slug:{$term->slug}", $term->term_id, $group );
     969                wp_cache_add( "name:" . md5( $term->name ), $term->term_id, $group );
    967970                $_term = $term;
    968971        } else {
    969972                if ( is_object($term) )
     
    974977                        $_term = $wpdb->get_row( $wpdb->prepare( "SELECT t.*, tt.* FROM $wpdb->terms AS t INNER JOIN $wpdb->term_taxonomy AS tt ON t.term_id = tt.term_id WHERE tt.taxonomy = %s AND t.term_id = %d LIMIT 1", $taxonomy, $term) );
    975978                        if ( ! $_term )
    976979                                return null;
    977                         wp_cache_add($term, $_term, $taxonomy);
     980                        wp_cache_add( $term, $_term, $taxonomy );
     981                        wp_cache_add( "slug:{$_term->slug}", $term, $group );
     982                        wp_cache_add( "name:" . md5( $_term->name ), $term, $group );
    978983                }
    979984        }
    980985
     
    10461051        if ( ! taxonomy_exists($taxonomy) )
    10471052                return false;
    10481053
     1054        $cache = false;
     1055        $group = $taxonomy . ':' . wp_get_last_changed( 'terms' );
    10491056        if ( 'slug' == $field ) {
    10501057                $field = 't.slug';
    10511058                $value = sanitize_title($value);
    10521059                if ( empty($value) )
    10531060                        return false;
     1061
     1062                $term_id = wp_cache_get( "slug:{$value}", $group );
     1063                if ( $term_id ) {
     1064                        $value = $term_id;
     1065                        $cache = true;
     1066                }
    10541067        } else if ( 'name' == $field ) {
    10551068                // Assume already escaped
    10561069                $value = wp_unslash($value);
    10571070                $field = 't.name';
     1071                $term_id = wp_cache_get( "name:" . md5( $value ), $group );
     1072                if ( $term_id ) {
     1073                        $value = $term_id;
     1074                        $cache = true;
     1075                }
    10581076        } else if ( 'term_taxonomy_id' == $field ) {
    10591077                $value = (int) $value;
    10601078                $field = 'tt.term_taxonomy_id';
    10611079        } else {
     1080                $cache = true;
     1081        }
     1082
     1083        if ( $cache ) {
    10621084                $term = get_term( (int) $value, $taxonomy, $output, $filter);
    10631085                if ( is_wp_error( $term ) )
    10641086                        $term = false;
     
    10691091        if ( !$term )
    10701092                return false;
    10711093
    1072         wp_cache_add($term->term_id, $term, $taxonomy);
    1073 
    10741094        /** This filter is documented in wp-includes/taxonomy.php */
    10751095        $term = apply_filters( 'get_term', $term, $taxonomy );
    10761096
     
    10791099
    10801100        $term = sanitize_term($term, $taxonomy, $filter);
    10811101
     1102        wp_cache_add( $term->term_id, $term, $taxonomy );
     1103        wp_cache_add( "slug:{$term->slug}", $term->term_id, $group );
     1104        wp_cache_add( "name:" . md5( $term->name ), $term->term_id, $group );
     1105
    10821106        if ( $output == OBJECT ) {
    10831107                return $term;
    10841108        } elseif ( $output == ARRAY_A ) {
     
    32603284 * @param bool $clean_taxonomy Whether to clean taxonomy wide caches (true), or just individual term object caches (false). Default is true.
    32613285 */
    32623286function clean_term_cache($ids, $taxonomy = '', $clean_taxonomy = true) {
    3263         global $wpdb;
     3287        global $_wp_suspend_cache_invalidation, $wpdb;
    32643288
     3289        if ( ! empty( $_wp_suspend_cache_invalidation ) ) {
     3290                return;
     3291        }
     3292
    32653293        if ( !is_array($ids) )
    32663294                $ids = array($ids);
    32673295
     
    33073335                do_action( 'clean_term_cache', $ids, $taxonomy );
    33083336        }
    33093337
    3310         wp_cache_set( 'last_changed', microtime(), 'terms' );
     3338        wp_get_last_changed( 'terms', true );
    33113339}
    33123340
    33133341/**
     
    34073435                if ( empty($term_taxonomy) )
    34083436                        $term_taxonomy = $term->taxonomy;
    34093437
    3410                 wp_cache_add($term->term_id, $term, $term_taxonomy);
     3438                wp_cache_add( $term->term_id, $term, $term_taxonomy );
     3439                $group = $term_taxonomy . ':' . wp_get_last_changed( 'terms', true );
     3440                wp_cache_add( "slug:{$term->slug}", $term->term_id, $group );
     3441                wp_cache_add( "name:" . md5( $term->name ), $term->term_id, $group );
    34113442        }
    34123443}
    34133444
  • tests/phpunit/tests/term/cache.php

     
    9393
    9494                _unregister_taxonomy( $tax );
    9595        }
     96
     97        /**
     98         * @ticket 21760
     99         */
     100        function test_get_term_by_slug_cache() {
     101                global $wpdb;
     102                $term_id = $this->factory->term->create( array( 'slug' => 'burrito', 'taxonomy' => 'post_tag' ) );
     103
     104                $queries = $wpdb->num_queries;
     105                get_term_by( 'slug', 'burrito', 'post_tag' );
     106                $initial = $queries + 1;
     107                $this->assertEquals( $initial, $wpdb->num_queries );
     108                $term = get_term_by( 'slug', 'burrito', 'post_tag' );
     109                $this->assertEquals( $initial, $wpdb->num_queries );
     110
     111                $this->assertEquals( $term, wp_cache_get( $term_id, 'post_tag' ) );
     112
     113                $this->assertEquals( get_term( $term_id, 'post_tag' ), $term );
     114                $this->assertEquals( $initial, $wpdb->num_queries );
     115        }
     116
     117        /**
     118         * @ticket 21760
     119         */
     120        function test_get_term_by_slug_cache_update() {
     121                global $wpdb;
     122                $term_id = $this->factory->term->create( array( 'slug' => 'burrito', 'taxonomy' => 'post_tag' ) );
     123
     124                $queries = $wpdb->num_queries;
     125                get_term_by( 'slug', 'burrito', 'post_tag' );
     126                $initial = $queries + 1;
     127                $this->assertEquals( $initial, $wpdb->num_queries );
     128                $term = get_term_by( 'slug', 'burrito', 'post_tag' );
     129                $this->assertEquals( $initial, $wpdb->num_queries );
     130
     131                $this->assertEquals( $term, wp_cache_get( $term_id, 'post_tag' ) );
     132
     133                wp_update_term( $term_id, 'post_tag', array( 'name' => 'Taco' ) );
     134                $this->assertNotEquals( $term, get_term( $term_id, 'post_tag' ) );
     135                $after_queries = $wpdb->num_queries;
     136                get_term_by( 'slug', 'burrito', 'post_tag' );
     137                $this->assertEquals( $after_queries, $wpdb->num_queries );
     138        }
     139
     140        /**
     141         * @ticket 21760
     142         */
     143        function test_get_term_by_name_cache() {
     144                global $wpdb;
     145                $term_id = $this->factory->term->create( array( 'name' => 'burrito', 'taxonomy' => 'post_tag' ) );
     146
     147                $queries = $wpdb->num_queries;
     148                get_term_by( 'name', 'burrito', 'post_tag' );
     149                $initial = $queries + 1;
     150                $this->assertEquals( $initial, $wpdb->num_queries );
     151                $term = get_term_by( 'name', 'burrito', 'post_tag' );
     152                $this->assertEquals( $initial, $wpdb->num_queries );
     153
     154                $this->assertEquals( get_term( $term_id, 'post_tag' ), $term );
     155                $this->assertEquals( $initial, $wpdb->num_queries );
     156        }
     157
     158        /**
     159         * @ticket 21760
     160         */
     161        function test_get_term_by_name_cache_update() {
     162                global $wpdb;
     163                $term_id = $this->factory->term->create( array( 'name' => 'burrito', 'taxonomy' => 'post_tag' ) );
     164
     165                $queries = $wpdb->num_queries;
     166                get_term_by( 'name', 'burrito', 'post_tag' );
     167                $initial = $queries + 1;
     168                $this->assertEquals( $initial, $wpdb->num_queries );
     169                $term = get_term_by( 'name', 'burrito', 'post_tag' );
     170                $this->assertEquals( $initial, $wpdb->num_queries );
     171
     172                wp_update_term( $term_id, 'post_tag', array( 'slug' => 'Taco' ) );
     173                $this->assertNotEquals( $term, get_term( $term_id, 'post_tag' ) );
     174                $after_queries = $wpdb->num_queries;
     175                get_term_by( 'name', 'burrito', 'post_tag' );
     176                $this->assertEquals( $after_queries, $wpdb->num_queries );
     177        }
     178
     179        /**
     180         * @ticket 21760
     181         */
     182        function test_invalidating_term_caches_should_fail_when_invalidation_is_suspended() {
     183                $slug = 'taco';
     184                $name = 'Taco';
     185                $taxonomy = 'post_tag';
     186                $cache_key_slug = 'slug:' . $slug;
     187                $cache_key_name = 'name:' . md5( $name );
     188
     189                $term_id = $this->factory->term->create( array( 'slug' => $slug, 'name' => $name, 'taxonomy' => $taxonomy ) );
     190
     191                $last_changed = wp_get_last_changed( 'terms' );
     192                $group = $taxonomy . ':' . $last_changed;
     193
     194                $term = get_term_by( 'slug', $slug, $taxonomy );
     195
     196                // Verify the term is cached by ID, slug and name
     197                $this->assertEquals( $term, wp_cache_get( $term_id, $taxonomy ) );
     198                $this->assertEquals( $term_id, wp_cache_get( $cache_key_slug, $group ) );
     199                $this->assertEquals( $term_id, wp_cache_get( $cache_key_name, $group ) );
     200
     201                wp_suspend_cache_invalidation();
     202                clean_term_cache( $term_id, $taxonomy );
     203
     204                // Verify that the cached value still matches the correct value
     205                $this->assertEquals( $term, wp_cache_get( $term_id, $taxonomy ) );
     206                $this->assertEquals( $term_id, wp_cache_get( $cache_key_slug, $group ) );
     207                $this->assertEquals( $term_id, wp_cache_get( $cache_key_name, $group ) );
     208
     209                // Verify that last changed has not been updated as part of an invalidation routine
     210                $this->assertEquals( $last_changed, wp_get_last_changed( 'terms' ) );
     211        }
    96212}
     213 No newline at end of file