ToIso8601( gmdate( 'Y-m-d H:i:s', $timestamp ) ) : false; } return aioseo()->helpers->dateTimeToIso8601( gmdate( 'Y-m-d H:i:s', max( $pages ) ) ); } /** * Formats a given image URL for usage in the sitemap. * * @since 4.0.0 * * @param string $url The URL. * @return string The formatted URL. */ public function formatUrl( $url ) { // Remove URL parameters. $url = strtok( $url, '?' ); $url = htmlspecialchars( $url, ENT_COMPAT, 'UTF-8', false ); return aioseo()->helpers->makeUrlAbsolute( $url ); } /** * Logs the performance of the sitemap for debugging purposes. * * @since 4.0.0 * * @return void */ public function logPerformance() { // Start logging the performance. if ( ! $this->performance ) { $this->performance['time'] = microtime( true ); $this->performance['memory'] = ( memory_get_peak_usage( true ) / 1024 ) / 1024; return; } // Stop logging the performance. $time = microtime( true ) - $this->performance['time']; $memory = $this->performance['memory']; $type = aioseo()->sitemap->type; $indexName = aioseo()->sitemap->indexName; error_log( wp_json_encode( "$indexName index of $type sitemap generated in $time seconds using a maximum of $memory mb of memory." ) ); } /** * Returns the post types that should be included in the sitemap. * * @since 4.0.0 * * @param boolean $hasArchivesOnly Whether or not to only include post types which have archives. * @return array $postTypes The included post types. */ public function includedPostTypes( $hasArchivesOnly = false ) { $postTypes = []; if ( aioseo()->options->sitemap->{aioseo()->sitemap->type}->postTypes->all ) { $postTypes = aioseo()->helpers->getPublicPostTypes( true, $hasArchivesOnly ); } else { $postTypes = aioseo()->options->sitemap->{aioseo()->sitemap->type}->postTypes->included; } if ( ! $postTypes ) { return $postTypes; } $options = aioseo()->options->noConflict(); $dynamicOptions = aioseo()->dynamicOptions->noConflict(); $publicPostTypes = aioseo()->helpers->getPublicPostTypes( true, $hasArchivesOnly ); foreach ( $postTypes as $postType ) { // Check if post type is no longer registered. if ( ! in_array( $postType, $publicPostTypes, true ) || ! $dynamicOptions->searchAppearance->postTypes->has( $postType ) ) { $postTypes = aioseo()->helpers->unsetValue( $postTypes, $postType ); continue; } // Check if post type isn't noindexed. if ( aioseo()->helpers->isPostTypeNoindexed( $postType ) ) { if ( ! $this->checkForIndexedPost( $postType ) ) { $postTypes = aioseo()->helpers->unsetValue( $postTypes, $postType ); continue; } } if ( $dynamicOptions->searchAppearance->postTypes->$postType->advanced->robotsMeta->default && ! $options->searchAppearance->advanced->globalRobotsMeta->default && $options->searchAppearance->advanced->globalRobotsMeta->noindex ) { if ( ! $this->checkForIndexedPost( $postType ) ) { $postTypes = aioseo()->helpers->unsetValue( $postTypes, $postType ); continue; } } } return $postTypes; } /** * Checks if any post is explicitly indexed when the post type is noindexed. * * @since 4.0.0 * * @param string $postType The post type to check for. * @return bool Whether or not there is an indexed post. */ private function checkForIndexedPost( $postType ) { $db = aioseo()->core->db->noConflict(); $posts = $db->start( aioseo()->core->db->db->posts . ' as p', true ) ->select( 'p.ID' ) ->join( 'aioseo_posts as ap', '`ap`.`post_id` = `p`.`ID`' ) ->where( 'p.post_status', 'attachment' === $postType ? 'inherit' : 'publish' ) ->where( 'p.post_type', $postType ) ->whereRaw( '( `ap`.`robots_default` = 0 AND `ap`.`robots_noindex` = 0 )' ) ->limit( 1 ) ->run() ->result(); if ( $posts && count( $posts ) ) { return true; } return false; } /** * Returns the taxonomies that should be included in the sitemap. * * @since 4.0.0 * * @return array The included taxonomies. */ public function includedTaxonomies() { $taxonomies = []; if ( aioseo()->options->sitemap->{aioseo()->sitemap->type}->taxonomies->all ) { $taxonomies = get_taxonomies(); } else { $taxonomies = aioseo()->options->sitemap->{aioseo()->sitemap->type}->taxonomies->included; } if ( ! $taxonomies ) { return []; } $options = aioseo()->options->noConflict(); $dynamicOptions = aioseo()->dynamicOptions->noConflict(); $publicTaxonomies = aioseo()->helpers->getPublicTaxonomies( true ); foreach ( $taxonomies as $taxonomy ) { // Check if taxonomy is no longer registered. if ( ! in_array( $taxonomy, $publicTaxonomies, true ) || ! $dynamicOptions->searchAppearance->taxonomies->has( $taxonomy ) ) { $taxonomies = aioseo()->helpers->unsetValue( $taxonomies, $taxonomy ); continue; } // Check if taxonomy isn't noindexed. if ( aioseo()->helpers->isTaxonomyNoindexed( $taxonomy ) ) { $taxonomies = aioseo()->helpers->unsetValue( $taxonomies, $taxonomy ); continue; } if ( $dynamicOptions->searchAppearance->taxonomies->$taxonomy->advanced->robotsMeta->default && ! $options->searchAppearance->advanced->globalRobotsMeta->default && $options->searchAppearance->advanced->globalRobotsMeta->noindex ) { $taxonomies = aioseo()->helpers->unsetValue( $taxonomies, $taxonomy ); continue; } } return $taxonomies; } /** * Splits sitemap entries into chuncks based on the max. amount of URLs per index. * * @since 4.0.0 * * @param array $entries The sitemap entries. * @return array The chunked sitemap entries. */ public function chunkEntries( $entries ) { return array_chunk( $entries, aioseo()->sitemap->linksPerIndex, true ); } /** * Formats the last Modified date of a user-submitted additional page as an ISO 8601 date. * * @since 4.0.0 * * @param object $page The additional page object. * @return string The formatted datetime. */ public function lastModifiedAdditionalPage( $page ) { return gmdate( 'c', strtotime( $page->lastModified ) ); } /** * Returns a list of excluded post IDs. * * @since 4.0.0 * * @return string The excluded IDs. */ public function excludedPosts() { return $this->excludedObjectIds( 'excludePosts' ); } /** * Returns a list of excluded term IDs. * * @since 4.0.0 * * @return string The excluded IDs. */ public function excludedTerms() { return $this->excludedObjectIds( 'excludeTerms' ); } /** * Returns a list of excluded IDs for a given option as a comma separated string. * * Helper method for excludedPosts() and excludedTerms(). * * @since 4.0.0 * @version 4.4.7 Improved method name. * * @param string $option The option name. * @return string The excluded IDs. */ private function excludedObjectIds( $option ) { $type = aioseo()->sitemap->type; // The RSS Sitemap needs to exclude whatever is excluded in the general sitemap. if ( 'rss' === $type ) { $type = 'general'; } // Allow WPML to filter out hidden language posts/terms. $hiddenObjectIds = []; if ( aioseo()->helpers->isWpmlActive() ) { $hiddenLanguages = apply_filters( 'wpml_setting', [], 'hidden_languages' ); foreach ( $hiddenLanguages as $language ) { $objectTypes = []; if ( 'excludePosts' === $option ) { $objectTypes = aioseo()->sitemap->helpers->includedPostTypes(); $objectTypes = array_map( function( $postType ) { return "post_{$postType}"; }, $objectTypes ); } if ( 'excludeTerms' === $option ) { $objectTypes = aioseo()->sitemap->helpers->includedTaxonomies(); $objectTypes = array_map( function( $taxonomy ) { return "tax_{$taxonomy}"; }, $objectTypes ); } $dbNoConflict = aioseo()->core->db->noConflict(); $rows = $dbNoConflict->start( 'icl_translations' ) ->select( 'element_id' ) ->whereIn( 'element_type', $objectTypes ) ->where( 'language_code', $language ) ->run() ->result(); $ids = array_map( function( $row ) { return (int) $row->element_id; }, $rows ); $hiddenObjectIds = array_merge( $hiddenObjectIds, $ids ); } } $hasFilter = has_filter( 'aioseo_sitemap_' . aioseo()->helpers->toSnakeCase( $option ) ); $advanced = aioseo()->options->sitemap->$type->advancedSettings->enable; $excluded = array_merge( $hiddenObjectIds, aioseo()->options->sitemap->{$type}->advancedSettings->{$option} ); if ( ! $advanced && empty( $excluded ) && ! $hasFilter ) { return ''; } $ids = []; foreach ( $excluded as $object ) { if ( is_numeric( $object ) ) { $ids[] = (int) $object; continue; } $object = json_decode( $object ); if ( is_int( $object->value ) ) { $ids[] = $object->value; } } if ( 'excludePosts' === $option ) { $ids = apply_filters( 'aioseo_sitemap_exclude_posts', $ids, $type ); } if ( 'excludeTerms' === $option ) { $ids = apply_filters( 'aioseo_sitemap_exclude_terms', $ids, $type ); } return count( $ids ) ? esc_sql( implode( ', ', $ids ) ) : ''; } /** * Returns the URLs of all active sitemaps. * * @since 4.0.0 * @version 4.6.2 Removed the prefix from the list of URLs. * * @return array $urls The sitemap URLs. */ public function getSitemapUrls() { static $urls = []; if ( $urls ) { return $urls; } $addonsUrls = array_filter( aioseo()->addons->doAddonFunction( 'helpers', 'getSitemapUrls' ) ); foreach ( $addonsUrls as $addonUrls ) { $urls = array_merge( $urls, $addonUrls ); } if ( aioseo()->options->sitemap->general->enable ) { $urls[] = $this->getUrl( 'general' ); } if ( aioseo()->options->sitemap->rss->enable ) { $urls[] = $this->getUrl( 'rss' ); } return $urls; } /** * Returns the URLs of all active sitemaps with the 'Sitemap: ' prefix. * * @since 4.6.2 * * @return array $urls The sitemap URLs. */ public function getSitemapUrlsPrefixed() { $urls = $this->getSitemapUrls(); foreach ( $urls as &$url ) { $url = 'Sitemap: ' . $url; } return $urls; } /** * Extracts existing sitemap URLs from the robots.txt file. * We need this in case users have existing sitemap directives added to their robots.txt file. * * @since 4.0.10 * @version 4.4.9 * * @return array The sitemap URLs. */ public function extractSitemapUrlsFromRobotsTxt() { // First, we need to remove our filter, so that it doesn't run unintentionally. remove_filter( 'robots_txt', [ aioseo()->robotsTxt, 'buildRules' ], 10000 ); $robotsTxt = apply_filters( 'robots_txt', '', true ); add_filter( 'robots_txt', [ aioseo()->robotsTxt, 'buildRules' ], 10000 ); if ( ! $robotsTxt ) { return []; } $lines = explode( "\n", $robotsTxt ); if ( ! is_array( $lines ) || ! count( $lines ) ) { return []; } return aioseo()->robotsTxt->extractSitemapUrls( $robotsTxt ); } /** * Returns the URL of the given sitemap type. * * @since 4.1.5 * * @param string $type The sitemap type. * @return string The sitemap URL. */ public function getUrl( $type ) { $url = home_url( 'sitemap.xml' ); if ( 'rss' === $type ) { $url = home_url( 'sitemap.rss' ); } if ( 'general' === $type ) { // Check if user has a custom filename from the V3 migration. $filename = $this->filename( 'general' ) ?: 'sitemap'; $url = home_url( $filename . '.xml' ); } $addon = aioseo()->addons->getLoadedAddon( $type ); if ( ! empty( $addon->helpers ) && method_exists( $addon->helpers, 'getUrl' ) ) { $url = $addon->helpers->getUrl(); } return $url; } /** * Returns if images should be excluded from the sitemap. * * @since 4.2.2 * * @return bool */ public function excludeImages() { $shouldExclude = aioseo()->options->sitemap->general->advancedSettings->enable && aioseo()->options->sitemap->general->advancedSettings->excludeImages; return apply_filters( 'aioseo_sitemap_exclude_images', $shouldExclude ); } /** * Returns the post types to check against for the author sitemap. * * @since 4.4.4 * * @return array The post types. */ public function getAuthorPostTypes() { // By default, WP only considers posts for author archives, but users can include additional post types. $postTypes = [ 'post' ]; return apply_filters( 'aioseo_sitemap_author_post_types', $postTypes ); } }
Fatal error: Uncaught Error: Class "AIOSEO\Plugin\Common\Sitemap\Helpers" not found in /htdocs/wp-content/plugins/all-in-one-seo-pack/app/Common/Sitemap/Sitemap.php:94 Stack trace: #0 /htdocs/wp-content/plugins/all-in-one-seo-pack/app/AIOSEO.php(311): AIOSEO\Plugin\Common\Sitemap\Sitemap->__construct() #1 /htdocs/wp-content/plugins/all-in-one-seo-pack/app/AIOSEO.php(97): AIOSEO\Plugin\AIOSEO->load() #2 /htdocs/wp-content/plugins/all-in-one-seo-pack/app/AIOSEO.php(76): AIOSEO\Plugin\AIOSEO->init() #3 /htdocs/wp-content/plugins/all-in-one-seo-pack/app/AIOSEO.php(414): AIOSEO\Plugin\AIOSEO::instance() #4 /htdocs/wp-content/plugins/all-in-one-seo-pack/all_in_one_seo_pack.php(96): aioseo() #5 /htdocs/wp-settings.php(526): include_once('/htdocs/wp-cont...') #6 /htdocs/wp-config.php(85): require_once('/htdocs/wp-sett...') #7 /htdocs/wp-load.php(50): require_once('/htdocs/wp-conf...') #8 /htdocs/wp-blog-header.php(13): require_once('/htdocs/wp-load...') #9 /htdocs/index.php(17): require('/htdocs/wp-blog...') #10 {main} thrown in /htdocs/wp-content/plugins/all-in-one-seo-pack/app/Common/Sitemap/Sitemap.php on line 94