67 private $isDeletePageUnitTest =
false;
69 private $suppress =
false;
73 private $logSubtype =
'delete';
75 private $forceImmediate =
false;
77 private $associatedTalk;
80 private $legacyHookErrors =
'';
82 private $mergeLegacyHookErrors =
true;
88 private $successfulDeletionsIDs;
93 private $wasScheduled;
95 private $attemptedDeletion =
false;
105 private string $localWikiID;
106 private string $webRequestID;
129 string $webRequestID,
139 $this->hookRunner =
new HookRunner( $hookContainer );
140 $this->eventDispatcher = $eventDispatcher;
141 $this->revisionStore = $revisionStore;
142 $this->lbFactory = $lbFactory;
143 $this->jobQueueGroup = $jobQueueGroup;
144 $this->commentStore = $commentStore;
146 $this->options = $serviceOptions;
147 $this->recentDeletesCache = $recentDeletesCache;
148 $this->localWikiID = $localWikiID;
149 $this->webRequestID = $webRequestID;
150 $this->wikiPageFactory = $wikiPageFactory;
151 $this->userFactory = $userFactory;
152 $this->backlinkCacheFactory = $backlinkCacheFactory;
153 $this->namespaceInfo = $namespaceInfo;
154 $this->contLangMsgTextFormatter = $contLangMsgTextFormatter;
157 $this->deleter = $deleter;
158 $this->redirectStore = $redirectStore;
166 return $this->legacyHookErrors;
174 $this->mergeLegacyHookErrors = false;
186 $this->suppress = $suppress;
196 public function setTags( array $tags ): self {
208 $this->logSubtype = $logSubtype;
219 $this->forceImmediate = $forceImmediate;
228 if ( $this->namespaceInfo->isTalk( $this->page->getNamespace() ) ) {
229 return StatusValue::newFatal(
'delete-error-associated-alreadytalk' );
232 $talkPage = $this->wikiPageFactory->newFromLinkTarget(
233 $this->namespaceInfo->getTalkPage( $this->page->getTitle() )
235 if ( !$talkPage->exists() ) {
236 return StatusValue::newFatal(
'delete-error-associated-doesnotexist' );
238 return StatusValue::newGood();
254 $this->associatedTalk =
null;
258 if ( $this->namespaceInfo->isTalk( $this->page->getNamespace() ) ) {
259 throw new BadMethodCallException(
"Cannot delete associated talk page of a talk page! ($this->page)" );
262 $this->associatedTalk = $this->wikiPageFactory->newFromLinkTarget(
263 $this->namespaceInfo->getTalkPage( $this->page->getTitle() )
274 if ( !defined(
'MW_PHPUNIT_TEST' ) ) {
275 throw new LogicException( __METHOD__ .
' can only be used in tests!' );
277 $this->isDeletePageUnitTest = $test;
286 $this->attemptedDeletion = true;
287 $this->successfulDeletionsIDs = [ self::PAGE_BASE => null ];
288 $this->wasScheduled = [ self::PAGE_BASE => null ];
289 if ( $this->associatedTalk ) {
290 $this->successfulDeletionsIDs[self::PAGE_TALK] =
null;
291 $this->wasScheduled[self::PAGE_TALK] =
null;
300 private function assertDeletionAttempted(): void {
301 if ( !$this->attemptedDeletion ) {
302 throw new BadMethodCallException(
'No deletion was attempted' );
311 $this->assertDeletionAttempted();
312 return $this->successfulDeletionsIDs;
320 $this->assertDeletionAttempted();
321 return $this->wasScheduled;
331 $this->setDeletionAttempted();
332 $status = $this->authorizeDeletion();
333 if ( !$status->isGood() ) {
337 return $this->deleteUnsafe( $reason );
342 $this->deleter->authorizeWrite(
'delete', $this->page, $status );
343 if ( $this->associatedTalk ) {
344 $this->deleter->authorizeWrite(
'delete', $this->associatedTalk, $status );
346 if ( !$this->deleter->isAllowed(
'bigdelete' ) && $this->isBigDeletion() ) {
348 'delete-toomanyrevisions',
349 Message::numParam( $this->options->get( MainConfigNames::DeleteRevisionsLimit ) )
353 $status->merge( ChangeTags::canAddTagsAccompanyingChange( $this->tags, $this->deleter ) );
358 private function isBigDeletion(): bool {
359 $revLimit = $this->options->get( MainConfigNames::DeleteRevisionsLimit );
364 $dbr = $this->lbFactory->getReplicaDatabase();
365 $revCount = $this->revisionStore->countRevisionsByPageId( $dbr, $this->page->getId() );
366 if ( $this->associatedTalk ) {
367 $revCount += $this->revisionStore->countRevisionsByPageId( $dbr, $this->associatedTalk->getId() );
370 return $revCount > $revLimit;
386 $dbr = $this->lbFactory->getReplicaDatabase();
387 $revCount = $this->revisionStore->countRevisionsByPageId( $dbr, $this->page->getId() );
388 $revCount += $safetyMargin;
390 if ( $revCount >= $this->options->get( MainConfigNames::DeleteRevisionsBatchSize ) ) {
392 } elseif ( !$this->associatedTalk ) {
396 $talkRevCount = $this->revisionStore->countRevisionsByPageId( $dbr, $this->associatedTalk->getId() );
397 $talkRevCount += $safetyMargin;
399 return $talkRevCount >= $this->options->get( MainConfigNames::DeleteRevisionsBatchSize );
413 $this->setDeletionAttempted();
414 $origReason = $reason;
415 $hookStatus = $this->runPreDeleteHooks( $this->page, $reason );
416 if ( !$hookStatus->isGood() ) {
419 if ( $this->associatedTalk ) {
420 $talkReason = $this->contLangMsgTextFormatter->format(
421 MessageValue::new(
'delete-talk-summary-prefix' )->plaintextParams( $origReason )
423 $talkHookStatus = $this->runPreDeleteHooks( $this->associatedTalk, $talkReason );
424 if ( !$talkHookStatus->isGood() ) {
425 return $talkHookStatus;
429 $status = $this->deleteInternal( $this->page, self::PAGE_BASE, $reason );
430 if ( !$this->associatedTalk || !$status->isGood() ) {
437 $status->merge( $this->deleteInternal( $this->associatedTalk, self::PAGE_TALK, $talkReason ) );
446 private function runPreDeleteHooks( WikiPage $page,
string &$reason ): Status {
447 $status = Status::newGood();
449 $legacyDeleter = $this->userFactory->newFromAuthority( $this->deleter );
450 if ( !$this->hookRunner->onArticleDelete(
451 $page, $legacyDeleter, $reason, $this->legacyHookErrors, $status, $this->suppress )
453 if ( $this->mergeLegacyHookErrors && $this->legacyHookErrors !==
'' ) {
454 if ( is_string( $this->legacyHookErrors ) ) {
455 $this->legacyHookErrors = [ $this->legacyHookErrors ];
457 foreach ( $this->legacyHookErrors as $legacyError ) {
458 $status->fatal(
new RawMessage( $legacyError ) );
461 if ( $status->isOK() ) {
463 $status->fatal(
'delete-hook-aborted' );
469 $status = Status::newGood();
470 $hookRes = $this->hookRunner->onPageDelete( $page, $this->deleter, $reason, $status, $this->suppress );
471 if ( !$hookRes && !$status->isGood() ) {
475 return Status::newGood();
497 ?
string $webRequestId =
null,
501 $status = Status::newGood();
503 $dbw = $this->lbFactory->getPrimaryDatabase();
504 $dbw->startAtomic( __METHOD__ );
507 $id = $page->
getId();
513 if ( $id === 0 || $page->
getLatest() !== $lockedLatest ) {
514 $dbw->endAtomic( __METHOD__ );
516 $status->error(
'cannotdelete',
wfEscapeWikiText( $title->getPrefixedText() ) );
527 if ( !$revisionRecord ) {
528 throw new LogicException(
"No revisions for $page?" );
531 $content = $page->
getContent( RevisionRecord::RAW );
532 }
catch ( TimeoutException $e ) {
534 }
catch ( Exception $ex ) {
535 wfLogWarning( __METHOD__ .
': failed to load content during deletion! '
536 . $ex->getMessage() );
544 $done = $this->archiveRevisions( $page, $id );
545 if ( $done || !$this->forceImmediate ) {
548 $dbw->endAtomic( __METHOD__ );
549 $this->lbFactory->commitAndWaitForReplication( __METHOD__, $ticket );
550 $dbw->startAtomic( __METHOD__ );
554 $dbw->endAtomic( __METHOD__ );
557 'namespace' => $title->getNamespace(),
558 'title' => $title->getDBkey(),
560 'requestId' => $webRequestId ?? $this->webRequestID,
562 'suppress' => $this->suppress,
563 'userId' => $this->deleter->getUser()->getId(),
564 'tags' => json_encode( $this->tags ),
565 'logsubtype' => $this->logSubtype,
566 'pageRole' => $pageRole,
569 $job =
new DeletePageJob( $jobParams );
570 $this->jobQueueGroup->push(
$job );
571 $this->wasScheduled[$pageRole] =
true;
574 $this->wasScheduled[$pageRole] =
false;
583 $archivedRevisionCount = $dbw->newSelectQueryBuilder()
587 'ar_namespace' => $title->getNamespace(),
588 'ar_title' => $title->getDBkey(),
591 ->caller( __METHOD__ )->fetchRowCount();
599 $logTitle = clone $title;
600 $wikiPageBeforeDelete = clone $page;
604 $dbw->newDeleteQueryBuilder()
605 ->deleteFrom(
'page' )
606 ->where( [
'page_id' => $id ] )
607 ->caller( __METHOD__ )->execute();
610 $logtype = $this->suppress ?
'suppress' :
'delete';
612 $logEntry =
new ManualLogEntry( $logtype, $this->logSubtype );
613 $logEntry->setPerformer( $this->deleter->getUser() );
614 $logEntry->setTarget( $logTitle );
615 $logEntry->setComment( $reason );
616 $logEntry->addTags( $this->tags );
617 if ( !$this->isDeletePageUnitTest ) {
619 $logid = $logEntry->insert();
621 $dbw->onTransactionPreCommitOrIdle(
622 static function () use ( $logEntry, $logid ) {
624 $logEntry->publish( $logid );
632 $this->eventDispatcher->dispatch(
new PageDeletedEvent(
635 $this->deleter->getUser(),
637 [ PageDeletedEvent::FLAG_SUPPRESSED => $this->suppress ],
638 $logEntry->getTimestamp(),
640 $archivedRevisionCount
641 ), $this->lbFactory );
643 $dbw->endAtomic( __METHOD__ );
645 $this->doDeleteUpdates( $wikiPageBeforeDelete, $revisionRecord );
648 $page->loadFromRow(
false, IDBAccessObject::READ_LATEST );
651 Title::clearCaches();
653 $legacyDeleter = $this->userFactory->newFromAuthority( $this->deleter );
654 $this->hookRunner->onArticleDeleteComplete(
655 $wikiPageBeforeDelete,
661 $archivedRevisionCount
663 $this->hookRunner->onPageDeleteComplete(
670 $archivedRevisionCount
672 $this->successfulDeletionsIDs[$pageRole] = $logid;
675 $this->redirectStore->clearCache( $page );
678 $key = $this->recentDeletesCache->makeKey(
'page-recent-delete', md5( $logTitle->getPrefixedText() ) );
679 $this->recentDeletesCache->set( $key, 1, BagOStuff::TTL_DAY );
691 private function archiveRevisions( WikiPage $page,
int $id ): bool {
693 $namespace = $page->
getTitle()->getNamespace();
694 $dbKey = $page->getTitle()->getDBkey();
696 $dbw = $this->lbFactory->getPrimaryDatabase();
698 $revQuery = $this->revisionStore->getQueryInfo();
702 if ( $this->suppress ) {
703 $bitfield = RevisionRecord::SUPPRESSED_ALL;
704 $revQuery[
'fields'] = array_diff( $revQuery[
'fields'], [
'rev_deleted' ] );
718 $lockQuery = $revQuery;
719 $lockQuery[
'tables'] = array_intersect(
721 [
'revision',
'revision_comment_temp' ]
723 unset( $lockQuery[
'fields'] );
724 $dbw->newSelectQueryBuilder()
725 ->queryInfo( $lockQuery )
726 ->where( [
'rev_page' => $id ] )
728 ->caller( __METHOD__ )
731 $deleteBatchSize = $this->options->get( MainConfigNames::DeleteRevisionsBatchSize );
734 $res = $dbw->newSelectQueryBuilder()
735 ->queryInfo( $revQuery )
736 ->where( [
'rev_page' => $id ] )
737 ->orderBy( [
'rev_timestamp',
'rev_id' ] )
738 ->limit( $deleteBatchSize + 1 )
739 ->caller( __METHOD__ )
750 foreach ( $res as $row ) {
751 if ( count( $revids ) >= $deleteBatchSize ) {
756 $comment = $this->commentStore->getComment(
'rev_comment', $row );
758 'ar_namespace' => $namespace,
759 'ar_title' => $dbKey,
760 'ar_actor' => $row->rev_actor,
761 'ar_timestamp' => $row->rev_timestamp,
762 'ar_minor_edit' => $row->rev_minor_edit,
763 'ar_rev_id' => $row->rev_id,
764 'ar_parent_id' => $row->rev_parent_id,
765 'ar_len' => $row->rev_len,
767 'ar_deleted' => $this->suppress ? $bitfield : $row->rev_deleted,
768 'ar_sha1' => $row->rev_sha1,
769 ] + $this->commentStore->insert( $dbw,
'ar_comment', $comment );
771 $rowsInsert[] = $rowInsert;
772 $revids[] = $row->rev_id;
776 if ( (
int)$row->rev_user === 0 && IPUtils::isValid( $row->rev_user_text ) ) {
777 $ipRevIds[] = $row->rev_id;
781 if ( count( $revids ) > 0 ) {
783 $dbw->newInsertQueryBuilder()
784 ->insertInto(
'archive' )
785 ->rows( $rowsInsert )
786 ->caller( __METHOD__ )->execute();
788 $dbw->newDeleteQueryBuilder()
789 ->deleteFrom(
'revision' )
790 ->where( [
'rev_id' => $revids ] )
791 ->caller( __METHOD__ )->execute();
793 if ( count( $ipRevIds ) > 0 ) {
794 $dbw->newDeleteQueryBuilder()
795 ->deleteFrom(
'ip_changes' )
796 ->where( [
'ipc_rev_id' => $ipRevIds ] )
797 ->caller( __METHOD__ )->execute();
812 private function doDeleteUpdates( WikiPage $pageBeforeDelete, RevisionRecord $revRecord ): void {
814 $countable = $pageBeforeDelete->isCountable();
815 }
catch ( TimeoutException $e ) {
817 }
catch ( Exception $ex ) {
826 DeferredUpdates::addUpdate( SiteStatsUpdate::factory(
827 [
'edits' => 1,
'articles' => $countable ? -1 : 0,
'pages' => -1 ]
831 $updates = $this->getDeletionUpdates( $pageBeforeDelete, $revRecord );
832 foreach ( $updates as $update ) {
833 DeferredUpdates::addUpdate( $update );
837 LinksUpdate::queueRecursiveJobsForTable(
838 $pageBeforeDelete->getTitle(),
841 $this->deleter->getUser()->getName(),
842 $this->backlinkCacheFactory->getBacklinkCache( $pageBeforeDelete->getTitle() )
845 if ( $pageBeforeDelete->getTitle()->getNamespace() ===
NS_FILE ) {
846 LinksUpdate::queueRecursiveJobsForTable(
847 $pageBeforeDelete->getTitle(),
850 $this->deleter->getUser()->getName(),
851 $this->backlinkCacheFactory->getBacklinkCache( $pageBeforeDelete->getTitle() )
855 if ( !$this->isDeletePageUnitTest ) {
858 WikiPage::onArticleDelete( $pageBeforeDelete->getTitle() );
871 private function getDeletionUpdates( WikiPage $page, RevisionRecord $rev ): array {
872 if ( $this->isDeletePageUnitTest ) {
876 $slotContent = array_map(
static function ( SlotRecord $slot ) {
877 return $slot->getContent();
878 }, $rev->getSlots()->getSlots() );
880 $allUpdates = [
new LinksDeletionUpdate( $page ) ];
887 foreach ( $slotContent as $role => $content ) {
888 $handler = $content->getContentHandler();
890 $updates = $handler->getDeletionUpdates(
895 $allUpdates = array_merge( $allUpdates, $updates );
898 $this->hookRunner->onPageDeletionDataUpdates(
899 $page->getTitle(), $rev, $allUpdates );
902 $this->hookRunner->onWikiPageDeletionUpdates( $page, $content, $allUpdates );
__construct(HookContainer $hookContainer, DomainEventDispatcher $eventDispatcher, RevisionStore $revisionStore, LBFactory $lbFactory, JobQueueGroup $jobQueueGroup, CommentStore $commentStore, ServiceOptions $serviceOptions, BagOStuff $recentDeletesCache, string $localWikiID, string $webRequestID, WikiPageFactory $wikiPageFactory, UserFactory $userFactory, BacklinkCacheFactory $backlinkCacheFactory, NamespaceInfo $namespaceInfo, ITextFormatter $contLangMsgTextFormatter, RedirectStore $redirectStore, ProperPageIdentity $page, Authority $deleter)