FileDeleteForm.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490
  1. <?php
  2. /**
  3. * File deletion user interface.
  4. *
  5. * This program is free software; you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License as published by
  7. * the Free Software Foundation; either version 2 of the License, or
  8. * (at your option) any later version.
  9. *
  10. * This program is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License along
  16. * with this program; if not, write to the Free Software Foundation, Inc.,
  17. * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
  18. * http://www.gnu.org/copyleft/gpl.html
  19. *
  20. * @file
  21. * @author Rob Church <robchur@gmail.com>
  22. * @ingroup Media
  23. */
  24. use MediaWiki\MediaWikiServices;
  25. /**
  26. * File deletion user interface
  27. *
  28. * @ingroup Media
  29. */
  30. class FileDeleteForm {
  31. /**
  32. * @var Title
  33. */
  34. private $title = null;
  35. /**
  36. * @var LocalFile
  37. */
  38. private $file = null;
  39. /**
  40. * @var LocalFile
  41. */
  42. private $oldfile = null;
  43. private $oldimage = '';
  44. /**
  45. * @param LocalFile $file File object we're deleting
  46. */
  47. public function __construct( $file ) {
  48. $this->title = $file->getTitle();
  49. $this->file = $file;
  50. }
  51. /**
  52. * Fulfil the request; shows the form or deletes the file,
  53. * pending authentication, confirmation, etc.
  54. */
  55. public function execute() {
  56. global $wgOut, $wgRequest, $wgUser, $wgUploadMaintenance;
  57. $permissionErrors = $this->title->getUserPermissionsErrors( 'delete', $wgUser );
  58. if ( count( $permissionErrors ) ) {
  59. throw new PermissionsError( 'delete', $permissionErrors );
  60. }
  61. if ( wfReadOnly() ) {
  62. throw new ReadOnlyError;
  63. }
  64. if ( $wgUploadMaintenance ) {
  65. throw new ErrorPageError( 'filedelete-maintenance-title', 'filedelete-maintenance' );
  66. }
  67. $this->setHeaders();
  68. $this->oldimage = $wgRequest->getText( 'oldimage', false );
  69. $token = $wgRequest->getText( 'wpEditToken' );
  70. # Flag to hide all contents of the archived revisions
  71. $permissionManager = MediaWikiServices::getInstance()->getPermissionManager();
  72. $suppress = $wgRequest->getCheck( 'wpSuppress' ) &&
  73. $permissionManager->userHasRight( $wgUser, 'suppressrevision' );
  74. if ( $this->oldimage ) {
  75. $this->oldfile = RepoGroup::singleton()->getLocalRepo()->newFromArchiveName(
  76. $this->title,
  77. $this->oldimage
  78. );
  79. }
  80. if ( !self::haveDeletableFile( $this->file, $this->oldfile, $this->oldimage ) ) {
  81. $wgOut->addHTML( $this->prepareMessage( 'filedelete-nofile' ) );
  82. $wgOut->addReturnTo( $this->title );
  83. return;
  84. }
  85. // Perform the deletion if appropriate
  86. if ( $wgRequest->wasPosted() && $wgUser->matchEditToken( $token, $this->oldimage ) ) {
  87. $deleteReasonList = $wgRequest->getText( 'wpDeleteReasonList' );
  88. $deleteReason = $wgRequest->getText( 'wpReason' );
  89. if ( $deleteReasonList == 'other' ) {
  90. $reason = $deleteReason;
  91. } elseif ( $deleteReason != '' ) {
  92. // Entry from drop down menu + additional comment
  93. $reason = $deleteReasonList . wfMessage( 'colon-separator' )
  94. ->inContentLanguage()->text() . $deleteReason;
  95. } else {
  96. $reason = $deleteReasonList;
  97. }
  98. $status = self::doDelete(
  99. $this->title,
  100. $this->file,
  101. $this->oldimage,
  102. $reason,
  103. $suppress,
  104. $wgUser
  105. );
  106. if ( !$status->isGood() ) {
  107. $wgOut->addHTML( '<h2>' . $this->prepareMessage( 'filedeleteerror-short' ) . "</h2>\n" );
  108. $wgOut->wrapWikiTextAsInterface(
  109. 'error',
  110. $status->getWikiText( 'filedeleteerror-short', 'filedeleteerror-long' )
  111. );
  112. }
  113. if ( $status->isOK() ) {
  114. $wgOut->setPageTitle( wfMessage( 'actioncomplete' ) );
  115. $wgOut->addHTML( $this->prepareMessage( 'filedelete-success' ) );
  116. // Return to the main page if we just deleted all versions of the
  117. // file, otherwise go back to the description page
  118. $wgOut->addReturnTo( $this->oldimage ? $this->title : Title::newMainPage() );
  119. WatchAction::doWatchOrUnwatch( $wgRequest->getCheck( 'wpWatch' ), $this->title, $wgUser );
  120. }
  121. return;
  122. }
  123. $this->showForm();
  124. $this->showLogEntries();
  125. }
  126. /**
  127. * Really delete the file
  128. *
  129. * @param Title &$title
  130. * @param LocalFile &$file
  131. * @param string &$oldimage Archive name
  132. * @param string $reason Reason of the deletion
  133. * @param bool $suppress Whether to mark all deleted versions as restricted
  134. * @param User|null $user User object performing the request
  135. * @param array $tags Tags to apply to the deletion action
  136. * @throws MWException
  137. * @return Status
  138. */
  139. public static function doDelete( &$title, &$file, &$oldimage, $reason,
  140. $suppress, User $user = null, $tags = []
  141. ) {
  142. if ( $user === null ) {
  143. global $wgUser;
  144. $user = $wgUser;
  145. }
  146. if ( $oldimage ) {
  147. $page = null;
  148. $status = $file->deleteOld( $oldimage, $reason, $suppress, $user );
  149. if ( $status->isOK() ) {
  150. // Need to do a log item
  151. $logComment = wfMessage( 'deletedrevision', $oldimage )->inContentLanguage()->text();
  152. if ( trim( $reason ) != '' ) {
  153. $logComment .= wfMessage( 'colon-separator' )
  154. ->inContentLanguage()->text() . $reason;
  155. }
  156. $logtype = $suppress ? 'suppress' : 'delete';
  157. $logEntry = new ManualLogEntry( $logtype, 'delete' );
  158. $logEntry->setPerformer( $user );
  159. $logEntry->setTarget( $title );
  160. $logEntry->setComment( $logComment );
  161. $logEntry->addTags( $tags );
  162. $logid = $logEntry->insert();
  163. $logEntry->publish( $logid );
  164. $status->value = $logid;
  165. }
  166. } else {
  167. $status = Status::newFatal( 'cannotdelete',
  168. wfEscapeWikiText( $title->getPrefixedText() )
  169. );
  170. $page = WikiPage::factory( $title );
  171. $dbw = wfGetDB( DB_MASTER );
  172. $dbw->startAtomic( __METHOD__ );
  173. // delete the associated article first
  174. $error = '';
  175. $deleteStatus = $page->doDeleteArticleReal( $reason, $suppress, 0, false, $error,
  176. $user, $tags );
  177. // doDeleteArticleReal() returns a non-fatal error status if the page
  178. // or revision is missing, so check for isOK() rather than isGood()
  179. if ( $deleteStatus->isOK() ) {
  180. $status = $file->delete( $reason, $suppress, $user );
  181. if ( $status->isOK() ) {
  182. if ( $deleteStatus->value === null ) {
  183. // No log ID from doDeleteArticleReal(), probably
  184. // because the page/revision didn't exist, so create
  185. // one here.
  186. $logtype = $suppress ? 'suppress' : 'delete';
  187. $logEntry = new ManualLogEntry( $logtype, 'delete' );
  188. $logEntry->setPerformer( $user );
  189. $logEntry->setTarget( clone $title );
  190. $logEntry->setComment( $reason );
  191. $logEntry->addTags( $tags );
  192. $logid = $logEntry->insert();
  193. $dbw->onTransactionPreCommitOrIdle(
  194. function () use ( $logEntry, $logid ) {
  195. $logEntry->publish( $logid );
  196. },
  197. __METHOD__
  198. );
  199. $status->value = $logid;
  200. } else {
  201. $status->value = $deleteStatus->value; // log id
  202. }
  203. $dbw->endAtomic( __METHOD__ );
  204. } else {
  205. // Page deleted but file still there? rollback page delete
  206. $lbFactory = MediaWikiServices::getInstance()->getDBLoadBalancerFactory();
  207. $lbFactory->rollbackMasterChanges( __METHOD__ );
  208. }
  209. } else {
  210. $dbw->endAtomic( __METHOD__ );
  211. }
  212. }
  213. if ( $status->isOK() ) {
  214. Hooks::run( 'FileDeleteComplete', [ &$file, &$oldimage, &$page, &$user, &$reason ] );
  215. }
  216. return $status;
  217. }
  218. /**
  219. * Show the confirmation form
  220. */
  221. private function showForm() {
  222. global $wgOut, $wgUser, $wgRequest;
  223. $permissionManager = MediaWikiServices::getInstance()->getPermissionManager();
  224. $wgOut->addModules( 'mediawiki.action.delete.file' );
  225. $checkWatch = $wgUser->getBoolOption( 'watchdeletion' ) || $wgUser->isWatched( $this->title );
  226. $wgOut->enableOOUI();
  227. $fields = [];
  228. $fields[] = new OOUI\LabelWidget( [ 'label' => new OOUI\HtmlSnippet(
  229. $this->prepareMessage( 'filedelete-intro' ) ) ]
  230. );
  231. $options = Xml::listDropDownOptions(
  232. $wgOut->msg( 'filedelete-reason-dropdown' )->inContentLanguage()->text(),
  233. [ 'other' => $wgOut->msg( 'filedelete-reason-otherlist' )->inContentLanguage()->text() ]
  234. );
  235. $options = Xml::listDropDownOptionsOoui( $options );
  236. $fields[] = new OOUI\FieldLayout(
  237. new OOUI\DropdownInputWidget( [
  238. 'name' => 'wpDeleteReasonList',
  239. 'inputId' => 'wpDeleteReasonList',
  240. 'tabIndex' => 1,
  241. 'infusable' => true,
  242. 'value' => '',
  243. 'options' => $options,
  244. ] ),
  245. [
  246. 'label' => $wgOut->msg( 'filedelete-comment' )->text(),
  247. 'align' => 'top',
  248. ]
  249. );
  250. // HTML maxlength uses "UTF-16 code units", which means that characters outside BMP
  251. // (e.g. emojis) count for two each. This limit is overridden in JS to instead count
  252. // Unicode codepoints.
  253. $fields[] = new OOUI\FieldLayout(
  254. new OOUI\TextInputWidget( [
  255. 'name' => 'wpReason',
  256. 'inputId' => 'wpReason',
  257. 'tabIndex' => 2,
  258. 'maxLength' => CommentStore::COMMENT_CHARACTER_LIMIT,
  259. 'infusable' => true,
  260. 'value' => $wgRequest->getText( 'wpReason' ),
  261. 'autofocus' => true,
  262. ] ),
  263. [
  264. 'label' => $wgOut->msg( 'filedelete-otherreason' )->text(),
  265. 'align' => 'top',
  266. ]
  267. );
  268. if ( $permissionManager->userHasRight( $wgUser, 'suppressrevision' ) ) {
  269. $fields[] = new OOUI\FieldLayout(
  270. new OOUI\CheckboxInputWidget( [
  271. 'name' => 'wpSuppress',
  272. 'inputId' => 'wpSuppress',
  273. 'tabIndex' => 3,
  274. 'selected' => false,
  275. ] ),
  276. [
  277. 'label' => $wgOut->msg( 'revdelete-suppress' )->text(),
  278. 'align' => 'inline',
  279. 'infusable' => true,
  280. ]
  281. );
  282. }
  283. if ( $wgUser->isLoggedIn() ) {
  284. $fields[] = new OOUI\FieldLayout(
  285. new OOUI\CheckboxInputWidget( [
  286. 'name' => 'wpWatch',
  287. 'inputId' => 'wpWatch',
  288. 'tabIndex' => 3,
  289. 'selected' => $checkWatch,
  290. ] ),
  291. [
  292. 'label' => $wgOut->msg( 'watchthis' )->text(),
  293. 'align' => 'inline',
  294. 'infusable' => true,
  295. ]
  296. );
  297. }
  298. $fields[] = new OOUI\FieldLayout(
  299. new OOUI\ButtonInputWidget( [
  300. 'name' => 'mw-filedelete-submit',
  301. 'inputId' => 'mw-filedelete-submit',
  302. 'tabIndex' => 4,
  303. 'value' => $wgOut->msg( 'filedelete-submit' )->text(),
  304. 'label' => $wgOut->msg( 'filedelete-submit' )->text(),
  305. 'flags' => [ 'primary', 'destructive' ],
  306. 'type' => 'submit',
  307. ] ),
  308. [
  309. 'align' => 'top',
  310. ]
  311. );
  312. $fieldset = new OOUI\FieldsetLayout( [
  313. 'label' => $wgOut->msg( 'filedelete-legend' )->text(),
  314. 'items' => $fields,
  315. ] );
  316. $form = new OOUI\FormLayout( [
  317. 'method' => 'post',
  318. 'action' => $this->getAction(),
  319. 'id' => 'mw-img-deleteconfirm',
  320. ] );
  321. $form->appendContent(
  322. $fieldset,
  323. new OOUI\HtmlSnippet(
  324. Html::hidden( 'wpEditToken', $wgUser->getEditToken( $this->oldimage ) )
  325. )
  326. );
  327. $wgOut->addHTML(
  328. new OOUI\PanelLayout( [
  329. 'classes' => [ 'deletepage-wrapper' ],
  330. 'expanded' => false,
  331. 'padded' => true,
  332. 'framed' => true,
  333. 'content' => $form,
  334. ] )
  335. );
  336. if ( $permissionManager->userHasRight( $wgUser, 'editinterface' ) ) {
  337. $linkRenderer = MediaWikiServices::getInstance()->getLinkRenderer();
  338. $link = $linkRenderer->makeKnownLink(
  339. $wgOut->msg( 'filedelete-reason-dropdown' )->inContentLanguage()->getTitle(),
  340. wfMessage( 'filedelete-edit-reasonlist' )->text(),
  341. [],
  342. [ 'action' => 'edit' ]
  343. );
  344. $wgOut->addHTML( '<p class="mw-filedelete-editreasons">' . $link . '</p>' );
  345. }
  346. }
  347. /**
  348. * Show deletion log fragments pertaining to the current file
  349. */
  350. private function showLogEntries() {
  351. global $wgOut;
  352. $deleteLogPage = new LogPage( 'delete' );
  353. $wgOut->addHTML( '<h2>' . $deleteLogPage->getName()->escaped() . "</h2>\n" );
  354. LogEventsList::showLogExtract( $wgOut, 'delete', $this->title );
  355. }
  356. /**
  357. * Prepare a message referring to the file being deleted,
  358. * showing an appropriate message depending upon whether
  359. * it's a current file or an old version
  360. *
  361. * @param string $message Message base
  362. * @return string
  363. */
  364. private function prepareMessage( $message ) {
  365. global $wgLang;
  366. if ( $this->oldimage ) {
  367. # Message keys used:
  368. # 'filedelete-intro-old', 'filedelete-nofile-old', 'filedelete-success-old'
  369. return wfMessage(
  370. "{$message}-old",
  371. wfEscapeWikiText( $this->title->getText() ),
  372. $wgLang->date( $this->getTimestamp(), true ),
  373. $wgLang->time( $this->getTimestamp(), true ),
  374. wfExpandUrl( $this->file->getArchiveUrl( $this->oldimage ), PROTO_CURRENT ) )->parseAsBlock();
  375. } else {
  376. return wfMessage(
  377. $message,
  378. wfEscapeWikiText( $this->title->getText() )
  379. )->parseAsBlock();
  380. }
  381. }
  382. /**
  383. * Set headers, titles and other bits
  384. */
  385. private function setHeaders() {
  386. global $wgOut;
  387. $wgOut->setPageTitle( wfMessage( 'filedelete', $this->title->getText() ) );
  388. $wgOut->setRobotPolicy( 'noindex,nofollow' );
  389. $wgOut->addBacklinkSubtitle( $this->title );
  390. }
  391. /**
  392. * Is the provided `oldimage` value valid?
  393. *
  394. * @param string $oldimage
  395. * @return bool
  396. */
  397. public static function isValidOldSpec( $oldimage ) {
  398. return strlen( $oldimage ) >= 16
  399. && strpos( $oldimage, '/' ) === false
  400. && strpos( $oldimage, '\\' ) === false;
  401. }
  402. /**
  403. * Could we delete the file specified? If an `oldimage`
  404. * value was provided, does it correspond to an
  405. * existing, local, old version of this file?
  406. *
  407. * @param LocalFile &$file
  408. * @param LocalFile &$oldfile
  409. * @param LocalFile $oldimage
  410. * @return bool
  411. */
  412. public static function haveDeletableFile( &$file, &$oldfile, $oldimage ) {
  413. return $oldimage
  414. ? $oldfile && $oldfile->exists() && $oldfile->isLocal()
  415. : $file && $file->exists() && $file->isLocal();
  416. }
  417. /**
  418. * Prepare the form action
  419. *
  420. * @return string
  421. */
  422. private function getAction() {
  423. $q = [];
  424. $q['action'] = 'delete';
  425. if ( $this->oldimage ) {
  426. $q['oldimage'] = $this->oldimage;
  427. }
  428. return $this->title->getLocalURL( $q );
  429. }
  430. /**
  431. * Extract the timestamp of the old version
  432. *
  433. * @return string
  434. */
  435. private function getTimestamp() {
  436. return $this->oldfile->getTimestamp();
  437. }
  438. }