2012-01-17 06:13:46 +00:00
|
|
|
<?php
|
|
|
|
|
2016-05-08 08:17:29 +00:00
|
|
|
use MediaWiki\Auth\AuthManager;
|
|
|
|
|
2012-01-17 06:13:46 +00:00
|
|
|
/**
|
|
|
|
* Hooks for the spam blacklist extension
|
|
|
|
*/
|
|
|
|
class SpamBlacklistHooks {
|
2015-05-05 20:44:21 +00:00
|
|
|
|
|
|
|
public static function registerExtension() {
|
2016-05-08 08:17:29 +00:00
|
|
|
global $wgDisableAuthManager, $wgAuthManagerAutoConfig;
|
2015-05-05 20:44:21 +00:00
|
|
|
|
2016-05-08 08:17:29 +00:00
|
|
|
if ( class_exists( AuthManager::class ) && !$wgDisableAuthManager ) {
|
|
|
|
$wgAuthManagerAutoConfig['preauth'][SpamBlacklistPreAuthenticationProvider::class] =
|
|
|
|
[ 'class' => SpamBlacklistPreAuthenticationProvider::class ];
|
|
|
|
} else {
|
|
|
|
Hooks::register( 'AbortNewAccount', 'SpamBlacklistHooks::abortNewAccount' );
|
|
|
|
}
|
2015-05-05 20:44:21 +00:00
|
|
|
}
|
|
|
|
|
2012-01-17 06:13:46 +00:00
|
|
|
/**
|
2013-07-25 14:05:13 +00:00
|
|
|
* Hook function for EditFilterMergedContent
|
|
|
|
*
|
|
|
|
* @param IContextSource $context
|
2017-09-01 04:57:27 +00:00
|
|
|
* @param Content $content
|
|
|
|
* @param Status $status
|
|
|
|
* @param string $summary
|
|
|
|
* @param User $user
|
|
|
|
* @param bool $minoredit
|
2012-01-17 06:13:46 +00:00
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
2017-06-02 16:02:09 +00:00
|
|
|
static function filterMergedContent(
|
|
|
|
IContextSource $context,
|
|
|
|
Content $content,
|
|
|
|
Status $status,
|
|
|
|
$summary,
|
|
|
|
User $user,
|
|
|
|
$minoredit
|
|
|
|
) {
|
2013-07-25 14:05:13 +00:00
|
|
|
$title = $context->getTitle();
|
|
|
|
|
|
|
|
// get the link from the not-yet-saved page content.
|
2014-01-07 22:00:30 +00:00
|
|
|
$editInfo = $context->getWikiPage()->prepareContentForEdit( $content );
|
|
|
|
$pout = $editInfo->output;
|
2013-07-25 14:05:13 +00:00
|
|
|
$links = array_keys( $pout->getExternalLinks() );
|
|
|
|
|
2015-11-28 04:02:57 +00:00
|
|
|
// HACK: treat the edit summary as a link if it contains anything
|
|
|
|
// that looks like it could be a URL or e-mail address.
|
|
|
|
if ( preg_match( '/\S(\.[^\s\d]{2,}|[\/@]\S)/', $summary ) ) {
|
2013-07-25 14:05:13 +00:00
|
|
|
$links[] = $summary;
|
|
|
|
}
|
|
|
|
|
2012-01-17 06:13:46 +00:00
|
|
|
$spamObj = BaseBlacklist::getInstance( 'spam' );
|
2013-07-25 14:05:13 +00:00
|
|
|
$matches = $spamObj->filter( $links, $title );
|
|
|
|
|
|
|
|
if ( $matches !== false ) {
|
|
|
|
$status->fatal( 'spamprotectiontext' );
|
|
|
|
|
|
|
|
foreach ( $matches as $match ) {
|
|
|
|
$status->fatal( 'spamprotectionmatch', $match );
|
|
|
|
}
|
2016-06-23 17:48:06 +00:00
|
|
|
|
|
|
|
$status->apiHookResult = [
|
|
|
|
'spamblacklist' => implode( '|', $matches ),
|
|
|
|
];
|
2012-01-17 06:13:46 +00:00
|
|
|
}
|
2013-07-25 14:05:13 +00:00
|
|
|
|
|
|
|
// Always return true, EditPage will look at $status->isOk().
|
|
|
|
return true;
|
2012-01-17 06:13:46 +00:00
|
|
|
}
|
|
|
|
|
2016-06-30 21:26:36 +00:00
|
|
|
public static function onParserOutputStashForEdit(
|
2017-06-02 16:02:09 +00:00
|
|
|
WikiPage $page,
|
|
|
|
Content $content,
|
|
|
|
ParserOutput $output
|
2016-06-30 21:26:36 +00:00
|
|
|
) {
|
2016-09-09 00:44:26 +00:00
|
|
|
$links = array_keys( $output->getExternalLinks() );
|
2016-02-16 21:39:06 +00:00
|
|
|
$spamObj = BaseBlacklist::getInstance( 'spam' );
|
2016-06-30 21:26:36 +00:00
|
|
|
$spamObj->warmCachesForFilter( $page->getTitle(), $links );
|
2016-02-16 21:39:06 +00:00
|
|
|
}
|
|
|
|
|
2012-01-18 23:29:37 +00:00
|
|
|
/**
|
|
|
|
* Verify that the user can send emails
|
|
|
|
*
|
|
|
|
* @param $user User
|
|
|
|
* @param $hookErr array
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public static function userCanSendEmail( &$user, &$hookErr ) {
|
|
|
|
/** @var $blacklist EmailBlacklist */
|
|
|
|
$blacklist = BaseBlacklist::getInstance( 'email' );
|
|
|
|
if ( $blacklist->checkUser( $user ) ) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-06-06 16:29:27 +00:00
|
|
|
$hookErr = [ 'spam-blacklisted-email', 'spam-blacklisted-email-text', null ];
|
2012-01-18 23:29:37 +00:00
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2013-02-27 11:50:47 +00:00
|
|
|
* Processes new accounts for valid email addresses
|
2012-01-18 23:29:37 +00:00
|
|
|
*
|
|
|
|
* @param $user User
|
|
|
|
* @param $abortError
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public static function abortNewAccount( $user, &$abortError ) {
|
|
|
|
/** @var $blacklist EmailBlacklist */
|
|
|
|
$blacklist = BaseBlacklist::getInstance( 'email' );
|
|
|
|
if ( $blacklist->checkUser( $user ) ) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
$abortError = wfMessage( 'spam-blacklisted-email-signup' )->escaped();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-01-17 06:13:46 +00:00
|
|
|
/**
|
|
|
|
* Hook function for EditFilter
|
|
|
|
* Confirm that a local blacklist page being saved is valid,
|
|
|
|
* and toss back a warning to the user if it isn't.
|
|
|
|
*
|
2017-09-01 20:24:40 +00:00
|
|
|
* @param EditPage $editPage
|
|
|
|
* @param string $text
|
|
|
|
* @param string $section
|
|
|
|
* @param string $hookError
|
2012-01-17 06:13:46 +00:00
|
|
|
* @return bool
|
|
|
|
*/
|
2017-09-01 20:24:40 +00:00
|
|
|
static function validate( EditPage $editPage, $text, $section, &$hookError ) {
|
|
|
|
$title = $editPage->getTitle();
|
|
|
|
$thisPageName = $title->getPrefixedDBkey();
|
2012-01-17 06:13:46 +00:00
|
|
|
|
2017-09-01 20:24:40 +00:00
|
|
|
if ( !BaseBlacklist::isLocalSource( $title ) ) {
|
2017-06-02 16:02:09 +00:00
|
|
|
wfDebugLog( 'SpamBlacklist',
|
|
|
|
"Spam blacklist validator: [[$thisPageName]] not a local blacklist\n"
|
|
|
|
);
|
2012-01-17 06:13:46 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-09-01 20:24:40 +00:00
|
|
|
$type = BaseBlacklist::getTypeFromTitle( $title );
|
2012-01-18 23:29:37 +00:00
|
|
|
if ( $type === false ) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-01-17 06:13:46 +00:00
|
|
|
$lines = explode( "\n", $text );
|
|
|
|
|
2012-01-18 23:29:37 +00:00
|
|
|
$badLines = SpamRegexBatch::getBadLines( $lines, BaseBlacklist::getInstance( $type ) );
|
2017-06-06 16:29:27 +00:00
|
|
|
if ( $badLines ) {
|
2017-06-02 16:02:09 +00:00
|
|
|
wfDebugLog( 'SpamBlacklist',
|
|
|
|
"Spam blacklist validator: [[$thisPageName]] given invalid input lines: " .
|
|
|
|
implode( ', ', $badLines ) . "\n"
|
|
|
|
);
|
2012-01-17 06:13:46 +00:00
|
|
|
|
2012-09-02 15:41:39 +00:00
|
|
|
$badList = "*<code>" .
|
|
|
|
implode( "</code>\n*<code>",
|
2012-01-17 06:13:46 +00:00
|
|
|
array_map( 'wfEscapeWikiText', $badLines ) ) .
|
2012-09-02 15:41:39 +00:00
|
|
|
"</code>\n";
|
2012-01-17 06:13:46 +00:00
|
|
|
$hookError =
|
|
|
|
"<div class='errorbox'>" .
|
2012-09-02 15:41:39 +00:00
|
|
|
wfMessage( 'spam-invalid-lines' )->numParams( $badLines )->text() . "<br />" .
|
2012-01-17 06:13:46 +00:00
|
|
|
$badList .
|
|
|
|
"</div>\n" .
|
|
|
|
"<br clear='all' />\n";
|
|
|
|
} else {
|
2017-06-02 16:02:09 +00:00
|
|
|
wfDebugLog( 'SpamBlacklist',
|
|
|
|
"Spam blacklist validator: [[$thisPageName]] ok or empty blacklist\n"
|
|
|
|
);
|
2012-01-17 06:13:46 +00:00
|
|
|
}
|
2012-01-18 23:29:37 +00:00
|
|
|
|
|
|
|
return true;
|
2012-01-17 06:13:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2013-07-25 14:05:13 +00:00
|
|
|
* Hook function for PageContentSaveComplete
|
2012-01-17 06:13:46 +00:00
|
|
|
* Clear local spam blacklist caches on page save.
|
|
|
|
*
|
2013-07-25 14:05:13 +00:00
|
|
|
* @param Page $wikiPage
|
2017-09-01 04:57:27 +00:00
|
|
|
* @param User $user
|
|
|
|
* @param Content $content
|
|
|
|
* @param string $summary
|
|
|
|
* @param bool $isMinor
|
|
|
|
* @param bool $isWatch
|
|
|
|
* @param string $section
|
|
|
|
* @param int $flags
|
2016-01-08 21:58:59 +00:00
|
|
|
* @param Revision|null $revision
|
2017-09-01 04:57:27 +00:00
|
|
|
* @param Status $status
|
|
|
|
* @param int $baseRevId
|
2013-07-25 14:05:13 +00:00
|
|
|
*
|
2012-01-17 06:13:46 +00:00
|
|
|
* @return bool
|
|
|
|
*/
|
2013-07-25 14:05:13 +00:00
|
|
|
static function pageSaveContent(
|
|
|
|
Page $wikiPage,
|
|
|
|
User $user,
|
|
|
|
Content $content,
|
|
|
|
$summary,
|
|
|
|
$isMinor,
|
|
|
|
$isWatch,
|
|
|
|
$section,
|
|
|
|
$flags,
|
|
|
|
$revision,
|
|
|
|
Status $status,
|
|
|
|
$baseRevId
|
|
|
|
) {
|
2016-06-22 10:18:11 +00:00
|
|
|
if ( $revision ) {
|
|
|
|
BaseBlacklist::getInstance( 'spam' )
|
2016-09-27 20:35:58 +00:00
|
|
|
->doLogging( $user, $wikiPage->getTitle(), $revision->getId() );
|
2016-06-22 10:18:11 +00:00
|
|
|
}
|
|
|
|
|
2015-07-31 21:02:14 +00:00
|
|
|
if ( !BaseBlacklist::isLocalSource( $wikiPage->getTitle() ) ) {
|
2012-02-15 14:58:26 +00:00
|
|
|
return true;
|
2012-01-17 06:13:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// This sucks because every Blacklist needs to be cleared
|
|
|
|
foreach ( BaseBlacklist::getBlacklistTypes() as $type => $class ) {
|
2015-07-31 21:02:14 +00:00
|
|
|
$blacklist = BaseBlacklist::getInstance( $type );
|
|
|
|
$blacklist->clearCache();
|
2012-01-17 06:13:46 +00:00
|
|
|
}
|
2015-07-31 21:02:14 +00:00
|
|
|
|
2016-07-22 14:02:31 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param UploadBase $upload
|
|
|
|
* @param User $user
|
|
|
|
* @param array $props
|
|
|
|
* @param string $comment
|
|
|
|
* @param string $pageText
|
|
|
|
* @param array|ApiMessage &$error
|
|
|
|
* @return bool
|
|
|
|
*/
|
2017-06-02 16:02:09 +00:00
|
|
|
public static function onUploadVerifyUpload(
|
|
|
|
UploadBase $upload,
|
|
|
|
User $user,
|
|
|
|
array $props,
|
|
|
|
$comment,
|
|
|
|
$pageText,
|
|
|
|
&$error
|
|
|
|
) {
|
2016-07-22 14:02:31 +00:00
|
|
|
$title = $upload->getTitle();
|
|
|
|
|
|
|
|
// get the link from the not-yet-saved page content.
|
|
|
|
$content = ContentHandler::makeContent( $pageText, $title );
|
|
|
|
$parserOptions = $content->getContentHandler()->makeParserOptions( 'canonical' );
|
|
|
|
$output = $content->getParserOutput( $title, null, $parserOptions );
|
|
|
|
$links = array_keys( $output->getExternalLinks() );
|
|
|
|
|
|
|
|
// HACK: treat comment as a link if it contains anything
|
|
|
|
// that looks like it could be a URL or e-mail address.
|
|
|
|
if ( preg_match( '/\S(\.[^\s\d]{2,}|[\/@]\S)/', $comment ) ) {
|
|
|
|
$links[] = $comment;
|
|
|
|
}
|
|
|
|
if ( !$links ) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
$spamObj = BaseBlacklist::getInstance( 'spam' );
|
|
|
|
$matches = $spamObj->filter( $links, $title );
|
|
|
|
|
|
|
|
if ( $matches !== false ) {
|
|
|
|
$error = new ApiMessage(
|
|
|
|
wfMessage( 'spamprotectiontext' ),
|
|
|
|
'spamblacklist',
|
2017-06-06 16:29:27 +00:00
|
|
|
[
|
|
|
|
'spamblacklist' => [ 'matches' => $matches ],
|
|
|
|
'message' => [
|
2016-08-17 09:53:43 +00:00
|
|
|
'key' => 'spamprotectionmatch',
|
|
|
|
'params' => $matches[0],
|
2017-06-06 16:29:27 +00:00
|
|
|
],
|
|
|
|
]
|
2016-07-22 14:02:31 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2012-02-02 22:12:43 +00:00
|
|
|
return true;
|
2012-01-17 06:13:46 +00:00
|
|
|
}
|
2016-09-27 20:35:58 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param WikiPage $article
|
|
|
|
* @param User $user
|
|
|
|
* @param $reason
|
|
|
|
* @param $error
|
|
|
|
*/
|
|
|
|
public static function onArticleDelete( WikiPage &$article, User &$user, &$reason, &$error ) {
|
|
|
|
/** @var SpamBlacklist $spam */
|
|
|
|
$spam = BaseBlacklist::getInstance( 'spam' );
|
|
|
|
if ( !$spam->isLoggingEnabled() ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Log the changes, but we only commit them once the deletion has happened.
|
|
|
|
// We do that since the external links table could get cleared before the
|
|
|
|
// ArticleDeleteComplete hook runs
|
|
|
|
$spam->logUrlChanges( $spam->getCurrentLinks( $article->getTitle() ), [], [] );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param WikiPage $page
|
|
|
|
* @param User $user
|
|
|
|
* @param $reason
|
|
|
|
* @param $id
|
|
|
|
* @param Content|null $content
|
|
|
|
* @param LogEntry $logEntry
|
|
|
|
*/
|
2017-06-02 16:02:09 +00:00
|
|
|
public static function onArticleDeleteComplete(
|
|
|
|
&$page,
|
|
|
|
User &$user,
|
|
|
|
$reason,
|
|
|
|
$id,
|
|
|
|
Content $content = null,
|
|
|
|
LogEntry $logEntry
|
2016-09-27 20:35:58 +00:00
|
|
|
) {
|
|
|
|
/** @var SpamBlacklist $spam */
|
|
|
|
$spam = BaseBlacklist::getInstance( 'spam' );
|
|
|
|
$spam->doLogging( $user, $page->getTitle(), $page->getLatest() );
|
|
|
|
}
|
2012-01-17 06:13:46 +00:00
|
|
|
}
|