$v) { /* check for any matches to user-added cookies to no-cache */ foreach ( $nocache_cookies as $nocache_cookie ) { if( strpos( $n, $nocache_cookie ) === 0 ) { __wp_ffpc_debug__ ( "Cookie exception matched: {$n}, skipping"); return false; } } } } } /* no cache for excluded URL patterns */ if ( isset($wp_ffpc_config['nocache_url']) && trim($wp_ffpc_config['nocache_url']) ) { $pattern = sprintf('#%s#', trim($wp_ffpc_config['nocache_url'])); if ( preg_match($pattern, $wp_ffpc_uri) ) { __wp_ffpc_debug__ ( "Cache exception based on URL regex pattern matched, skipping"); return false; } } /* canonical redirect storage */ $wp_ffpc_redirect = null; /* fires up the backend storage array with current config */ include_once ('wp-ffpc-backend.php'); $backend_class = 'WP_FFPC_Backend_' . $wp_ffpc_config['cache_type']; $wp_ffpc_backend = new $backend_class ( $wp_ffpc_config ); //$wp_ffpc_backend = new WP_FFPC_Backend( $wp_ffpc_config ); /* no cache for for logged in users unless it's set identifier cookies are listed in backend as var for easier usage */ if ( !isset($wp_ffpc_config['cache_loggedin']) || $wp_ffpc_config['cache_loggedin'] == 0 || empty($wp_ffpc_config['cache_loggedin']) ) { foreach ($_COOKIE as $n=>$v) { foreach ( $wp_ffpc_backend->cookies as $nocache_cookie ) { if( strpos( $n, $nocache_cookie ) === 0 ) { __wp_ffpc_debug__ ( "No cache for cookie: {$n}, skipping"); return false; } } } } /* will store time of page generation */ $wp_ffpc_gentime = 0; /* backend connection failed, no caching :( */ if ( $wp_ffpc_backend->status() === false ) { __wp_ffpc_debug__ ( "Backend offline"); return false; } /* try to get data & meta keys for current page */ /* include the mobile detect */ include_once ('backends/mobile-detect.php'); $mobile_detect = new Mobile_Detect; /* verify if mobile device (phones or tablets). */ $wp_ffpc_keys = ($mobile_detect->isMobile() ? array ( 'meta' => $wp_ffpc_config['prefix_meta_mobile'], 'data' => $wp_ffpc_config['prefix_data_mobile'] ) : array ( 'meta' => $wp_ffpc_config['prefix_meta'], 'data' => $wp_ffpc_config['prefix_data']); $wp_ffpc_values = array(); __wp_ffpc_debug__ ( "Trying to fetch entries"); foreach ( $wp_ffpc_keys as $internal => $key ) { $key = $wp_ffpc_backend->key ( $key ); $value = $wp_ffpc_backend->get ( $key ); if ( ! $value ) { __wp_ffpc_debug__("No cached data found"); /* does not matter which is missing, we need both, if one fails, no caching */ wp_ffpc_start(); return; } else { /* store results */ $wp_ffpc_values[ $internal ] = $value; __wp_ffpc_debug__('Got value for ' . $internal); } } /* serve cache 404 status */ if ( isset( $wp_ffpc_values['meta']['status'] ) && $wp_ffpc_values['meta']['status'] == 404 ) { __wp_ffpc_debug__("Serving 404"); header("HTTP/1.1 404 Not Found"); /* if I kill the page serving here, the 404 page will not be showed at all, so we do not do that * flush(); * die(); */ } /* server redirect cache */ if ( isset( $wp_ffpc_values['meta']['redirect'] ) && $wp_ffpc_values['meta']['redirect'] ) { __wp_ffpc_debug__("Serving redirect to {$wp_ffpc_values['meta']['redirect']}"); header('Location: ' . $wp_ffpc_values['meta']['redirect'] ); /* cut the connection as fast as possible */ flush(); die(); } /* page is already cached on client side (chrome likes to do this, anyway, it's quite efficient) */ if ( array_key_exists( "HTTP_IF_MODIFIED_SINCE" , $_SERVER ) && !empty( $wp_ffpc_values['meta']['lastmodified'] ) ) { $if_modified_since = strtotime(preg_replace('/;.*$/', '', $_SERVER["HTTP_IF_MODIFIED_SINCE"])); /* check is cache is still valid */ if ( $if_modified_since >= $wp_ffpc_values['meta']['lastmodified'] ) { __wp_ffpc_debug__("Serving 304 Not Modified"); header("HTTP/1.0 304 Not Modified"); /* connection cut for faster serving */ flush(); die(); } } /*** SERVING CACHED PAGE ***/ /* if we reach this point it means data was found & correct, serve it */ if (!empty ( $wp_ffpc_values['meta']['mime'] ) ) header('Content-Type: ' . $wp_ffpc_values['meta']['mime']); /* set expiry date */ if (isset($wp_ffpc_values['meta']['expire']) && !empty ( $wp_ffpc_values['meta']['expire'] ) ) { $hash = md5 ( $wp_ffpc_uri . $wp_ffpc_values['meta']['expire'] ); switch ($wp_ffpc_values['meta']['type']) { case 'home': case 'feed': $expire = $wp_ffpc_config['browsercache_home']; break; case 'archive': $expire = $wp_ffpc_config['browsercache_taxonomy']; break; case 'single': $expire = $wp_ffpc_config['browsercache']; break; default: $expire = 0; } header('Cache-Control: public,max-age='.$expire.',s-maxage='.$expire.',must-revalidate'); header('Expires: ' . gmdate("D, d M Y H:i:s", $wp_ffpc_values['meta']['expire'] ) . " GMT"); header('ETag: '. $hash); unset($expire, $hash); } else { /* in case there is no expiry set, expire immediately and don't serve Etag; browser cache is disabled */ header('Expires: ' . gmdate("D, d M Y H:i:s", time() ) . " GMT"); /* if I set these, the 304 not modified will never, ever kick in, so not setting these * leaving here as a reminder why it should not be set */ //header('Cache-Control: no-store, no-cache, must-revalidate, max-age=0, s-maxage=0, post-check=0, pre-check=0'); //header('Pragma: no-cache'); } /* if shortlinks were set */ if (isset($wp_ffpc_values['meta']['shortlink']) && !empty ( $wp_ffpc_values['meta']['shortlink'] ) ) header( 'Link:<'. $wp_ffpc_values['meta']['shortlink'] .'>; rel=shortlink' ); /* if last modifications were set (for posts & pages) */ if (isset($wp_ffpc_values['meta']['lastmodified']) && !empty($wp_ffpc_values['meta']['lastmodified']) ) header( 'Last-Modified: ' . gmdate("D, d M Y H:i:s", $wp_ffpc_values['meta']['lastmodified'] ). " GMT" ); /* pingback urls, if existx */ if ( isset($wp_ffpc_values['meta']['pingback']) && !empty( $wp_ffpc_values['meta']['pingback'] ) && isset($wp_ffpc_config['pingback_header']) && $wp_ffpc_config['pingback_header'] ) header( 'X-Pingback: ' . $wp_ffpc_values['meta']['pingback'] ); /* for debugging */ if ( isset($wp_ffpc_config['response_header']) && $wp_ffpc_config['response_header'] ) header( 'X-Cache-Engine: WP-FFPC with ' . $wp_ffpc_config['cache_type'] .' via PHP'); /* HTML data */ if ( isset($wp_ffpc_config['generate_time']) && $wp_ffpc_config['generate_time'] == '1' && stripos($wp_ffpc_values['data'], '') ) { $mtime = explode ( " ", microtime() ); $wp_ffpc_gentime = ( $mtime[1] + $mtime[0] ) - $wp_ffpc_gentime; $insertion = "\n\n"; $index = stripos( $wp_ffpc_values['data'] , '' ); $wp_ffpc_values['data'] = substr_replace( $wp_ffpc_values['data'], $insertion, $index, 0); } __wp_ffpc_debug__("Serving data"); echo trim($wp_ffpc_values['data']); flush(); die(); /*** END SERVING CACHED PAGE ***/ /*** GENERATING CACHE ENTRY ***/ /** * starts caching function * */ function wp_ffpc_start( ) { /* set start time */ global $wp_ffpc_gentime; $mtime = explode ( " ", microtime() ); $wp_ffpc_gentime = $mtime[1] + $mtime[0]; /* start object "colleting" and pass it the the actual storer function */ ob_start('wp_ffpc_callback'); } /** * callback function for WordPress redirect urls * */ function wp_ffpc_redirect_callback ($redirect_url, $requested_url) { global $wp_ffpc_redirect; $wp_ffpc_redirect = $redirect_url; return $redirect_url; } /** * write cache function, called when page generation ended */ function wp_ffpc_callback( $buffer ) { /* use global config */ global $wp_ffpc_config; /* backend was already set up, try to use it */ global $wp_ffpc_backend; /* check is it's a redirect */ global $wp_ffpc_redirect; /* check is it's a mobile version*/ global $mobile_detect; /* no is_home = error, WordPress functions are not availabe */ if (!function_exists('is_home')) return $buffer; /* no close tag = not HTML, also no , not feed, don't cache */ if ( stripos($buffer, '') === false && stripos($buffer, '') === false ) return $buffer; /* reset meta to solve conflicts */ $meta = array(); /* trim unneeded whitespace from beginning / ending of buffer */ $buffer = trim( $buffer ); /* Can be a trackback or other things without a body. We do not cache them, WP needs to get those calls. */ if (strlen($buffer) == 0) return ''; if ( isset($wp_ffpc_config[ 'nocache_comment' ]) && !empty($wp_ffpc_config[ 'nocache_comment' ]) && trim($wp_ffpc_config[ 'nocache_comment' ])) { $pattern = sprintf('#%s#', trim($wp_ffpc_config['nocache_comment'])); __wp_ffpc_debug__ ( sprintf("Testing comment with pattern: %s", $pattern)); if ( preg_match($pattern, $buffer) ) { __wp_ffpc_debug__ ( "Cache exception based on content regex pattern matched, skipping"); return $buffer; } } if ( is_home() || is_feed() ) { if (is_home()) $meta['type'] = 'home'; elseif(is_feed()) $meta['type'] = 'feed'; if (isset($wp_ffpc_config['browsercache_home']) && !empty($wp_ffpc_config['browsercache_home']) && $wp_ffpc_config['browsercache_home'] > 0) { $meta['expire'] = time() + $wp_ffpc_config['browsercache_home']; } __wp_ffpc_debug__( 'Getting latest post for for home & feed'); /* get newest post and set last modified accordingly */ $args = array( 'numberposts' => 1, 'orderby' => 'modified', 'order' => 'DESC', 'post_status' => 'publish', ); $recent_post = wp_get_recent_posts( $args, OBJECT ); if ( !empty($recent_post)) { $recent_post = array_pop($recent_post); if (!empty ( $recent_post->post_modified_gmt ) ) { $meta['lastmodified'] = strtotime ( $recent_post->post_modified_gmt ); } } } elseif ( is_archive() ) { $meta['type'] = 'archive'; if (isset($wp_ffpc_config['browsercache_taxonomy']) && !empty($wp_ffpc_config['browsercache_taxonomy']) && $wp_ffpc_config['browsercache_taxonomy'] > 0) { $meta['expire'] = time() + $wp_ffpc_config['browsercache_taxonomy']; } global $wp_query; if ( null != $wp_query->tax_query && !empty($wp_query->tax_query)) { __wp_ffpc_debug__( 'Getting latest post for taxonomy: ' . json_encode($wp_query->tax_query)); $args = array( 'numberposts' => 1, 'orderby' => 'modified', 'order' => 'DESC', 'post_status' => 'publish', 'tax_query' => $wp_query->tax_query, ); $recent_post = get_posts( $args, OBJECT ); if ( !empty($recent_post)) { $recent_post = array_pop($recent_post); if (!empty ( $recent_post->post_modified_gmt ) ) { $meta['lastmodified'] = strtotime ( $recent_post->post_modified_gmt ); } } } } elseif ( is_single() || is_page() ) { $meta['type'] = 'single'; if (isset($wp_ffpc_config['browsercache']) && !empty($wp_ffpc_config['browsercache']) && $wp_ffpc_config['browsercache'] > 0) { $meta['expire'] = time() + $wp_ffpc_config['browsercache']; } /* try if post is available if made with archieve, last listed post can make this go bad */ global $post; if ( !empty($post) && !empty ( $post->post_modified_gmt ) ) { /* get last modification data */ $meta['lastmodified'] = strtotime ( $post->post_modified_gmt ); /* get shortlink, if possible */ if (function_exists('wp_get_shortlink')) { $shortlink = wp_get_shortlink( ); if (!empty ( $shortlink ) ) $meta['shortlink'] = $shortlink; } } } else { $meta['type'] = 'unknown'; } if ( $meta['type'] != 'unknown' ) { /* check if caching is disabled for page type */ $nocache_key = 'nocache_'. $meta['type']; /* don't cache if prevented by rule */ if ( $wp_ffpc_config[ $nocache_key ] == 1 ) { return $buffer; } } if ( is_404() ) $meta['status'] = 404; /* redirect page */ if ( $wp_ffpc_redirect != null) $meta['redirect'] = $wp_ffpc_redirect; /* feed is xml, all others forced to be HTML */ if ( is_feed() ) $meta['mime'] = 'text/xml;charset='; else $meta['mime'] = 'text/html;charset='; /* set mimetype */ $meta['mime'] = $meta['mime'] . $wp_ffpc_config['charset']; /* store pingback url if pingbacks are enabled */ if ( get_option ( 'default_ping_status' ) == 'open' ) $meta['pingback'] = get_bloginfo('pingback_url'); $to_store = $buffer; /* add generation info is option is set, but only to HTML */ if ( $wp_ffpc_config['generate_time'] == '1' && stripos($buffer, '') ) { global $wp_ffpc_gentime; /* verify the device type to output into the generation stats */ $device_type = $mobile_detect -> isMobile() ? 'mobile': 'desktop'; $mtime = explode ( " ", microtime() ); $wp_ffpc_gentime = ( $mtime[1] + $mtime[0] )- $wp_ffpc_gentime; $insertion = "\n\n"; $index = stripos( $buffer , '' ); $to_store = substr_replace( $buffer, $insertion, $index, 0); } /** * Allows to edit the content to be stored in cache. * * This hooks allows the user to edit the page content right before it is about * to be stored in the cache. This could be useful for alterations like * minification. * * @since 1.10.2 * * @param string $to_store The content to be stored in cache. */ $to_store = apply_filters( 'wp-ffpc-to-store', $to_store ); $prefix_meta = ($mobile_detect -> isMobile())? $wp_ffpc_backend->key ( $wp_ffpc_config['prefix_meta_mobile'] ): $wp_ffpc_backend->key ( $wp_ffpc_config['prefix_meta'] ); $wp_ffpc_backend->set ( $prefix_meta, $meta ); $prefix_data = ($mobile_detect -> isMobile())? $wp_ffpc_backend->key ( $wp_ffpc_config['prefix_data_mobile'] ): $wp_ffpc_backend->key ( $wp_ffpc_config['prefix_data'] ); $wp_ffpc_backend->set ( $prefix_data , $to_store ); if ( !empty( $meta['status'] ) && $meta['status'] == 404 ) { header("HTTP/1.1 404 Not Found"); } else { /* vital for nginx, make no problem at other places */ header("HTTP/1.1 200 OK"); } /* echoes HTML out */ return trim($buffer); } /*** END GENERATING CACHE ENTRY ***/