Post being redirected to attachment post

The question:

That’s strange issue I have on this one website. Similar to this problem and I am trying to get this fixed from last couple of weeks.

This website have latest WordPress installed and /%postname%/ permalink structure.

Sometimes, when I try to view a post, I get redirected to attachment post /?attachment_id=358 instead of the post itself. This happens on posts when post and the attachment have the same title.

Although I can make the post working again if I change the permalink of the post. But that’s too late since this website publish and share posts automatically. And users get error when they do not get to the right post.

I already tried disabling all plugins for a while. None worked.

The Solutions:

Below are the methods you can try. The first solution is probably the best. Try others if the first one doesn’t work. Senior developers aren’t just copying/pasting – they read the methods carefully & apply them wisely to each case.

Method 1

I think the reason for this is the get_page_by_path() check inside the WP_Query, because it’s checking for both page and attachment slugs, in this case.

So if we have both an attachment and a post with the someslug postname and visit:

example.tdl/someslug

then get_page_by_path() will generate the following query:

SELECT ID, post_name, post_parent, post_type
FROM wp_posts
WHERE post_name IN ('someslug')
AND post_type IN ('page','attachment')

and we’re therefore served the attachment’s page instead of the post’s page.

So this seems to be a core feature but not a bug generated by your plugins.

Workaround

Because of the permalink settings, the post request is handled like a page request.

We therefore have the pagename query variable constructed instead of the name query variable.

We can adjust the query variables, generated for the post request:

/**
 * Handle post requests correctly, in the case of the /%postname%/ permalink settings
 */
! is_admin() && add_filter( 'request', function( $query_vars )  
{
    if( 
           isset( $query_vars['pagename'] ) 
        && $post = get_page_by_path( $query_vars['pagename'], OBJECT, 'post' )
    ) {
        if( is_a( $post, 'WP_Post' ) )
        {
            // Add the 'name' query variable 
            $query_vars['name'] = $post->post_name;     

            // Unset the 'pagename' query variable
            unset( $query_vars['pagename'] );
        }
    }
    return $query_vars;
}, PHP_INT_MAX );

Here we check if a post exists with the pagename slug and if it exists, we construct the name query variable and finally unset the pagename query variable.

Now the post should be served instead of the attachment.


All methods was sourced from stackoverflow.com or stackexchange.com, is licensed under cc by-sa 2.5, cc by-sa 3.0 and cc by-sa 4.0

Leave a Comment