wikimedia/mediawiki-core

View on GitHub
maintenance/deleteEqualMessages.php

Summary

Maintainability
C
1 day
Test Coverage
<?php
/**
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * 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, write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
 * http://www.gnu.org/copyleft/gpl.html
 *
 * @file
 * @ingroup Maintenance
 */

use MediaWiki\Languages\LanguageNameUtils;
use MediaWiki\Pager\AllMessagesTablePager;
use MediaWiki\StubObject\StubGlobalUser;
use MediaWiki\Title\Title;
use MediaWiki\User\User;

require_once __DIR__ . '/Maintenance.php';

/**
 * Maintenance script that deletes all pages in the MediaWiki namespace
 * of which the content is equal to the system default.
 *
 * @ingroup Maintenance
 */
class DeleteEqualMessages extends Maintenance {
    public function __construct() {
        parent::__construct();
        $this->addDescription( 'Deletes all pages in the MediaWiki namespace that are equal to '
            . 'the default message' );
        $this->addOption( 'delete', 'Actually delete the pages (default: dry run)' );
        $this->addOption( 'delete-talk', 'Don\'t leave orphaned talk pages behind during deletion' );
        $this->addOption( 'lang-code', 'Check for subpages of this language code (default: root '
            . 'page against content language). Use value "*" to run for all mwfile language code '
            . 'subpages (including the base pages that override content language).', false, true );
    }

    /**
     * @param string|false $langCode See --lang-code option.
     * @param array &$messageInfo
     */
    protected function fetchMessageInfo( $langCode, array &$messageInfo ) {
        $services = $this->getServiceContainer();
        $contLang = $services->getContentLanguage();
        if ( $langCode ) {
            $this->output( "\n... fetching message info for language: $langCode" );
            $nonContentLanguage = true;
        } else {
            $this->output( "\n... fetching message info for content language" );
            $langCode = $contLang->getCode();
            $nonContentLanguage = false;
        }

        /* Based on SpecialAllmessages::reallyDoQuery #filter=modified */

        $l10nCache = $services->getLocalisationCache();
        $messageNames = $l10nCache->getSubitemList( 'en', 'messages' );
        // Normalise message names for NS_MEDIAWIKI page_title
        $messageNames = array_map( [ $contLang, 'ucfirst' ], $messageNames );

        $statuses = AllMessagesTablePager::getCustomisedStatuses(
            $messageNames,
            $langCode,
            $nonContentLanguage,
            $this->getReplicaDB()
        );
        // getCustomisedStatuses is stripping the sub page from the page titles, add it back
        $titleSuffix = $nonContentLanguage ? "/$langCode" : '';

        foreach ( $messageNames as $key ) {
            $customised = isset( $statuses['pages'][$key] );
            if ( $customised ) {
                $actual = wfMessage( $key )->inLanguage( $langCode )->plain();
                $default = wfMessage( $key )->inLanguage( $langCode )->useDatabase( false )->plain();

                $messageInfo['relevantPages']++;

                if (
                    // Exclude messages that are empty by default, such as sitenotice, specialpage
                    // summaries and accesskeys.
                    $default !== '' && $default !== '-' &&
                        $actual === $default
                ) {
                    $hasTalk = isset( $statuses['talks'][$key] );
                    $messageInfo['results'][] = [
                        'title' => $key . $titleSuffix,
                        'hasTalk' => $hasTalk,
                    ];
                    $messageInfo['equalPages']++;
                    if ( $hasTalk ) {
                        $messageInfo['equalPagesTalks']++;
                    }
                }
            }
        }
    }

    public function execute() {
        $doDelete = $this->hasOption( 'delete' );
        $doDeleteTalk = $this->hasOption( 'delete-talk' );
        $langCode = $this->getOption( 'lang-code' );

        $messageInfo = [
            'relevantPages' => 0,
            'equalPages' => 0,
            'equalPagesTalks' => 0,
            'results' => [],
        ];

        $this->output( 'Checking for pages with default message...' );

        $services = $this->getServiceContainer();
        // Load message information
        if ( $langCode ) {
            $langCodes = $services->getLanguageNameUtils()
                ->getLanguageNames( LanguageNameUtils::AUTONYMS, LanguageNameUtils::SUPPORTED );
            if ( $langCode === '*' ) {
                // All valid lang-code subpages in NS_MEDIAWIKI that
                // override the messages in that language
                foreach ( $langCodes as $key => $value ) {
                    $this->fetchMessageInfo( $key, $messageInfo );
                }
                // Lastly, the base pages in NS_MEDIAWIKI that override
                // messages in content language
                $this->fetchMessageInfo( false, $messageInfo );
            } else {
                if ( !isset( $langCodes[$langCode] ) ) {
                    $this->fatalError( 'Invalid language code: ' . $langCode );
                }
                $this->fetchMessageInfo( $langCode, $messageInfo );
            }
        } else {
            $this->fetchMessageInfo( false, $messageInfo );
        }

        if ( $messageInfo['equalPages'] === 0 ) {
            // No more equal messages left
            $this->output( "\ndone.\n" );

            return;
        }

        $this->output( "\n{$messageInfo['relevantPages']} pages in the MediaWiki namespace "
            . "override messages." );
        $this->output( "\n{$messageInfo['equalPages']} pages are equal to the default message "
            . "(+ {$messageInfo['equalPagesTalks']} talk pages).\n" );

        if ( !$doDelete ) {
            $list = '';
            foreach ( $messageInfo['results'] as $result ) {
                $title = Title::makeTitle( NS_MEDIAWIKI, $result['title'] );
                $list .= "* [[$title]]\n";
                if ( $result['hasTalk'] ) {
                    $title = Title::makeTitle( NS_MEDIAWIKI_TALK, $result['title'] );
                    $list .= "* [[$title]]\n";
                }
            }
            $this->output( "\nList:\n$list\nRun the script again with --delete to delete these pages" );
            if ( $messageInfo['equalPagesTalks'] !== 0 ) {
                $this->output( " (include --delete-talk to also delete the talk pages)" );
            }
            $this->output( "\n" );

            return;
        }

        $user = User::newSystemUser( 'MediaWiki default', [ 'steal' => true ] );
        if ( !$user ) {
            $this->fatalError( "Invalid username" );
        }
        StubGlobalUser::setUser( $user );

        // Hide deletions from RecentChanges
        $userGroupManager = $services->getUserGroupManager();
        $userGroupManager->addUserToGroup( $user, 'bot' );

        // Handle deletion
        $this->output( "\n...deleting equal messages (this may take a long time!)..." );
        $dbw = $this->getPrimaryDB();
        $wikiPageFactory = $services->getWikiPageFactory();
        $delPageFactory = $services->getDeletePageFactory();
        foreach ( $messageInfo['results'] as $result ) {
            $this->waitForReplication();
            $dbw->ping();
            $title = Title::makeTitle( NS_MEDIAWIKI, $result['title'] );
            $this->output( "\n* [[$title]]" );
            $page = $wikiPageFactory->newFromTitle( $title );
            $status = $delPageFactory->newDeletePage( $page, $user )->deleteUnsafe( 'No longer required' );
            if ( !$status->isOK() ) {
                $this->output( " (Failed!)" );
            }
            if ( $result['hasTalk'] && $doDeleteTalk ) {
                $title = Title::makeTitle( NS_MEDIAWIKI_TALK, $result['title'] );
                $this->output( "\n* [[$title]]" );
                $page = $wikiPageFactory->newFromTitle( $title );
                $status = $delPageFactory->newDeletePage( $page, $user )
                    ->deleteUnsafe( 'Orphaned talk page of no longer required message' );
                if ( !$status->isOK() ) {
                    $this->output( " (Failed!)" );
                }
            }
        }
        $this->output( "\n\ndone!\n" );
    }
}

$maintClass = DeleteEqualMessages::class;
require_once RUN_MAINTENANCE_IF_MAIN;