1 | <?php |
---|
2 | |
---|
3 | /* |
---|
4 | Plugin Name: Advanced Post Caching |
---|
5 | Description: Cache post queries. |
---|
6 | Version: 0.2 |
---|
7 | Author: Automattic |
---|
8 | Author URI: http://automattic.com/ |
---|
9 | */ |
---|
10 | |
---|
11 | class Advanced_Post_Cache { |
---|
12 | var $CACHE_GROUP_PREFIX = 'advanced_post_cache_'; |
---|
13 | |
---|
14 | // Flag for temp (within one page load) turning invalidations on and off |
---|
15 | // @see dont_clear_advanced_post_cache() |
---|
16 | // @see do_clear_advanced_post_cache() |
---|
17 | // Used to prevent invalidation during new comment |
---|
18 | var $do_flush_cache = true; |
---|
19 | |
---|
20 | // Flag for preventing multiple invalidations in a row: clean_post_cache() calls itself recursively for post children. |
---|
21 | var $need_to_flush_cache = true; // Currently disabled |
---|
22 | |
---|
23 | /* Per cache-clear data */ |
---|
24 | var $cache_incr = 0; // Increments the cache group (advanced_post_cache_0, advanced_post_cache_1, ...) |
---|
25 | var $cache_group = ''; // CACHE_GROUP_PREFIX . $cache_incr |
---|
26 | |
---|
27 | /* Per query data */ |
---|
28 | var $cache_key = ''; // md5 of current SQL query |
---|
29 | var $all_post_ids = false; // IDs of all posts current SQL query returns |
---|
30 | var $cached_post_ids = array(); // subset of $all_post_ids whose posts are currently in cache |
---|
31 | var $cached_posts = array(); |
---|
32 | var $found_posts = false; // The result of the FOUND_ROWS() query |
---|
33 | var $cache_func = 'wp_cache_add'; // Turns to set if there seems to be inconsistencies |
---|
34 | |
---|
35 | function __construct() { |
---|
36 | wp_cache_add_group_prefix_map( $this->CACHE_GROUP_PREFIX, 'advanced_post_cache' ); |
---|
37 | |
---|
38 | $this->cache_incr = wp_cache_get( 'advanced_post_cache', 'cache_incrementors' ); // Get and construct current cache group name |
---|
39 | if ( !is_numeric( $this->cache_incr ) ) { |
---|
40 | $now = time(); |
---|
41 | wp_cache_set( 'advanced_post_cache', $now, 'cache_incrementors' ); |
---|
42 | $this->cache_incr = $now; |
---|
43 | } |
---|
44 | $this->cache_group = $this->CACHE_GROUP_PREFIX . $this->cache_incr; |
---|
45 | |
---|
46 | add_filter( 'posts_request', array( &$this, 'posts_request' ) ); // Short circuits if cached |
---|
47 | add_filter( 'posts_results', array( &$this, 'posts_results' ) ); // Collates if cached, primes cache if not |
---|
48 | |
---|
49 | add_filter( 'post_limits_request', array( &$this, 'post_limits_request' ), 999, 2 ); // Checks to see if we need to worry about found_posts |
---|
50 | |
---|
51 | add_filter( 'found_posts_query', array( &$this, 'found_posts_query' ) ); // Short circuits if cached |
---|
52 | add_filter( 'found_posts', array( &$this, 'found_posts' ) ); // Reads from cache if cached, primes cache if not |
---|
53 | } |
---|
54 | |
---|
55 | /* Advanced Post Cache API */ |
---|
56 | |
---|
57 | /** |
---|
58 | * Flushes the cache by incrementing the cache group |
---|
59 | */ |
---|
60 | function flush_cache() { |
---|
61 | // Cache flushes have been disabled |
---|
62 | if ( !$this->do_flush_cache ) |
---|
63 | return; |
---|
64 | |
---|
65 | // Bail on post preview |
---|
66 | if ( is_admin() && isset( $_POST['wp-preview'] ) && 'dopreview' == $_POST['wp-preview'] ) |
---|
67 | return; |
---|
68 | |
---|
69 | // Bail on autosave |
---|
70 | if ( defined( 'DOING_AUTOSAVE' ) && DOING_AUTOSAVE ) |
---|
71 | return; |
---|
72 | |
---|
73 | // We already flushed once this page load, and have not put anything into the cache since. |
---|
74 | // OTHER processes may have put something into the cache! In theory, this could cause stale caches. |
---|
75 | // We do this since clean_post_cache() (which fires the action this method attaches to) is called RECURSIVELY for all descendants. |
---|
76 | // if ( !$this->need_to_flush_cache ) |
---|
77 | // return; |
---|
78 | |
---|
79 | $this->cache_incr = wp_cache_incr( 'advanced_post_cache', 1, 'cache_incrementors' ); |
---|
80 | if ( 10 < strlen( $this->cache_incr ) ) { |
---|
81 | wp_cache_set( 'advanced_post_cache', 0, 'cache_incrementors' ); |
---|
82 | $this->cache_incr = 0; |
---|
83 | } |
---|
84 | $this->cache_group = $this->CACHE_GROUP_PREFIX . $this->cache_incr; |
---|
85 | $this->need_to_flush_cache = false; |
---|
86 | } |
---|
87 | |
---|
88 | |
---|
89 | /* Cache Reading/Priming Functions */ |
---|
90 | |
---|
91 | /** |
---|
92 | * Determines (by hash of SQL) if query is cached. |
---|
93 | * If cached: Return query of needed post IDs. |
---|
94 | * Otherwise: Returns query unchanged. |
---|
95 | */ |
---|
96 | function posts_request( $sql ) { |
---|
97 | global $wpdb; |
---|
98 | |
---|
99 | $this->cache_key = md5( $sql ); // init |
---|
100 | $this->all_post_ids = wp_cache_get( $this->cache_key, $this->cache_group ); |
---|
101 | if ( 'NA' !== $this->found_posts ) |
---|
102 | $this->found_posts = wp_cache_get( "{$this->cache_key}_found", $this->cache_group ); |
---|
103 | |
---|
104 | if ( $this->all_post_ids xor $this->found_posts ) |
---|
105 | $this->cache_func = 'wp_cache_set'; |
---|
106 | else |
---|
107 | $this->cache_func = 'wp_cache_add'; |
---|
108 | |
---|
109 | $this->cached_post_ids = array(); // re-init |
---|
110 | $this->cached_posts = array(); // re-init |
---|
111 | |
---|
112 | // Query is cached |
---|
113 | if ( $this->found_posts && is_array( $this->all_post_ids ) ) { |
---|
114 | $this->cached_posts = array_filter( wp_cache_get_multi( array( 'posts' => $this->all_post_ids ) ) ); |
---|
115 | foreach ( $this->cached_posts as $post ) { |
---|
116 | if ( !empty( $post ) ) |
---|
117 | $this->cached_post_ids[] = $post->ID; |
---|
118 | } |
---|
119 | $uncached_post_ids = array_diff( $this->all_post_ids, $this->cached_post_ids ); |
---|
120 | |
---|
121 | if ( $uncached_post_ids ) |
---|
122 | return "SELECT * FROM $wpdb->posts WHERE ID IN(" . join( ',', array_map( 'absint', $uncached_post_ids ) ) . ")"; |
---|
123 | return ''; |
---|
124 | } |
---|
125 | |
---|
126 | return $sql; |
---|
127 | } |
---|
128 | |
---|
129 | /** |
---|
130 | * If cached: Collates posts returned by SQL query with posts that are already cached. Orders correctly. |
---|
131 | * Otherwise: Primes cache with data for current posts WP_Query. |
---|
132 | */ |
---|
133 | function posts_results( $posts ) { |
---|
134 | if ( $this->found_posts && is_array( $this->all_post_ids ) ) { // is cached |
---|
135 | $collated_posts = array(); |
---|
136 | foreach ( $this->cached_posts as $post ) |
---|
137 | $posts[] = $post; |
---|
138 | |
---|
139 | foreach ( $posts as $post ) { |
---|
140 | $loc = array_search( $post->ID, $this->all_post_ids ); |
---|
141 | if ( is_numeric( $loc ) && -1 < $loc ) |
---|
142 | $collated_posts[$loc] = $post; |
---|
143 | } |
---|
144 | ksort( $collated_posts ); |
---|
145 | return array_values( $collated_posts ); |
---|
146 | } |
---|
147 | |
---|
148 | $post_ids = array(); |
---|
149 | foreach ( $posts as $post ) |
---|
150 | $post_ids[] = $post->ID; |
---|
151 | |
---|
152 | if ( !$post_ids ) |
---|
153 | return array(); |
---|
154 | |
---|
155 | call_user_func( $this->cache_func, $this->cache_key, $post_ids, $this->cache_group ); |
---|
156 | $this->need_to_flush_cache = true; |
---|
157 | |
---|
158 | return $posts; |
---|
159 | } |
---|
160 | |
---|
161 | /** |
---|
162 | * If $limits is empty, WP_Query never calls the found_rows stuff, so we set $this->found_rows to 'NA' |
---|
163 | */ |
---|
164 | function post_limits_request( $limits, &$query ) { |
---|
165 | if ( empty( $limits ) || ( isset( $query->query_vars['no_found_rows'] ) && $query->query_vars['no_found_rows'] ) ) |
---|
166 | $this->found_posts = 'NA'; |
---|
167 | else |
---|
168 | $this->found_posts = false; // re-init |
---|
169 | return $limits; |
---|
170 | } |
---|
171 | |
---|
172 | /** |
---|
173 | * If cached: Blanks SELECT FOUND_ROWS() query. This data is already stored in cache. |
---|
174 | * Otherwise: Returns query unchanged. |
---|
175 | */ |
---|
176 | function found_posts_query( $sql ) { |
---|
177 | if ( $this->found_posts && is_array( $this->all_post_ids ) ) // is cached |
---|
178 | return ''; |
---|
179 | return $sql; |
---|
180 | } |
---|
181 | |
---|
182 | /** |
---|
183 | * If cached: Returns cached result of FOUND_ROWS() query. |
---|
184 | * Otherwise: Returs result unchanged |
---|
185 | */ |
---|
186 | function found_posts( $found_posts ) { |
---|
187 | if ( $this->found_posts && is_array( $this->all_post_ids ) ) // is cached |
---|
188 | return (int) $this->found_posts; |
---|
189 | |
---|
190 | call_user_func( $this->cache_func, "{$this->cache_key}_found", (int) $found_posts, $this->cache_group ); |
---|
191 | $this->need_to_flush_cache = true; |
---|
192 | |
---|
193 | return $found_posts; |
---|
194 | } |
---|
195 | } |
---|
196 | |
---|
197 | global $advanced_post_cache_object; |
---|
198 | $advanced_post_cache_object = new Advanced_Post_Cache; |
---|
199 | |
---|
200 | function clear_advanced_post_cache() { |
---|
201 | global $advanced_post_cache_object; |
---|
202 | $advanced_post_cache_object->flush_cache(); |
---|
203 | } |
---|
204 | |
---|
205 | function do_clear_advanced_post_cache() { |
---|
206 | $GLOBALS['advanced_post_cache_object']->do_flush_cache = true; |
---|
207 | } |
---|
208 | |
---|
209 | function dont_clear_advanced_post_cache() { |
---|
210 | $GLOBALS['advanced_post_cache_object']->do_flush_cache = false; |
---|
211 | } |
---|
212 | |
---|
213 | add_action( 'clean_term_cache', 'clear_advanced_post_cache' ); |
---|
214 | add_action( 'clean_post_cache', 'clear_advanced_post_cache' ); |
---|
215 | |
---|
216 | // Don't clear Advanced Post Cache for a new comment - temp core hack |
---|
217 | // http://core.trac.wordpress.org/ticket/15565 |
---|
218 | add_action( 'wp_updating_comment_count', 'dont_clear_advanced_post_cache' ); |
---|
219 | add_action( 'wp_update_comment_count' , 'do_clear_advanced_post_cache' ); |
---|