2008-06-27 06:18:51 +00:00
|
|
|
<?php
|
2010-08-19 21:12:09 +00:00
|
|
|
if ( !defined( 'MEDIAWIKI' ) ) {
|
2008-06-27 06:18:51 +00:00
|
|
|
die();
|
2010-08-19 21:12:09 +00:00
|
|
|
}
|
2008-06-27 06:18:51 +00:00
|
|
|
|
|
|
|
class AbuseFilterHooks {
|
2011-08-26 20:12:34 +00:00
|
|
|
// So far, all of the error message out-params for these hooks accept HTML.
|
|
|
|
// Hooray!
|
2011-08-24 22:11:52 +00:00
|
|
|
|
|
|
|
/**
|
2011-10-25 09:41:25 +00:00
|
|
|
* Entry points for MediaWiki hook 'EditFilterMerged'
|
|
|
|
*
|
|
|
|
* @param $editor EditPage instance (object)
|
|
|
|
* @param $text Content of the edit box
|
|
|
|
* @param &$error Error message to return
|
|
|
|
* @param $summary Edit summary for page
|
2011-08-24 22:11:52 +00:00
|
|
|
* @return bool
|
|
|
|
*/
|
2009-10-07 13:57:06 +00:00
|
|
|
public static function onEditFilterMerged( $editor, $text, &$error, $summary ) {
|
2008-06-27 06:18:51 +00:00
|
|
|
// Load vars
|
2009-02-26 12:15:14 +00:00
|
|
|
$vars = new AbuseFilterVariableHolder;
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2009-07-03 14:01:43 +00:00
|
|
|
// Check for null edits.
|
2009-09-17 09:52:40 +00:00
|
|
|
$oldtext = '';
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2011-11-09 08:36:26 +00:00
|
|
|
$article = $editor->getArticle();
|
|
|
|
if ( $article->exists() ) {
|
2011-10-25 12:46:05 +00:00
|
|
|
// Make sure we load the latest text saved in database (bug 31656)
|
2011-11-09 08:36:26 +00:00
|
|
|
$oldtext = $article->getRevision()->getRawText();
|
2009-09-17 09:52:40 +00:00
|
|
|
}
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2011-10-25 12:46:05 +00:00
|
|
|
// Cache article object so we can share a parse operation
|
|
|
|
$title = $editor->mTitle;
|
|
|
|
$articleCacheKey = $title->getNamespace() . ':' . $title->getText();
|
2011-11-09 08:36:26 +00:00
|
|
|
AFComputedVariable::$articleCache[$articleCacheKey] = $article;
|
2011-10-25 12:46:05 +00:00
|
|
|
|
2011-10-25 12:12:02 +00:00
|
|
|
if ( strcmp( $oldtext, $text ) == 0 ) {
|
|
|
|
// Don't trigger for null edits.
|
2009-07-03 14:01:43 +00:00
|
|
|
return true;
|
|
|
|
}
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2008-06-27 06:18:51 +00:00
|
|
|
global $wgUser;
|
2009-02-26 12:15:14 +00:00
|
|
|
$vars->addHolder( AbuseFilter::generateUserVars( $wgUser ) );
|
2011-11-09 08:36:26 +00:00
|
|
|
$vars->addHolder( AbuseFilter::generateTitleVars( $title , 'ARTICLE' ) );
|
2009-02-26 12:15:14 +00:00
|
|
|
$vars->setVar( 'ACTION', 'edit' );
|
|
|
|
$vars->setVar( 'SUMMARY', $summary );
|
2009-03-04 02:04:48 +00:00
|
|
|
$vars->setVar( 'minor_edit', $editor->minoredit );
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2009-07-03 14:01:43 +00:00
|
|
|
$vars->setVar( 'old_wikitext', $oldtext );
|
2009-03-17 20:44:53 +00:00
|
|
|
$vars->setVar( 'new_wikitext', $text );
|
2008-06-27 06:18:51 +00:00
|
|
|
|
2011-11-09 08:36:26 +00:00
|
|
|
$vars->addHolder( AbuseFilter::getEditVars( $title, $article ) );
|
2009-01-28 01:12:34 +00:00
|
|
|
|
2011-11-09 08:36:26 +00:00
|
|
|
$filter_result = AbuseFilter::filterAction( $vars, $title );
|
2008-10-24 08:58:32 +00:00
|
|
|
|
2010-02-13 14:10:36 +00:00
|
|
|
if ( $filter_result !== true ) {
|
2009-01-31 01:59:13 +00:00
|
|
|
global $wgOut;
|
|
|
|
$wgOut->addHTML( $filter_result );
|
|
|
|
$editor->showEditForm();
|
|
|
|
return false;
|
2008-06-27 09:26:54 +00:00
|
|
|
}
|
2008-06-27 06:18:51 +00:00
|
|
|
return true;
|
|
|
|
}
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2008-06-27 06:18:51 +00:00
|
|
|
public static function onGetAutoPromoteGroups( $user, &$promote ) {
|
|
|
|
global $wgMemc;
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2008-06-27 06:18:51 +00:00
|
|
|
$key = AbuseFilter::autoPromoteBlockKey( $user );
|
2009-10-07 13:57:06 +00:00
|
|
|
|
|
|
|
if ( $wgMemc->get( $key ) ) {
|
2008-06-27 06:18:51 +00:00
|
|
|
$promote = array();
|
|
|
|
}
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2008-06-27 06:18:51 +00:00
|
|
|
return true;
|
|
|
|
}
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2008-08-07 13:53:18 +00:00
|
|
|
public static function onAbortMove( $oldTitle, $newTitle, $user, &$error, $reason ) {
|
2009-02-26 12:15:14 +00:00
|
|
|
$vars = new AbuseFilterVariableHolder;
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2008-06-27 06:18:51 +00:00
|
|
|
global $wgUser;
|
2009-10-07 13:57:06 +00:00
|
|
|
$vars->addHolder(
|
|
|
|
AbuseFilterVariableHolder::merge(
|
|
|
|
AbuseFilter::generateUserVars( $wgUser ),
|
|
|
|
AbuseFilter::generateTitleVars( $oldTitle, 'MOVED_FROM' ),
|
|
|
|
AbuseFilter::generateTitleVars( $newTitle, 'MOVED_TO' )
|
|
|
|
)
|
|
|
|
);
|
2009-02-26 12:15:14 +00:00
|
|
|
$vars->setVar( 'SUMMARY', $reason );
|
|
|
|
$vars->setVar( 'ACTION', 'move' );
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2008-06-27 06:18:51 +00:00
|
|
|
$filter_result = AbuseFilter::filterAction( $vars, $oldTitle );
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2008-06-27 08:34:34 +00:00
|
|
|
$error = $filter_result;
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2008-06-27 09:18:45 +00:00
|
|
|
return $filter_result == '' || $filter_result === true;
|
2008-06-27 06:18:51 +00:00
|
|
|
}
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2008-08-07 13:53:18 +00:00
|
|
|
public static function onArticleDelete( &$article, &$user, &$reason, &$error ) {
|
2009-02-26 12:15:14 +00:00
|
|
|
$vars = new AbuseFilterVariableHolder;
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2008-06-27 06:18:51 +00:00
|
|
|
global $wgUser;
|
2009-02-26 12:15:14 +00:00
|
|
|
$vars->addHolder( AbuseFilter::generateUserVars( $wgUser ) );
|
|
|
|
$vars->addHolder( AbuseFilter::generateTitleVars( $article->mTitle, 'ARTICLE' ) );
|
|
|
|
$vars->setVar( 'SUMMARY', $reason );
|
|
|
|
$vars->setVar( 'ACTION', 'delete' );
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2008-06-27 08:16:11 +00:00
|
|
|
$filter_result = AbuseFilter::filterAction( $vars, $article->mTitle );
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2008-06-27 08:34:34 +00:00
|
|
|
$error = $filter_result;
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2008-06-27 09:18:45 +00:00
|
|
|
return $filter_result == '' || $filter_result === true;
|
2008-06-27 06:18:51 +00:00
|
|
|
}
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2008-08-07 13:53:18 +00:00
|
|
|
public static function onAbortNewAccount( $user, &$message ) {
|
2009-10-07 13:57:06 +00:00
|
|
|
if ( $user->getName() == wfMsgForContent( 'abusefilter-blocker' ) ) {
|
2008-06-27 09:49:26 +00:00
|
|
|
$message = wfMsg( 'abusefilter-accountreserved' );
|
|
|
|
return false;
|
|
|
|
}
|
2009-03-01 01:49:34 +00:00
|
|
|
$vars = new AbuseFilterVariableHolder;
|
2011-08-24 00:29:26 +00:00
|
|
|
// Add variables only for a registered user, so IP addresses of
|
|
|
|
// new users won't be exposed
|
|
|
|
global $wgUser;
|
|
|
|
if ( $wgUser->getId() ) {
|
|
|
|
$vars->addHolder( AbuseFilter::generateUserVars( $wgUser ) );
|
|
|
|
}
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2009-02-26 12:15:14 +00:00
|
|
|
$vars->setVar( 'ACTION', 'createaccount' );
|
|
|
|
$vars->setVar( 'ACCOUNTNAME', $user->getName() );
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2010-02-13 14:10:36 +00:00
|
|
|
$filter_result = AbuseFilter::filterAction(
|
2009-02-07 09:34:11 +00:00
|
|
|
$vars, SpecialPage::getTitleFor( 'Userlogin' ) );
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2008-06-27 08:34:34 +00:00
|
|
|
$message = $filter_result;
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2008-06-27 09:18:45 +00:00
|
|
|
return $filter_result == '' || $filter_result === true;
|
2008-06-27 06:18:51 +00:00
|
|
|
}
|
2009-01-23 19:23:19 +00:00
|
|
|
|
2009-01-28 19:08:18 +00:00
|
|
|
public static function onRecentChangeSave( $recentChange ) {
|
2009-10-07 13:57:06 +00:00
|
|
|
$title = Title::makeTitle(
|
2010-02-13 14:10:36 +00:00
|
|
|
$recentChange->mAttribs['rc_namespace'],
|
2009-10-07 13:57:06 +00:00
|
|
|
$recentChange->mAttribs['rc_title']
|
|
|
|
);
|
2010-02-13 14:10:36 +00:00
|
|
|
$action = $recentChange->mAttribs['rc_log_type'] ?
|
2009-02-07 09:34:11 +00:00
|
|
|
$recentChange->mAttribs['rc_log_type'] : 'edit';
|
2009-01-28 19:08:18 +00:00
|
|
|
$actionID = implode( '-', array(
|
|
|
|
$title->getPrefixedText(), $recentChange->mAttribs['rc_user_text'], $action
|
|
|
|
) );
|
2009-01-23 19:23:19 +00:00
|
|
|
|
2010-02-13 14:10:36 +00:00
|
|
|
if ( !empty( AbuseFilter::$tagsToSet[$actionID] )
|
|
|
|
&& count( $tags = AbuseFilter::$tagsToSet[$actionID] ) )
|
2009-02-07 09:34:11 +00:00
|
|
|
{
|
2009-10-07 13:57:06 +00:00
|
|
|
ChangeTags::addTags(
|
|
|
|
$tags,
|
|
|
|
$recentChange->mAttribs['rc_id'],
|
2009-02-07 09:34:11 +00:00
|
|
|
$recentChange->mAttribs['rc_this_oldid'],
|
2009-10-07 13:57:06 +00:00
|
|
|
$recentChange->mAttribs['rc_logid']
|
|
|
|
);
|
2009-01-28 19:08:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static function onListDefinedTags( &$emptyTags ) {
|
2010-02-13 14:10:36 +00:00
|
|
|
# This is a pretty awful hack.
|
2009-01-28 19:08:18 +00:00
|
|
|
$dbr = wfGetDB( DB_SLAVE );
|
|
|
|
|
2009-10-07 13:57:06 +00:00
|
|
|
$res = $dbr->select(
|
|
|
|
array( 'abuse_filter_action', 'abuse_filter' ),
|
2010-02-13 14:10:36 +00:00
|
|
|
'afa_parameters',
|
2009-10-07 13:57:06 +00:00
|
|
|
array( 'afa_consequence' => 'tag', 'af_enabled' => true ),
|
|
|
|
__METHOD__,
|
|
|
|
array(),
|
|
|
|
array( 'abuse_filter' => array( 'INNER JOIN', 'afa_filter=af_id' ) )
|
|
|
|
);
|
2009-01-28 19:08:18 +00:00
|
|
|
|
2011-02-10 17:25:25 +00:00
|
|
|
foreach ( $res as $row ) {
|
2009-10-07 13:57:06 +00:00
|
|
|
$emptyTags = array_filter(
|
|
|
|
array_merge( explode( "\n", $row->afa_parameters ), $emptyTags )
|
2009-02-07 09:34:11 +00:00
|
|
|
);
|
2009-01-28 19:08:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2009-01-29 00:37:53 +00:00
|
|
|
|
2011-02-10 17:32:57 +00:00
|
|
|
/**
|
|
|
|
* @param $updater DatabaseUpdater
|
|
|
|
* @return bool
|
|
|
|
*/
|
2010-10-03 15:51:04 +00:00
|
|
|
public static function onLoadExtensionSchemaUpdates( $updater = null ) {
|
2009-01-30 18:51:37 +00:00
|
|
|
$dir = dirname( __FILE__ );
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2011-12-27 16:35:30 +00:00
|
|
|
if ( $updater->getDB()->getType() == 'mysql' || $updater->getDB()->getType() == 'sqlite' ) {
|
|
|
|
if ( $updater->getDB()->getType() == 'mysql' ) {
|
|
|
|
$updater->addExtensionUpdate( array( 'addTable', 'abuse_filter', "$dir/abusefilter.tables.sql", true ) );
|
|
|
|
$updater->addExtensionUpdate( array( 'addTable', 'abuse_filter_history', "$dir/db_patches/patch-abuse_filter_history.sql", true ) );
|
2011-11-08 03:07:01 +00:00
|
|
|
} else {
|
2011-12-27 16:35:30 +00:00
|
|
|
$updater->addExtensionUpdate( array( 'addTable', 'abuse_filter', "$dir/abusefilter.tables.sqlite.sql", true ) );
|
|
|
|
$updater->addExtensionUpdate( array( 'addTable', 'abuse_filter_history', "$dir/db_patches/patch-abuse_filter_history.sqlite.sql", true ) );
|
2010-10-03 15:51:04 +00:00
|
|
|
}
|
2011-12-27 16:35:30 +00:00
|
|
|
$updater->addExtensionUpdate( array( 'addField', 'abuse_filter_history', 'afh_changed_fields', "$dir/db_patches/patch-afh_changed_fields.sql", true ) );
|
|
|
|
$updater->addExtensionUpdate( array( 'addField', 'abuse_filter', 'af_deleted', "$dir/db_patches/patch-af_deleted.sql", true ) );
|
|
|
|
$updater->addExtensionUpdate( array( 'addField', 'abuse_filter', 'af_actions', "$dir/db_patches/patch-af_actions.sql", true ) );
|
|
|
|
$updater->addExtensionUpdate( array( 'addField', 'abuse_filter', 'af_global', "$dir/db_patches/patch-global_filters.sql", true ) );
|
|
|
|
if ( $updater->getDB()->getType() == 'mysql' ) {
|
|
|
|
$updater->addExtensionUpdate( array( 'addIndex', 'abuse_filter_log', 'filter_timestamp', "$dir/db_patches/patch-fix-indexes.sql", true ) );
|
|
|
|
} else {
|
|
|
|
$updater->addExtensionUpdate( array( 'addIndex', 'abuse_filter_log', 'afl_filter_timestamp', "$dir/db_patches/patch-fix-indexes.sqlite.sql", true ) );
|
|
|
|
}
|
|
|
|
} elseif ( $updater->getDB()->getType() == 'postgres' ) {
|
|
|
|
$updater->addExtensionUpdate( array( 'addTable', 'abuse_filter', "$dir/abusefilter.tables.pg.sql", true ) );
|
|
|
|
$updater->addExtensionUpdate( array( 'addTable', 'abuse_filter_history', "$dir/db_patches/patch-abuse_filter_history.pg.sql", true ) );
|
|
|
|
$updater->addExtensionUpdate( array( 'addPgField', 'abuse_filter', 'af_actions', "TEXT NOT NULL DEFAULT ''" ) );
|
|
|
|
$updater->addExtensionUpdate( array( 'addPgField', 'abuse_filter', 'af_deleted', 'SMALLINT NOT NULL DEFAULT 0' ) );
|
|
|
|
$updater->addExtensionUpdate( array( 'addPgField', 'abuse_filter', 'af_global', 'SMALLINT NOT NULL DEFAULT 0' ) );
|
|
|
|
$updater->addExtensionUpdate( array( 'addPgField', 'abuse_filter_log', 'afl_wiki', 'TEXT' ) );
|
|
|
|
$updater->addExtensionUpdate( array( 'addPgField', 'abuse_filter_log', 'afl_deleted', 'SMALLINT' ) );
|
|
|
|
$updater->addExtensionUpdate( array( 'changeField', 'abuse_filter_log', 'afl_filter', 'TEXT' ) );
|
|
|
|
$updater->addExtensionUpdate( array( 'addPgExtIndex', 'abuse_filter_log', 'abuse_filter_log_ip', "(afl_ip)" ) );
|
2010-10-03 15:51:04 +00:00
|
|
|
} else {
|
2011-12-27 16:35:30 +00:00
|
|
|
throw new MWException("No known Schema updates.");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create the Abuse Filter user.
|
|
|
|
$user = User::newFromName( wfMsgForContent( 'abusefilter-blocker' ) );
|
|
|
|
|
|
|
|
if ( !$updater->updateRowExists( 'create abusefilter-blocker-user' ) ) {
|
|
|
|
if ( !$user->getId() ) {
|
|
|
|
$user->addToDatabase();
|
|
|
|
$user->saveSettings();
|
|
|
|
# Increment site_stats.ss_users
|
|
|
|
$ssu = new SiteStatsUpdate( 0, 0, 0, 0, 1 );
|
|
|
|
$ssu->doUpdate();
|
2011-11-08 03:07:01 +00:00
|
|
|
} else {
|
2011-12-27 16:35:30 +00:00
|
|
|
// Sorry dude, we need this account.
|
|
|
|
$user->setPassword( null );
|
|
|
|
$user->setEmail( null );
|
|
|
|
$user->saveSettings();
|
2010-10-03 15:51:04 +00:00
|
|
|
}
|
2011-12-27 16:35:30 +00:00
|
|
|
$updater->insertUpdateRow( 'create abusefilter-blocker-user' );
|
|
|
|
# Promote user so it doesn't look too crazy.
|
|
|
|
$user->addGroup( 'sysop' );
|
2009-03-23 14:18:35 +00:00
|
|
|
}
|
2011-12-27 16:35:30 +00:00
|
|
|
|
2009-01-29 00:37:53 +00:00
|
|
|
return true;
|
|
|
|
}
|
2009-10-07 13:57:06 +00:00
|
|
|
|
|
|
|
public static function onContributionsToolLinks( $id, $nt, &$tools ) {
|
|
|
|
global $wgUser;
|
2010-02-13 14:10:36 +00:00
|
|
|
if ( $wgUser->isAllowed( 'abusefilter-log' ) ) {
|
2009-10-07 13:57:06 +00:00
|
|
|
$sk = $wgUser->getSkin();
|
|
|
|
$tools[] = $sk->link(
|
|
|
|
SpecialPage::getTitleFor( 'AbuseLog' ),
|
|
|
|
wfMsg( 'abusefilter-log-linkoncontribs' ),
|
|
|
|
array( 'title' =>
|
|
|
|
wfMsgExt( 'abusefilter-log-linkoncontribs-text', 'parseinline' ) ),
|
|
|
|
array( 'wpSearchUser' => $nt->getText() )
|
|
|
|
);
|
2009-04-24 01:53:12 +00:00
|
|
|
}
|
|
|
|
return true;
|
2009-10-07 13:57:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public static function onUploadVerification( $saveName, $tempName, &$error ) {
|
2009-06-03 15:10:44 +00:00
|
|
|
$vars = new AbuseFilterVariableHolder;
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2009-06-03 15:10:44 +00:00
|
|
|
global $wgUser;
|
2009-10-07 13:57:06 +00:00
|
|
|
$title = Title::makeTitle( NS_FILE, $saveName );
|
|
|
|
$vars->addHolder(
|
|
|
|
AbuseFilterVariableHolder::merge(
|
|
|
|
AbuseFilter::generateUserVars( $wgUser ),
|
|
|
|
AbuseFilter::generateTitleVars( $title, 'FILE' )
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2009-06-03 15:10:44 +00:00
|
|
|
$vars->setVar( 'ACTION', 'upload' );
|
|
|
|
$vars->setVar( 'file_sha1', sha1_file( $tempName ) ); // TODO share with save
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2009-07-21 16:29:12 +00:00
|
|
|
$filter_result = AbuseFilter::filterAction( $vars, $title );
|
2009-10-07 13:57:06 +00:00
|
|
|
|
|
|
|
if ( is_string( $filter_result ) ) {
|
2009-06-03 15:10:44 +00:00
|
|
|
$error = $filter_result;
|
|
|
|
}
|
2009-10-07 13:57:06 +00:00
|
|
|
|
2009-06-03 15:10:44 +00:00
|
|
|
return $filter_result == '' || $filter_result === true;
|
2009-10-07 13:57:06 +00:00
|
|
|
}
|
2011-08-26 20:12:34 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds global variables to the Javascript as needed
|
|
|
|
*
|
|
|
|
* @param array $vars
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public static function onMakeGlobalVariablesScript( array &$vars ) {
|
|
|
|
if ( AbuseFilter::$editboxName !== null ) {
|
2011-08-26 20:26:57 +00:00
|
|
|
$vars['abuseFilterBoxName'] = AbuseFilter::$editboxName;
|
2011-08-26 20:12:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( AbuseFilterViewExamine::$examineType !== null ) {
|
2011-08-26 20:26:57 +00:00
|
|
|
$vars['abuseFilterExamine'] = array(
|
|
|
|
'type' => AbuseFilterViewExamine::$examineType,
|
|
|
|
'id' => AbuseFilterViewExamine::$examineId,
|
|
|
|
);
|
2011-08-26 20:12:34 +00:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2008-06-27 06:18:51 +00:00
|
|
|
}
|