PageRenderTime 43ms CodeModel.GetById 17ms RepoModel.GetById 0ms app.codeStats 0ms

/extensions/AbuseFilter/AbuseFilter.hooks.php

https://github.com/ChuguluGames/mediawiki-svn
PHP | 298 lines | 220 code | 50 blank | 28 comment | 31 complexity | 11f81b503b705f8b259d003afe9e5157 MD5 | raw file
  1. <?php
  2. if ( !defined( 'MEDIAWIKI' ) ) {
  3. die();
  4. }
  5. class AbuseFilterHooks {
  6. // So far, all of the error message out-params for these hooks accept HTML.
  7. // Hooray!
  8. /**
  9. * @param $editor
  10. * @param $text
  11. * @param $error
  12. * @param $summary
  13. * @return bool
  14. */
  15. public static function onEditFilterMerged( $editor, $text, &$error, $summary ) {
  16. // Load vars
  17. $vars = new AbuseFilterVariableHolder;
  18. // Cache article object so we can share a parse operation
  19. $title = $editor->mTitle;
  20. $articleCacheKey = $title->getNamespace() . ':' . $title->getText();
  21. AFComputedVariable::$articleCache[$articleCacheKey] = $editor->mArticle;
  22. // Check for null edits.
  23. $oldtext = '';
  24. if ( $editor->mArticle->exists() ) {
  25. $oldtext = $editor->mArticle->getContent();
  26. }
  27. if ( strcmp( $oldtext, $text ) == 0 ) {
  28. // Don't trigger for null edits.
  29. return true;
  30. }
  31. global $wgUser;
  32. $vars->addHolder( AbuseFilter::generateUserVars( $wgUser ) );
  33. $vars->addHolder( AbuseFilter::generateTitleVars( $editor->mTitle , 'ARTICLE' ) );
  34. $vars->setVar( 'ACTION', 'edit' );
  35. $vars->setVar( 'SUMMARY', $summary );
  36. $vars->setVar( 'minor_edit', $editor->minoredit );
  37. $vars->setVar( 'old_wikitext', $oldtext );
  38. $vars->setVar( 'new_wikitext', $text );
  39. $vars->addHolder( AbuseFilter::getEditVars( $editor->mTitle ) );
  40. $filter_result = AbuseFilter::filterAction( $vars, $editor->mTitle );
  41. if ( $filter_result !== true ) {
  42. global $wgOut;
  43. $wgOut->addHTML( $filter_result );
  44. $editor->showEditForm();
  45. return false;
  46. }
  47. return true;
  48. }
  49. public static function onGetAutoPromoteGroups( $user, &$promote ) {
  50. global $wgMemc;
  51. $key = AbuseFilter::autoPromoteBlockKey( $user );
  52. if ( $wgMemc->get( $key ) ) {
  53. $promote = array();
  54. }
  55. return true;
  56. }
  57. public static function onAbortMove( $oldTitle, $newTitle, $user, &$error, $reason ) {
  58. $vars = new AbuseFilterVariableHolder;
  59. global $wgUser;
  60. $vars->addHolder(
  61. AbuseFilterVariableHolder::merge(
  62. AbuseFilter::generateUserVars( $wgUser ),
  63. AbuseFilter::generateTitleVars( $oldTitle, 'MOVED_FROM' ),
  64. AbuseFilter::generateTitleVars( $newTitle, 'MOVED_TO' )
  65. )
  66. );
  67. $vars->setVar( 'SUMMARY', $reason );
  68. $vars->setVar( 'ACTION', 'move' );
  69. $filter_result = AbuseFilter::filterAction( $vars, $oldTitle );
  70. $error = $filter_result;
  71. return $filter_result == '' || $filter_result === true;
  72. }
  73. public static function onArticleDelete( &$article, &$user, &$reason, &$error ) {
  74. $vars = new AbuseFilterVariableHolder;
  75. global $wgUser;
  76. $vars->addHolder( AbuseFilter::generateUserVars( $wgUser ) );
  77. $vars->addHolder( AbuseFilter::generateTitleVars( $article->mTitle, 'ARTICLE' ) );
  78. $vars->setVar( 'SUMMARY', $reason );
  79. $vars->setVar( 'ACTION', 'delete' );
  80. $filter_result = AbuseFilter::filterAction( $vars, $article->mTitle );
  81. $error = $filter_result;
  82. return $filter_result == '' || $filter_result === true;
  83. }
  84. public static function onAbortNewAccount( $user, &$message ) {
  85. if ( $user->getName() == wfMsgForContent( 'abusefilter-blocker' ) ) {
  86. $message = wfMsg( 'abusefilter-accountreserved' );
  87. return false;
  88. }
  89. $vars = new AbuseFilterVariableHolder;
  90. // Add variables only for a registered user, so IP addresses of
  91. // new users won't be exposed
  92. global $wgUser;
  93. if ( $wgUser->getId() ) {
  94. $vars->addHolder( AbuseFilter::generateUserVars( $wgUser ) );
  95. }
  96. $vars->setVar( 'ACTION', 'createaccount' );
  97. $vars->setVar( 'ACCOUNTNAME', $user->getName() );
  98. $filter_result = AbuseFilter::filterAction(
  99. $vars, SpecialPage::getTitleFor( 'Userlogin' ) );
  100. $message = $filter_result;
  101. return $filter_result == '' || $filter_result === true;
  102. }
  103. public static function onRecentChangeSave( $recentChange ) {
  104. $title = Title::makeTitle(
  105. $recentChange->mAttribs['rc_namespace'],
  106. $recentChange->mAttribs['rc_title']
  107. );
  108. $action = $recentChange->mAttribs['rc_log_type'] ?
  109. $recentChange->mAttribs['rc_log_type'] : 'edit';
  110. $actionID = implode( '-', array(
  111. $title->getPrefixedText(), $recentChange->mAttribs['rc_user_text'], $action
  112. ) );
  113. if ( !empty( AbuseFilter::$tagsToSet[$actionID] )
  114. && count( $tags = AbuseFilter::$tagsToSet[$actionID] ) )
  115. {
  116. ChangeTags::addTags(
  117. $tags,
  118. $recentChange->mAttribs['rc_id'],
  119. $recentChange->mAttribs['rc_this_oldid'],
  120. $recentChange->mAttribs['rc_logid']
  121. );
  122. }
  123. return true;
  124. }
  125. public static function onListDefinedTags( &$emptyTags ) {
  126. # This is a pretty awful hack.
  127. $dbr = wfGetDB( DB_SLAVE );
  128. $res = $dbr->select(
  129. array( 'abuse_filter_action', 'abuse_filter' ),
  130. 'afa_parameters',
  131. array( 'afa_consequence' => 'tag', 'af_enabled' => true ),
  132. __METHOD__,
  133. array(),
  134. array( 'abuse_filter' => array( 'INNER JOIN', 'afa_filter=af_id' ) )
  135. );
  136. foreach ( $res as $row ) {
  137. $emptyTags = array_filter(
  138. array_merge( explode( "\n", $row->afa_parameters ), $emptyTags )
  139. );
  140. }
  141. return true;
  142. }
  143. /**
  144. * @static
  145. * @param $updater DatabaseUpdater
  146. * @return bool
  147. */
  148. public static function onLoadExtensionSchemaUpdates( $updater = null ) {
  149. $dir = dirname( __FILE__ );
  150. if ( $updater === null ) {
  151. global $wgExtNewTables, $wgExtNewFields, $wgExtPGNewFields, $wgExtPGAlteredFields, $wgExtNewIndexes, $wgDBtype;
  152. // DB updates
  153. if ( $wgDBtype == 'mysql' ) {
  154. $wgExtNewTables[] = array( 'abuse_filter', "$dir/abusefilter.tables.sql" );
  155. $wgExtNewTables[] = array( 'abuse_filter_history', "$dir/db_patches/patch-abuse_filter_history.sql" );
  156. $wgExtNewFields[] = array( 'abuse_filter_history', 'afh_changed_fields', "$dir/db_patches/patch-afh_changed_fields.sql" );
  157. $wgExtNewFields[] = array( 'abuse_filter', 'af_deleted', "$dir/db_patches/patch-af_deleted.sql" );
  158. $wgExtNewFields[] = array( 'abuse_filter', 'af_actions', "$dir/db_patches/patch-af_actions.sql" );
  159. $wgExtNewFields[] = array( 'abuse_filter', 'af_global', "$dir/db_patches/patch-global_filters.sql" );
  160. $wgExtNewIndexes[] = array( 'abuse_filter_log', 'filter_timestamp', "$dir/db_patches/patch-fix-indexes.sql" );
  161. } elseif ( $wgDBtype == 'postgres' ) {
  162. $wgExtNewTables = array_merge( $wgExtNewTables,
  163. array(
  164. array( 'abuse_filter', "$dir/abusefilter.tables.pg.sql" ),
  165. array( 'abuse_filter_history', "$dir/db_patches/patch-abuse_filter_history.pg.sql" ),
  166. ) );
  167. $wgExtPGNewFields[] = array( 'abuse_filter', 'af_actions', "TEXT NOT NULL DEFAULT ''" );
  168. $wgExtPGNewFields[] = array( 'abuse_filter', 'af_deleted', 'SMALLINT NOT NULL DEFAULT 0' );
  169. $wgExtPGNewFields[] = array( 'abuse_filter', 'af_global', 'SMALLINT NOT NULL DEFAULT 0' );
  170. $wgExtPGNewFields[] = array( 'abuse_filter_log', 'afl_wiki', 'TEXT' );
  171. $wgExtPGNewFields[] = array( 'abuse_filter_log', 'afl_deleted', 'SMALLINT' );
  172. $wgExtPGAlteredFields[] = array( 'abuse_filter_log', 'afl_filter', 'TEXT' );
  173. $wgExtNewIndexes[] = array( 'abuse_filter_log', 'abuse_filter_log_ip', "(afl_ip)" );
  174. }
  175. } else {
  176. if ( $updater->getDB()->getType() == 'mysql' ) {
  177. $updater->addExtensionUpdate( array( 'addTable', 'abuse_filter', "$dir/abusefilter.tables.sql", true ) );
  178. $updater->addExtensionUpdate( array( 'addTable', 'abuse_filter_history', "$dir/db_patches/patch-abuse_filter_history.sql", true ) );
  179. $updater->addExtensionUpdate( array( 'addField', 'abuse_filter_history', 'afh_changed_fields', "$dir/db_patches/patch-afh_changed_fields.sql", true ) );
  180. $updater->addExtensionUpdate( array( 'addField', 'abuse_filter', 'af_deleted', "$dir/db_patches/patch-af_deleted.sql", true ) );
  181. $updater->addExtensionUpdate( array( 'addField', 'abuse_filter', 'af_actions', "$dir/db_patches/patch-af_actions.sql", true ) );
  182. $updater->addExtensionUpdate( array( 'addField', 'abuse_filter', 'af_global', "$dir/db_patches/patch-global_filters.sql", true ) );
  183. $updater->addExtensionUpdate( array( 'addIndex', 'abuse_filter_log', 'filter_timestamp', "$dir/db_patches/patch-fix-indexes.sql", true ) );
  184. } elseif ( $updater->getDB()->getType() == 'postgres' ) {
  185. $updater->addExtensionUpdate( array( 'addTable', 'abuse_filter', "$dir/abusefilter.tables.pg.sql", true ) );
  186. $updater->addExtensionUpdate( array( 'addTable', 'abuse_filter_history', "$dir/db_patches/patch-abuse_filter_history.pg.sql", true ) );
  187. $updater->addExtensionUpdate( array( 'addPgField', 'abuse_filter', 'af_actions', "TEXT NOT NULL DEFAULT ''" ) );
  188. $updater->addExtensionUpdate( array( 'addPgField', 'abuse_filter', 'af_deleted', 'SMALLINT NOT NULL DEFAULT 0' ) );
  189. $updater->addExtensionUpdate( array( 'addPgField', 'abuse_filter', 'af_global', 'SMALLINT NOT NULL DEFAULT 0' ) );
  190. $updater->addExtensionUpdate( array( 'addPgField', 'abuse_filter_log', 'afl_wiki', 'TEXT' ) );
  191. $updater->addExtensionUpdate( array( 'addPgField', 'abuse_filter_log', 'afl_deleted', 'SMALLINT' ) );
  192. $updater->addExtensionUpdate( array( 'changeField', 'abuse_filter_log', 'afl_filter', 'TEXT' ) );
  193. $updater->addExtensionUpdate( array( 'addPgExtIndex', 'abuse_filter_log', 'abuse_filter_log_ip', "(afl_ip)" ) );
  194. }
  195. }
  196. return true;
  197. }
  198. public static function onContributionsToolLinks( $id, $nt, &$tools ) {
  199. global $wgUser;
  200. if ( $wgUser->isAllowed( 'abusefilter-log' ) ) {
  201. $sk = $wgUser->getSkin();
  202. $tools[] = $sk->link(
  203. SpecialPage::getTitleFor( 'AbuseLog' ),
  204. wfMsg( 'abusefilter-log-linkoncontribs' ),
  205. array( 'title' =>
  206. wfMsgExt( 'abusefilter-log-linkoncontribs-text', 'parseinline' ) ),
  207. array( 'wpSearchUser' => $nt->getText() )
  208. );
  209. }
  210. return true;
  211. }
  212. public static function onUploadVerification( $saveName, $tempName, &$error ) {
  213. $vars = new AbuseFilterVariableHolder;
  214. global $wgUser;
  215. $title = Title::makeTitle( NS_FILE, $saveName );
  216. $vars->addHolder(
  217. AbuseFilterVariableHolder::merge(
  218. AbuseFilter::generateUserVars( $wgUser ),
  219. AbuseFilter::generateTitleVars( $title, 'FILE' )
  220. )
  221. );
  222. $vars->setVar( 'ACTION', 'upload' );
  223. $vars->setVar( 'file_sha1', sha1_file( $tempName ) ); // TODO share with save
  224. $filter_result = AbuseFilter::filterAction( $vars, $title );
  225. if ( is_string( $filter_result ) ) {
  226. $error = $filter_result;
  227. }
  228. return $filter_result == '' || $filter_result === true;
  229. }
  230. /**
  231. * Adds global variables to the Javascript as needed
  232. *
  233. * @param array $vars
  234. * @return bool
  235. */
  236. public static function onMakeGlobalVariablesScript( array &$vars ) {
  237. if ( AbuseFilter::$editboxName !== null ) {
  238. $vars['abuseFilterBoxName'] = AbuseFilter::$editboxName;
  239. }
  240. if ( AbuseFilterViewExamine::$examineType !== null ) {
  241. $vars['abuseFilterExamine'] = array(
  242. 'type' => AbuseFilterViewExamine::$examineType,
  243. 'id' => AbuseFilterViewExamine::$examineId,
  244. );
  245. }
  246. return true;
  247. }
  248. }