mirror of
https://github.com/silverstripe/silverstripe-versionfeed
synced 2024-10-22 11:05:31 +02:00
API Better implementation of caching / rate limiting
This commit is contained in:
parent
4946376793
commit
a243a3510a
@ -1,7 +1,4 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
SiteTree::add_extension('VersionFeed');
|
|
||||||
ContentController::add_extension('VersionFeed_Controller');
|
|
||||||
|
|
||||||
// Set the cache lifetime to 5 mins.
|
// Set the cache lifetime to 5 mins.
|
||||||
SS_Cache::set_cache_lifetime('VersionFeed_Controller', 5*60);
|
SS_Cache::set_cache_lifetime('VersionFeed_Controller', 5*60);
|
||||||
|
18
_config/versionfeed.yml
Normal file
18
_config/versionfeed.yml
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
---
|
||||||
|
Name: versionedfeedconfig
|
||||||
|
---
|
||||||
|
Injector:
|
||||||
|
RateLimitFilter: RateLimitFilter
|
||||||
|
ContentFilter:
|
||||||
|
class: CachedContentFilter
|
||||||
|
constructor:
|
||||||
|
- %$RateLimitFilter
|
||||||
|
SiteTree:
|
||||||
|
extensions:
|
||||||
|
- VersionFeed
|
||||||
|
ContentController:
|
||||||
|
extensions:
|
||||||
|
- VersionFeed_Controller
|
||||||
|
VersionFeed_Controller:
|
||||||
|
dependencies:
|
||||||
|
ContentFilter: %$ContentFilter
|
@ -3,7 +3,7 @@
|
|||||||
class VersionFeed extends SiteTreeExtension {
|
class VersionFeed extends SiteTreeExtension {
|
||||||
|
|
||||||
private static $db = array(
|
private static $db = array(
|
||||||
'PublicHistory' => 'Boolean'
|
'PublicHistory' => 'Boolean(true)'
|
||||||
);
|
);
|
||||||
|
|
||||||
private static $defaults = array(
|
private static $defaults = array(
|
||||||
|
@ -7,7 +7,38 @@ class VersionFeed_Controller extends Extension {
|
|||||||
'allchanges'
|
'allchanges'
|
||||||
);
|
);
|
||||||
|
|
||||||
function onAfterInit() {
|
/**
|
||||||
|
* Content handler
|
||||||
|
*
|
||||||
|
* @var ContentFilter
|
||||||
|
*/
|
||||||
|
protected $contentFilter;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the content filter
|
||||||
|
*
|
||||||
|
* @param ContentFilter $contentFilter
|
||||||
|
*/
|
||||||
|
public function setContentFilter(ContentFilter $contentFilter) {
|
||||||
|
$this->contentFilter = $contentFilter;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Evaluates the result of the given callback
|
||||||
|
*
|
||||||
|
* @param string $key Unique key for this
|
||||||
|
* @param callable $callback Callback for evaluating the content
|
||||||
|
* @return mixed Result of $callback()
|
||||||
|
*/
|
||||||
|
protected function filterContent($key, $callback) {
|
||||||
|
if($this->contentFilter) {
|
||||||
|
return $this->contentFilter->getContent($key, $callback);
|
||||||
|
} else {
|
||||||
|
return call_user_func($callback);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function onAfterInit() {
|
||||||
// RSS feed for per-page changes.
|
// RSS feed for per-page changes.
|
||||||
if ($this->owner->PublicHistory) {
|
if ($this->owner->PublicHistory) {
|
||||||
RSSFeed::linkToFeed($this->owner->Link() . 'changes',
|
RSSFeed::linkToFeed($this->owner->Link() . 'changes',
|
||||||
@ -26,18 +57,15 @@ class VersionFeed_Controller extends Extension {
|
|||||||
/**
|
/**
|
||||||
* Get page-specific changes in a RSS feed.
|
* Get page-specific changes in a RSS feed.
|
||||||
*/
|
*/
|
||||||
function changes() {
|
public function changes() {
|
||||||
if(!$this->owner->PublicHistory) throw new SS_HTTPResponse_Exception('Page history not viewable', 404);;
|
if(!$this->owner->PublicHistory) throw new SS_HTTPResponse_Exception('Page history not viewable', 404);
|
||||||
|
|
||||||
// Cache the diffs to remove DOS possibility.
|
// Cache the diffs to remove DOS possibility.
|
||||||
$cache = SS_Cache::factory('VersionFeed_Controller');
|
$target = $this->owner;
|
||||||
$cache->setOption('automatic_serialization', true);
|
|
||||||
$key = implode('_', array('changes', $this->owner->ID, $this->owner->Version));
|
$key = implode('_', array('changes', $this->owner->ID, $this->owner->Version));
|
||||||
$entries = $cache->load($key);
|
$entries = $this->filterContent($key, function() use ($target) {
|
||||||
if(!$entries || isset($_GET['flush'])) {
|
return $target->getDiffedChanges();
|
||||||
$entries = $this->owner->getDiffedChanges();
|
});
|
||||||
$cache->save($entries, $key);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Generate the output.
|
// Generate the output.
|
||||||
$title = sprintf(_t('RSSHistory.SINGLEPAGEFEEDTITLE', 'Updates to %s page'), $this->owner->Title);
|
$title = sprintf(_t('RSSHistory.SINGLEPAGEFEEDTITLE', 'Updates to %s page'), $this->owner->Title);
|
||||||
@ -49,42 +77,47 @@ class VersionFeed_Controller extends Extension {
|
|||||||
/**
|
/**
|
||||||
* Get all changes from the site in a RSS feed.
|
* Get all changes from the site in a RSS feed.
|
||||||
*/
|
*/
|
||||||
function allchanges() {
|
public function allchanges() {
|
||||||
|
|
||||||
$latestChanges = DB::query('SELECT * FROM "SiteTree_versions" WHERE "WasPublished"=\'1\' AND "CanViewType" IN (\'Anyone\', \'Inherit\') AND "ShowInSearch"=1 AND ("PublicHistory" IS NULL OR "PublicHistory" = \'1\') ORDER BY "LastEdited" DESC LIMIT 20');
|
$latestChanges = DB::query('
|
||||||
|
SELECT * FROM "SiteTree_versions"
|
||||||
|
WHERE "WasPublished" = \'1\'
|
||||||
|
AND "CanViewType" IN (\'Anyone\', \'Inherit\')
|
||||||
|
AND "ShowInSearch" = 1
|
||||||
|
AND ("PublicHistory" IS NULL OR "PublicHistory" = \'1\')
|
||||||
|
ORDER BY "LastEdited" DESC LIMIT 20'
|
||||||
|
);
|
||||||
$lastChange = $latestChanges->record();
|
$lastChange = $latestChanges->record();
|
||||||
$latestChanges->rewind();
|
$latestChanges->rewind();
|
||||||
|
|
||||||
if ($lastChange) {
|
if ($lastChange) {
|
||||||
|
|
||||||
// Cache the diffs to remove DOS possibility.
|
// Cache the diffs to remove DOS possibility.
|
||||||
$member = Member::currentUser();
|
$key = 'allchanges'
|
||||||
$cache = SS_Cache::factory('VersionFeed_Controller');
|
. preg_replace('#[^a-zA-Z0-9_]#', '', $lastChange['LastEdited'])
|
||||||
$cache->setOption('automatic_serialization', true);
|
. (Member::currentUserID() ?: 'public');
|
||||||
$key = 'allchanges' . preg_replace('#[^a-zA-Z0-9_]#', '', $lastChange['LastEdited']) .
|
$changeList = $this->filterContent($key, function() use ($latestChanges) {
|
||||||
($member ? $member->ID : 'public');
|
|
||||||
|
|
||||||
$changeList = $cache->load($key);
|
|
||||||
if(!$changeList || isset($_GET['flush'])) {
|
|
||||||
|
|
||||||
$changeList = new ArrayList();
|
$changeList = new ArrayList();
|
||||||
|
$canView = array();
|
||||||
foreach ($latestChanges as $record) {
|
foreach ($latestChanges as $record) {
|
||||||
|
|
||||||
// Check if the page should be visible.
|
// Check if the page should be visible.
|
||||||
// WARNING: although we are providing historical details, we check the current configuration.
|
// WARNING: although we are providing historical details, we check the current configuration.
|
||||||
$page = SiteTree::get()->filter(array('ID'=>$record['RecordID']))->First();
|
$id = $record['RecordID'];
|
||||||
if (!$page->canView(new Member())) continue;
|
if(!isset($canView[$id])) {
|
||||||
|
$page = SiteTree::get()->byID($id);
|
||||||
|
$canView[$id] = $page && $page->canView(new Member());
|
||||||
|
}
|
||||||
|
if (!$canView[$id]) continue;
|
||||||
|
|
||||||
// Get the diff to the previous version.
|
// Get the diff to the previous version.
|
||||||
$version = new Versioned_Version($record);
|
$version = new Versioned_Version($record);
|
||||||
|
|
||||||
$changes = $version->getDiffedChanges($version->Version, false);
|
$changes = $version->getDiffedChanges($version->Version, false);
|
||||||
if ($changes && $changes->Count()) $changeList->push($changes->First());
|
if ($changes && $changes->Count()) $changeList->push($changes->First());
|
||||||
}
|
}
|
||||||
|
|
||||||
$cache->save($changeList, $key);
|
return $changeList;
|
||||||
}
|
});
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
$changeList = new ArrayList();
|
$changeList = new ArrayList();
|
||||||
}
|
}
|
||||||
|
22
code/caching/CachedContentFilter.php
Normal file
22
code/caching/CachedContentFilter.php
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Caches results of a callback
|
||||||
|
*/
|
||||||
|
class CachedContentFilter extends ContentFilter {
|
||||||
|
|
||||||
|
public function getContent($key, $callback) {
|
||||||
|
$cache = $this->getCache();
|
||||||
|
|
||||||
|
// Return cached value if available
|
||||||
|
$result = isset($_GET['flush'])
|
||||||
|
? null
|
||||||
|
: $cache->load($key);
|
||||||
|
if($result) return $result;
|
||||||
|
|
||||||
|
// Fallback to generate result
|
||||||
|
$result = parent::getContent($key, $callback);
|
||||||
|
$cache->save($result, $key);
|
||||||
|
return $result;
|
||||||
|
}
|
||||||
|
}
|
45
code/caching/ContentFilter.php
Normal file
45
code/caching/ContentFilter.php
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Conditionally executes a given callback, attempting to return the desired results
|
||||||
|
* of its execution.
|
||||||
|
*/
|
||||||
|
abstract class ContentFilter {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Nested content filter
|
||||||
|
*
|
||||||
|
* @var ContentFilter
|
||||||
|
*/
|
||||||
|
protected $nestedContentFilter;
|
||||||
|
|
||||||
|
public function __construct($nestedContentFilter = null) {
|
||||||
|
$this->nestedContentFilter = $nestedContentFilter;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the cache to use
|
||||||
|
*
|
||||||
|
* @return Zend_Cache_Frontend
|
||||||
|
*/
|
||||||
|
protected function getCache() {
|
||||||
|
$cache = SS_Cache::factory('VersionFeed_Controller');
|
||||||
|
$cache->setOption('automatic_serialization', true);
|
||||||
|
return $cache;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Evaluates the result of the given callback
|
||||||
|
*
|
||||||
|
* @param string $key Unique key for this
|
||||||
|
* @param callable $callback Callback for evaluating the content
|
||||||
|
* @return mixed Result of $callback()
|
||||||
|
*/
|
||||||
|
public function getContent($key, $callback) {
|
||||||
|
if($this->nestedContentFilter) {
|
||||||
|
return $this->nestedContentFilter->getContent($key, $callback);
|
||||||
|
} else {
|
||||||
|
return call_user_func($callback);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
92
code/caching/RateLimitFilter.php
Normal file
92
code/caching/RateLimitFilter.php
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provides rate limiting of execution of a callback
|
||||||
|
*/
|
||||||
|
class RateLimitFilter extends ContentFilter {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Time duration (in second) to allow for generation of cached results. Requests to
|
||||||
|
* pages that within this time period that do not hit the cache (and would otherwise trigger
|
||||||
|
* a version query) will be presented with a 429 (rate limit) HTTP error
|
||||||
|
*
|
||||||
|
* @config
|
||||||
|
* @var int
|
||||||
|
*/
|
||||||
|
private static $lock_timeout = 10;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determine if the cache generation should be locked on a per-page basis. If true, concurrent page versions
|
||||||
|
* may be generated without rate interference.
|
||||||
|
*
|
||||||
|
* Suggested to turn this to false on small sites that will not have many concurrent views of page versions
|
||||||
|
*
|
||||||
|
* @config
|
||||||
|
* @var bool
|
||||||
|
*/
|
||||||
|
private static $lock_bypage = true;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determine if rate limiting should be applied independently to each IP address. This method is not
|
||||||
|
* reliable, as most DDoS attacks use multiple IP addresses.
|
||||||
|
*
|
||||||
|
* @config
|
||||||
|
* @var bool
|
||||||
|
*/
|
||||||
|
private static $lock_byuserip = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cache key prefix
|
||||||
|
*/
|
||||||
|
const CACHE_PREFIX = 'RateLimitBegin';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determines the key to use for saving the current rate
|
||||||
|
*
|
||||||
|
* @param string $itemkey Input key
|
||||||
|
* @return string Result key
|
||||||
|
*/
|
||||||
|
protected function getCacheKey($itemkey) {
|
||||||
|
$key = self::CACHE_PREFIX;
|
||||||
|
|
||||||
|
// Add global identifier
|
||||||
|
if(Config::inst()->get(get_class(), 'lock_bypage')) {
|
||||||
|
$key .= '_' . md5($itemkey);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add user-specific identifier
|
||||||
|
if(Config::inst()->get(get_class(), 'lock_byuserip') && Controller::has_curr()) {
|
||||||
|
$ip = Controller::curr()->getRequest()->getIP();
|
||||||
|
$key .= '_' . md5($ip);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $key;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public function getContent($key, $callback) {
|
||||||
|
// Bypass rate limiting if flushing, or timeout isn't set
|
||||||
|
$timeout = Config::inst()->get(get_class(), 'lock_timeout');
|
||||||
|
if(isset($_GET['flush']) || !$timeout) {
|
||||||
|
return parent::getContent($key, $callback);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generate result with rate limiting enabled
|
||||||
|
$limitKey = $this->getCacheKey($key);
|
||||||
|
$cache = $this->getCache();
|
||||||
|
if($cacheBegin = $cache->load($limitKey)) {
|
||||||
|
if(time() - $cacheBegin < $timeout) {
|
||||||
|
// Politely inform visitor of limit
|
||||||
|
$response = new SS_HTTPResponse_Exception('Too Many Requests.', 429);
|
||||||
|
$response->getResponse()->addHeader('Retry-After', 1 + time() - $cacheBegin);
|
||||||
|
throw $response;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generate result with rate limit locked
|
||||||
|
$cache->save(time(), $limitKey);
|
||||||
|
$result = parent::getContent($key, $callback);
|
||||||
|
$cache->remove($limitKey);
|
||||||
|
return $result;
|
||||||
|
}
|
||||||
|
}
|
7
docs/en/changelogs/1.0.3.md
Normal file
7
docs/en/changelogs/1.0.3.md
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
# 1.0.3
|
||||||
|
|
||||||
|
## Upgrading
|
||||||
|
|
||||||
|
* Caching and rate limiting rules now apply in this version, and it may be necessary to adjust the values
|
||||||
|
for `RateLimitFilter.lock_timeout`, `RateLimitFilter.lock_bypage` and `RateLimitFilter.lock_byuserip` configs.
|
||||||
|
See the [Developer Documentation](../developer.md) for information on how these affect cache performance and behaviour.
|
@ -21,3 +21,35 @@ and returning the URL of your desired RSS feed:
|
|||||||
}
|
}
|
||||||
|
|
||||||
This can be used in templates as `$DefaultRSSLink`.
|
This can be used in templates as `$DefaultRSSLink`.
|
||||||
|
|
||||||
|
### Rate limiting and caching
|
||||||
|
|
||||||
|
By default all content is filtered based on the rules specified in `versionfeed/_config/versionfeed.yml`.
|
||||||
|
Two filters are applied on top of one another:
|
||||||
|
|
||||||
|
* `CachedContentFilter` provides caching of versions based on an identifier built up of the record ID and the
|
||||||
|
most recently saved version number. There is no configuration required for this class.
|
||||||
|
* `RateLimitFilter` provides rate limiting to ensure that requests to uncached data does not overload the
|
||||||
|
server. This filter will only be applied if the `CachedContentFilter` does not have any cached record
|
||||||
|
for a request.
|
||||||
|
|
||||||
|
Either one of these can be replaced, added to, or removed, by adjusting the `VersionFeed_Controller.dependencies`
|
||||||
|
config to point to a replacement (or no) filter.
|
||||||
|
|
||||||
|
For large servers with a high level of traffic (more than 1 visits every 10 seconds) the default settings should
|
||||||
|
be sufficient.
|
||||||
|
|
||||||
|
For smaller servers where it's reasonable to apply a more strict approach to rate limiting, consider setting the
|
||||||
|
`RateLimitFilter.lock_bypage` config to false, meaning that a single limit will be applied to all URLs.
|
||||||
|
If left on true, then each URL will have its own rate limit, and on smaller servers with lots of
|
||||||
|
concurrent requests this can still overwhelm capacity. This will also leave smaller servers vulnerable to DDoS
|
||||||
|
attacks which target many URLs simultaneously. This config will have no effect on the `allchanges` method.
|
||||||
|
|
||||||
|
`RateLimitFilter.lock_byuserip` can also be set to true in order to prevent requests from different users
|
||||||
|
interfering with one another. However, this can provide an ineffective safeguard against malicious DDoS attacks
|
||||||
|
which use multiple IP addresses.
|
||||||
|
|
||||||
|
Another important variable is the `RateLimitFilter.lock_timeout` config, which is set to 10 seconds by default.
|
||||||
|
This should be increased on sites which may be slow to generate page versions, whether due to lower
|
||||||
|
server capacity or volume of content (number of page versions). Requests to this page after the timeout
|
||||||
|
will not trigger any rate limit safeguard, so you should be sure that this is set to an appropriate level.
|
||||||
|
@ -6,11 +6,28 @@ class VersionFeedFunctionalTest extends FunctionalTest {
|
|||||||
'Page_Controller' => array('VersionFeed_Controller'),
|
'Page_Controller' => array('VersionFeed_Controller'),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
protected $userIP;
|
||||||
|
|
||||||
public function setUp() {
|
public function setUp() {
|
||||||
parent::setUp();
|
parent::setUp();
|
||||||
|
|
||||||
$cache = SS_Cache::factory('VersionFeed_Controller');
|
$cache = SS_Cache::factory('VersionFeed_Controller');
|
||||||
$cache->clean(Zend_Cache::CLEANING_MODE_ALL);
|
$cache->clean(Zend_Cache::CLEANING_MODE_ALL);
|
||||||
|
|
||||||
|
$this->userIP = isset($_SERVER['HTTP_CLIENT_IP']) ? $_SERVER['HTTP_CLIENT_IP'] : null;
|
||||||
|
|
||||||
|
Config::nest();
|
||||||
|
Config::inst()->update('RateLimitFilter', 'lock_timeout', 20);
|
||||||
|
Config::inst()->update('RateLimitFilter', 'lock_bypage', false);
|
||||||
|
Config::inst()->update('RateLimitFilter', 'lock_byuserip', false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function tearDown() {
|
||||||
|
Config::unnest();
|
||||||
|
|
||||||
|
$_SERVER['HTTP_CLIENT_IP'] = $this->userIP;
|
||||||
|
|
||||||
|
parent::tearDown();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testPublicHistory() {
|
public function testPublicHistory() {
|
||||||
@ -37,6 +54,59 @@ class VersionFeedFunctionalTest extends FunctionalTest {
|
|||||||
$this->assertTrue((bool)$xml->channel->item);
|
$this->assertTrue((bool)$xml->channel->item);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testRateLimiting() {
|
||||||
|
|
||||||
|
$page1 = $this->createPageWithChanges(array('PublicHistory' => true, 'Title' => 'Page1'));
|
||||||
|
$page2 = $this->createPageWithChanges(array('PublicHistory' => true, 'Title' => 'Page2'));
|
||||||
|
|
||||||
|
// Artifically set cache lock
|
||||||
|
Config::inst()->update('RateLimitFilter', 'lock_byuserip', false);
|
||||||
|
$cache = SS_Cache::factory('VersionFeed_Controller');
|
||||||
|
$cache->setOption('automatic_serialization', true);
|
||||||
|
$cache->save(time() - 2, RateLimitFilter::CACHE_PREFIX);
|
||||||
|
|
||||||
|
// Test normal hit
|
||||||
|
$response = $this->get($page1->RelativeLink('changes'));
|
||||||
|
$this->assertEquals(429, $response->getStatusCode());
|
||||||
|
$this->assertGreaterThan(0, $response->getHeader('Retry-After'));
|
||||||
|
$response = $this->get($page2->RelativeLink('changes'));
|
||||||
|
$this->assertEquals(429, $response->getStatusCode());
|
||||||
|
$this->assertGreaterThan(0, $response->getHeader('Retry-After'));
|
||||||
|
|
||||||
|
// Test page specific lock
|
||||||
|
Config::inst()->update('RateLimitFilter', 'lock_bypage', true);
|
||||||
|
$key = implode('_', array(
|
||||||
|
'changes',
|
||||||
|
$page1->ID,
|
||||||
|
Versioned::get_versionnumber_by_stage('SiteTree', 'Live', $page1->ID, false)
|
||||||
|
));
|
||||||
|
$key = RateLimitFilter::CACHE_PREFIX . '_' . md5($key);
|
||||||
|
$cache->save(time() - 2, $key);
|
||||||
|
$response = $this->get($page1->RelativeLink('changes'));
|
||||||
|
$this->assertEquals(429, $response->getStatusCode());
|
||||||
|
$this->assertGreaterThan(0, $response->getHeader('Retry-After'));
|
||||||
|
$response = $this->get($page2->RelativeLink('changes'));
|
||||||
|
$this->assertEquals(200, $response->getStatusCode());
|
||||||
|
Config::inst()->update('RateLimitFilter', 'lock_bypage', false);
|
||||||
|
|
||||||
|
// Test rate limit hit by IP
|
||||||
|
Config::inst()->update('RateLimitFilter', 'lock_byuserip', true);
|
||||||
|
$_SERVER['HTTP_CLIENT_IP'] = '127.0.0.1';
|
||||||
|
$cache->save(time() - 2, RateLimitFilter::CACHE_PREFIX . '_' . md5('127.0.0.1'));
|
||||||
|
$response = $this->get($page1->RelativeLink('changes'));
|
||||||
|
$this->assertEquals(429, $response->getStatusCode());
|
||||||
|
$this->assertGreaterThan(0, $response->getHeader('Retry-After'));
|
||||||
|
|
||||||
|
// Test rate limit doesn't hit other IP
|
||||||
|
$_SERVER['HTTP_CLIENT_IP'] = '127.0.0.20';
|
||||||
|
$cache->save(time() - 2, RateLimitFilter::CACHE_PREFIX . '_' . md5('127.0.0.1'));
|
||||||
|
$response = $this->get($page1->RelativeLink('changes'));
|
||||||
|
$this->assertEquals(200, $response->getStatusCode());
|
||||||
|
|
||||||
|
// Restore setting
|
||||||
|
Config::inst()->update('RateLimitFilter', 'lock_byuserip', false);
|
||||||
|
}
|
||||||
|
|
||||||
public function testContainsChangesForPageOnly() {
|
public function testContainsChangesForPageOnly() {
|
||||||
$page1 = $this->createPageWithChanges(array('Title' => 'Page1'));
|
$page1 = $this->createPageWithChanges(array('Title' => 'Page1'));
|
||||||
$page2 = $this->createPageWithChanges(array('Title' => 'Page2'));
|
$page2 = $this->createPageWithChanges(array('Title' => 'Page2'));
|
||||||
|
Loading…
Reference in New Issue
Block a user