MediaWiki 1.40.4
deleteEqualMessages.php
Go to the documentation of this file.
1<?php
26
27require_once __DIR__ . '/Maintenance.php';
28
36 public function __construct() {
37 parent::__construct();
38 $this->addDescription( 'Deletes all pages in the MediaWiki namespace that are equal to '
39 . 'the default message' );
40 $this->addOption( 'delete', 'Actually delete the pages (default: dry run)' );
41 $this->addOption( 'delete-talk', 'Don\'t leave orphaned talk pages behind during deletion' );
42 $this->addOption( 'lang-code', 'Check for subpages of this language code (default: root '
43 . 'page against content language). Use value "*" to run for all mwfile language code '
44 . 'subpages (including the base pages that override content language).', false, true );
45 }
46
51 protected function fetchMessageInfo( $langCode, array &$messageInfo ) {
52 $services = MediaWikiServices::getInstance();
53 $contLang = $services->getContentLanguage();
54 if ( $langCode ) {
55 $this->output( "\n... fetching message info for language: $langCode" );
56 $nonContentLanguage = true;
57 } else {
58 $this->output( "\n... fetching message info for content language" );
59 $langCode = $contLang->getCode();
60 $nonContentLanguage = false;
61 }
62
63 /* Based on SpecialAllmessages::reallyDoQuery #filter=modified */
64
65 $l10nCache = $services->getLocalisationCache();
66 $messageNames = $l10nCache->getSubitemList( 'en', 'messages' );
67 // Normalise message names for NS_MEDIAWIKI page_title
68 $messageNames = array_map( [ $contLang, 'ucfirst' ], $messageNames );
69
71 $messageNames,
72 $langCode,
73 $nonContentLanguage,
74 $this->getDB( DB_REPLICA )
75 );
76 // getCustomisedStatuses is stripping the sub page from the page titles, add it back
77 $titleSuffix = $nonContentLanguage ? "/$langCode" : '';
78
79 foreach ( $messageNames as $key ) {
80 $customised = isset( $statuses['pages'][$key] );
81 if ( $customised ) {
82 $actual = wfMessage( $key )->inLanguage( $langCode )->plain();
83 $default = wfMessage( $key )->inLanguage( $langCode )->useDatabase( false )->plain();
84
85 $messageInfo['relevantPages']++;
86
87 if (
88 // Exclude messages that are empty by default, such as sitenotice, specialpage
89 // summaries and accesskeys.
90 $default !== '' && $default !== '-' &&
91 $actual === $default
92 ) {
93 $hasTalk = isset( $statuses['talks'][$key] );
94 $messageInfo['results'][] = [
95 'title' => $key . $titleSuffix,
96 'hasTalk' => $hasTalk,
97 ];
98 $messageInfo['equalPages']++;
99 if ( $hasTalk ) {
100 $messageInfo['equalPagesTalks']++;
101 }
102 }
103 }
104 }
105 }
106
107 public function execute() {
108 $doDelete = $this->hasOption( 'delete' );
109 $doDeleteTalk = $this->hasOption( 'delete-talk' );
110 $langCode = $this->getOption( 'lang-code' );
111
112 $messageInfo = [
113 'relevantPages' => 0,
114 'equalPages' => 0,
115 'equalPagesTalks' => 0,
116 'results' => [],
117 ];
118
119 $this->output( 'Checking for pages with default message...' );
120
121 $services = MediaWikiServices::getInstance();
122 // Load message information
123 if ( $langCode ) {
124 $langCodes = $services->getLanguageNameUtils()
125 ->getLanguageNames( LanguageNameUtils::AUTONYMS, LanguageNameUtils::SUPPORTED );
126 if ( $langCode === '*' ) {
127 // All valid lang-code subpages in NS_MEDIAWIKI that
128 // override the messsages in that language
129 foreach ( $langCodes as $key => $value ) {
130 $this->fetchMessageInfo( $key, $messageInfo );
131 }
132 // Lastly, the base pages in NS_MEDIAWIKI that override
133 // messages in content language
134 $this->fetchMessageInfo( false, $messageInfo );
135 } else {
136 if ( !isset( $langCodes[$langCode] ) ) {
137 $this->fatalError( 'Invalid language code: ' . $langCode );
138 }
139 $this->fetchMessageInfo( $langCode, $messageInfo );
140 }
141 } else {
142 $this->fetchMessageInfo( false, $messageInfo );
143 }
144
145 if ( $messageInfo['equalPages'] === 0 ) {
146 // No more equal messages left
147 $this->output( "\ndone.\n" );
148
149 return;
150 }
151
152 $this->output( "\n{$messageInfo['relevantPages']} pages in the MediaWiki namespace "
153 . "override messages." );
154 $this->output( "\n{$messageInfo['equalPages']} pages are equal to the default message "
155 . "(+ {$messageInfo['equalPagesTalks']} talk pages).\n" );
156
157 if ( !$doDelete ) {
158 $list = '';
159 foreach ( $messageInfo['results'] as $result ) {
160 $title = Title::makeTitle( NS_MEDIAWIKI, $result['title'] );
161 $list .= "* [[$title]]\n";
162 if ( $result['hasTalk'] ) {
163 $title = Title::makeTitle( NS_MEDIAWIKI_TALK, $result['title'] );
164 $list .= "* [[$title]]\n";
165 }
166 }
167 $this->output( "\nList:\n$list\nRun the script again with --delete to delete these pages" );
168 if ( $messageInfo['equalPagesTalks'] !== 0 ) {
169 $this->output( " (include --delete-talk to also delete the talk pages)" );
170 }
171 $this->output( "\n" );
172
173 return;
174 }
175
176 $user = User::newSystemUser( 'MediaWiki default', [ 'steal' => true ] );
177 if ( !$user ) {
178 $this->fatalError( "Invalid username" );
179 }
180 StubGlobalUser::setUser( $user );
181
182 // Hide deletions from RecentChanges
183 $userGroupManager = $services->getUserGroupManager();
184 $userGroupManager->addUserToGroup( $user, 'bot' );
185
186 // Handle deletion
187 $this->output( "\n...deleting equal messages (this may take a long time!)..." );
188 $dbw = $this->getDB( DB_PRIMARY );
189 $wikiPageFactory = $services->getWikiPageFactory();
190 foreach ( $messageInfo['results'] as $result ) {
191 $this->waitForReplication();
192 $dbw->ping();
193 $title = Title::makeTitle( NS_MEDIAWIKI, $result['title'] );
194 $this->output( "\n* [[$title]]" );
195 $page = $wikiPageFactory->newFromTitle( $title );
196 $status = $page->doDeleteArticleReal( 'No longer required', $user );
197 if ( !$status->isOK() ) {
198 $this->output( " (Failed!)" );
199 }
200 if ( $result['hasTalk'] && $doDeleteTalk ) {
201 $title = Title::makeTitle( NS_MEDIAWIKI_TALK, $result['title'] );
202 $this->output( "\n* [[$title]]" );
203 $page = $wikiPageFactory->newFromTitle( $title );
204 $status = $page->doDeleteArticleReal(
205 'Orphaned talk page of no longer required message',
206 $user
207 );
208 if ( !$status->isOK() ) {
209 $this->output( " (Failed!)" );
210 }
211 }
212 }
213 $this->output( "\n\ndone!\n" );
214 }
215}
216
217$maintClass = DeleteEqualMessages::class;
218require_once RUN_MAINTENANCE_IF_MAIN;
getDB()
const NS_MEDIAWIKI_TALK
Definition Defines.php:73
const NS_MEDIAWIKI
Definition Defines.php:72
wfMessage( $key,... $params)
This is the function for getting translated interface messages.
static getCustomisedStatuses( $messageNames, $langcode='en', $foreign=false, IDatabase $dbr=null)
Determine which of the MediaWiki and MediaWiki_talk namespace pages exist.
Maintenance script that deletes all pages in the MediaWiki namespace of which the content is equal to...
__construct()
Default constructor.
execute()
Do the actual work.
fetchMessageInfo( $langCode, array &$messageInfo)
Abstract maintenance class for quickly writing and churning out maintenance scripts with minimal effo...
output( $out, $channel=null)
Throw some output to the user.
waitForReplication()
Wait for replica DBs to catch up.
hasOption( $name)
Checks to see if a particular option was set.
addDescription( $text)
Set the description text.
addOption( $name, $description, $required=false, $withArg=false, $shortName=false, $multiOccurrence=false)
Add a parameter to the script.
getOption( $name, $default=null)
Get an option, or return the default.
fatalError( $msg, $exitCode=1)
Output a message and terminate the current script.
A service that provides utilities to do with language names and codes.
Service locator for MediaWiki core services.
Stub object for the global user ($wgUser) that makes it possible to change the relevant underlying ob...
Represents a title within MediaWiki.
Definition Title.php:82
static newSystemUser( $name, $options=[])
Static factory method for creation of a "system" user from username.
Definition User.php:793
const DB_REPLICA
Definition defines.php:26
const DB_PRIMARY
Definition defines.php:28