2008-05-09 18:53:09 +00:00
|
|
|
<?php
|
|
|
|
|
2021-02-11 02:37:19 +00:00
|
|
|
namespace MediaWiki\Extension\Nuke;
|
|
|
|
|
2021-08-28 23:00:15 +00:00
|
|
|
use DeletePageJob;
|
2024-10-15 17:28:15 +00:00
|
|
|
use ErrorPageError;
|
2022-11-04 11:03:00 +00:00
|
|
|
use JobQueueGroup;
|
2024-10-15 17:28:15 +00:00
|
|
|
use MediaWiki\CheckUser\Services\CheckUserTemporaryAccountsByIPLookup;
|
2023-10-02 21:02:12 +00:00
|
|
|
use MediaWiki\CommentStore\CommentStore;
|
2021-02-11 02:37:19 +00:00
|
|
|
use MediaWiki\Extension\Nuke\Hooks\NukeHookRunner;
|
2023-10-02 21:02:12 +00:00
|
|
|
use MediaWiki\Html\Html;
|
|
|
|
use MediaWiki\Html\ListToggle;
|
2024-06-09 16:58:16 +00:00
|
|
|
use MediaWiki\HTMLForm\HTMLForm;
|
2024-10-20 07:31:41 +00:00
|
|
|
use MediaWiki\Language\Language;
|
2024-11-12 01:56:47 +00:00
|
|
|
use MediaWiki\MainConfigNames;
|
2023-05-05 20:23:03 +00:00
|
|
|
use MediaWiki\Page\File\FileDeleteForm;
|
2022-11-04 11:03:00 +00:00
|
|
|
use MediaWiki\Permissions\PermissionManager;
|
2023-10-02 21:02:12 +00:00
|
|
|
use MediaWiki\Request\WebRequest;
|
|
|
|
use MediaWiki\SpecialPage\SpecialPage;
|
2024-05-03 13:41:59 +00:00
|
|
|
use MediaWiki\Title\NamespaceInfo;
|
2023-08-19 04:17:58 +00:00
|
|
|
use MediaWiki\Title\Title;
|
2024-10-15 17:28:15 +00:00
|
|
|
use MediaWiki\User\Options\UserOptionsLookup;
|
|
|
|
use MediaWiki\User\User;
|
2021-09-18 08:56:33 +00:00
|
|
|
use MediaWiki\User\UserFactory;
|
|
|
|
use MediaWiki\User\UserNamePrefixSearch;
|
|
|
|
use MediaWiki\User\UserNameUtils;
|
2024-10-20 07:31:41 +00:00
|
|
|
use MediaWiki\Xml\Xml;
|
2022-01-29 13:17:50 +00:00
|
|
|
use OOUI\DropdownInputWidget;
|
|
|
|
use OOUI\FieldLayout;
|
|
|
|
use OOUI\TextInputWidget;
|
2021-02-11 02:37:19 +00:00
|
|
|
use PermissionsError;
|
2022-11-04 11:03:00 +00:00
|
|
|
use RepoGroup;
|
2021-02-11 02:37:19 +00:00
|
|
|
use UserBlockedError;
|
2024-10-15 17:28:15 +00:00
|
|
|
use Wikimedia\IPUtils;
|
2023-12-19 15:59:32 +00:00
|
|
|
use Wikimedia\Rdbms\IConnectionProvider;
|
2024-05-03 13:41:59 +00:00
|
|
|
use Wikimedia\Rdbms\IExpression;
|
|
|
|
use Wikimedia\Rdbms\LikeMatch;
|
|
|
|
use Wikimedia\Rdbms\LikeValue;
|
2023-12-19 15:59:32 +00:00
|
|
|
use Wikimedia\Rdbms\SelectQueryBuilder;
|
2020-02-11 21:24:24 +00:00
|
|
|
|
2008-05-09 18:53:09 +00:00
|
|
|
class SpecialNuke extends SpecialPage {
|
2011-09-27 16:17:06 +00:00
|
|
|
|
2021-09-18 08:56:33 +00:00
|
|
|
/** @var NukeHookRunner|null */
|
2020-06-05 09:39:50 +00:00
|
|
|
private $hookRunner;
|
|
|
|
|
2024-04-19 13:10:15 +00:00
|
|
|
private JobQueueGroup $jobQueueGroup;
|
|
|
|
private IConnectionProvider $dbProvider;
|
|
|
|
private PermissionManager $permissionManager;
|
|
|
|
private RepoGroup $repoGroup;
|
|
|
|
private UserFactory $userFactory;
|
2024-10-15 17:28:15 +00:00
|
|
|
private UserOptionsLookup $userOptionsLookup;
|
2024-04-19 13:10:15 +00:00
|
|
|
private UserNamePrefixSearch $userNamePrefixSearch;
|
|
|
|
private UserNameUtils $userNameUtils;
|
2024-05-03 13:41:59 +00:00
|
|
|
private NamespaceInfo $namespaceInfo;
|
|
|
|
private Language $contentLanguage;
|
2024-10-15 17:28:15 +00:00
|
|
|
/** @var CheckUserTemporaryAccountsByIPLookup|null */
|
|
|
|
private $checkUserTemporaryAccountsByIPLookup = null;
|
2022-11-04 11:03:00 +00:00
|
|
|
|
2024-10-15 17:28:15 +00:00
|
|
|
/**
|
|
|
|
* @inheritDoc
|
|
|
|
*/
|
2022-11-04 11:03:00 +00:00
|
|
|
public function __construct(
|
|
|
|
JobQueueGroup $jobQueueGroup,
|
2023-12-19 15:59:32 +00:00
|
|
|
IConnectionProvider $dbProvider,
|
2022-11-04 11:03:00 +00:00
|
|
|
PermissionManager $permissionManager,
|
2021-09-18 08:56:33 +00:00
|
|
|
RepoGroup $repoGroup,
|
|
|
|
UserFactory $userFactory,
|
2024-10-15 17:28:15 +00:00
|
|
|
UserOptionsLookup $userOptionsLookup,
|
2021-09-18 08:56:33 +00:00
|
|
|
UserNamePrefixSearch $userNamePrefixSearch,
|
2024-05-03 13:41:59 +00:00
|
|
|
UserNameUtils $userNameUtils,
|
|
|
|
NamespaceInfo $namespaceInfo,
|
2024-10-15 17:28:15 +00:00
|
|
|
Language $contentLanguage,
|
|
|
|
$checkUserTemporaryAccountsByIPLookup = null
|
2022-11-04 11:03:00 +00:00
|
|
|
) {
|
2008-05-09 18:53:09 +00:00
|
|
|
parent::__construct( 'Nuke', 'nuke' );
|
2022-11-04 11:03:00 +00:00
|
|
|
$this->jobQueueGroup = $jobQueueGroup;
|
2023-12-19 15:59:32 +00:00
|
|
|
$this->dbProvider = $dbProvider;
|
2022-11-04 11:03:00 +00:00
|
|
|
$this->permissionManager = $permissionManager;
|
|
|
|
$this->repoGroup = $repoGroup;
|
2021-09-18 08:56:33 +00:00
|
|
|
$this->userFactory = $userFactory;
|
2024-10-15 17:28:15 +00:00
|
|
|
$this->userOptionsLookup = $userOptionsLookup;
|
2021-09-18 08:56:33 +00:00
|
|
|
$this->userNamePrefixSearch = $userNamePrefixSearch;
|
|
|
|
$this->userNameUtils = $userNameUtils;
|
2024-05-03 13:41:59 +00:00
|
|
|
$this->namespaceInfo = $namespaceInfo;
|
|
|
|
$this->contentLanguage = $contentLanguage;
|
2024-10-15 17:28:15 +00:00
|
|
|
$this->checkUserTemporaryAccountsByIPLookup = $checkUserTemporaryAccountsByIPLookup;
|
2008-05-09 18:53:09 +00:00
|
|
|
}
|
|
|
|
|
2024-05-11 06:48:28 +00:00
|
|
|
/**
|
|
|
|
* @inheritDoc
|
|
|
|
* @codeCoverageIgnore
|
|
|
|
*/
|
2016-01-19 20:06:35 +00:00
|
|
|
public function doesWrites() {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-02-04 22:50:54 +00:00
|
|
|
/**
|
|
|
|
* @param null|string $par
|
|
|
|
*/
|
2011-12-03 12:28:29 +00:00
|
|
|
public function execute( $par ) {
|
2008-05-09 18:53:09 +00:00
|
|
|
$this->setHeaders();
|
2016-05-25 06:12:17 +00:00
|
|
|
$this->checkPermissions();
|
|
|
|
$this->checkReadOnly();
|
2008-05-09 18:53:09 +00:00
|
|
|
$this->outputHeader();
|
2019-10-08 13:46:17 +00:00
|
|
|
$this->addHelpLink( 'Help:Extension:Nuke' );
|
2011-09-27 16:17:06 +00:00
|
|
|
|
2016-05-25 06:12:17 +00:00
|
|
|
$currentUser = $this->getUser();
|
2020-03-07 07:57:10 +00:00
|
|
|
$block = $currentUser->getBlock();
|
|
|
|
|
|
|
|
// appliesToRight is presently a no-op, since there is no handling for `delete`,
|
|
|
|
// and so will return `null`. `true` will be returned if the block actively
|
|
|
|
// applies to `delete`, and both `null` and `true` should result in an error
|
|
|
|
if ( $block && ( $block->isSitewide() ||
|
|
|
|
( $block->appliesToRight( 'delete' ) !== false ) )
|
|
|
|
) {
|
2012-03-14 01:58:01 +00:00
|
|
|
throw new UserBlockedError( $block );
|
|
|
|
}
|
2012-06-02 15:08:09 +00:00
|
|
|
|
2011-11-10 17:22:05 +00:00
|
|
|
$req = $this->getRequest();
|
2022-07-17 19:43:16 +00:00
|
|
|
$target = trim( $req->getText( 'target', $par ?? '' ) );
|
2011-11-25 00:20:54 +00:00
|
|
|
|
2011-10-18 22:46:31 +00:00
|
|
|
// Normalise name
|
|
|
|
if ( $target !== '' ) {
|
2021-09-18 08:56:33 +00:00
|
|
|
$user = $this->userFactory->newFromName( $target );
|
2012-10-20 17:41:44 +00:00
|
|
|
if ( $user ) {
|
|
|
|
$target = $user->getName();
|
|
|
|
}
|
2011-10-18 22:46:31 +00:00
|
|
|
}
|
2011-11-25 00:20:54 +00:00
|
|
|
|
2022-01-29 13:17:50 +00:00
|
|
|
$reason = $this->getDeleteReason( $this->getRequest(), $target );
|
2011-11-25 00:20:54 +00:00
|
|
|
|
2017-02-22 13:54:14 +00:00
|
|
|
$limit = $req->getInt( 'limit', 500 );
|
2022-11-04 11:28:02 +00:00
|
|
|
$namespace = $req->getIntOrNull( 'namespace' );
|
2012-10-05 13:49:43 +00:00
|
|
|
|
2012-08-16 13:19:22 +00:00
|
|
|
if ( $req->wasPosted()
|
2016-05-25 06:12:17 +00:00
|
|
|
&& $currentUser->matchEditToken( $req->getVal( 'wpEditToken' ) )
|
2015-10-02 07:48:42 +00:00
|
|
|
) {
|
2022-08-30 08:40:10 +00:00
|
|
|
if ( $req->getRawVal( 'action' ) === 'delete' ) {
|
2024-11-17 09:53:02 +00:00
|
|
|
$pages = $req->getArray( 'pages' ) ?? [];
|
|
|
|
$originalPageList
|
|
|
|
= explode( '|', $req->getText( 'originalPageList' ) );
|
2011-10-18 22:46:31 +00:00
|
|
|
|
2024-11-17 09:53:02 +00:00
|
|
|
if ( count( $originalPageList ) === 1 && !$originalPageList[0] ) {
|
|
|
|
// No page list was provided.
|
|
|
|
$originalPageList = [];
|
2011-11-10 17:04:38 +00:00
|
|
|
}
|
2024-11-17 09:53:02 +00:00
|
|
|
|
|
|
|
$this->doDelete( $pages, $originalPageList, $reason, $target );
|
2022-08-30 08:40:10 +00:00
|
|
|
} elseif ( $req->getRawVal( 'action' ) === 'submit' ) {
|
2024-10-15 17:28:15 +00:00
|
|
|
// if the target is an ip addresss and temp account lookup is available,
|
|
|
|
// list pages created by the ip user or by temp accounts associated with the ip address
|
|
|
|
if (
|
|
|
|
$this->checkUserTemporaryAccountsByIPLookup &&
|
|
|
|
IPUtils::isValid( $target )
|
|
|
|
) {
|
|
|
|
$this->assertUserCanAccessTemporaryAccounts( $currentUser );
|
|
|
|
$tempnames = $this->getTempAccountData( $target );
|
|
|
|
$reason = $this->getDeleteReason( $this->getRequest(), $target, true );
|
|
|
|
$this->listForm( $target, $reason, $limit, $namespace, $tempnames );
|
|
|
|
} else {
|
|
|
|
// otherwise just list pages normally
|
|
|
|
$this->listForm( $target, $reason, $limit, $namespace );
|
|
|
|
}
|
2012-01-13 22:45:24 +00:00
|
|
|
} else {
|
2011-11-10 17:04:38 +00:00
|
|
|
$this->promptForm();
|
2008-05-09 18:53:09 +00:00
|
|
|
}
|
2012-01-13 22:45:24 +00:00
|
|
|
} elseif ( $target === '' ) {
|
2008-05-09 18:53:09 +00:00
|
|
|
$this->promptForm();
|
2012-01-13 22:45:24 +00:00
|
|
|
} else {
|
2012-10-05 13:49:43 +00:00
|
|
|
$this->listForm( $target, $reason, $limit, $namespace );
|
2011-10-18 22:46:31 +00:00
|
|
|
}
|
2008-05-09 18:53:09 +00:00
|
|
|
}
|
|
|
|
|
2024-10-15 17:28:15 +00:00
|
|
|
/**
|
|
|
|
* Does the user have the appropriate permissions and have they enabled in preferences?
|
|
|
|
* Adapted from MediaWiki\CheckUser\Api\Rest\Handler\AbstractTemporaryAccountHandler::checkPermissions
|
|
|
|
*
|
|
|
|
* @param User $currentUser
|
|
|
|
*
|
|
|
|
* @throws PermissionsError if the user does not have the 'checkuser-temporary-account' right
|
|
|
|
* @throws ErrorPageError if the user has not enabled the 'checkuser-temporary-account-enabled' preference
|
|
|
|
*/
|
|
|
|
private function assertUserCanAccessTemporaryAccounts( User $currentUser ) {
|
|
|
|
if (
|
|
|
|
!$currentUser->isAllowed( 'checkuser-temporary-account-no-preference' )
|
|
|
|
) {
|
|
|
|
if (
|
|
|
|
!$currentUser->isAllowed( 'checkuser-temporary-account' )
|
|
|
|
) {
|
|
|
|
throw new PermissionsError( 'checkuser-temporary-account' );
|
|
|
|
}
|
|
|
|
if (
|
|
|
|
!$this->userOptionsLookup->getOption(
|
|
|
|
$currentUser,
|
|
|
|
'checkuser-temporary-account-enable'
|
|
|
|
)
|
|
|
|
) {
|
|
|
|
throw new ErrorPageError(
|
|
|
|
$this->msg( 'checkuser-ip-contributions-permission-error-title' ),
|
|
|
|
$this->msg( 'checkuser-ip-contributions-permission-error-description' )
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Given an IP address, return a list of temporary accounts that are known to have edited from the IP.
|
|
|
|
*
|
|
|
|
* Calls to this method result in a log entry being generated for the logged-in user account making the request.
|
|
|
|
* @param string $ip The IP address used for looking up temporary account names.
|
|
|
|
* The address will be normalized in the IP lookup service.
|
|
|
|
* @return string[] A list of temporary account usernames associated with the IP address
|
|
|
|
*/
|
|
|
|
private function getTempAccountData( string $ip ): array {
|
|
|
|
// Requires CheckUserTemporaryAccountsByIPLookup service
|
|
|
|
if ( !$this->checkUserTemporaryAccountsByIPLookup ) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
$status = $this->checkUserTemporaryAccountsByIPLookup->get(
|
|
|
|
$ip,
|
|
|
|
$this->getAuthority(),
|
|
|
|
true
|
|
|
|
);
|
|
|
|
if ( $status->isGood() ) {
|
|
|
|
return $status->getValue();
|
|
|
|
}
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
2010-12-01 20:46:05 +00:00
|
|
|
/**
|
2011-02-14 20:21:02 +00:00
|
|
|
* Prompt for a username or IP address.
|
2011-12-03 12:28:29 +00:00
|
|
|
*
|
2017-08-09 21:47:49 +00:00
|
|
|
* @param string $userName
|
2010-12-01 20:46:05 +00:00
|
|
|
*/
|
2024-05-11 06:47:56 +00:00
|
|
|
protected function promptForm( string $userName = '' ): void {
|
2011-11-10 17:22:05 +00:00
|
|
|
$out = $this->getOutput();
|
2011-11-25 00:20:54 +00:00
|
|
|
|
2024-10-15 17:28:15 +00:00
|
|
|
if ( $this->checkUserTemporaryAccountsByIPLookup ) {
|
|
|
|
$out->addWikiMsg( 'nuke-tools-tempaccount' );
|
|
|
|
} else {
|
|
|
|
$out->addWikiMsg( 'nuke-tools' );
|
|
|
|
}
|
2011-09-27 16:17:06 +00:00
|
|
|
|
2016-12-22 23:42:01 +00:00
|
|
|
$formDescriptor = [
|
|
|
|
'nuke-target' => [
|
|
|
|
'id' => 'nuke-target',
|
|
|
|
'default' => $userName,
|
|
|
|
'label' => $this->msg( 'nuke-userorip' )->text(),
|
|
|
|
'type' => 'user',
|
2018-03-30 20:52:50 +00:00
|
|
|
'name' => 'target',
|
|
|
|
'autofocus' => true
|
2016-12-22 23:42:01 +00:00
|
|
|
],
|
|
|
|
'nuke-pattern' => [
|
|
|
|
'id' => 'nuke-pattern',
|
|
|
|
'label' => $this->msg( 'nuke-pattern' )->text(),
|
|
|
|
'maxLength' => 40,
|
2017-02-22 13:54:14 +00:00
|
|
|
'type' => 'text',
|
|
|
|
'name' => 'pattern'
|
2016-12-22 23:42:01 +00:00
|
|
|
],
|
|
|
|
'namespace' => [
|
|
|
|
'id' => 'nuke-namespace',
|
|
|
|
'type' => 'namespaceselect',
|
|
|
|
'label' => $this->msg( 'nuke-namespace' )->text(),
|
2017-02-22 13:54:14 +00:00
|
|
|
'all' => 'all',
|
|
|
|
'name' => 'namespace'
|
2016-12-22 23:42:01 +00:00
|
|
|
],
|
|
|
|
'limit' => [
|
|
|
|
'id' => 'nuke-limit',
|
|
|
|
'maxLength' => 7,
|
|
|
|
'default' => 500,
|
|
|
|
'label' => $this->msg( 'nuke-maxpages' )->text(),
|
2017-02-22 13:54:14 +00:00
|
|
|
'type' => 'int',
|
|
|
|
'name' => 'limit'
|
2016-12-22 23:42:01 +00:00
|
|
|
]
|
|
|
|
];
|
|
|
|
|
|
|
|
HTMLForm::factory( 'ooui', $formDescriptor, $this->getContext() )
|
|
|
|
->setName( 'massdelete' )
|
|
|
|
->setFormIdentifier( 'massdelete' )
|
|
|
|
->setWrapperLegendMsg( 'nuke' )
|
|
|
|
->setSubmitTextMsg( 'nuke-submit-user' )
|
|
|
|
->setSubmitName( 'nuke-submit-user' )
|
|
|
|
->setAction( $this->getPageTitle()->getLocalURL( 'action=submit' ) )
|
|
|
|
->prepareForm()
|
|
|
|
->displayForm( false );
|
2008-05-09 18:53:09 +00:00
|
|
|
}
|
|
|
|
|
2010-12-01 20:46:05 +00:00
|
|
|
/**
|
2011-02-14 20:21:02 +00:00
|
|
|
* Display list of pages to delete.
|
2011-09-27 16:17:06 +00:00
|
|
|
*
|
2011-02-14 20:21:02 +00:00
|
|
|
* @param string $username
|
|
|
|
* @param string $reason
|
2017-08-09 21:47:49 +00:00
|
|
|
* @param int $limit
|
|
|
|
* @param int|null $namespace
|
2024-10-15 17:28:15 +00:00
|
|
|
* @param string[] $tempnames
|
2010-12-01 20:46:05 +00:00
|
|
|
*/
|
2024-10-15 17:28:15 +00:00
|
|
|
protected function listForm( $username, $reason, $limit, $namespace = null, $tempnames = [] ): void {
|
2011-11-10 17:22:05 +00:00
|
|
|
$out = $this->getOutput();
|
2011-11-25 00:20:54 +00:00
|
|
|
|
2024-10-15 17:28:15 +00:00
|
|
|
$pages = $this->getNewPages( $username, $limit, $namespace, $tempnames );
|
2009-01-12 18:23:47 +00:00
|
|
|
|
2024-04-19 13:10:15 +00:00
|
|
|
if ( !$pages ) {
|
2011-11-10 17:04:38 +00:00
|
|
|
if ( $username === '' ) {
|
2011-11-10 17:22:05 +00:00
|
|
|
$out->addWikiMsg( 'nuke-nopages-global' );
|
2012-01-13 22:45:24 +00:00
|
|
|
} else {
|
2011-11-10 17:22:05 +00:00
|
|
|
$out->addWikiMsg( 'nuke-nopages', $username );
|
2011-11-10 17:04:38 +00:00
|
|
|
}
|
2011-11-25 00:20:54 +00:00
|
|
|
|
2012-01-13 22:45:24 +00:00
|
|
|
$this->promptForm( $username );
|
|
|
|
return;
|
2008-05-09 18:53:09 +00:00
|
|
|
}
|
2011-09-27 16:17:06 +00:00
|
|
|
|
2016-12-28 08:15:54 +00:00
|
|
|
$out->addModules( 'ext.nuke.confirm' );
|
2024-05-05 07:35:45 +00:00
|
|
|
$out->addModuleStyles( [ 'ext.nuke.styles', 'mediawiki.interface.helpers.styles' ] );
|
2016-12-28 08:15:54 +00:00
|
|
|
|
2011-11-10 17:04:38 +00:00
|
|
|
if ( $username === '' ) {
|
2011-11-10 17:22:05 +00:00
|
|
|
$out->addWikiMsg( 'nuke-list-multiple' );
|
2024-10-15 17:28:15 +00:00
|
|
|
} elseif ( $tempnames ) {
|
|
|
|
$out->addWikiMsg( 'nuke-list-tempaccount', $username );
|
2011-09-27 16:17:06 +00:00
|
|
|
} else {
|
2011-11-10 17:22:05 +00:00
|
|
|
$out->addWikiMsg( 'nuke-list', $username );
|
2011-02-14 20:21:02 +00:00
|
|
|
}
|
2008-05-09 18:53:09 +00:00
|
|
|
|
2015-10-02 07:48:42 +00:00
|
|
|
$nuke = $this->getPageTitle();
|
2009-01-12 18:23:47 +00:00
|
|
|
|
2024-03-09 14:45:40 +00:00
|
|
|
$options = Xml::listDropdownOptions(
|
2022-01-29 13:17:50 +00:00
|
|
|
$this->msg( 'deletereason-dropdown' )->inContentLanguage()->text(),
|
|
|
|
[ 'other' => $this->msg( 'deletereasonotherlist' )->inContentLanguage()->text() ]
|
|
|
|
);
|
|
|
|
|
|
|
|
$dropdown = new FieldLayout(
|
|
|
|
new DropdownInputWidget( [
|
|
|
|
'name' => 'wpDeleteReasonList',
|
|
|
|
'inputId' => 'wpDeleteReasonList',
|
|
|
|
'tabIndex' => 1,
|
|
|
|
'infusable' => true,
|
|
|
|
'value' => '',
|
2024-03-09 14:45:40 +00:00
|
|
|
'options' => Xml::listDropdownOptionsOoui( $options ),
|
2022-01-29 13:17:50 +00:00
|
|
|
] ),
|
|
|
|
[
|
|
|
|
'label' => $this->msg( 'deletecomment' )->text(),
|
|
|
|
'align' => 'top',
|
|
|
|
]
|
|
|
|
);
|
|
|
|
$reasonField = new FieldLayout(
|
|
|
|
new TextInputWidget( [
|
|
|
|
'name' => 'wpReason',
|
|
|
|
'inputId' => 'wpReason',
|
|
|
|
'tabIndex' => 2,
|
|
|
|
'maxLength' => CommentStore::COMMENT_CHARACTER_LIMIT,
|
|
|
|
'infusable' => true,
|
|
|
|
'value' => $reason,
|
|
|
|
'autofocus' => true,
|
|
|
|
] ),
|
|
|
|
[
|
|
|
|
'label' => $this->msg( 'deleteotherreason' )->text(),
|
|
|
|
'align' => 'top',
|
|
|
|
]
|
|
|
|
);
|
|
|
|
|
|
|
|
$out->enableOOUI();
|
2011-11-10 17:22:05 +00:00
|
|
|
$out->addHTML(
|
2024-04-19 12:52:44 +00:00
|
|
|
Html::openElement( 'form', [
|
2015-10-02 07:48:42 +00:00
|
|
|
'action' => $nuke->getLocalURL( 'action=delete' ),
|
|
|
|
'method' => 'post',
|
2016-03-07 14:53:32 +00:00
|
|
|
'name' => 'nukelist' ]
|
2009-01-12 18:23:47 +00:00
|
|
|
) .
|
2012-08-17 14:43:17 +00:00
|
|
|
Html::hidden( 'wpEditToken', $this->getUser()->getEditToken() ) .
|
2024-11-17 09:53:02 +00:00
|
|
|
Html::hidden( 'target', $username ) .
|
2024-04-19 13:10:15 +00:00
|
|
|
$dropdown .
|
|
|
|
$reasonField .
|
|
|
|
// Select: All, None, Invert
|
|
|
|
( new ListToggle( $this->getOutput() ) )->getHTML() .
|
2016-09-13 15:38:35 +00:00
|
|
|
'<ul>'
|
2009-01-12 18:23:47 +00:00
|
|
|
);
|
|
|
|
|
2024-11-17 09:53:02 +00:00
|
|
|
$titles = [];
|
|
|
|
|
2015-02-07 13:39:43 +00:00
|
|
|
$wordSeparator = $this->msg( 'word-separator' )->escaped();
|
|
|
|
$commaSeparator = $this->msg( 'comma-separator' )->escaped();
|
2024-05-05 07:35:45 +00:00
|
|
|
$pipeSeparator = $this->msg( 'pipe-separator' )->escaped();
|
2012-08-16 13:19:22 +00:00
|
|
|
|
2017-02-25 10:18:18 +00:00
|
|
|
$linkRenderer = $this->getLinkRenderer();
|
2022-11-04 11:03:00 +00:00
|
|
|
$localRepo = $this->repoGroup->getLocalRepo();
|
2024-04-19 13:10:15 +00:00
|
|
|
foreach ( $pages as [ $title, $userName ] ) {
|
2012-01-13 22:45:24 +00:00
|
|
|
/**
|
2012-01-13 22:59:39 +00:00
|
|
|
* @var $title Title
|
2012-01-13 22:45:24 +00:00
|
|
|
*/
|
2024-11-17 09:53:02 +00:00
|
|
|
$titles[] = $title->getPrefixedDBkey();
|
2011-11-25 00:20:54 +00:00
|
|
|
|
2020-03-14 13:43:23 +00:00
|
|
|
$image = $title->inNamespace( NS_FILE ) ? $localRepo->newFile( $title ) : false;
|
2015-10-02 07:48:42 +00:00
|
|
|
$thumb = $image && $image->exists() ?
|
2016-03-07 14:53:32 +00:00
|
|
|
$image->transform( [ 'width' => 120, 'height' => 120 ], 0 ) :
|
2015-10-02 07:48:42 +00:00
|
|
|
false;
|
2009-01-12 18:23:47 +00:00
|
|
|
|
2015-10-02 07:48:42 +00:00
|
|
|
$userNameText = $userName ?
|
2024-05-05 07:35:45 +00:00
|
|
|
' <span class="mw-changeslist-separator"></span> ' . $this->msg( 'nuke-editby', $userName )->parse() :
|
2015-10-02 07:48:42 +00:00
|
|
|
'';
|
2017-02-25 10:18:18 +00:00
|
|
|
$changesLink = $linkRenderer->makeKnownLink(
|
2012-08-16 13:19:22 +00:00
|
|
|
$title,
|
2017-02-25 10:18:18 +00:00
|
|
|
$this->msg( 'nuke-viewchanges' )->text(),
|
2016-03-07 14:53:32 +00:00
|
|
|
[],
|
|
|
|
[ 'action' => 'history' ]
|
2012-08-16 13:19:22 +00:00
|
|
|
);
|
2024-05-05 07:35:45 +00:00
|
|
|
|
|
|
|
$talkPageText = $this->namespaceInfo->isTalk( $title->getNamespace() ) ?
|
|
|
|
'' :
|
|
|
|
$linkRenderer->makeLink(
|
|
|
|
$this->namespaceInfo->getTalkPage( $title ),
|
|
|
|
$this->msg( 'sp-contributions-talk' )->text(),
|
|
|
|
[],
|
|
|
|
[],
|
|
|
|
) . $wordSeparator . $pipeSeparator;
|
|
|
|
|
2024-04-19 13:10:15 +00:00
|
|
|
$query = $title->isRedirect() ? [ 'redirect' => 'no' ] : [];
|
2024-05-05 07:35:45 +00:00
|
|
|
$attributes = $title->isRedirect() ? [ 'class' => 'ext-nuke-italicize' ] : [];
|
2011-11-10 17:22:05 +00:00
|
|
|
$out->addHTML( '<li>' .
|
2024-04-19 12:52:44 +00:00
|
|
|
Html::check(
|
2012-02-12 14:45:29 +00:00
|
|
|
'pages[]',
|
|
|
|
true,
|
2016-03-07 14:53:32 +00:00
|
|
|
[ 'value' => $title->getPrefixedDBkey() ]
|
2022-08-30 13:24:16 +00:00
|
|
|
) . "\u{00A0}" .
|
2016-03-07 14:53:32 +00:00
|
|
|
( $thumb ? $thumb->toHtml( [ 'desc-link' => true ] ) : '' ) .
|
2024-05-04 15:17:51 +00:00
|
|
|
$linkRenderer->makeKnownLink( $title, null, $attributes, $query ) . $wordSeparator .
|
2024-05-05 07:35:45 +00:00
|
|
|
$this->msg( 'parentheses' )->rawParams( $talkPageText . $changesLink )->escaped() . $wordSeparator .
|
|
|
|
"<span class='ext-nuke-italicize'>" . $userNameText . "</span>" .
|
2012-08-16 13:19:22 +00:00
|
|
|
"</li>\n" );
|
2008-05-09 18:53:09 +00:00
|
|
|
}
|
2011-09-27 16:17:06 +00:00
|
|
|
|
2011-11-10 17:22:05 +00:00
|
|
|
$out->addHTML(
|
2009-01-12 18:23:47 +00:00
|
|
|
"</ul>\n" .
|
2024-11-17 09:53:02 +00:00
|
|
|
Html::hidden( 'originalPageList', implode( '|', $titles ) ) .
|
2024-04-19 12:52:44 +00:00
|
|
|
Html::submitButton( $this->msg( 'nuke-submit-delete' )->text() ) .
|
2012-01-11 09:34:47 +00:00
|
|
|
'</form>'
|
2009-01-12 18:23:47 +00:00
|
|
|
);
|
2008-05-09 18:53:09 +00:00
|
|
|
}
|
|
|
|
|
2011-02-14 20:21:02 +00:00
|
|
|
/**
|
|
|
|
* Gets a list of new pages by the specified user or everyone when none is specified.
|
2011-09-27 16:17:06 +00:00
|
|
|
*
|
2011-02-14 20:21:02 +00:00
|
|
|
* @param string $username
|
2017-08-09 21:47:49 +00:00
|
|
|
* @param int $limit
|
|
|
|
* @param int|null $namespace
|
2024-10-15 17:28:15 +00:00
|
|
|
* @param string[] $tempnames
|
2011-09-27 16:17:06 +00:00
|
|
|
*
|
2024-04-19 13:10:15 +00:00
|
|
|
* @return array{0:Title,1:string|false}[]
|
2011-02-14 20:21:02 +00:00
|
|
|
*/
|
2024-10-15 17:28:15 +00:00
|
|
|
protected function getNewPages( $username, $limit, $namespace = null, $tempnames = [] ): array {
|
2023-12-19 15:59:32 +00:00
|
|
|
$dbr = $this->dbProvider->getReplicaDatabase();
|
2024-11-12 01:56:47 +00:00
|
|
|
|
|
|
|
$maxAge = $this->getConfig()->get( "NukeMaxAge" );
|
|
|
|
// If no Nuke-specific max age was set, this should match the value of `$wgRCMaxAge`.
|
|
|
|
if ( !$maxAge ) {
|
|
|
|
$maxAge = $this->getConfig()->get( MainConfigNames::RCMaxAge );
|
|
|
|
}
|
|
|
|
|
2023-12-19 15:59:32 +00:00
|
|
|
$queryBuilder = $dbr->newSelectQueryBuilder()
|
2024-05-03 10:27:28 +00:00
|
|
|
->select( [ 'page_title', 'page_namespace' ] )
|
2024-11-12 01:56:47 +00:00
|
|
|
->from( 'revision' )
|
|
|
|
->join( 'actor', null, 'actor_id=rev_actor' )
|
|
|
|
->join( 'page', null, 'page_id=rev_page' )
|
|
|
|
->where( [
|
|
|
|
$dbr->expr( 'rev_parent_id', '=', 0 ),
|
|
|
|
$dbr->expr( 'rev_timestamp', '>', $dbr->timestamp(
|
|
|
|
time() - $maxAge
|
|
|
|
) )
|
|
|
|
] )
|
|
|
|
->orderBy( 'rev_timestamp', SelectQueryBuilder::SORT_DESC )
|
|
|
|
->distinct()
|
|
|
|
->limit( $limit )
|
|
|
|
->setMaxExecutionTime(
|
|
|
|
$this->getConfig()->get( MainConfigNames::MaxExecutionTimeForExpensiveQueries )
|
|
|
|
);
|
2011-09-27 16:17:06 +00:00
|
|
|
|
2024-11-12 01:56:47 +00:00
|
|
|
$queryBuilder->field( 'actor_name' );
|
2024-10-17 12:50:48 +00:00
|
|
|
$actornames = array_filter( [ $username, ...$tempnames ] );
|
|
|
|
if ( $actornames ) {
|
|
|
|
$queryBuilder->andWhere( [ 'actor_name' => $actornames ] );
|
2011-02-14 20:21:02 +00:00
|
|
|
}
|
2011-11-25 00:20:54 +00:00
|
|
|
|
2012-10-05 13:49:43 +00:00
|
|
|
if ( $namespace !== null ) {
|
2024-05-03 10:27:28 +00:00
|
|
|
$queryBuilder->andWhere( [ 'page_namespace' => $namespace ] );
|
2012-10-05 13:49:43 +00:00
|
|
|
}
|
|
|
|
|
2017-02-22 13:54:14 +00:00
|
|
|
$pattern = $this->getRequest()->getText( 'pattern' );
|
2020-01-14 08:25:54 +00:00
|
|
|
if ( $pattern !== null && trim( $pattern ) !== '' ) {
|
2024-05-03 13:41:59 +00:00
|
|
|
$addedWhere = false;
|
2024-05-11 06:47:56 +00:00
|
|
|
|
2024-05-03 13:41:59 +00:00
|
|
|
$pattern = trim( $pattern );
|
|
|
|
$pattern = preg_replace( '/ +/', '`_', $pattern );
|
|
|
|
$pattern = preg_replace( '/\\\\([%_])/', '`$1', $pattern );
|
2024-05-11 06:47:56 +00:00
|
|
|
|
2024-05-03 13:41:59 +00:00
|
|
|
if ( $namespace !== null ) {
|
2024-05-11 06:47:56 +00:00
|
|
|
// Custom namespace requested
|
|
|
|
// If that namespace capitalizes titles, capitalize the first character
|
|
|
|
// to match the DB title.
|
2024-05-03 13:41:59 +00:00
|
|
|
$pattern = $this->namespaceInfo->isCapitalized( $namespace ) ?
|
|
|
|
$this->contentLanguage->ucfirst( $pattern ) : $pattern;
|
|
|
|
} else {
|
2024-05-11 06:47:56 +00:00
|
|
|
// All namespaces requested
|
|
|
|
|
2024-05-03 13:41:59 +00:00
|
|
|
$overriddenNamespaces = [];
|
|
|
|
$capitalLinks = $this->getConfig()->get( 'CapitalLinks' );
|
|
|
|
$capitalLinkOverrides = $this->getConfig()->get( 'CapitalLinkOverrides' );
|
2024-05-11 06:47:56 +00:00
|
|
|
// If there are any capital-overridden namespaces, keep track of them. "overridden"
|
|
|
|
// here means the namespace-specific value is not equal to $wgCapitalLinks.
|
2024-05-03 13:41:59 +00:00
|
|
|
foreach ( $capitalLinkOverrides as $k => $v ) {
|
|
|
|
if ( $v !== $capitalLinks ) {
|
|
|
|
$overriddenNamespaces[] = $k;
|
|
|
|
}
|
|
|
|
}
|
2024-05-11 06:47:56 +00:00
|
|
|
|
2024-05-03 13:41:59 +00:00
|
|
|
if ( count( $overriddenNamespaces ) ) {
|
2024-05-11 06:47:56 +00:00
|
|
|
// If there are overridden namespaces, they have to be converted
|
|
|
|
// on a case-by-case basis.
|
|
|
|
|
2024-05-03 13:41:59 +00:00
|
|
|
$validNamespaces = $this->namespaceInfo->getValidNamespaces();
|
|
|
|
$nonOverriddenNamespaces = [];
|
|
|
|
foreach ( $validNamespaces as $ns ) {
|
|
|
|
if ( !in_array( $ns, $overriddenNamespaces ) ) {
|
2024-05-11 06:47:56 +00:00
|
|
|
// Put all namespaces that aren't overridden in $nonOverriddenNamespaces
|
2024-05-03 13:41:59 +00:00
|
|
|
$nonOverriddenNamespaces[] = $ns;
|
|
|
|
}
|
|
|
|
}
|
2024-05-11 06:47:56 +00:00
|
|
|
|
2024-05-03 13:41:59 +00:00
|
|
|
$patternSpecific = $this->namespaceInfo->isCapitalized( $overriddenNamespaces[0] ) ?
|
|
|
|
$this->contentLanguage->ucfirst( $pattern ) : $pattern;
|
|
|
|
$orConditions = [
|
|
|
|
$dbr->expr(
|
|
|
|
'page_title', IExpression::LIKE, new LikeValue(
|
|
|
|
new LikeMatch( $patternSpecific )
|
|
|
|
)
|
|
|
|
)->and(
|
2024-05-11 06:47:56 +00:00
|
|
|
// IN condition
|
2024-05-03 13:41:59 +00:00
|
|
|
'page_namespace', '=', $overriddenNamespaces
|
|
|
|
)
|
|
|
|
];
|
|
|
|
if ( count( $nonOverriddenNamespaces ) ) {
|
|
|
|
$patternStandard = $this->namespaceInfo->isCapitalized( $nonOverriddenNamespaces[0] ) ?
|
|
|
|
$this->contentLanguage->ucfirst( $pattern ) : $pattern;
|
|
|
|
$orConditions[] = $dbr->expr(
|
|
|
|
'page_title', IExpression::LIKE, new LikeValue(
|
|
|
|
new LikeMatch( $patternStandard )
|
|
|
|
)
|
|
|
|
)->and(
|
2024-05-11 06:47:56 +00:00
|
|
|
// IN condition, with the non-overridden namespaces.
|
|
|
|
// If the default is case-sensitive namespaces, $pattern's first
|
|
|
|
// character is turned lowercase. Otherwise, it is turned uppercase.
|
2024-05-03 13:41:59 +00:00
|
|
|
'page_namespace', '=', $nonOverriddenNamespaces
|
|
|
|
);
|
|
|
|
}
|
2024-07-11 16:36:16 +00:00
|
|
|
$queryBuilder->andWhere( $dbr->orExpr( $orConditions ) );
|
2024-05-03 13:41:59 +00:00
|
|
|
$addedWhere = true;
|
|
|
|
} else {
|
2024-05-11 06:47:56 +00:00
|
|
|
// No overridden namespaces; just convert all titles.
|
2024-05-03 13:41:59 +00:00
|
|
|
$pattern = $this->namespaceInfo->isCapitalized( NS_MAIN ) ?
|
|
|
|
$this->contentLanguage->ucfirst( $pattern ) : $pattern;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( !$addedWhere ) {
|
|
|
|
$queryBuilder->andWhere(
|
|
|
|
$dbr->expr(
|
|
|
|
'page_title',
|
|
|
|
IExpression::LIKE,
|
|
|
|
new LikeValue(
|
|
|
|
new LikeMatch( $pattern )
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
2011-11-10 17:04:38 +00:00
|
|
|
}
|
2011-09-27 16:17:06 +00:00
|
|
|
|
2023-12-19 15:59:32 +00:00
|
|
|
$result = $queryBuilder->caller( __METHOD__ )->fetchResultSet();
|
2024-04-19 13:10:15 +00:00
|
|
|
/** @var array{0:Title,1:string|false}[] $pages */
|
2016-03-07 14:53:32 +00:00
|
|
|
$pages = [];
|
2010-10-29 21:30:20 +00:00
|
|
|
foreach ( $result as $row ) {
|
2016-03-07 14:53:32 +00:00
|
|
|
$pages[] = [
|
2024-05-03 10:27:28 +00:00
|
|
|
Title::makeTitle( $row->page_namespace, $row->page_title ),
|
2024-11-12 01:56:47 +00:00
|
|
|
$row->actor_name
|
2016-03-07 14:53:32 +00:00
|
|
|
];
|
2008-05-09 18:53:09 +00:00
|
|
|
}
|
2011-09-27 16:17:06 +00:00
|
|
|
|
2015-11-30 11:30:08 +00:00
|
|
|
// Allows other extensions to provide pages to be nuked that don't use
|
|
|
|
// the recentchanges table the way mediawiki-core does
|
2021-09-18 08:56:33 +00:00
|
|
|
$this->getNukeHookRunner()->onNukeGetNewPages( $username, $pattern, $namespace, $limit, $pages );
|
2015-11-30 11:30:08 +00:00
|
|
|
|
|
|
|
// Re-enforcing the limit *after* the hook because other extensions
|
|
|
|
// may add and/or remove pages. We need to make sure we don't end up
|
|
|
|
// with more pages than $limit.
|
|
|
|
if ( count( $pages ) > $limit ) {
|
|
|
|
$pages = array_slice( $pages, 0, $limit );
|
|
|
|
}
|
|
|
|
|
2008-05-09 18:53:09 +00:00
|
|
|
return $pages;
|
|
|
|
}
|
|
|
|
|
2011-02-14 20:21:02 +00:00
|
|
|
/**
|
|
|
|
* Does the actual deletion of the pages.
|
2011-09-27 16:17:06 +00:00
|
|
|
*
|
2011-02-14 20:21:02 +00:00
|
|
|
* @param array $pages The pages to delete
|
2024-11-17 09:53:02 +00:00
|
|
|
* @param array $originalPageList The original list of pages shown to the user
|
2011-02-14 20:21:02 +00:00
|
|
|
* @param string $reason
|
2024-11-17 09:53:02 +00:00
|
|
|
* @param string $target
|
2012-08-16 13:19:22 +00:00
|
|
|
* @throws PermissionsError
|
2011-02-14 20:21:02 +00:00
|
|
|
*/
|
2024-11-17 09:53:02 +00:00
|
|
|
protected function doDelete(
|
|
|
|
array $pages, array $originalPageList, string $reason, string $target
|
|
|
|
): void {
|
2016-03-07 14:53:32 +00:00
|
|
|
$res = [];
|
2021-08-28 23:00:15 +00:00
|
|
|
$jobs = [];
|
2024-11-17 09:53:02 +00:00
|
|
|
$skippedRes = [];
|
2020-03-07 08:05:47 +00:00
|
|
|
$user = $this->getUser();
|
2024-11-17 09:53:02 +00:00
|
|
|
$queuedCount = 0;
|
2011-11-25 00:20:54 +00:00
|
|
|
|
2024-11-17 09:53:02 +00:00
|
|
|
// Get a list of all pages involved and what to do with them.
|
|
|
|
// Pages in $pages will always be deleted, even if they are not present in
|
|
|
|
// $originalPageList.
|
|
|
|
$willDeleteList = [];
|
2012-08-16 13:19:22 +00:00
|
|
|
foreach ( $pages as $page ) {
|
2024-11-17 09:53:02 +00:00
|
|
|
$willDeleteList[$page] = true;
|
|
|
|
}
|
|
|
|
foreach ( $originalPageList as $originalPage ) {
|
|
|
|
if ( !isset( $willDeleteList[$originalPage] ) ) {
|
|
|
|
$willDeleteList[$originalPage] = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$localRepo = $this->repoGroup->getLocalRepo();
|
|
|
|
foreach ( $willDeleteList as $page => $willDelete ) {
|
2015-12-11 23:09:54 +00:00
|
|
|
$title = Title::newFromText( $page );
|
2012-03-14 01:58:01 +00:00
|
|
|
|
2024-11-17 09:53:02 +00:00
|
|
|
if ( !$willDelete ) {
|
|
|
|
// If this page was skipped, add it to the list of skipped pages and move on.
|
|
|
|
$skippedRes[] = $this->msg(
|
|
|
|
'nuke-skipped',
|
|
|
|
wfEscapeWikiText( $title->getPrefixedText() ),
|
|
|
|
wfEscapeWikiText( $title->getTalkPageIfDefined() )
|
|
|
|
)->parse();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-11-30 11:30:08 +00:00
|
|
|
$deletionResult = false;
|
2021-09-18 08:56:33 +00:00
|
|
|
if ( !$this->getNukeHookRunner()->onNukeDeletePage( $title, $reason, $deletionResult ) ) {
|
2022-06-07 02:46:39 +00:00
|
|
|
$res[] = $this->msg(
|
|
|
|
$deletionResult ? 'nuke-deleted' : 'nuke-not-deleted',
|
|
|
|
wfEscapeWikiText( $title->getPrefixedText() )
|
|
|
|
)->parse();
|
2015-11-30 11:30:08 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2022-11-04 11:03:00 +00:00
|
|
|
$permission_errors = $this->permissionManager->getPermissionErrors( 'delete', $user, $title );
|
2012-03-14 01:58:01 +00:00
|
|
|
|
2016-03-07 14:53:32 +00:00
|
|
|
if ( $permission_errors !== [] ) {
|
2012-03-14 01:58:01 +00:00
|
|
|
throw new PermissionsError( 'delete', $permission_errors );
|
|
|
|
}
|
|
|
|
|
2021-08-11 18:29:27 +00:00
|
|
|
$file = $title->getNamespace() === NS_FILE ? $localRepo->newFile( $title ) : false;
|
2008-06-01 19:34:23 +00:00
|
|
|
if ( $file ) {
|
2022-11-04 13:09:25 +00:00
|
|
|
// Must be passed by reference
|
|
|
|
$oldimage = null;
|
2020-03-19 15:31:08 +00:00
|
|
|
$status = FileDeleteForm::doDelete(
|
2020-02-19 22:33:11 +00:00
|
|
|
$title,
|
|
|
|
$file,
|
|
|
|
$oldimage,
|
|
|
|
$reason,
|
|
|
|
false,
|
|
|
|
$user
|
2020-03-19 15:31:08 +00:00
|
|
|
);
|
2011-07-10 07:00:09 +00:00
|
|
|
} else {
|
2021-08-28 23:00:15 +00:00
|
|
|
$job = new DeletePageJob( [
|
|
|
|
'namespace' => $title->getNamespace(),
|
2021-10-30 13:08:06 +00:00
|
|
|
'title' => $title->getDBKey(),
|
2021-08-28 23:00:15 +00:00
|
|
|
'reason' => $reason,
|
2021-10-30 13:08:06 +00:00
|
|
|
'userId' => $user->getId(),
|
|
|
|
'wikiPageId' => $title->getId(),
|
|
|
|
'suppress' => false,
|
2024-11-08 21:26:15 +00:00
|
|
|
'tags' => '["nuke"]',
|
2021-10-30 13:08:06 +00:00
|
|
|
'logsubtype' => 'delete',
|
2021-08-28 23:00:15 +00:00
|
|
|
] );
|
|
|
|
$jobs[] = $job;
|
|
|
|
$status = 'job';
|
2011-07-10 07:00:09 +00:00
|
|
|
}
|
2012-03-14 01:58:01 +00:00
|
|
|
|
2022-08-30 08:40:10 +00:00
|
|
|
if ( $status === 'job' ) {
|
2022-06-07 02:46:39 +00:00
|
|
|
$res[] = $this->msg(
|
|
|
|
'nuke-deletion-queued',
|
|
|
|
wfEscapeWikiText( $title->getPrefixedText() )
|
|
|
|
)->parse();
|
2024-11-17 09:53:02 +00:00
|
|
|
$queuedCount++;
|
2008-06-01 19:34:23 +00:00
|
|
|
} else {
|
2022-06-07 02:46:39 +00:00
|
|
|
$res[] = $this->msg(
|
|
|
|
$status->isOK() ? 'nuke-deleted' : 'nuke-not-deleted',
|
|
|
|
wfEscapeWikiText( $title->getPrefixedText() )
|
|
|
|
)->parse();
|
2024-11-17 09:53:02 +00:00
|
|
|
if ( $status->isOK() ) {
|
|
|
|
$queuedCount++;
|
|
|
|
}
|
2008-06-01 19:34:23 +00:00
|
|
|
}
|
2008-05-09 18:53:09 +00:00
|
|
|
}
|
2011-11-25 00:20:54 +00:00
|
|
|
|
2021-08-28 23:00:15 +00:00
|
|
|
if ( $jobs ) {
|
2022-11-04 11:03:00 +00:00
|
|
|
$this->jobQueueGroup->push( $jobs );
|
2021-08-28 23:00:15 +00:00
|
|
|
}
|
|
|
|
|
2024-11-17 09:53:02 +00:00
|
|
|
// Show the main summary, regardless of whether we deleted pages or not.
|
|
|
|
if ( $target ) {
|
|
|
|
$this->getOutput()->addWikiMsg( 'nuke-delete-summary-user', $queuedCount, $target );
|
|
|
|
} else {
|
|
|
|
$this->getOutput()->addWikiMsg( 'nuke-delete-summary', $queuedCount );
|
|
|
|
}
|
|
|
|
if ( $queuedCount ) {
|
|
|
|
$this->getOutput()->addHTML(
|
|
|
|
"<ul>\n<li>" .
|
|
|
|
implode( "</li>\n<li>", $res ) .
|
|
|
|
"</li>\n</ul>\n"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if ( count( $skippedRes ) ) {
|
|
|
|
$this->getOutput()->addWikiMsg( 'nuke-skipped-summary', count( $skippedRes ) );
|
|
|
|
$this->getOutput()->addHTML(
|
|
|
|
"<ul>\n<li>" .
|
|
|
|
implode( "</li>\n<li>", $skippedRes ) .
|
|
|
|
"</li>\n</ul>\n"
|
|
|
|
);
|
|
|
|
}
|
2011-11-10 17:22:05 +00:00
|
|
|
$this->getOutput()->addWikiMsg( 'nuke-delete-more' );
|
2008-05-09 18:53:09 +00:00
|
|
|
}
|
2015-05-15 19:49:35 +00:00
|
|
|
|
2016-01-30 20:01:03 +00:00
|
|
|
/**
|
|
|
|
* Return an array of subpages beginning with $search that this special page will accept.
|
|
|
|
*
|
|
|
|
* @param string $search Prefix to search for
|
|
|
|
* @param int $limit Maximum number of results to return (usually 10)
|
|
|
|
* @param int $offset Number of results to skip (usually 0)
|
|
|
|
* @return string[] Matching subpages
|
|
|
|
*/
|
|
|
|
public function prefixSearchSubpages( $search, $limit, $offset ) {
|
2021-09-18 08:56:33 +00:00
|
|
|
$search = $this->userNameUtils->getCanonical( $search );
|
|
|
|
if ( !$search ) {
|
2016-01-30 20:01:03 +00:00
|
|
|
// No prefix suggestion for invalid user
|
2016-03-07 14:53:32 +00:00
|
|
|
return [];
|
2016-01-30 20:01:03 +00:00
|
|
|
}
|
2020-02-17 23:54:13 +00:00
|
|
|
|
2016-01-30 20:01:03 +00:00
|
|
|
// Autocomplete subpage as user list - public to allow caching
|
2021-09-18 08:56:33 +00:00
|
|
|
return $this->userNamePrefixSearch
|
|
|
|
->search( UserNamePrefixSearch::AUDIENCE_PUBLIC, $search, $limit, $offset );
|
2016-01-30 20:01:03 +00:00
|
|
|
}
|
|
|
|
|
2020-03-07 08:26:53 +00:00
|
|
|
/**
|
|
|
|
* Group Special:Nuke with pagetools
|
|
|
|
*
|
2024-05-11 06:48:28 +00:00
|
|
|
* @codeCoverageIgnore
|
2020-03-07 08:26:53 +00:00
|
|
|
* @return string
|
|
|
|
*/
|
2015-05-15 19:49:35 +00:00
|
|
|
protected function getGroupName() {
|
|
|
|
return 'pagetools';
|
|
|
|
}
|
2022-01-29 13:17:50 +00:00
|
|
|
|
2024-10-15 17:28:15 +00:00
|
|
|
private function getDeleteReason( WebRequest $request, string $target, bool $tempaccount = false ): string {
|
|
|
|
if ( $tempaccount ) {
|
|
|
|
$defaultReason = $this->msg( 'nuke-defaultreason-tempaccount' );
|
|
|
|
} else {
|
|
|
|
$defaultReason = $target === ''
|
|
|
|
? $this->msg( 'nuke-multiplepeople' )->inContentLanguage()->text()
|
|
|
|
: $this->msg( 'nuke-defaultreason', $target )->inContentLanguage()->text();
|
|
|
|
}
|
2022-01-29 13:17:50 +00:00
|
|
|
|
|
|
|
$dropdownSelection = $request->getText( 'wpDeleteReasonList', 'other' );
|
|
|
|
$reasonInput = $request->getText( 'wpReason', $defaultReason );
|
|
|
|
|
|
|
|
if ( $dropdownSelection === 'other' ) {
|
|
|
|
return $reasonInput;
|
|
|
|
} elseif ( $reasonInput !== '' ) {
|
|
|
|
// Entry from drop down menu + additional comment
|
|
|
|
$separator = $this->msg( 'colon-separator' )->inContentLanguage()->text();
|
|
|
|
return $dropdownSelection . $separator . $reasonInput;
|
|
|
|
} else {
|
|
|
|
return $dropdownSelection;
|
|
|
|
}
|
|
|
|
}
|
2021-09-18 08:56:33 +00:00
|
|
|
|
2024-04-19 13:10:15 +00:00
|
|
|
private function getNukeHookRunner(): NukeHookRunner {
|
|
|
|
$this->hookRunner ??= new NukeHookRunner( $this->getHookContainer() );
|
2021-09-18 08:56:33 +00:00
|
|
|
return $this->hookRunner;
|
|
|
|
}
|
2008-05-09 18:53:09 +00:00
|
|
|
}
|