deleteEqualMessages.php 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. <?php
  2. /**
  3. * This program is free software; you can redistribute it and/or modify
  4. * it under the terms of the GNU General Public License as published by
  5. * the Free Software Foundation; either version 2 of the License, or
  6. * (at your option) any later version.
  7. *
  8. * This program is distributed in the hope that it will be useful,
  9. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. * GNU General Public License for more details.
  12. *
  13. * You should have received a copy of the GNU General Public License along
  14. * with this program; if not, write to the Free Software Foundation, Inc.,
  15. * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
  16. * http://www.gnu.org/copyleft/gpl.html
  17. *
  18. * @file
  19. * @ingroup Maintenance
  20. */
  21. require_once __DIR__ . '/Maintenance.php';
  22. /**
  23. * Maintenance script that deletes all pages in the MediaWiki namespace
  24. * of which the content is equal to the system default.
  25. *
  26. * @ingroup Maintenance
  27. */
  28. class DeleteEqualMessages extends Maintenance {
  29. public function __construct() {
  30. parent::__construct();
  31. $this->addDescription( 'Deletes all pages in the MediaWiki namespace that are equal to '
  32. . 'the default message' );
  33. $this->addOption( 'delete', 'Actually delete the pages (default: dry run)' );
  34. $this->addOption( 'delete-talk', 'Don\'t leave orphaned talk pages behind during deletion' );
  35. $this->addOption( 'lang-code', 'Check for subpages of this language code (default: root '
  36. . 'page against content language). Use value "*" to run for all mwfile language code '
  37. . 'subpages (including the base pages that override content language).', false, true );
  38. }
  39. /**
  40. * @param string|bool $langCode See --lang-code option.
  41. * @param array &$messageInfo
  42. */
  43. protected function fetchMessageInfo( $langCode, array &$messageInfo ) {
  44. global $wgContLang;
  45. if ( $langCode ) {
  46. $this->output( "\n... fetching message info for language: $langCode" );
  47. $nonContLang = true;
  48. } else {
  49. $this->output( "\n... fetching message info for content language" );
  50. $langCode = $wgContLang->getCode();
  51. $nonContLang = false;
  52. }
  53. /* Based on SpecialAllmessages::reallyDoQuery #filter=modified */
  54. $l10nCache = Language::getLocalisationCache();
  55. $messageNames = $l10nCache->getSubitemList( 'en', 'messages' );
  56. // Normalise message names for NS_MEDIAWIKI page_title
  57. $messageNames = array_map( [ $wgContLang, 'ucfirst' ], $messageNames );
  58. $statuses = AllMessagesTablePager::getCustomisedStatuses(
  59. $messageNames, $langCode, $nonContLang );
  60. // getCustomisedStatuses is stripping the sub page from the page titles, add it back
  61. $titleSuffix = $nonContLang ? "/$langCode" : '';
  62. foreach ( $messageNames as $key ) {
  63. $customised = isset( $statuses['pages'][$key] );
  64. if ( $customised ) {
  65. $actual = wfMessage( $key )->inLanguage( $langCode )->plain();
  66. $default = wfMessage( $key )->inLanguage( $langCode )->useDatabase( false )->plain();
  67. $messageInfo['relevantPages']++;
  68. if (
  69. // Exclude messages that are empty by default, such as sitenotice, specialpage
  70. // summaries and accesskeys.
  71. $default !== '' && $default !== '-' &&
  72. $actual === $default
  73. ) {
  74. $hasTalk = isset( $statuses['talks'][$key] );
  75. $messageInfo['results'][] = [
  76. 'title' => $key . $titleSuffix,
  77. 'hasTalk' => $hasTalk,
  78. ];
  79. $messageInfo['equalPages']++;
  80. if ( $hasTalk ) {
  81. $messageInfo['equalPagesTalks']++;
  82. }
  83. }
  84. }
  85. }
  86. }
  87. public function execute() {
  88. $doDelete = $this->hasOption( 'delete' );
  89. $doDeleteTalk = $this->hasOption( 'delete-talk' );
  90. $langCode = $this->getOption( 'lang-code' );
  91. $messageInfo = [
  92. 'relevantPages' => 0,
  93. 'equalPages' => 0,
  94. 'equalPagesTalks' => 0,
  95. 'results' => [],
  96. ];
  97. $this->output( 'Checking for pages with default message...' );
  98. // Load message information
  99. if ( $langCode ) {
  100. $langCodes = Language::fetchLanguageNames( null, 'mwfile' );
  101. if ( $langCode === '*' ) {
  102. // All valid lang-code subpages in NS_MEDIAWIKI that
  103. // override the messsages in that language
  104. foreach ( $langCodes as $key => $value ) {
  105. $this->fetchMessageInfo( $key, $messageInfo );
  106. }
  107. // Lastly, the base pages in NS_MEDIAWIKI that override
  108. // messages in content language
  109. $this->fetchMessageInfo( false, $messageInfo );
  110. } else {
  111. if ( !isset( $langCodes[$langCode] ) ) {
  112. $this->error( 'Invalid language code: ' . $langCode, 1 );
  113. }
  114. $this->fetchMessageInfo( $langCode, $messageInfo );
  115. }
  116. } else {
  117. $this->fetchMessageInfo( false, $messageInfo );
  118. }
  119. if ( $messageInfo['equalPages'] === 0 ) {
  120. // No more equal messages left
  121. $this->output( "\ndone.\n" );
  122. return;
  123. }
  124. $this->output( "\n{$messageInfo['relevantPages']} pages in the MediaWiki namespace "
  125. . "override messages." );
  126. $this->output( "\n{$messageInfo['equalPages']} pages are equal to the default message "
  127. . "(+ {$messageInfo['equalPagesTalks']} talk pages).\n" );
  128. if ( !$doDelete ) {
  129. $list = '';
  130. foreach ( $messageInfo['results'] as $result ) {
  131. $title = Title::makeTitle( NS_MEDIAWIKI, $result['title'] );
  132. $list .= "* [[$title]]\n";
  133. if ( $result['hasTalk'] ) {
  134. $title = Title::makeTitle( NS_MEDIAWIKI_TALK, $result['title'] );
  135. $list .= "* [[$title]]\n";
  136. }
  137. }
  138. $this->output( "\nList:\n$list\nRun the script again with --delete to delete these pages" );
  139. if ( $messageInfo['equalPagesTalks'] !== 0 ) {
  140. $this->output( " (include --delete-talk to also delete the talk pages)" );
  141. }
  142. $this->output( "\n" );
  143. return;
  144. }
  145. $user = User::newSystemUser( 'MediaWiki default', [ 'steal' => true ] );
  146. if ( !$user ) {
  147. $this->error( "Invalid username", true );
  148. }
  149. global $wgUser;
  150. $wgUser = $user;
  151. // Hide deletions from RecentChanges
  152. $user->addGroup( 'bot' );
  153. // Handle deletion
  154. $this->output( "\n...deleting equal messages (this may take a long time!)..." );
  155. $dbw = $this->getDB( DB_MASTER );
  156. foreach ( $messageInfo['results'] as $result ) {
  157. wfWaitForSlaves();
  158. $dbw->ping();
  159. $title = Title::makeTitle( NS_MEDIAWIKI, $result['title'] );
  160. $this->output( "\n* [[$title]]" );
  161. $page = WikiPage::factory( $title );
  162. $error = ''; // Passed by ref
  163. $success = $page->doDeleteArticle( 'No longer required', false, 0, true, $error, $user );
  164. if ( !$success ) {
  165. $this->output( " (Failed!)" );
  166. }
  167. if ( $result['hasTalk'] && $doDeleteTalk ) {
  168. $title = Title::makeTitle( NS_MEDIAWIKI_TALK, $result['title'] );
  169. $this->output( "\n* [[$title]]" );
  170. $page = WikiPage::factory( $title );
  171. $error = ''; // Passed by ref
  172. $success = $page->doDeleteArticle( 'Orphaned talk page of no longer required message',
  173. false, 0, true, $error, $user );
  174. if ( !$success ) {
  175. $this->output( " (Failed!)" );
  176. }
  177. }
  178. }
  179. $this->output( "\n\ndone!\n" );
  180. }
  181. }
  182. $maintClass = "DeleteEqualMessages";
  183. require_once RUN_MAINTENANCE_IF_MAIN;