Difference between revisions of "MediaWiki/NukePlus/Nuke body.php"

from HTYP, the free directory anyone can edit if they can prove to me that they're not a spambot
Jump to navigation Jump to search
(version from HostGator server)
 
(→‎Updates: to do)
 
(2 intermediate revisions by the same user not shown)
Line 1: Line 1:
 +
==Updates==
 +
* '''2013-01-20'''
 +
** now lists uploads that are not in recentchanges
 +
** edits belonging to users with "nonuke" ability are left unchecked in the list
 +
*** '''to do''': figure out how to leave them out altogether
 +
 +
==Code==
 
<php>
 
<php>
 
<?php
 
<?php
 
 
class SpecialNuke extends SpecialPage {
 
class SpecialNuke extends SpecialPage {
  
Line 103: Line 109:
 
*/
 
*/
 
protected function listForm( $username, $reason, $limit ) {
 
protected function listForm( $username, $reason, $limit ) {
global $wgUser;
+
global $wgUser; // need this for displaying "mark as patrolled" link
  
 
$out = $this->getOutput();
 
$out = $this->getOutput();
Line 170: Line 176:
 
*/
 
*/
 
list( $title, $userName, $idRev ) = $info;
 
list( $title, $userName, $idRev ) = $info;
 +
 +
// (Wzl) look up user rights -- are they trusted?
 +
$mwoUser = User::newFromName($userName);
 +
// later: query should fetch user ID as well as name -- names can change, in theory -- and this should use that
 +
$doCheck = !$mwoUser->isAllowed('nonuke');
  
 
$image = $title->getNamespace() == NS_IMAGE ? wfLocalFile( $title ) : false;
 
$image = $title->getNamespace() == NS_IMAGE ? wfLocalFile( $title ) : false;
 
$thumb = $image && $image->exists() ? $image->transform( array( 'width' => 120, 'height' => 120 ), 0 ) : false;
 
$thumb = $image && $image->exists() ? $image->transform( array( 'width' => 120, 'height' => 120 ), 0 ) : false;
// for marking as patrolled
+
// (Wzl) for marking as patrolled
$token = $wgUser->getEditToken( $idRev );
+
if (is_null($idRev)) {
 +
$token = NULL;
 +
} else {
 +
$token = $wgUser->getEditToken( $idRev );
 +
}
  
 
$out->addHTML( '<li>' .
 
$out->addHTML( '<li>' .
 
Xml::check(
 
Xml::check(
 
'pages[]',
 
'pages[]',
true,
+
$doCheck,
 
array( 'value' =>  $title->getPrefixedDbKey() )
 
array( 'value' =>  $title->getPrefixedDbKey() )
 
) .
 
) .
Line 192: Line 207:
 
array(),
 
array(),
 
array( 'action' => 'history' )
 
array( 'action' => 'history' )
) .
+
)
// show "mark as patrolled" link
+
);
',&nbsp;' . Linker::link(
+
if (!is_null($token)) {
$title,
+
// (Wzl) show "mark as patrolled" link
wfMsgHtml( 'markaspatrolledtext' ),
+
$out->addHTML(',&nbsp;' . Linker::link(
array(),
+
$title,
array(
+
wfMsgHtml( 'markaspatrolledtext' ),
'action' => 'markpatrolled',
+
array(),
'rcid' => $idRev,
+
array(
'token' => $token,
+
'action' => 'markpatrolled',
),
+
'rcid' => $idRev,
array( 'known', 'noclasses' )
+
'token' => $token,
) .
+
),
+
array( 'known', 'noclasses' )
// END
+
)
")</li>\n" );
+
);
 +
}
 +
// (Wzl) END
 +
$out->addHTML(")</li>\n");
 
}
 
}
  
Line 237: Line 255:
 
// original
 
// original
 
//$where = array( "(rc_new = 1) OR (rc_log_type = 'upload' AND rc_log_action = 'upload')" );
 
//$where = array( "(rc_new = 1) OR (rc_log_type = 'upload' AND rc_log_action = 'upload')" );
// Wzl improvement
+
// Wzl improvement #1
$where = array( "(page_id IS NOT NULL) AND (NOT rc_patrolled) AND ((rc_new = 1) OR (rc_log_type = 'upload' AND rc_log_action = 'upload'))" );
+
$sqlFilt =  
 +
"(page_id IS NOT NULL)"
 +
." AND (NOT rc_patrolled)"
 +
." AND ((rc_new = 1)"
 +
." OR (rc_log_type = 'upload' AND rc_log_action = 'upload'))";
 +
$where = array( $sqlFilt );
  
 
+
$hasUserFilter = ($username !== '');
if ( $username === '' ) {
+
if ( $hasUserFilter ) {
 +
$where['rc_user_text'] = $username;
 +
} else {
 
$what[] = 'rc_user_text';
 
$what[] = 'rc_user_text';
} else {
 
$where['rc_user_text'] = $username;
 
 
}
 
}
  
Line 270: Line 293:
 
$pages[] = array(
 
$pages[] = array(
 
Title::makeTitle( $row->rc_namespace, $row->rc_title ),
 
Title::makeTitle( $row->rc_namespace, $row->rc_title ),
$username === '' ? $row->rc_user_text : false,
+
$hasUserFilter ? FALSE : $row->rc_user_text,
$row->rc_id // for marking as patrolled
+
$row->rc_id // (Wzl) for marking as patrolled
 +
);
 +
}
 +
 
 +
// (Wzl) if limit not exceeded, get new uploads too
 +
$qPages = count($pages);
 +
if ($qPages < $limit) {
 +
$what = array(
 +
'img_name',
 +
'img_timestamp',
 +
'rc_id', // for marking as patrolled
 +
);
 +
$where = array('(NOT rc_patrolled) OR (rc_id IS NULL)');
 +
if ( $hasUserFilter ) {
 +
$where['img_user_text'] = $username;
 +
} else {
 +
$what[] = 'img_user_text';
 +
}
 +
$group = implode( ', ', $what );
 +
 
 +
$result = $dbr->select( 'image LEFT JOIN recentchanges ON img_name=rc_title',
 +
$what,
 +
$where,
 +
__METHOD__,
 +
array(
 +
'ORDER BY' => 'img_timestamp DESC',
 +
'GROUP BY' => $group,
 +
'LIMIT' => $limit - $qPages
 +
)
 
);
 
);
 +
foreach ( $result as $row ) {
 +
$pages[] = array(
 +
Title::makeTitle( NS_FILE, $row->img_name ),
 +
$hasUserFilter ? FALSE : $row->img_user_text,
 +
$row->rc_id
 +
);
 +
}
 
}
 
}
 +
// (Wzl) END
 +
 
return $pages;
 
return $pages;
 
}
 
}
Line 316: Line 376:
 
}
 
}
  
}
+
}</php>
</php>
 

Latest revision as of 13:02, 21 January 2013

Updates

  • 2013-01-20
    • now lists uploads that are not in recentchanges
    • edits belonging to users with "nonuke" ability are left unchecked in the list
      • to do: figure out how to leave them out altogether

Code

<php> <?php class SpecialNuke extends SpecialPage {

public function __construct() { parent::__construct( 'Nuke', 'nuke' ); }

public function execute( $par ) { if ( !$this->userCanExecute( $this->getUser() ) ) { $this->displayRestrictionError(); } $this->setHeaders(); $this->outputHeader();

if ( $this->getUser()->isBlocked() ) { $block = $this->getUser()->getBlock(); throw new UserBlockedError( $block ); } $this->checkReadOnly();

$req = $this->getRequest();

$target = trim( $req->getText( 'target', $par ) );

// Normalise name if ( $target !== ) { $user = User::newFromName( $target ); if ( $user ) $target = $user->getName(); }

$reason = $req->getText( 'wpReason', $target ===  ? wfMsgForContent( 'nuke-multiplepeople' ) : wfMsgForContent( 'nuke-defaultreason', "$target" ) );

if( $req->wasPosted() && $this->getUser()->matchEditToken( $req->getVal( 'wpEditToken' ) ) ) {

if ( $req->getVal( 'action' ) == 'delete' ) { $pages = $req->getArray( 'pages' );

if( $pages ) { $this->doDelete( $pages, $reason ); return; } } elseif ( $req->getVal( 'action' ) == 'submit' ) { $this->listForm( $target, $reason, $req->getInt( 'limit', 500 ) ); } else { $this->promptForm(); } } elseif ( $target === ) { $this->promptForm(); } else { $this->listForm( $target, $reason, $req->getInt( 'limit', 500 ) ); } }

/** * Prompt for a username or IP address. * * @param $userName string */ protected function promptForm( $userName = ) { $out = $this->getOutput();

$out->addWikiMsg( 'nuke-tools' );

$out->addHTML( Xml::openElement( 'form', array( 'action' => $this->getTitle()->getLocalURL( 'action=submit' ), 'method' => 'post' ) )

. '

' . '' . '' . '' . '' . '' . '' . '' . '' . '' . '' . '' .'
' . Xml::label( wfMsg( 'nuke-userorip' ), 'nuke-target' ) . '' . Xml::input( 'target', 40, $userName, array( 'id' => 'nuke-target' ) ) . '
' . Xml::label( wfMsg( 'nuke-pattern' ), 'nuke-pattern' ) . '' . Xml::input( 'pattern', 40, , array( 'id' => 'nuke-pattern' ) ) . '
' . Xml::label( wfMsg( 'nuke-maxpages' ), 'nuke-limit' ) . '' . Xml::input( 'limit', 7, '500', array( 'id' => 'nuke-limit' ) ) . '
' . Xml::submitButton( wfMsg( 'nuke-submit-user' ) ) . '

'

. Html::hidden( 'wpEditToken', $this->getUser()->editToken() ) . Xml::closeElement( 'form' ) ); }

/** * Display list of pages to delete. * * @param string $username * @param string $reason * @param integer $limit */ protected function listForm( $username, $reason, $limit ) { global $wgUser; // need this for displaying "mark as patrolled" link

$out = $this->getOutput();

$pages = $this->getNewPages( $username, $limit );

if( count( $pages ) == 0 ) { if ( $username === ) { $out->addWikiMsg( 'nuke-nopages-global' ); } else { $out->addWikiMsg( 'nuke-nopages', $username ); }

$this->promptForm( $username ); return; }

if ( $username === ) { $out->addWikiMsg( 'nuke-list-multiple' ); } else { $out->addWikiMsg( 'nuke-list', $username ); }

$nuke = $this->getTitle();

$out->addModules( 'ext.nuke' );

$out->addHTML( Xml::openElement( 'form', array( 'action' => $nuke->getLocalURL( 'action=delete' ), 'method' => 'post', 'name' => 'nukelist') ) . Html::hidden( 'wpEditToken', $this->getUser()->editToken() ) . Xml::tags( 'p', null, Xml::inputLabel( wfMsg( 'deletecomment' ), 'wpReason', 'wpReason', 70, $reason ) ) );

// Select: All, None $links = array(); $links[] = '<a href="#" id="toggleall">' . wfMsg( 'powersearch-toggleall' ) . '</a>'; $links[] = '<a href="#" id="togglenone">' . wfMsg( 'powersearch-togglenone' ) . '</a>'; $out->addHTML( Xml::tags( 'p', null, wfMsg( 'nuke-select', $this->getLanguage()->commaList( $links ) ) ) );

// Delete button $out->addHTML( Xml::submitButton( wfMsg( 'nuke-submit-delete' ) ) );

$out->addHTML( '

    ' ); foreach( $pages as $info ) { /** * @var $title Title */ list( $title, $userName, $idRev ) = $info; // (Wzl) look up user rights -- are they trusted? $mwoUser = User::newFromName($userName); // later: query should fetch user ID as well as name -- names can change, in theory -- and this should use that $doCheck = !$mwoUser->isAllowed('nonuke'); $image = $title->getNamespace() == NS_IMAGE ? wfLocalFile( $title ) : false; $thumb = $image && $image->exists() ? $image->transform( array( 'width' => 120, 'height' => 120 ), 0 ) : false; // (Wzl) for marking as patrolled if (is_null($idRev)) { $token = NULL; } else { $token = $wgUser->getEditToken( $idRev ); } $out->addHTML( '
  • ' . Xml::check( 'pages[]', $doCheck, array( 'value' => $title->getPrefixedDbKey() ) ) . ' ' . ( $thumb ? $thumb->toHtml( array( 'desc-link' => true ) ) : ) . Linker::linkKnown( $title ) . ' (' . ( $userName ? wfMsgExt( 'nuke-editby', 'parseinline', $userName ) . ', ' : ) . Linker::linkKnown( $title, wfMsg( 'nuke-viewchanges' ), array(), array( 'action' => 'history' ) ) ); if (!is_null($token)) { // (Wzl) show "mark as patrolled" link $out->addHTML(', ' . Linker::link( $title, wfMsgHtml( 'markaspatrolledtext' ), array(), array( 'action' => 'markpatrolled', 'rcid' => $idRev, 'token' => $token, ), array( 'known', 'noclasses' ) ) ); } // (Wzl) END $out->addHTML(")
  • \n"); } $out->addHTML( "

\n" .

Xml::submitButton( wfMsg( 'nuke-submit-delete' ) ) . '</form>' ); }

/** * Gets a list of new pages by the specified user or everyone when none is specified. * * @param string $username * @param integer $limit * * @return array */ protected function getNewPages( $username, $limit ) { $dbr = wfGetDB( DB_SLAVE );

$what = array( 'rc_namespace', 'rc_title', 'rc_timestamp', 'rc_id', // (Wzl) for marking as patrolled );

// original //$where = array( "(rc_new = 1) OR (rc_log_type = 'upload' AND rc_log_action = 'upload')" ); // Wzl improvement #1 $sqlFilt = "(page_id IS NOT NULL)" ." AND (NOT rc_patrolled)" ." AND ((rc_new = 1)" ." OR (rc_log_type = 'upload' AND rc_log_action = 'upload'))"; $where = array( $sqlFilt );

$hasUserFilter = ($username !== ); if ( $hasUserFilter ) { $where['rc_user_text'] = $username; } else { $what[] = 'rc_user_text'; }

$pattern = $this->getRequest()->getText( 'pattern' ); if ( !is_null( $pattern ) && trim( $pattern ) !== ) { $where[] = 'rc_title LIKE ' . $dbr->addQuotes( $pattern ); } $group = implode( ', ', $what );

// (Wzl) was just 'recentchanges', but we need to look at 'page' to make sure the page actually still exists (is this a MW bug?) $result = $dbr->select( 'recentchanges LEFT JOIN page ON (rc_title=page_title) AND (rc_namespace=page_namespace)', $what, $where, __METHOD__, array( 'ORDER BY' => 'rc_timestamp DESC', 'GROUP BY' => $group, 'LIMIT' => $limit ) );

$pages = array();

foreach ( $result as $row ) { $pages[] = array( Title::makeTitle( $row->rc_namespace, $row->rc_title ), $hasUserFilter ? FALSE : $row->rc_user_text, $row->rc_id // (Wzl) for marking as patrolled ); }

// (Wzl) if limit not exceeded, get new uploads too $qPages = count($pages); if ($qPages < $limit) { $what = array( 'img_name', 'img_timestamp', 'rc_id', // for marking as patrolled ); $where = array('(NOT rc_patrolled) OR (rc_id IS NULL)'); if ( $hasUserFilter ) { $where['img_user_text'] = $username; } else { $what[] = 'img_user_text'; } $group = implode( ', ', $what );

$result = $dbr->select( 'image LEFT JOIN recentchanges ON img_name=rc_title', $what, $where, __METHOD__, array( 'ORDER BY' => 'img_timestamp DESC', 'GROUP BY' => $group, 'LIMIT' => $limit - $qPages ) ); foreach ( $result as $row ) { $pages[] = array( Title::makeTitle( NS_FILE, $row->img_name ), $hasUserFilter ? FALSE : $row->img_user_text, $row->rc_id ); } } // (Wzl) END

return $pages; }

/** * Does the actual deletion of the pages. * * @param array $pages The pages to delete * @param string $reason */ protected function doDelete( array $pages, $reason ) { $res = array();

foreach( $pages as $page ) { $title = Title::newFromURL( $page ); $file = $title->getNamespace() == NS_FILE ? wfLocalFile( $title ) : false;

$permission_errors = $title->getUserPermissionsErrors( 'delete', $this->getUser());

if ( count( $permission_errors )) { throw new PermissionsError( 'delete', $permission_errors ); }

if ( $file ) { $oldimage = null; // Must be passed by reference $ok = FileDeleteForm::doDelete( $title, $file, $oldimage, $reason, false )->isOK(); } else { $article = new Article( $title, 0 ); $ok = $article->doDeleteArticle( $reason ); }

if ( $ok ) { $res[] = wfMsgExt( 'nuke-deleted', array( 'parseinline' ), $title->getPrefixedText() ); } else { $res[] = wfMsgExt( 'nuke-not-deleted', array( 'parseinline' ), $title->getPrefixedText() ); } }

$this->getOutput()->addHTML( "

    \n
  • " . implode( "
  • \n
  • ", $res ) . "
  • \n

\n" );

$this->getOutput()->addWikiMsg( 'nuke-delete-more' ); }

}</php>