File: //proc/self/cwd/wp-content/plugins/autodescription/inc/classes/admin/seobar/builder/term.class.php
<?php
/**
* @package The_SEO_Framework\Classes\Admin\SEOBar\Builder\Term
* @subpackage The_SEO_Framework\SEOBar
*/
namespace The_SEO_Framework\Admin\SEOBar\Builder;
\defined( 'THE_SEO_FRAMEWORK_PRESENT' ) or die;
use const \The_SEO_Framework\ROBOTS_ASSERT;
use \The_SEO_Framework\{
Data,
Data\Filter\Sanitize,
Meta,
RobotsTXT,
Admin\SEOBar\Builder,
};
use \The_SEO_Framework\Helper\{
Guidelines,
Format\Strings,
Query,
Migrate,
Taxonomy,
};
/**
* The SEO Framework plugin
* Copyright (C) 2019 - 2024 Sybre Waaijer, CyberWire B.V. (https://cyberwire.nl/)
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 3 as published
* by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
/**
* Generates the SEO Bar for terms.
*
* @since 4.0.0
* @since 4.2.0 Renamed from `SeoBar_Term`.
* @since 5.0.0 Moved from `\The_SEO_Framework\Builders\SEOBar`.
*
* @access private
* @see \The_SEO_Framework\Admin\SEOBar\Builder
*/
final class Term extends Main {
/**
* @since 4.0.0
* @abstract
* @var array All known tests.
*/
public static $tests = [ 'title', 'description', 'indexing', 'following', 'archiving', 'redirect' ];
/**
* Primes the cache.
*
* @since 4.0.0
* @abstract
*/
protected function prime_cache() {
// phpcs:disable, PEAR.Functions.FunctionCallSignature.Indent -- False negative.
static::get_cache( 'general/i18n/textsizeguidelines' )
or static::set_cache(
'general/i18n/textsizeguidelines',
Guidelines::get_text_size_guidelines_i18n()
);
static::get_cache( 'general/detect/robotsglobal' )
or static::set_cache(
'general/detect/robotsglobal',
[
'hasrobotstxt' => RobotsTXT\Utils::has_root_robots_txt(),
'blogpublic' => Data\Blog::is_public(),
'site' => [
'noindex' => Data\Plugin::get_option( 'site_noindex' ),
'nofollow' => Data\Plugin::get_option( 'site_nofollow' ),
'noarchive' => Data\Plugin::get_option( 'site_noarchive' ),
],
'posttype' => [
'noindex' => Data\Plugin::get_option( Data\Plugin\Helper::get_robots_option_index( 'post_type', 'noindex' ) ),
'nofollow' => Data\Plugin::get_option( Data\Plugin\Helper::get_robots_option_index( 'post_type', 'nofollow' ) ),
'noarchive' => Data\Plugin::get_option( Data\Plugin\Helper::get_robots_option_index( 'post_type', 'noarchive' ) ),
],
'taxonomy' => [
'noindex' => Data\Plugin::get_option( Data\Plugin\Helper::get_robots_option_index( 'taxonomy', 'noindex' ) ),
'nofollow' => Data\Plugin::get_option( Data\Plugin\Helper::get_robots_option_index( 'taxonomy', 'nofollow' ) ),
'noarchive' => Data\Plugin::get_option( Data\Plugin\Helper::get_robots_option_index( 'taxonomy', 'noarchive' ) ),
],
],
);
// phpcs:enable, PEAR.Functions.FunctionCallSignature.Indent
}
/**
* Primes the current query cache.
*
* @since 4.0.0
* @since 5.0.0 Removed first parameter.
* @abstract
*/
protected function prime_query_cache() {
$this->query_cache = [
'term' => \get_term( static::$query['id'], static::$query['tax'] ),
'meta' => Data\Plugin\Term::get_meta( static::$query['id'] ),
'states' => [
'locale' => \get_locale(),
'isempty' => ! Data\Term::is_term_populated( static::$query['id'], static::$query['tax'] ),
'posttypes' => Taxonomy::get_post_types( static::$query['tax'] ),
'robotsmeta' => array_merge(
[
'noindex' => false,
'nofollow' => false,
'noarchive' => false,
],
Meta\Robots::get_generated_meta(
[
'id' => static::$query['id'],
'tax' => static::$query['tax'],
],
[ 'noindex', 'nofollow', 'noarchive' ],
ROBOTS_ASSERT,
),
),
// We don't use this... yet. I couldn't find a way to properly implement the assertions in the right order.
// The asserter should be leading, but the SEO Bar should be readable.
'robotsassert' => Meta\Robots::get_collected_meta_assertions(),
],
];
}
/**
* Tests for blocking redirection.
*
* @since 4.0.0
* @abstract
*
* @return bool True if there's a blocking redirect, false otherwise.
*/
protected function has_blocking_redirect() {
return ! empty( $this->query_cache['meta']['redirect'] );
}
/**
* Runs title tests.
*
* @since 4.0.0
* @since 4.0.5 1. Removed `['params']['prefixed'] from cache.
* 2. Now tests for term title prefix per state.
* 3. Added syntax test.
*
* @return array $item {
* The SEO Bar title item.
*
* @type string $symbol The displayed symbol that identifies your bar.
* @type string $title The title of the assessment.
* @type int $status Power of two. See SEOBar's class constants.
* @type string $reason The final assessment: The reason for the $status. The latest state-changing reason is used.
* @type string $assess The assessments on why the reason is set. Keep it short and concise!
* Does not accept HTML for performant ARIA support.
* }
*/
protected function test_title() {
$cache = static::get_cache( 'term/title/defaults' ) ?: static::set_cache(
'term/title/defaults',
[
'params' => [
'untitled' => Meta\Title::get_untitled_title(),
'blogname_quoted' => preg_quote(
Sanitize::normalize_metadata_content_for_strcmp( Data\Blog::get_public_blog_name() ),
'/',
),
/* translators: 1 = An assessment, 2 = Disclaimer, e.g. "take it with a grain of salt" */
'disclaim' => \__( '%1$s (%2$s)', 'autodescription' ),
'estimated' => \__( 'Estimated from the number of characters found. The pixel counter asserts the true length.', 'autodescription' ),
],
'assess' => [
'empty' => \__( 'No title could be fetched.', 'autodescription' ),
'untitled' => \sprintf(
/* translators: %s = "Untitled" */
\__( 'No title could be fetched, "%s" is used instead.', 'autodescription' ),
Meta\Title::get_untitled_title(),
),
'prefixed' => \__( 'A term label prefix is automatically added which increases the length.', 'autodescription' ),
'branding' => [
'not' => \__( "It's not branded. Search engines may ignore your title. Consider adding back the site title.", 'autodescription' ),
'manual' => \__( "It's manually branded.", 'autodescription' ),
'automatic' => \__( "It's automatically branded.", 'autodescription' ),
],
'duplicated' => \__( 'The site title is found multiple times.', 'autodescription' ),
'syntax' => \__( "Markup syntax was found that isn't transformed. Consider rewriting the custom title.", 'autodescription' ),
],
'reason' => [
'incomplete' => \__( 'Incomplete.', 'autodescription' ),
'duplicated' => \__( 'The branding is repeated.', 'autodescription' ),
'notbranded' => \__( 'Not branded.', 'autodescription' ),
'syntax' => \__( 'Found markup syntax.', 'autodescription' ),
],
'defaults' => [
'generated' => [
'symbol' => \_x( 'TG', 'Title Generated', 'autodescription' ),
'title' => \__( 'Title, generated', 'autodescription' ),
'status' => Builder::STATE_GOOD,
'reason' => \__( 'Automatically generated.', 'autodescription' ),
'assess' => [
'base' => \__( "It's built from the term name.", 'autodescription' ),
],
],
'custom' => [
'symbol' => \_x( 'T', 'Title', 'autodescription' ),
'title' => \__( 'Title', 'autodescription' ),
'status' => Builder::STATE_GOOD,
'reason' => \__( 'Obtained from term SEO meta input.', 'autodescription' ),
'assess' => [
'base' => \__( "It's built from term SEO meta input.", 'autodescription' ),
],
],
],
],
);
$generator_args = [
'id' => static::$query['id'],
'tax' => static::$query['tax'],
];
// TODO instead of getting values from the options API, why don't we store the parameters and allow them to be modified?
// This way, we can implement real-time live-edit AJAX SEO bar items...
$title_part = Meta\Title::get_bare_custom_title( $generator_args );
if ( \strlen( $title_part ) ) {
$item = $cache['defaults']['custom'];
if ( Migrate::text_has_unprocessed_syntax( $title_part, false ) ) {
$item['status'] = Builder::STATE_BAD;
$item['reason'] = $cache['reason']['syntax'];
$item['assess']['syntax'] = $cache['assess']['syntax'];
// Further assessments must be made later. Halt assertion here to prevent confusion.
return $item;
}
} else {
$item = $cache['defaults']['generated'];
if ( Meta\Title\Conditions::use_generated_archive_prefix( $this->query_cache['term'] ) ) {
$item['assess']['prefixed'] = $cache['assess']['prefixed'];
}
$title_part = Meta\Title::get_bare_generated_title( $generator_args );
}
if ( ! \strlen( $title_part ) ) {
$item['status'] = Builder::STATE_BAD;
$item['reason'] = $cache['reason']['incomplete'];
$item['assess']['empty'] = $cache['assess']['empty'];
// Further assessments must be made later. Halt assertion here to prevent confusion.
return $item;
} elseif ( $title_part === $cache['params']['untitled'] ) {
$item['status'] = Builder::STATE_BAD;
$item['reason'] = $cache['reason']['incomplete'];
$item['assess']['untitled'] = $cache['assess']['untitled'];
// Further assessments must be made later. Halt assertion here to prevent confusion.
return $item;
}
$title = $title_part;
if ( Meta\Title\Conditions::use_branding( $generator_args ) ) {
$_title_before = $title;
$title = Meta\Title::add_branding( $title, $generator_args );
// Absence assertion is done after this.
if ( $title === $_title_before ) {
// Title didn't change, so no automatic branding was added.
$item['assess']['branding'] = $cache['assess']['branding']['manual'];
} else {
$item['assess']['branding'] = $cache['assess']['branding']['automatic'];
}
} else {
$item['assess']['branding'] = $cache['assess']['branding']['manual'];
}
$strcmp_title = Sanitize::normalize_metadata_content_for_strcmp( $title );
$brand_count = \strlen( $cache['params']['blogname_quoted'] )
? preg_match_all(
"/{$cache['params']['blogname_quoted']}/ui",
$strcmp_title,
$matches,
)
: 0;
if ( ! $brand_count ) {
// Override branding state.
$item['status'] = Builder::STATE_UNKNOWN;
$item['reason'] = $cache['reason']['notbranded'];
$item['assess']['branding'] = $cache['assess']['branding']['not'];
} elseif ( $brand_count > 1 ) {
$item['status'] = Builder::STATE_BAD;
$item['reason'] = $cache['reason']['duplicated'];
$item['assess']['duplicated'] = $cache['assess']['duplicated'];
// Further assessments must be made later. Halt assertion here to prevent confusion.
return $item;
}
$title_len = mb_strlen( $strcmp_title );
$guidelines = Guidelines::get_text_size_guidelines(
$this->query_cache['states']['locale']
)['title']['search']['chars'];
$guidelines_i18n = static::get_cache( 'general/i18n/textsizeguidelines' );
if ( $title_len < $guidelines['lower'] ) {
$item['status'] = Builder::STATE_BAD;
$item['reason'] = $guidelines_i18n['shortdot']['farTooShort'];
$length_i18n = $guidelines_i18n['long']['farTooShort'];
} elseif ( $title_len < $guidelines['goodLower'] ) {
$item['status'] = Builder::STATE_OKAY;
$item['reason'] = $guidelines_i18n['shortdot']['tooShort'];
$length_i18n = $guidelines_i18n['long']['tooShort'];
} elseif ( $title_len > $guidelines['upper'] ) {
$item['status'] = Builder::STATE_BAD;
$item['reason'] = $guidelines_i18n['shortdot']['farTooLong'];
$length_i18n = $guidelines_i18n['long']['farTooLong'];
} elseif ( $title_len > $guidelines['goodUpper'] ) {
$item['status'] = Builder::STATE_OKAY;
$item['reason'] = $guidelines_i18n['shortdot']['tooLong'];
$length_i18n = $guidelines_i18n['long']['tooLong'];
} else {
// Use unaltered reason and status.
$length_i18n = $guidelines_i18n['long']['good'];
}
$item['assess']['length'] = \sprintf(
$cache['params']['disclaim'],
$length_i18n,
$cache['params']['estimated'],
);
return $item;
}
/**
* Runs description tests.
*
* @since 4.0.0
* @since 4.0.5 Added syntax test.
*
* @return array $item {
* The SEO Bar description item.
*
* @type string $symbol The displayed symbol that identifies your bar.
* @type string $title The title of the assessment.
* @type int $status Power of two. See SEOBar's class constants.
* @type string $reason The final assessment: The reason for the $status. The latest state-changing reason is used.
* @type string $assess The assessments on why the reason is set. Keep it short and concise!
* Does not accept HTML for performant ARIA support.
* }
*/
protected function test_description() {
$cache = static::get_cache( 'term/description/defaults' ) ?: static::set_cache(
'term/description/defaults',
[
'params' => [
/* translators: 1 = An assessment, 2 = Disclaimer, e.g. "take it with a grain of salt" */
'disclaim' => \__( '%1$s (%2$s)', 'autodescription' ),
'estimated' => \__( 'Estimated from the number of characters found. The pixel counter asserts the true length.', 'autodescription' ),
/**
* @since 2.6.0
* @param int $short_word_length The minimum stringlength of words to find as dupes.
*/
'dupe_short' => (int) \apply_filters( 'the_seo_framework_bother_me_desc_length', 3 ),
],
'assess' => [
'empty' => \__( 'No description could be generated.', 'autodescription' ),
/* translators: %s = list of repeated words */
'dupes' => \__( 'Found repeated words: %s', 'autodescription' ),
'syntax' => \__( "Markup syntax was found that isn't transformed. Consider rewriting the custom description.", 'autodescription' ),
],
'reason' => [
'empty' => \__( 'Empty.', 'autodescription' ),
'founddupe' => \__( 'Found repeated words.', 'autodescription' ),
'foundmanydupe' => \__( 'Found too many repeated words.', 'autodescription' ),
'syntax' => \__( 'Found markup syntax.', 'autodescription' ),
],
'defaults' => [
'generated' => [
'symbol' => \_x( 'DG', 'Description Generated', 'autodescription' ),
'title' => \__( 'Description, generated', 'autodescription' ),
'status' => Builder::STATE_GOOD,
'reason' => \__( 'Automatically generated.', 'autodescription' ),
'assess' => [
'base' => \__( "It's built from the term description field.", 'autodescription' ),
],
],
'emptynoauto' => [
'symbol' => \_x( 'D', 'Description', 'autodescription' ),
'title' => \__( 'Description', 'autodescription' ),
'status' => Builder::STATE_UNKNOWN,
'reason' => \__( 'Empty.', 'autodescription' ),
'assess' => [
'noauto' => \__( 'No term description is set.', 'autodescription' ),
],
],
'custom' => [
'symbol' => \_x( 'D', 'Description', 'autodescription' ),
'title' => \__( 'Description', 'autodescription' ),
'status' => Builder::STATE_GOOD,
'reason' => \__( 'Obtained from the term SEO meta input.', 'autodescription' ),
'assess' => [
'base' => \__( "It's built from the term SEO meta input.", 'autodescription' ),
],
],
],
],
);
$generator_args = [
'id' => static::$query['id'],
'tax' => static::$query['tax'],
];
// TODO instead of getting values from the options API, why don't we store the parameters and allow them to be modified?
// This way, we can implement real-time live-edit AJAX SEO bar items...
$desc = Meta\Description::get_custom_description( $generator_args );
if ( \strlen( $desc ) ) {
$item = $cache['defaults']['custom'];
if ( Migrate::text_has_unprocessed_syntax( $desc ) ) {
$item['status'] = Builder::STATE_BAD;
$item['reason'] = $cache['reason']['syntax'];
$item['assess']['syntax'] = $cache['assess']['syntax'];
// Further assessments must be made later. Halt assertion here to prevent confusion.
return $item;
}
} elseif ( ! Meta\Description::may_generate( $generator_args ) ) {
$item = $cache['defaults']['emptynoauto'];
// No description is found. There's no need to continue parsing.
return $item;
} else {
$item = $cache['defaults']['generated'];
$desc = Meta\Description::get_generated_description( $generator_args );
if ( ! \strlen( $desc ) ) {
$item['status'] = Builder::STATE_UNDEFINED;
$item['reason'] = $cache['reason']['empty'];
// This is now inaccurate, purge it.
// TODO consider alternative? "It TRIED to build it from...."?
unset( $item['assess']['base'] );
$item['assess']['empty'] = $cache['assess']['empty'];
// No description is found. There's no need to continue parsing.
return $item;
}
}
// Fetch words that are outputted more than 3 times.
$repeated_words = Strings::get_word_count( $desc, [ 'short_word_length' => $cache['params']['dupe_short'] ] );
if ( $repeated_words ) {
$dupes = [];
foreach ( $repeated_words as $_repeated_word ) {
// Keep abbreviations... WordPress, make multibyte support mandatory already.
// $_word = ctype_upper( reset( $_repeated_word ) ) ? reset( $_repeated_word ) : mb_strtolower( reset( $_repeated_word ) );
$dupes[] = \sprintf(
/* translators: 1: Word found, 2: Occurrences */
\esc_attr__( '“%1$s” is used %2$d times.', 'autodescription' ),
\esc_attr( key( $_repeated_word ) ),
reset( $_repeated_word ), // escaped in sprintf %d.
);
}
$item['assess']['dupe'] = implode( ' ', $dupes );
$max = max( $repeated_words );
$max = reset( $max );
// Warn when more than 3x triplet+/quintet+ words are found.
if ( $max > 3 || \count( $repeated_words ) > 1 ) {
// This must be resolved.
$item['reason'] = $cache['reason']['foundmanydupe'];
$item['status'] = Builder::STATE_BAD;
return $item;
} else {
$item['reason'] = $cache['reason']['founddupe'];
$item['status'] = Builder::STATE_OKAY;
}
}
$guidelines = Guidelines::get_text_size_guidelines(
$this->query_cache['states']['locale']
)['description']['search']['chars'];
$guidelines_i18n = static::get_cache( 'general/i18n/textsizeguidelines' );
$desc_len = mb_strlen( Sanitize::normalize_metadata_content_for_strcmp( $desc ) );
if ( $desc_len < $guidelines['lower'] ) {
$item['status'] = Builder::STATE_BAD;
$item['reason'] = $guidelines_i18n['shortdot']['farTooShort'];
$length_i18n = $guidelines_i18n['long']['farTooShort'];
} elseif ( $desc_len < $guidelines['goodLower'] ) {
$item['status'] = Builder::STATE_OKAY;
$item['reason'] = $guidelines_i18n['shortdot']['tooShort'];
$length_i18n = $guidelines_i18n['long']['tooShort'];
} elseif ( $desc_len > $guidelines['upper'] ) {
$item['status'] = Builder::STATE_BAD;
$item['reason'] = $guidelines_i18n['shortdot']['farTooLong'];
$length_i18n = $guidelines_i18n['long']['farTooLong'];
} elseif ( $desc_len > $guidelines['goodUpper'] ) {
$item['status'] = Builder::STATE_OKAY;
$item['reason'] = $guidelines_i18n['shortdot']['tooLong'];
$length_i18n = $guidelines_i18n['long']['tooLong'];
} else {
// Use unaltered reason and status.
$length_i18n = $guidelines_i18n['long']['good'];
}
$item['assess']['length'] = \sprintf(
$cache['params']['disclaim'],
$length_i18n,
$cache['params']['estimated'],
);
return $item;
}
/**
* Runs indexing tests.
*
* @since 4.0.0
* @since 4.1.0 Now asserts all taxonomy robots settings.
*
* @return array $item {
* The SEO Bar robots indexing item.
*
* @type string $symbol The displayed symbol that identifies your bar.
* @type string $title The title of the assessment.
* @type int $status Power of two. See SEOBar's class constants.
* @type string $reason The final assessment: The reason for the $status. The latest state-changing reason is used.
* @type string $assess The assessments on why the reason is set. Keep it short and concise!
* Does not accept HTML for performant ARIA support.
* }
*/
protected function test_indexing() {
$cache = static::get_cache( 'term/indexing/defaults' ) ?: static::set_cache(
'term/indexing/defaults',
[
'params' => [],
'assess' => [
'robotstxt' => \__( 'The robots.txt file is nonstandard, and may still direct search engines differently.', 'autodescription' ),
'notpublic' => \__( 'WordPress discourages crawling via the Reading Settings.', 'autodescription' ),
'site' => \__( 'Indexing is discouraged for the whole site at the SEO Settings screen.', 'autodescription' ),
'posttypes' => \__( 'Indexing is discouraged for all bound post types to this term at the SEO Settings screen.', 'autodescription' ),
'taxonomy' => \__( 'Indexing is discouraged for this taxonomy at the SEO Settings screen.', 'autodescription' ),
'override' => \__( 'The term SEO meta input overrides the indexing state.', 'autodescription' ),
'empty' => \__( 'No posts are attached to this term, so indexing is disabled.', 'autodescription' ),
'emptyoverride' => \__( 'No posts are attached to this term, so indexing should be disabled.', 'autodescription' ),
'canonicalurl' => \__( 'A custom canonical URL is set that points to another page.', 'autodescription' ),
],
'reason' => [
'notpublic' => \__( 'WordPress overrides the robots directive.', 'autodescription' ),
'empty' => \__( 'The term is empty.', 'autodescription' ),
'emptyoverride' => \__( 'The term is empty yet still indexed.', 'autodescription' ),
'canonicalurl' => \__( 'The canonical URL points to another page.', 'autodescription' ),
],
'defaults' => [
'index' => [
'symbol' => \_x( 'I', 'Indexing', 'autodescription' ),
'title' => \__( 'Indexing', 'autodescription' ),
'status' => Builder::STATE_GOOD,
'reason' => \__( 'Term may be indexed.', 'autodescription' ),
'assess' => [
'base' => \__( 'The robots meta tag allows indexing.', 'autodescription' ),
],
],
'noindex' => [
'symbol' => \_x( 'I', 'Indexing', 'autodescription' ),
'title' => \__( 'Indexing', 'autodescription' ),
'status' => Builder::STATE_UNKNOWN,
'reason' => \__( 'Term may not be indexed.', 'autodescription' ),
'assess' => [
'base' => \__( 'The robots meta tag does not allow indexing.', 'autodescription' ),
],
],
],
],
);
$robots_global = static::get_cache( 'general/detect/robotsglobal' );
if ( $this->query_cache['states']['robotsmeta']['noindex'] ) {
$item = $cache['defaults']['noindex'];
} else {
$item = $cache['defaults']['index'];
}
if ( ! $robots_global['blogpublic'] ) {
$item['status'] = Builder::STATE_BAD;
$item['reason'] = $cache['reason']['notpublic'];
unset( $item['assess']['base'] );
$item['assess']['notpublic'] = $cache['assess']['notpublic'];
// Change symbol to grab attention
$item['symbol'] = '!!!';
// Let the user resolve this first, everything's moot hereafter.
return $item;
}
if ( $robots_global['site']['noindex'] ) {
// Status is already set.
$item['assess']['site'] = $cache['assess']['site'];
}
// Test all post types bound to the term. Only if all post types are excluded, set this option.
$_post_type_noindex_set = [];
foreach ( $this->query_cache['states']['posttypes'] as $_post_type ) {
$_post_type_noindex_set[] = ! empty( $robots_global['posttype']['noindex'][ $_post_type ] );
}
if ( ! \in_array( false, $_post_type_noindex_set, true ) ) {
// Status is already set.
$item['assess']['posttypes'] = $cache['assess']['posttypes'];
}
if ( ! empty( $robots_global['taxonomy']['noindex'][ static::$query['tax'] ] ) ) {
// Status is already set.
$item['assess']['taxonomy'] = $cache['assess']['taxonomy'];
}
if ( 0 !== Sanitize::qubit( $this->query_cache['meta']['noindex'] ) ) {
// Status is already set.
// Don't assert posttype nor site as "blocking" if there's an override.
unset( $item['assess']['site'], $item['assess']['posttypes'], $item['assess']['taxonomy'] );
$item['assess']['override'] = $cache['assess']['override'];
}
if ( $this->query_cache['meta']['canonical'] ) {
$permalink = Meta\URI::get_generated_url( [
'id' => static::$query['id'],
'tax' => static::$query['tax'],
] );
$canonical = Meta\URI::get_canonical_url( [
'id' => static::$query['id'],
'tax' => static::$query['tax'],
] );
if ( $permalink !== $canonical ) {
$item['status'] = Builder::STATE_UNKNOWN;
$item['reason'] = $cache['reason']['canonicalurl'];
$item['assess']['protected'] = $cache['assess']['canonicalurl'];
}
}
if ( $this->query_cache['states']['isempty'] ) {
if ( $this->query_cache['states']['robotsmeta']['noindex'] ) {
// Everything's as intended...
$item['status'] = Builder::STATE_UNKNOWN;
$item['reason'] = $cache['reason']['empty'];
$item['assess']['empty'] = $cache['assess']['empty'];
} else {
// Something's wrong. Maybe override, maybe filter, maybe me.
$item['status'] = Builder::STATE_BAD;
$item['reason'] = $cache['reason']['emptyoverride'];
$item['assess']['empty'] = $cache['assess']['emptyoverride'];
}
}
if ( ! $this->query_cache['states']['robotsmeta']['noindex'] && $robots_global['hasrobotstxt'] ) {
// Don't change status, we do not parse the robots.txt file. Merely disclaim.
$item['assess']['robotstxt'] = $cache['assess']['robotstxt'];
}
return $item;
}
/**
* Runs following tests.
*
* @since 4.0.0
* @since 4.1.0 Now asserts all taxonomy robots settings.
*
* @return array $item {
* The SEO Bar robots following item.
*
* @type string $symbol The displayed symbol that identifies your bar.
* @type string $title The title of the assessment.
* @type int $status Power of two. See SEOBar's class constants.
* @type string $reason The final assessment: The reason for the $status. The latest state-changing reason is used.
* @type string $assess The assessments on why the reason is set. Keep it short and concise!
* Does not accept HTML for performant ARIA support.
* }
*/
protected function test_following() {
$cache = static::get_cache( 'term/following/defaults' ) ?: static::set_cache(
'term/following/defaults',
[
'params' => [],
'assess' => [
'robotstxt' => \__( 'The robots.txt file is nonstandard, and may still direct search engines differently.', 'autodescription' ),
'notpublic' => \__( 'WordPress discourages crawling via the Reading Settings.', 'autodescription' ),
'site' => \__( 'Link following is discouraged for the whole site at the SEO Settings screen.', 'autodescription' ),
'posttypes' => \__( 'Link following is discouraged for all bound post types to this term at the SEO Settings screen.', 'autodescription' ),
'taxonomy' => \__( 'Link following is discouraged for this taxonomy at the SEO Settings screen.', 'autodescription' ),
'override' => \__( 'The term SEO meta input overrides the link following state.', 'autodescription' ),
'noindex' => \__( 'The term may not be indexed, this may also discourage link following.', 'autodescription' ),
],
'reason' => [
'notpublic' => \__( 'WordPress overrides the robots directive.', 'autodescription' ),
],
'defaults' => [
'follow' => [
'symbol' => \_x( 'F', 'Following', 'autodescription' ),
'title' => \__( 'Following', 'autodescription' ),
'status' => Builder::STATE_GOOD,
'reason' => \__( 'Term links may be followed.', 'autodescription' ),
'assess' => [
'base' => \__( 'The robots meta tag allows link following.', 'autodescription' ),
],
],
'nofollow' => [
'symbol' => \_x( 'F', 'Following', 'autodescription' ),
'title' => \__( 'Following', 'autodescription' ),
'status' => Builder::STATE_UNKNOWN,
'reason' => \__( 'Term links may not be followed.', 'autodescription' ),
'assess' => [
'base' => \__( 'The robots meta tag does not allow link following.', 'autodescription' ),
],
],
],
],
);
$robots_global = static::get_cache( 'general/detect/robotsglobal' );
if ( $this->query_cache['states']['robotsmeta']['nofollow'] ) {
$item = $cache['defaults']['nofollow'];
} else {
$item = $cache['defaults']['follow'];
}
if ( ! $robots_global['blogpublic'] ) {
$item['status'] = Builder::STATE_BAD;
$item['reason'] = $cache['reason']['notpublic'];
unset( $item['assess']['base'] );
$item['assess']['notpublic'] = $cache['assess']['notpublic'];
// Change symbol to grab attention
$item['symbol'] = '!!!';
// Let the user resolve this first, everything's moot hereafter.
return $item;
}
if ( $robots_global['site']['nofollow'] ) {
// Status is already set.
$item['assess']['site'] = $cache['assess']['site'];
}
// Test all post types bound to the term. Only if all post types are excluded, set this option.
$_post_type_nofollow_set = [];
foreach ( $this->query_cache['states']['posttypes'] as $_post_type ) {
$_post_type_nofollow_set[] = ! empty( $robots_global['posttype']['nofollow'][ $_post_type ] );
}
if ( ! \in_array( false, $_post_type_nofollow_set, true ) ) {
// Status is already set.
$item['assess']['posttypes'] = $cache['assess']['posttypes'];
}
if ( ! empty( $robots_global['taxonomy']['nofollow'][ static::$query['tax'] ] ) ) {
// Status is already set.
$item['assess']['taxonomy'] = $cache['assess']['taxonomy'];
}
if ( 0 !== Sanitize::qubit( $this->query_cache['meta']['nofollow'] ) ) {
// Status is already set.
// Don't assert posttype nor site as "blocking" if there's an override.
unset( $item['assess']['site'], $item['assess']['posttypes'], $item['assess']['taxonomy'] );
$item['assess']['override'] = $cache['assess']['override'];
}
if ( ! $this->query_cache['states']['robotsmeta']['nofollow'] ) {
if ( $this->query_cache['states']['robotsmeta']['noindex'] ) {
$item['status'] = Builder::STATE_OKAY;
$item['assess']['noindex'] = $cache['assess']['noindex'];
}
if ( $robots_global['hasrobotstxt'] ) {
// Don't change status, we do not parse the robots.txt file. Merely disclaim.
$item['assess']['robotstxt'] = $cache['assess']['robotstxt'];
}
}
return $item;
}
/**
* Runs archiving tests.
*
* @since 4.0.0
* @since 4.1.0 Now asserts all taxonomy robots settings.
*
* @return array $item {
* The SEO Bar robots archiving item.
*
* @type string $symbol The displayed symbol that identifies your bar.
* @type string $title The title of the assessment.
* @type int $status Power of two. See SEOBar's class constants.
* @type string $reason The final assessment: The reason for the $status. The latest state-changing reason is used.
* @type string $assess The assessments on why the reason is set. Keep it short and concise!
* Does not accept HTML for performant ARIA support.
* }
*/
protected function test_archiving() {
$cache = static::get_cache( 'term/archiving/defaults' ) ?: static::set_cache(
'term/archiving/defaults',
[
'params' => [],
'assess' => [
'robotstxt' => \__( 'The robots.txt file is nonstandard, and may still direct search engines differently.', 'autodescription' ),
'notpublic' => \__( 'WordPress discourages crawling via the Reading Settings.', 'autodescription' ),
'site' => \__( 'Archiving is discouraged for the whole site at the SEO Settings screen.', 'autodescription' ),
'posttypes' => \__( 'Archiving is discouraged for all bound post types to this term at the SEO Settings screen.', 'autodescription' ),
'taxonomy' => \__( 'Archiving is discouraged for this taxonomy at the SEO Settings screen.', 'autodescription' ),
'override' => \__( 'The term SEO meta input overrides the archiving state.', 'autodescription' ),
'noindex' => \__( 'The term may not be indexed, this may also discourage archiving.', 'autodescription' ),
],
'reason' => [
'notpublic' => \__( 'WordPress overrides the robots directive.', 'autodescription' ),
],
'defaults' => [
'archive' => [
'symbol' => \_x( 'A', 'Archiving', 'autodescription' ),
'title' => \__( 'Archiving', 'autodescription' ),
'status' => Builder::STATE_GOOD,
'reason' => \__( 'Term may be archived.', 'autodescription' ),
'assess' => [
'base' => \__( 'The robots meta tag allows archiving.', 'autodescription' ),
],
],
'noarchive' => [
'symbol' => \_x( 'A', 'Archiving', 'autodescription' ),
'title' => \__( 'Archiving', 'autodescription' ),
'status' => Builder::STATE_UNKNOWN,
'reason' => \__( 'Term may not be archived.', 'autodescription' ),
'assess' => [
'base' => \__( 'The robots meta tag does not allow archiving.', 'autodescription' ),
],
],
],
],
);
$robots_global = static::get_cache( 'general/detect/robotsglobal' );
if ( $this->query_cache['states']['robotsmeta']['noarchive'] ) {
$item = $cache['defaults']['noarchive'];
} else {
$item = $cache['defaults']['archive'];
}
if ( ! $robots_global['blogpublic'] ) {
$item['status'] = Builder::STATE_BAD;
$item['reason'] = $cache['reason']['notpublic'];
unset( $item['assess']['base'] );
$item['assess']['notpublic'] = $cache['assess']['notpublic'];
// Change symbol to grab attention
$item['symbol'] = '!!!';
// Let the user resolve this first, everything's moot hereafter.
return $item;
}
if ( $robots_global['site']['noarchive'] ) {
// Status is already set.
$item['assess']['site'] = $cache['assess']['site'];
}
// Test all post types bound to the term. Only if all post types are excluded, set this option.
$_post_type_noarchive_set = [];
foreach ( $this->query_cache['states']['posttypes'] as $_post_type ) {
$_post_type_noarchive_set[] = ! empty( $robots_global['posttype']['noarchive'][ $_post_type ] );
}
if ( ! \in_array( false, $_post_type_noarchive_set, true ) ) {
// Status is already set.
$item['assess']['posttypes'] = $cache['assess']['posttypes'];
}
if ( ! empty( $robots_global['taxonomy']['noarchive'][ static::$query['tax'] ] ) ) {
// Status is already set.
$item['assess']['taxonomy'] = $cache['assess']['taxonomy'];
}
if ( 0 !== Sanitize::qubit( $this->query_cache['meta']['noarchive'] ) ) {
// Status is already set.
// Don't assert posttype nor site as "blocking" if there's an override.
unset( $item['assess']['site'], $item['assess']['posttypes'], $item['assess']['taxonomy'] );
$item['assess']['override'] = $cache['assess']['override'];
}
if ( ! $this->query_cache['states']['robotsmeta']['noarchive'] ) {
if ( $this->query_cache['states']['robotsmeta']['noindex'] ) {
$item['status'] = Builder::STATE_OKAY;
$item['assess']['noindex'] = $cache['assess']['noindex'];
}
if ( $robots_global['hasrobotstxt'] ) {
// Don't change status, we do not parse the robots.txt file. Merely disclaim.
$item['assess']['robotstxt'] = $cache['assess']['robotstxt'];
}
}
return $item;
}
/**
* Runs redirect tests.
*
* @since 4.0.0
*
* @return array $item {
* The SEO Bar redirect item.
*
* @type string $symbol The displayed symbol that identifies your bar.
* @type string $title The title of the assessment.
* @type int $status Power of two. See SEOBar's class constants.
* @type string $reason The final assessment: The reason for the $status. The latest state-changing reason is used.
* @type string $assess The assessments on why the reason is set. Keep it short and concise!
* Does not accept HTML for performant ARIA support.
* }
*/
protected function test_redirect() {
if ( empty( $this->query_cache['meta']['redirect'] ) ) {
return static::get_cache( 'term/redirect/default/0' ) ?: static::set_cache(
'term/redirect/default/0',
[
'symbol' => \_x( 'R', 'Redirect', 'autodescription' ),
'title' => \__( 'Redirection', 'autodescription' ),
'status' => Builder::STATE_GOOD,
'reason' => \__( 'Term does not redirect visitors.', 'autodescription' ),
'assess' => [
'redirect' => \__( 'All visitors and crawlers may access this page.', 'autodescription' ),
],
'meta' => [
'blocking' => false,
],
],
);
} else {
return static::get_cache( 'term/redirect/default/1' ) ?: static::set_cache(
'term/redirect/default/1',
[
'symbol' => \_x( 'R', 'Redirect', 'autodescription' ),
'title' => \__( 'Redirection', 'autodescription' ),
'status' => Builder::STATE_UNKNOWN,
'reason' => \__( 'Term redirects visitors.', 'autodescription' ),
'assess' => [
'redirect' => \__( 'All visitors and crawlers are being redirected. So, no other SEO enhancements are effective.', 'autodescription' ),
],
'meta' => [
'blocking' => true,
],
],
);
}
}
}