addToDatabase(); $user->addGroup( 'filterEditor' ); RequestContext::getMain()->setUser( $user ); self::$mUser = $user; // Make sure that the config we're using is the one we're expecting $this->setMwGlobals( [ 'wgUser' => $user, 'wgAbuseFilterRestrictions' => [ 'degroup' => true ], 'wgAbuseFilterIsCentral' => true, 'wgAbuseFilterActions' => [ 'throttle' => true, 'warn' => true, 'disallow' => true, 'blockautopromote' => true, 'block' => true, 'rangeblock' => true, 'degroup' => true, 'tag' => true ], 'wgAbuseFilterValidGroups' => [ 'default', 'flow' ] ] ); $this->setGroupPermissions( [ 'filterEditor' => [ 'abusefilter-modify' => true, 'abusefilter-modify-restricted' => false, 'abusefilter-modify-global' => false, ], 'filterEditorGlobal' => [ 'abusefilter-modify' => true, 'abusefilter-modify-global' => true, ] ] ); $this->overrideMwServices(); } /** * Gets an instance of AbuseFilterViewEdit ready for creating or editing filter * * @param string $filter 'new' for a new filter, its ID otherwise * @return AbuseFilterViewEdit */ private static function getViewEdit( $filter ) { $special = new SpecialAbuseFilter(); $context = RequestContext::getMain(); $context->setRequest( self::getRequest() ); $special->setContext( $context ); $special->mFilter = $filter; $viewEdit = new AbuseFilterViewEdit( $special, [ $filter ] ); // Being a static property, it's not deleted between tests $viewEdit::$mLoadedRow = null; return $viewEdit; } /** * Creates a FauxRequest object * * @return FauxRequest */ private static function getRequest() { $params = [ 'wpFilterRules' => self::$mParameters['rules'], 'wpFilterDescription' => self::$mParameters['description'], 'wpFilterNotes' => self::$mParameters['notes'], 'wpFilterGroup' => self::$mParameters['group'], 'wpFilterEnabled' => self::$mParameters['enabled'], 'wpFilterHidden' => self::$mParameters['hidden'], 'wpFilterDeleted' => self::$mParameters['deleted'], 'wpFilterGlobal' => self::$mParameters['global'], 'wpFilterActionThrottle' => self::$mParameters['throttleEnabled'], 'wpFilterThrottleCount' => self::$mParameters['throttleCount'], 'wpFilterThrottlePeriod' => self::$mParameters['throttlePeriod'], 'wpFilterThrottleGroups' => self::$mParameters['throttleGroups'], 'wpFilterActionWarn' => self::$mParameters['warnEnabled'], 'wpFilterWarnMessage' => self::$mParameters['warnMessage'], 'wpFilterWarnMessageOther' => self::$mParameters['warnMessageOther'], 'wpFilterActionDisallow' => self::$mParameters['disallowEnabled'], 'wpFilterDisallowMessage' => self::$mParameters['disallowMessage'], 'wpFilterDisallowMessageOther' => self::$mParameters['disallowMessageOther'], 'wpFilterActionBlockautopromote' => self::$mParameters['blockautopromoteEnabled'], 'wpFilterActionDegroup' => self::$mParameters['degroupEnabled'], 'wpFilterActionBlock' => self::$mParameters['blockEnabled'], 'wpFilterBlockTalk' => self::$mParameters['blockTalk'], 'wpBlockAnonDuration' => self::$mParameters['blockAnons'], 'wpBlockUserDuration' => self::$mParameters['blockUsers'], 'wpFilterActionRangeblock' => self::$mParameters['rangeblockEnabled'], 'wpFilterActionTag' => self::$mParameters['tagEnabled'], 'wpFilterTags' => self::$mParameters['tagTags'], ]; // Checkboxes aren't included at all if they aren't selected. We can remove them // this way (instead of iterating a hardcoded list) since they're the only false values $params = array_filter( $params, function ( $el ) { return $el !== false; } ); $request = new FauxRequest( $params, true ); return $request; } /** * Creates $amount new filters, in case we need to test updating an existing one * * @param int $amount How many filters to create */ private static function createNewFilters( $amount ) { $defaultRow = [ 'af_pattern' => '/**/', 'af_user' => 0, 'af_user_text' => 'FilterTester', 'af_timestamp' => wfTimestampNow(), 'af_enabled' => 1, 'af_comments' => '', 'af_public_comments' => 'Mock filter', 'af_hidden' => 0, 'af_hit_count' => 0, 'af_throttled' => 0, 'af_deleted' => 0, 'af_actions' => '', 'af_global' => 0, 'af_group' => 'default' ]; $dbw = wfGetDB( DB_MASTER ); for ( $i = 1; $i <= $amount; $i++ ) { $dbw->replace( 'abuse_filter', [ 'af_id' ], $defaultRow, __METHOD__ ); } } /** * Validate and save a filter given its parameters * * @param array $args Parameters of the filter and metadata for the test * @covers AbuseFilter::saveFilter * @dataProvider provideFilters */ public function testSaveFilter( $args ) { // Preliminar stuff for the test if ( $args['testData']['customUserGroup'] ) { self::$mUser->addGroup( $args['testData']['customUserGroup'] ); } if ( $args['testData']['needsOtherFilters'] ) { self::createNewFilters( $args['testData']['needsOtherFilters'] ); } $fixedParameters = [ 'id' => 'new', 'notes' => '', 'group' => 'default', 'enabled' => true, 'hidden' => false, 'global' => false, 'deleted' => false, 'throttled' => 0, 'throttleEnabled' => false, 'throttleCount' => 0, 'throttlePeriod' => 0, 'throttleGroups' => '', 'warnEnabled' => false, 'warnMessage' => 'abusefilter-warning', 'warnMessageOther' => '', 'disallowEnabled' => false, 'disallowMessage' => 'abusefilter-disallowed', 'disallowMessageOther' => '', 'blockautopromoteEnabled' => false, 'degroupEnabled' => false, 'blockEnabled' => false, 'blockTalk' => false, 'blockAnons' => 'infinity', 'blockUsers' => 'infinity', 'rangeblockEnabled' => false, 'tagEnabled' => false, 'tagTags' => '' ]; // Extract parameters from testset and build what we need to save a filter // The values specified in the testset will overwrite the fixed ones. self::$mParameters = $args['filterParameters'] + $fixedParameters; $filter = self::$mParameters['id']; $viewEdit = self::getViewEdit( $filter ); list( $newRow, $actions ) = $viewEdit->loadRequest( $filter ); self::$mParameters['rowActions'] = implode( ',', array_keys( $actions ) ); // Send data for validation and saving $status = AbuseFilter::saveFilter( $viewEdit, $filter, $newRow, $actions ); // Must be removed for the next test if ( $args['testData']['customUserGroup'] ) { self::$mUser->removeGroup( $args['testData']['customUserGroup'] ); } $shouldFail = $args['testData']['shouldFail']; $shouldBeSaved = $args['testData']['shouldBeSaved']; $furtherInfo = null; $expected = true; if ( $shouldFail ) { if ( $status->isGood() ) { $furtherInfo = 'The filter validation returned a valid status.'; $result = false; } else { $result = $status->getErrors(); $result = $result[0]['message']; $expected = $args['testData']['expectedMessage']; } } else { if ( $shouldBeSaved ) { $value = $status->getValue(); $result = $status->isGood() && is_array( $value ) && count( $value ) === 2 && is_numeric( $value[0] ) && is_numeric( $value[1] ); } else { $result = $status->isGood() && $status->getValue() === false; } } $errorMessage = $args['testData']['doingWhat'] . '. Expected: ' . $args['testData']['expectedResult'] . '.'; if ( $furtherInfo ) { $errorMessage .= "\nFurther info: " . $furtherInfo; } $this->assertEquals( $expected, $result, $errorMessage ); } /** * Data provider for creating and editing filters. * @return array */ public function provideFilters() { return [ [ [ 'filterParameters' => [ 'rules' => '', 'description' => '', 'blockautopromoteEnabled' => true, ], 'testData' => [ 'doingWhat' => 'Trying to save a filter without description and rules', 'expectedResult' => 'a "missing required fields" error message', 'expectedMessage' => 'abusefilter-edit-missingfields', 'shouldFail' => true, 'shouldBeSaved' => false, 'customUserGroup' => '', 'needsOtherFilters' => false ] ] ], [ [ 'filterParameters' => [ 'rules' => '/* My rules */', 'description' => 'Some new filter', 'enabled' => false, 'deleted' => true, ], 'testData' => [ 'doingWhat' => 'Trying to save a filter with only rules and description', 'expectedResult' => 'the saving to be successful', 'expectedMessage' => '', 'shouldFail' => false, 'shouldBeSaved' => true, 'customUserGroup' => '', 'needsOtherFilters' => false ] ] ], [ [ 'filterParameters' => [ 'rules' => 'rlike', 'description' => 'This syntax aint good', 'blockEnabled' => true, 'blockTalk' => true, 'blockAnons' => '8 hours', ], 'testData' => [ 'doingWhat' => 'Trying to save a filter with wrong syntax', 'expectedResult' => 'a "wrong syntax" error message', 'expectedMessage' => 'abusefilter-edit-badsyntax', 'shouldFail' => true, 'shouldBeSaved' => false, 'customUserGroup' => '', 'needsOtherFilters' => false ] ] ], [ [ 'filterParameters' => [ 'rules' => '1==1', 'description' => 'Enabled and deleted', 'deleted' => true, 'blockEnabled' => true, 'blockTalk' => true, 'blockAnons' => '8 hours', ], 'testData' => [ 'doingWhat' => 'Trying to save a filter marking it both enabled and deleted', 'expectedResult' => 'an error message', 'expectedMessage' => 'abusefilter-edit-deleting-enabled', 'shouldFail' => true, 'shouldBeSaved' => false, 'customUserGroup' => '', 'needsOtherFilters' => false ] ] ], [ [ 'filterParameters' => [ 'rules' => '1==1', 'description' => 'Reserved tag', 'notes' => 'Some notes', 'hidden' => true, 'tagEnabled' => true, 'tagTags' => 'mw-undo' ], 'testData' => [ 'doingWhat' => 'Trying to save a filter with a reserved tag', 'expectedResult' => 'an error message saying that the tag cannot be used', 'expectedMessage' => 'abusefilter-edit-bad-tags', 'shouldFail' => true, 'shouldBeSaved' => false, 'customUserGroup' => '', 'needsOtherFilters' => false ] ] ], [ [ 'filterParameters' => [ 'rules' => '1==1', 'description' => 'Invalid tag', 'notes' => 'Some notes', 'tagEnabled' => true, 'tagTags' => 'some|tag' ], 'testData' => [ 'doingWhat' => 'Trying to save a filter with an invalid tag', 'expectedResult' => 'an error message saying that the tag cannot be used', 'expectedMessage' => 'tags-create-invalid-chars', 'shouldFail' => true, 'shouldBeSaved' => false, 'customUserGroup' => '', 'needsOtherFilters' => false ] ] ], [ [ 'filterParameters' => [ 'rules' => '1!=0', 'description' => 'Empty tag', 'notes' => '', 'tagEnabled' => true, 'tagTags' => '' ], 'testData' => [ 'doingWhat' => 'Trying to save a filter with an empty tag', 'expectedResult' => 'an error message saying that the tag name must be specified', 'expectedMessage' => 'tags-create-no-name', 'shouldFail' => true, 'shouldBeSaved' => false, 'customUserGroup' => '', 'needsOtherFilters' => false ] ] ], [ [ 'filterParameters' => [ 'rules' => '1==1', 'description' => 'Global without perms', 'global' => true, 'disallowEnabled' => true, ], 'testData' => [ 'doingWhat' => 'Trying to save a global filter without enough rights', 'expectedResult' => 'an error message saying that I do not have the required rights', 'expectedMessage' => 'abusefilter-edit-notallowed-global', 'shouldFail' => true, 'shouldBeSaved' => false, 'customUserGroup' => '', 'needsOtherFilters' => false ] ] ], [ [ 'filterParameters' => [ 'rules' => '1==1', 'description' => 'Global with invalid warn message', 'global' => true, 'warnEnabled' => true, 'warnMessage' => 'abusefilter-beautiful-warning', ], 'testData' => [ 'doingWhat' => 'Trying to save a global filter with a custom warn message', 'expectedResult' => 'an error message saying that custom warn messages ' . 'cannot be used for global rules', 'expectedMessage' => 'abusefilter-edit-notallowed-global-custom-msg', 'shouldFail' => true, 'shouldBeSaved' => false, 'customUserGroup' => 'filterEditorGlobal', 'needsOtherFilters' => false ] ] ], [ [ 'filterParameters' => [ 'rules' => '1==1', 'description' => 'Global with invalid disallow message', 'global' => true, 'disallowEnabled' => true, 'disallowMessage' => 'abusefilter-disallowed-something', ], 'testData' => [ 'doingWhat' => 'Trying to save a global filter with a custom disallow message', 'expectedResult' => 'an error message saying that custom disallow messages ' . 'cannot be used for global rules', 'expectedMessage' => 'abusefilter-edit-notallowed-global-custom-msg', 'shouldFail' => true, 'shouldBeSaved' => false, 'customUserGroup' => 'filterEditorGlobal', 'needsOtherFilters' => false ] ] ], [ [ 'filterParameters' => [ 'rules' => '1==1', 'description' => 'Restricted action', 'degroupEnabled' => true, ], 'testData' => [ 'doingWhat' => 'Trying to save a filter with a restricted action', 'expectedResult' => 'an error message saying that the action is restricted', 'expectedMessage' => 'abusefilter-edit-restricted', 'shouldFail' => true, 'shouldBeSaved' => false, 'customUserGroup' => '', 'needsOtherFilters' => false ] ] ], [ [ 'filterParameters' => [ 'id' => '1', 'rules' => '/**/', 'description' => 'Mock filter', ], 'testData' => [ 'doingWhat' => 'Trying to save a filter without changing anything', 'expectedResult' => 'the validation to pass without the filter being saved', 'expectedMessage' => '', 'shouldFail' => false, 'shouldBeSaved' => false, 'customUserGroup' => '', 'needsOtherFilters' => 1 ] ] ], [ [ 'filterParameters' => [ 'rules' => '1==1', 'description' => 'Invalid throttle groups', 'notes' => 'Throttle... Again', 'throttleEnabled' => true, 'throttleCount' => 11, 'throttlePeriod' => 111, 'throttleGroups' => 'user\nfoo' ], 'testData' => [ 'doingWhat' => 'Trying to save a filter with invalid throttle groups', 'expectedResult' => 'an error message saying that throttle groups are invalid', 'expectedMessage' => 'abusefilter-edit-invalid-throttlegroups', 'shouldFail' => true, 'shouldBeSaved' => false, 'customUserGroup' => '', 'needsOtherFilters' => false ] ] ], [ [ 'filterParameters' => [ 'rules' => '1==1', 'description' => 'Empty warning message', 'warnEnabled' => true, 'warnMessage' => '', ], 'testData' => [ 'doingWhat' => 'Trying to save a filter with empty warning message', 'expectedResult' => 'an error message saying that the warning message cannot be empty', 'expectedMessage' => 'abusefilter-edit-invalid-warn-message', 'shouldFail' => true, 'shouldBeSaved' => false, 'customUserGroup' => '', 'needsOtherFilters' => false ] ] ], [ [ 'filterParameters' => [ 'rules' => '1==1', 'description' => 'Empty disallow message', 'disallowEnabled' => true, 'disallowMessage' => '', ], 'testData' => [ 'doingWhat' => 'Trying to save a filter with empty disallow message', 'expectedResult' => 'an error message saying that the disallow message cannot be empty', 'expectedMessage' => 'abusefilter-edit-invalid-disallow-message', 'shouldFail' => true, 'shouldBeSaved' => false, 'customUserGroup' => '', 'needsOtherFilters' => false ] ] ] ]; } /** * Check that our tag validation is working properly. Note that we only need one test * for each called function. Consistency within ChangeTags functions should be * assured by tests in core. The test for canAddTagsAccompanyingChange and canCreateTag * are missing because they won't actually fail, never. Resolving T173917 would * greatly improve the situation and could help writing better tests. * * @param string $tag The tag to validate * @param string|null $error The expected error message. Null if validations should pass * @covers AbuseFilter::isAllowedTag * @dataProvider provideTags */ public function testIsAllowedTag( $tag, $error ) { $status = AbuseFilter::isAllowedTag( $tag ); if ( !$status->isGood() ) { $actualError = $status->getErrors(); $actualError = $actualError[0]['message']; } else { $actualError = null; if ( $error !== null ) { $this->fail( "Tag validation returned a valid status instead of the expected '$error' error." ); } } $this->assertSame( $error, $actualError, "Expected message '$error', got '$actualError' while validating the tag '$tag'." ); } /** * Data provider for testIsAllowedTag * @return array */ public function provideTags() { return [ [ 'a|b', 'tags-create-invalid-chars' ], [ 'mw-undo', 'abusefilter-edit-bad-tags' ], [ 'abusefilter-condition-limit', 'abusefilter-tag-reserved' ], [ 'my_tag', null ], ]; } /** * Check that throttle parameters validation works fine * * @param array $params Throttle parameters * @param string|null $error The expected error message. Null if validations should pass * @covers AbuseFilter::checkThrottleParameters * @dataProvider provideThrottleParameters */ public function testCheckThrottleParameters( $params, $error ) { $result = AbuseFilter::checkThrottleParameters( $params ); $this->assertSame( $error, $result, 'Throttle parameter validation does not work as expected.' ); } /** * Data provider for testCheckThrottleParameters * @return array */ public function provideThrottleParameters() { return [ [ [ '1', '5,23', 'user', 'ip', 'page,range', 'ip,user', 'range,ip' ], null ], [ [ '1', '5.3,23', 'user', 'ip' ], 'abusefilter-edit-invalid-throttlecount' ], [ [ '1', '-3,23', 'user', 'ip' ], 'abusefilter-edit-invalid-throttlecount' ], [ [ '1', '5,2.3', 'user', 'ip' ], 'abusefilter-edit-invalid-throttleperiod' ], [ [ '1', '4,-14', 'user', 'ip' ], 'abusefilter-edit-invalid-throttleperiod' ], [ [ '1', '3,33,44', 'user', 'ip' ], 'abusefilter-edit-invalid-throttleperiod' ], [ [ '1', '3,33' ], 'abusefilter-edit-empty-throttlegroups' ], [ [ '1', '3,33', 'user', 'ip,foo,user' ], 'abusefilter-edit-invalid-throttlegroups' ], [ [ '1', '3,33', 'foo', 'ip,user' ], 'abusefilter-edit-invalid-throttlegroups' ], [ [ '1', '3,33', 'foo', 'ip,user,bar' ], 'abusefilter-edit-invalid-throttlegroups' ], [ [ '1', '3,33', 'user', 'ip,page,user' ], null ], [ [ '1', '3,33', 'ip', 'user','user,ip', 'ip,user', 'user,ip,user', 'user', 'ip,ip,user' ], 'abusefilter-edit-duplicated-throttlegroups' ], [ [ '1', '3,33', 'ip,ip,user' ], 'abusefilter-edit-duplicated-throttlegroups' ], [ [ '1', '3,33', 'user,ip', 'ip,user' ], 'abusefilter-edit-duplicated-throttlegroups' ], ]; } }