Убираем кривыми руками rel=shortlink в WordPress

Достался мне значит очередной сайт

Смотрим логи, а тама

[warn] [pid 25449] sapi_apache2.c(362): [client 2a03:6f00:1::5c35:6085:43826] PHP Warning:  call_user_func_array() expects parameter 1 to be a valid callback, function 'wp_shortlink_header' not found or invalid function name in /home/c/......./public_html/wp-includes/class-wp-hook.php on line 284

Ага, в скрипте class-wp-hook.php при помощи call_user_func_array() вызывается не объявленная функция wp_shortlink_header. Как так? Функция ядра и не объявлена?

Ищем объявление через Find in files и вуаля:

/**
 * Injects rel=shortlink into the head if a shortlink is defined for the current page.
 *
 * Attached to the {@see 'wp_head'} action.
 *
 * @since 3.0.0
 
function wp_shortlink_wp_head() {
	$shortlink = wp_get_shortlink( 0, 'query' );

	if ( empty( $shortlink ) )
		return;

	echo "<link rel='shortlink' href='" . esc_url( $shortlink ) . "' />\n";
}*/

/**
 * Sends a Link: rel=shortlink header if a shortlink is defined for the current page.
 *
 * Attached to the {@see 'wp'} action.
 *
 * @since 3.0.0
 
function wp_shortlink_header() {
	if ( headers_sent() )
		return;

	$shortlink = wp_get_shortlink(0, 'query');

	if ( empty($shortlink) )
		return;

	header('Link: <' . $shortlink . '>; rel=shortlink', false);
}
*/

Находим даже две закомментированные функции в файле /wp-includes/link-template.php — одна выводит в head тэг <link rel=’shortlink’… а вторая посылает такой же заголовок браузеру.

Гуглим эту дичь и находим вот что — https://www.seowind.ru/wordpress/ubiraem-link-relshortlink-iz-zagolovka-otveta-servera-wp/

Надеюсь Юрий морально вырос над собой с далекого 2013 года и больше таких советов не раздает, к тому же в комментах есть правильное решение. Вот оно :

remove_action( 'wp_head','wp_shortlink_wp_head',10);
remove_action( 'template_redirect', 'wp_shortlink_header', 11);

Добавляем эти строчки в functions.php, и возвращаем все на место. Обратите внимание на приоритеты! Их нужно расставить так, как они были при объявлении хука. Вот на каме подробнее об этом https://wp-kama.ru/question/udalenie-shortlink

Все, лог ошибок PHP девственно чист.

Как получить рубрики кастомных записей в WordPress

В вордпрессе часто приходится работать с кастомными записями — их делают как сами вэбмастера для удобства, так и различные плагины. Это очень удобно, так как типы записей по умолчанию ориентированы на блог, а если нужно что-то посерьезней, то без кастомных типов не обойтись.

Самым ярким примером может быть портфолио. Например фирма делает каркасные дома, и для владельца сайта было бы намного удобней, понятней добавлять в систему объект «Дом» у которого есть привычные ему свойства (например стоимость, срок строительства, изображение), а не объект типа «Запись»

Итак, регистрируем в плагине или в functions.php новый тип записи при помощи

function custom_post_type_faq() {

	$labels = array(
		'name'                  => _x( 'Полезно знать', 'Post Type General Name', 'text_domain' ),
		'singular_name'         => _x( 'Полезно знать', 'Post Type Singular Name', 'text_domain' ),
		'menu_name'             => __( 'Статьи / Полезно знать', 'text_domain' ),
		'name_admin_bar'        => __( 'Post Type', 'text_domain' ),
		'archives'              => __( 'Item Archives', 'text_domain' ),
		'attributes'            => __( 'Item Attributes', 'text_domain' ),
		'parent_item_colon'     => __( 'Parent Item:', 'text_domain' ),
		'all_items'             => __( 'Все статьи', 'text_domain' ),
		'add_new_item'          => __( 'Добавить статью', 'text_domain' ),
		'add_new'               => __( 'Добавить статью', 'text_domain' ),
		'new_item'              => __( 'Добавить статью', 'text_domain' ),
		'edit_item'             => __( 'Edit Item', 'text_domain' ),
		'update_item'           => __( 'Update Item', 'text_domain' ),
		'view_item'             => __( 'View Item', 'text_domain' ),
		'view_items'            => __( 'View Items', 'text_domain' ),
		'search_items'          => __( 'Search Item', 'text_domain' ),
		'not_found'             => __( 'Not found', 'text_domain' ),
		'not_found_in_trash'    => __( 'Not found in Trash', 'text_domain' ),
		'featured_image'        => __( 'Featured Image', 'text_domain' ),
		'set_featured_image'    => __( 'Set featured image', 'text_domain' ),
		'remove_featured_image' => __( 'Remove featured image', 'text_domain' ),
		'use_featured_image'    => __( 'Use as featured image', 'text_domain' ),
		'insert_into_item'      => __( 'Insert into item', 'text_domain' ),
		'uploaded_to_this_item' => __( 'Uploaded to this item', 'text_domain' ),
		'items_list'            => __( 'Items list', 'text_domain' ),
		'items_list_navigation' => __( 'Items list navigation', 'text_domain' ),
		'filter_items_list'     => __( 'Filter items list', 'text_domain' ),
	);
	$args = array(
		'label'                 => __( 'faq', 'text_domain' ),
		'description'           => __( 'faqPost Type Description', 'text_domain' ),
		'labels'                => $labels,
		'supports'              => array( ),
		'taxonomies'            => array('faq-cats'),
		'hierarchical'          => false,
		'public'                => true,
		'show_ui'               => true,
		'show_in_menu'          => true,
		'menu_position'         => 5,
		'show_in_admin_bar'     => true,
		'show_in_nav_menus'     => true,
		'can_export'            => true,
		'has_archive'           => true,
		'exclude_from_search'   => false,
		'publicly_queryable'    => true,
		'capability_type'       => 'post',
	);
	register_post_type( 'faq', $args );

}
add_action( 'init', 'custom_post_type_faq', 0 );

Зарегистрировали. Теперь добавляем новую таксономию

add_action( 'init', 'create_taxonomy',99 );
function create_taxonomy(){

	// список параметров: wp-kama.ru/function/get_taxonomy_labels
	register_taxonomy( 'faq-cats', [ 'faq' ], [ 
		'label'                 => '', // определяется параметром $labels->name
		'labels'                => [
			'name'              => 'Рубрики faq',
			'singular_name'     => 'Рубрика faq',
			'search_items'      => 'Search Genres',
			'all_items'         => 'Все рубрики faq',
			'view_item '        => 'Смотреть рубрики faq',
			'parent_item'       => 'Родительская рубрика',
			'parent_item_colon' => 'Родительская рубрика:',
			'edit_item'         => 'Редактировать рубрику',
			'update_item'       => 'Обновить рубрику',
			'add_new_item'      => 'Добавить новую рубрику',
			'new_item_name'     => 'Имя новой рубрики',
			'menu_name'         => 'Рубрики Faq',
		],
		'description'           => '', // описание таксономии
		'public'                => true,
		// 'publicly_queryable'    => null, // равен аргументу public
		'show_in_nav_menus'     => true, // равен аргументу public
		// 'show_ui'               => true, // равен аргументу public
		// 'show_in_menu'          => true, // равен аргументу show_ui
		// 'show_tagcloud'         => true, // равен аргументу show_ui
		// 'show_in_quick_edit'    => null, // равен аргументу show_ui
		'hierarchical'          => true,

		'rewrite'               => true,
		//'query_var'             => $taxonomy, // название параметра запроса
		'capabilities'          => array(),
		'meta_box_cb'           => null, // html метабокса. callback: `post_categories_meta_box` или `post_tags_meta_box`. false — метабокс отключен.
		'show_admin_column'     => true, // авто-создание колонки таксы в таблице ассоциированного типа записи. (с версии 3.5)
		'show_in_rest'          => null, // добавить в REST API
		'rest_base'             => null, // $taxonomy
		// '_builtin'              => false,
		//'update_count_callback' => '_update_post_term_count',
	] );
}

Теперь можно создавать новые рубрики новых записей и затем в цикле WordPress получить их так:

$terms = get_the_terms($post->ID,’taxonomy’);

WordPress уникализируем title и description на страницах пагинации на сайтах с Yoast SEO

add_filter('wpseo_title', 'filter_product_wpseo_title', 500);
function filter_product_wpseo_title($title) {
    if(  is_paged() ) {
		$paged = get_query_var('paged') ? get_query_var('paged') : 1;
        return $title.' Страница '.$paged;
    }
    else return $title;
}

add_filter('wpseo_metadesc', 'filter_product_wpseo_metadesc', 500);
function filter_product_wpseo_metadesc($desc) {
    if(  is_paged() ) {
		$paged = get_query_var('paged') ? get_query_var('paged') : 1;
        return $desc.' Страница '.$paged;
    }
    else return $desc;
}

WordPress отдает не 404, а 302

При запросе несуществующего адреса (напимер, «https://yoursite.com/вывывр») сайт выдает 302 ошибку. А должен выдавать 404 и высвечивать соответствующую страничку. 302 у нас ваще то «временно перемещен» или что-то там подобное.

В браузере же любую несуществующую страничку перекидывало на главную. Ну да, не очень правильно это, если вдуматься. А сеошника любого и того в жар бросит. Инфаркт микарда — во такой рубец!©

Первым делом смотрим .htaccess У чистого вордпресса должно быть что-то вроде

# BEGIN WordPress
<IfModule mod_rewrite.c>
RewriteEngine On
RewriteBase /
RewriteRule ^index\.php$ - [L]
RewriteCond %{REQUEST_FILENAME} !-f
RewriteCond %{REQUEST_FILENAME} !-d
RewriteRule . /index.php [L]
</IfModule>
# END WordPress

После сеошников может еще что-нибудь добавиться. Я например, нашел свои же вставки:

## Vash Webmaster edit start 2019.11.01
#### www -> without www
RewriteCond %{HTTP_HOST} ^www\.(.*)$
RewriteRule ^(.*)$ https://%1/$1 [L,R=301]

RewriteCond %{HTTPS} !=on
RewriteRule ^(.*)$ https://%{HTTP_HOST}/$1 [R=301,L]
## Vash Webmaster edit end 2019.11.01

Редирект c www и принудительный https. Тоже ничего криминального. Смотрим дальше.

За отображение страницы ошибки 404 в Вордпрессе отвечает файл 404.php в теме. Смотрим его и видим такую дичь:

Бинго! Редирект на домашнуюю страницу с любой несуществующей. Правим на что-нибудь адекватное

Правка шаблона ошибки 404.php  в wordpress

Проверяем:

Fixed

Яндекс.Метрика