options_helper = $options_helper; } /** * Returns the conditionals based on which this loadable should be active. * * @return array The conditionals. */ public static function get_conditionals() { return [ Front_End_Conditional::class ]; } /** * Register our RSS related hooks. * * @return void */ public function register_hooks() { if ( $this->is_true( 'remove_feed_global' ) ) { \add_filter( 'feed_links_show_posts_feed', '__return_false' ); } if ( $this->is_true( 'remove_feed_global_comments' ) ) { \add_filter( 'feed_links_show_comments_feed', '__return_false' ); } \add_action( 'wp', [ $this, 'maybe_disable_feeds' ] ); \add_action( 'wp', [ $this, 'maybe_redirect_feeds' ], -10000 ); } /** * Disable feeds on selected cases. * * @return void */ public function maybe_disable_feeds() { if ( $this->is_true( 'remove_feed_post_comments' ) ) { \add_filter( 'feed_links_extra_show_post_comments_feed', '__return_false' ); } if ( $this->is_true( 'remove_feed_authors' ) ) { \add_filter( 'feed_links_extra_show_author_feed', '__return_false' ); } if ( $this->is_true( 'remove_feed_categories' ) ) { \add_filter( 'feed_links_extra_show_category_feed', '__return_false' ); } if ( $this->is_true( 'remove_feed_tags' ) ) { \add_filter( 'feed_links_extra_show_tag_feed', '__return_false' ); } if ( $this->is_true( 'remove_feed_custom_taxonomies' ) ) { \add_filter( 'feed_links_extra_show_tax_feed', '__return_false' ); } if ( $this->is_true( 'remove_feed_post_types' ) ) { \add_filter( 'feed_links_extra_show_post_type_archive_feed', '__return_false' ); } if ( $this->is_true( 'remove_feed_search' ) ) { \add_filter( 'feed_links_extra_show_search_feed', '__return_false' ); } } /** * Redirect feeds we don't want away. * * @return void */ public function maybe_redirect_feeds() { global $wp_query; if ( ! \is_feed() ) { return; } if ( \in_array( \get_query_var( 'feed' ), [ 'atom', 'rdf' ], true ) && $this->is_true( 'remove_atom_rdf_feeds' ) ) { $this->redirect_feed( \home_url(), 'We disable Atom/RDF feeds for performance reasons.' ); } // Only if we're on the global feed, the query is _just_ `'feed' => 'feed'`, hence this check. if ( ( $wp_query->query === [ 'feed' => 'feed' ] || $wp_query->query === [ 'feed' => 'atom' ] || $wp_query->query === [ 'feed' => 'rdf' ] ) && $this->is_true( 'remove_feed_global' ) ) { $this->redirect_feed( \home_url(), 'We disable the RSS feed for performance reasons.' ); } if ( \is_comment_feed() && ! ( \is_singular() || \is_attachment() ) && $this->is_true( 'remove_feed_global_comments' ) ) { $this->redirect_feed( \home_url(), 'We disable comment feeds for performance reasons.' ); } elseif ( \is_comment_feed() && \is_singular() && ( $this->is_true( 'remove_feed_post_comments' ) || $this->is_true( 'remove_feed_global_comments' ) ) ) { $url = \get_permalink( \get_queried_object() ); $this->redirect_feed( $url, 'We disable post comment feeds for performance reasons.' ); } if ( \is_author() && $this->is_true( 'remove_feed_authors' ) ) { $author_id = (int) \get_query_var( 'author' ); $url = \get_author_posts_url( $author_id ); $this->redirect_feed( $url, 'We disable author feeds for performance reasons.' ); } if ( ( \is_category() && $this->is_true( 'remove_feed_categories' ) ) || ( \is_tag() && $this->is_true( 'remove_feed_tags' ) ) || ( \is_tax() && $this->is_true( 'remove_feed_custom_taxonomies' ) ) ) { $term = \get_queried_object(); $url = \get_term_link( $term, $term->taxonomy ); if ( \is_wp_error( $url ) ) { $url = \home_url(); } $this->redirect_feed( $url, 'We disable taxonomy feeds for performance reasons.' ); } if ( ( \is_post_type_archive() ) && $this->is_true( 'remove_feed_post_types' ) ) { $url = \get_post_type_archive_link( $this->get_queried_post_type() ); $this->redirect_feed( $url, 'We disable post type feeds for performance reasons.' ); } if ( \is_search() && $this->is_true( 'remove_feed_search' ) ) { $url = \trailingslashit( \home_url() ) . '?s=' . \get_search_query(); $this->redirect_feed( $url, 'We disable search RSS feeds for performance reasons.' ); } } /** * Sends a cache control header. * * @param int $expiration The expiration time. * * @return void */ public function cache_control_header( $expiration ) { \header_remove( 'Expires' ); // The cacheability of the current request. 'public' allows caching, 'private' would not allow caching by proxies like CloudFlare. $cacheability = 'public'; $format = '%1$s, max-age=%2$d, s-maxage=%2$d, stale-while-revalidate=120, stale-if-error=14400'; if ( \is_user_logged_in() ) { $expiration = 0; $cacheability = 'private'; $format = '%1$s, max-age=%2$d'; } \header( \sprintf( 'Cache-Control: ' . $format, $cacheability, $expiration ), true ); } /** * Redirect a feed result to somewhere else. * * @param string $url The location we're redirecting to. * @param string $reason The reason we're redirecting. * * @return void */ private function redirect_feed( $url, $reason ) { \header_remove( 'Content-Type' ); \header_remove( 'Last-Modified' ); $this->cache_control_header( 7 * \DAY_IN_SECONDS ); \wp_safe_redirect( $url, 301, 'Yoast SEO: ' . $reason ); exit; } /** * Retrieves the queried post type. * * @return string The queried post type. */ private function get_queried_post_type() { $post_type = \get_query_var( 'post_type' ); if ( \is_array( $post_type ) ) { $post_type = \reset( $post_type ); } return $post_type; } /** * Checks if the value of an option is set to true. * * @param string $option_name The option name. * * @return bool */ private function is_true( $option_name ) { return $this->options_helper->get( $option_name ) === true; } }