관리-도구
편집 파일: Query.php
<?php namespace AIOSEO\Plugin\Common\Sitemap; // Exit if accessed directly. if ( ! defined( 'ABSPATH' ) ) { exit; } use AIOSEO\Plugin\Common\Utils as CommonUtils; /** * Handles all complex queries for the sitemap. * * @since 4.0.0 */ class Query { /** * Returns all eligble sitemap entries for a given post type. * * @since 4.0.0 * * @param mixed $postTypes The post type(s). Either a singular string or an array of strings. * @param array $additionalArgs Any additional arguments for the post query. * @return array|int The post objects or the post count. */ public function posts( $postTypes, $additionalArgs = [] ) { $includedPostTypes = $postTypes; $postTypesArray = ! is_array( $postTypes ) ? [ $postTypes ] : $postTypes; if ( is_array( $postTypes ) ) { $includedPostTypes = implode( "', '", $postTypes ); } if ( empty( $includedPostTypes ) || ( 'attachment' === $includedPostTypes && 'disabled' !== aioseo()->dynamicOptions->searchAppearance->postTypes->attachment->redirectAttachmentUrls ) ) { return []; } // Set defaults. $fields = '`p`.`ID`, `p`.`post_title`, `p`.`post_content`, `p`.`post_excerpt`, `p`.`post_type`, `p`.`post_password`, '; $fields .= '`p`.`post_parent`, `p`.`post_date_gmt`, `p`.`post_modified_gmt`, `ap`.`priority`, `ap`.`frequency`'; $maxAge = ''; if ( ! aioseo()->sitemap->helpers->excludeImages() ) { $fields .= ', `ap`.`images`'; } // Order by highest priority first (highest priority at the top), // then by post modified date (most recently updated at the top). $orderBy = '`ap`.`priority` DESC, `p`.`post_modified_gmt` DESC'; // Override defaults if passed as additional arg. foreach ( $additionalArgs as $name => $value ) { // Attachments need to be fetched with all their fields because we need to get their post parent further down the line. $$name = esc_sql( $value ); if ( 'root' === $name && $value && 'attachment' !== $includedPostTypes ) { $fields = '`p`.`ID`, `p`.`post_type`'; } if ( 'count' === $name && $value ) { $fields = 'count(`p`.`ID`) as total'; } } $query = aioseo()->core->db ->start( aioseo()->core->db->db->posts . ' as p', true ) ->select( $fields ) ->leftJoin( 'aioseo_posts as ap', '`ap`.`post_id` = `p`.`ID`' ) ->where( 'p.post_status', 'attachment' === $includedPostTypes ? 'inherit' : 'publish' ) ->whereRaw( "p.post_type IN ( '$includedPostTypes' )" ); $homePageId = (int) get_option( 'page_on_front' ); if ( ! is_array( $postTypes ) ) { if ( ! aioseo()->helpers->isPostTypeNoindexed( $includedPostTypes ) ) { $query->whereRaw( "( `ap`.`robots_noindex` IS NULL OR `ap`.`robots_default` = 1 OR `ap`.`robots_noindex` = 0 OR post_id = $homePageId )" ); } else { $query->whereRaw( "( `ap`.`robots_default` = 0 AND `ap`.`robots_noindex` = 0 OR post_id = $homePageId )" ); } } else { $robotsMetaSql = []; foreach ( $postTypes as $postType ) { if ( ! aioseo()->helpers->isPostTypeNoindexed( $postType ) ) { $robotsMetaSql[] = "( `p`.`post_type` = '$postType' AND ( `ap`.`robots_noindex` IS NULL OR `ap`.`robots_default` = 1 OR `ap`.`robots_noindex` = 0 OR post_id = $homePageId ) )"; } else { $robotsMetaSql[] = "( `p`.`post_type` = '$postType' AND ( `ap`.`robots_default` = 0 AND `ap`.`robots_noindex` = 0 OR post_id = $homePageId ) )"; } } $query->whereRaw( '( ' . implode( ' OR ', $robotsMetaSql ) . ' )' ); } $excludedPosts = aioseo()->sitemap->helpers->excludedPosts(); $isStaticHomepage = 'page' === get_option( 'show_on_front' ); if ( $isStaticHomepage ) { $excludedPostIds = explode( ',', $excludedPosts ); $blogPageId = (int) get_option( 'page_for_posts' ); if ( in_array( 'page', $postTypesArray, true ) ) { // Exclude the blog page from the pages post type. if ( $blogPageId ) { $query->whereRaw( "`p`.`ID` != $blogPageId" ); } // Custom order by statement to always move the home page to the top. if ( $homePageId ) { $orderBy = "case when `p`.`ID` = $homePageId then 0 else 1 end, $orderBy"; } } // Include the blog page in the posts post type unless manually excluded. if ( $blogPageId && ! in_array( $blogPageId, $excludedPostIds, true ) && in_array( 'post', $postTypesArray, true ) ) { // We are using a database class hack to get in an OR clause to // bypass all the other WHERE statements and just include the // blog page ID manually. $query->whereRaw( "1=1 OR `p`.`ID` = $blogPageId" ); // Custom order by statement to always move the blog posts page to the top. $orderBy = "case when `p`.`ID` = $blogPageId then 0 else 1 end, $orderBy"; } } if ( $excludedPosts ) { $query->whereRaw( "( `p`.`ID` NOT IN ( $excludedPosts ) OR post_id = $homePageId )" ); } // Exclude posts assigned to excluded terms. $excludedTerms = aioseo()->sitemap->helpers->excludedTerms(); if ( $excludedTerms ) { $termRelationshipsTable = aioseo()->core->db->db->prefix . 'term_relationships'; $query->whereRaw(" ( `p`.`ID` NOT IN ( SELECT `tr`.`object_id` FROM `$termRelationshipsTable` as tr WHERE `tr`.`term_taxonomy_id` IN ( $excludedTerms ) ) )" ); } if ( $maxAge ) { $query->whereRaw( "( `p`.`post_date_gmt` >= '$maxAge' )" ); } if ( 'rss' === aioseo()->sitemap->type || ( aioseo()->sitemap->indexes && empty( $additionalArgs['root'] ) && ( empty( $additionalArgs['count'] ) || ! $additionalArgs['count'] ) ) ) { $query->limit( aioseo()->sitemap->linksPerIndex, aioseo()->sitemap->offset ); } $query->orderBy( $orderBy ); $query = $this->filterPostQuery( $query, $postTypes ); // Return the total if we are just counting the posts. if ( ! empty( $additionalArgs['count'] ) && $additionalArgs['count'] ) { return (int) $query->run( true, 'var' ) ->result(); } $posts = $query->run() ->result(); // Convert ID from string to int. foreach ( $posts as $post ) { $post->ID = (int) $post->ID; } return $this->filterPosts( $posts ); } /** * Filters the post query. * * @since 4.1.4 * * @param \AIOSEO\Plugin\Common\Utils\Database $query The query. * @param string $postType The post type. * @return \AIOSEO\Plugin\Common\Utils\Database The filtered query. */ private function filterPostQuery( $query, $postType ) { switch ( $postType ) { case 'product': return $this->excludeHiddenProducts( $query ); default: break; } return $query; } /** * Adds a condition to the query to exclude hidden WooCommerce products. * * @since 4.1.4 * * @param \AIOSEO\Plugin\Common\Utils\Database $query The query. * @return \AIOSEO\Plugin\Common\Utils\Database The filtered query. */ private function excludeHiddenProducts( $query ) { if ( ! aioseo()->helpers->isWooCommerceActive() || ! apply_filters( 'aioseo_sitemap_woocommerce_exclude_hidden_products', true ) ) { return $query; } static $hiddenProductIds = null; if ( null === $hiddenProductIds ) { $tempDb = new CommonUtils\Database(); $hiddenProducts = $tempDb->start( 'term_relationships as tr' ) ->select( 'tr.object_id' ) ->join( 'term_taxonomy as tt', 'tr.term_taxonomy_id = tt.term_taxonomy_id' ) ->join( 'terms as t', 'tt.term_id = t.term_id' ) ->where( 't.name', 'exclude-from-catalog' ) ->run() ->result(); if ( empty( $hiddenProducts ) ) { return $query; } $hiddenProductIds = []; foreach ( $hiddenProducts as $hiddenProduct ) { $hiddenProductIds[] = (int) $hiddenProduct->object_id; } $hiddenProductIds = esc_sql( implode( ', ', $hiddenProductIds ) ); } $query->whereRaw( "p.ID NOT IN ( $hiddenProductIds )" ); return $query; } /** * Filters the queried posts. * * @since 4.0.0 * * @param array $posts The posts. * @return array $remainingPosts The remaining posts. */ public function filterPosts( $posts ) { $remainingPosts = []; foreach ( $posts as $post ) { switch ( $post->post_type ) { case 'attachment': if ( ! $this->isInvalidAttachment( $post ) ) { $remainingPosts[] = $post; } break; default: $remainingPosts[] = $post; break; } } return $remainingPosts; } /** * Excludes attachments if their post parent isn't published or parent post type isn't registered anymore. * * @since 4.0.0 * * @param Object $post The post. * @return boolean Whether the attachment is invalid. */ private function isInvalidAttachment( $post ) { if ( empty( $post->post_parent ) ) { return false; } $parent = get_post( $post->post_parent ); if ( ! is_object( $parent ) ) { return false; } if ( 'publish' !== $parent->post_status || ! in_array( $parent->post_type, get_post_types(), true ) || $parent->post_password ) { return true; } return false; } /** * Returns all eligible sitemap entries for a given taxonomy. * * @since 4.0.0 * * @param string $taxonomy The taxonomy. * @param array $additionalArgs Any additional arguments for the term query. * @return array|int The term objects or the term count. */ public function terms( $taxonomy, $additionalArgs = [] ) { // Set defaults. $fields = 't.term_id'; $offset = aioseo()->sitemap->offset; // Override defaults if passed as additional arg. foreach ( $additionalArgs as $name => $value ) { $$name = esc_sql( $value ); if ( 'root' === $name && $value ) { $fields = 't.term_id, tt.count'; } if ( 'count' === $name && $value ) { $fields = 'count(t.term_id) as total'; } } $termRelationshipsTable = aioseo()->core->db->db->prefix . 'term_relationships'; $termTaxonomyTable = aioseo()->core->db->db->prefix . 'term_taxonomy'; // Include all terms that have assigned posts or whose children have assigned posts. $query = aioseo()->core->db ->start( aioseo()->core->db->db->terms . ' as t', true ) ->select( $fields ) ->leftJoin( 'term_taxonomy as tt', '`tt`.`term_id` = `t`.`term_id`' ) ->whereRaw( " ( `t`.`term_id` IN ( SELECT `tt`.`term_id` FROM `$termTaxonomyTable` as tt WHERE `tt`.`taxonomy` = '$taxonomy' AND ( `tt`.`count` > 0 OR EXISTS ( SELECT 1 FROM `$termTaxonomyTable` as tt2 WHERE `tt2`.`parent` = `tt`.`term_id` AND `tt2`.`count` > 0 ) ) ) )" ); $excludedTerms = aioseo()->sitemap->helpers->excludedTerms(); if ( $excludedTerms ) { $query->whereRaw(" ( `t`.`term_id` NOT IN ( SELECT `tr`.`term_taxonomy_id` FROM `$termRelationshipsTable` as tr WHERE `tr`.`term_taxonomy_id` IN ( $excludedTerms ) ) )" ); } if ( aioseo()->sitemap->indexes && empty( $additionalArgs['root'] ) && ( empty( $additionalArgs['count'] ) || ! $additionalArgs['count'] ) ) { $query->limit( aioseo()->sitemap->linksPerIndex, $offset ); } // Return the total if we are just counting the terms. if ( ! empty( $additionalArgs['count'] ) && $additionalArgs['count'] ) { return (int) $query->run( true, 'var' ) ->result(); } $terms = $query->orderBy( '`t`.`term_id` ASC' ) ->run() ->result(); foreach ( $terms as $term ) { // Convert ID from string to int. $term->term_id = (int) $term->term_id; // Add taxonomy name to object manually instead of querying it to prevent redundant join. $term->taxonomy = $taxonomy; } return $terms; } /** * Wipes all data and forces the plugin to rescan the site for images. * * @since 4.0.13 * * @return void */ public function resetImages() { aioseo()->core->db ->update( 'aioseo_posts' ) ->set( [ 'images' => null, 'image_scan_date' => null ] ) ->run(); } }