mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 12:05:37 +00:00
ENHANCEMENT Making SSDatetime mockable through SSDatetime::set_mock_now()
ENHANCEMENT SSDatetime::now() git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@79503 467b73ca-7a2a-4603-9d3b-597d59a354a9
This commit is contained in:
parent
610232f19c
commit
6fa83caae6
@ -6,6 +6,11 @@
|
|||||||
* Alternatively you can set a timestamp that is evaluated through
|
* Alternatively you can set a timestamp that is evaluated through
|
||||||
* PHP's built-in date() and strtotime() function according to your system locale.
|
* PHP's built-in date() and strtotime() function according to your system locale.
|
||||||
*
|
*
|
||||||
|
* For all computations involving the current date and time,
|
||||||
|
* please use {@link SSDatetime::now()} instead of PHP's built-in date() and time()
|
||||||
|
* methods. This ensures that all time-based computations are testable with mock dates
|
||||||
|
* through {@link SSDatetime::set_mock_now()}.
|
||||||
|
*
|
||||||
* @todo Add localization support, see http://open.silverstripe.com/ticket/2931
|
* @todo Add localization support, see http://open.silverstripe.com/ticket/2931
|
||||||
*
|
*
|
||||||
* @package sapphire
|
* @package sapphire
|
||||||
@ -44,7 +49,7 @@ class SSDatetime extends Date {
|
|||||||
|
|
||||||
function requireField() {
|
function requireField() {
|
||||||
$parts=Array('datatype'=>'datetime');
|
$parts=Array('datatype'=>'datetime');
|
||||||
$values=Array('type'=>'ssdatetime', 'parts'=>$parts);
|
$values=Array('type'=>'SSDatetime', 'parts'=>$parts);
|
||||||
DB::requireField($this->tableName, $this->name, $values);
|
DB::requireField($this->tableName, $this->name, $values);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -55,6 +60,50 @@ class SSDatetime extends Date {
|
|||||||
public function scaffoldFormField($title = null, $params = null) {
|
public function scaffoldFormField($title = null, $params = null) {
|
||||||
return new PopupDateTimeField($this->name, $title);
|
return new PopupDateTimeField($this->name, $title);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
protected static $mock_now = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns either the current system date as determined
|
||||||
|
* by date(), or a mocked date through {@link set_mock_now()}.
|
||||||
|
*
|
||||||
|
* @return SSDatetime
|
||||||
|
*/
|
||||||
|
static function now() {
|
||||||
|
if(self::$mock_now) {
|
||||||
|
return self::$mock_now;
|
||||||
|
} else {
|
||||||
|
return DBField::create('SSDatetime', date('Y-m-d H:i:s'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Mock the system date temporarily, which is useful for time-based unit testing.
|
||||||
|
* Use {@link clear_mock_now()} to revert to the current system date.
|
||||||
|
* Caution: This sets a fixed date that doesn't increment with time.
|
||||||
|
*
|
||||||
|
* @param SSDatetime|string $datetime Either in object format, or as a SSDatetime compatible string.
|
||||||
|
*/
|
||||||
|
static function set_mock_now($datetime) {
|
||||||
|
if($datetime instanceof SSDatetime) {
|
||||||
|
self::$mock_now = $datetime;
|
||||||
|
} elseif(is_string($datetime)) {
|
||||||
|
self::$mock_now = DBField::create('SSDatetime', $datetime);
|
||||||
|
} else {
|
||||||
|
throw new Exception('SSDatetime::set_mock_now(): Wrong format: ' . $datetime);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Clear any mocked date, which causes
|
||||||
|
* {@link Now()} to return the current system date.
|
||||||
|
*/
|
||||||
|
static function clear_mock_now() {
|
||||||
|
self::$mock_now = null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
?>
|
?>
|
36
tests/model/SSDateTimeTest.php
Normal file
36
tests/model/SSDateTimeTest.php
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* Tests for {@link SSDatetime} class.
|
||||||
|
*
|
||||||
|
* @todo Current date comparisons are slightly dodgy, as they only compare
|
||||||
|
* the current date (not hour, minute, second) and assume that the date
|
||||||
|
* doesn't switch throughout the test execution. This means tests might
|
||||||
|
* fail when run at 23:59:59.
|
||||||
|
*
|
||||||
|
* @package sapphire
|
||||||
|
* @subpackage tests
|
||||||
|
*/
|
||||||
|
class SSDatetimeTest extends SapphireTest {
|
||||||
|
function testNowWithSystemDate() {
|
||||||
|
$systemDatetime = DBField::create('SSDatetime', date('Y-m-d H:i:s'));
|
||||||
|
$nowDatetime = SSDatetime::now();
|
||||||
|
|
||||||
|
$this->assertEquals($systemDatetime->Date(), $nowDatetime->Date());
|
||||||
|
}
|
||||||
|
|
||||||
|
function testNowWithMockDate() {
|
||||||
|
// Test setting
|
||||||
|
$mockDate = '2001-12-31 22:10:59';
|
||||||
|
SSDatetime::set_mock_now($mockDate);
|
||||||
|
$systemDatetime = DBField::create('SSDatetime', date('Y-m-d H:i:s'));
|
||||||
|
$nowDatetime = SSDatetime::now();
|
||||||
|
$this->assertNotEquals($systemDatetime->Date(), $nowDatetime->Date());
|
||||||
|
$this->assertEquals($nowDatetime->getValue(), $mockDate);
|
||||||
|
|
||||||
|
// Test clearing
|
||||||
|
SSDatetime::clear_mock_now();
|
||||||
|
$systemDatetime = DBField::create('SSDatetime', date('Y-m-d H:i:s'));
|
||||||
|
$nowDatetime = SSDatetime::now();
|
||||||
|
$this->assertEquals($systemDatetime->Date(), $nowDatetime->Date());
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user