mirror of
https://github.com/silverstripe/silverstripe-testsession
synced 2024-10-22 14:06:00 +02:00
API Persist state directly on disk
In memory state can get out of date between CLI and web requests, leading to hard to debug errors. The performance impact of this should be low, given the size of the JSON file.
This commit is contained in:
parent
c247392da0
commit
40dd841eb3
@ -10,19 +10,26 @@
|
|||||||
* for example: Behat CLI starts a testsession, then opens a web browser which
|
* for example: Behat CLI starts a testsession, then opens a web browser which
|
||||||
* makes a separate request picking up the same testsession.
|
* makes a separate request picking up the same testsession.
|
||||||
*
|
*
|
||||||
* See {@link $state} for default information stored in the test session.
|
* An environment can have an optional identifier ({@link id}), which allows
|
||||||
*/
|
* multiple environments to exist at the same time in the same webroot.
|
||||||
class TestSessionEnvironment extends Object {
|
* This enables parallel testing with (mostly) isolated state.
|
||||||
/**
|
*
|
||||||
* @var stdClass Test session state. For a valid test session to exist, this needs to contain at least:
|
* For a valid test session to exist, this needs to contain at least:
|
||||||
* - database: The alternate database name that is being used for this test session (e.g. ss_tmpdb_1234567)
|
* - database: The alternate database name that is being used for this test session (e.g. ss_tmpdb_1234567)
|
||||||
* It can optionally contain other details that should be passed through many separate requests:
|
* It can optionally contain other details that should be passed through many separate requests:
|
||||||
* - datetime: Mocked SS_DateTime ({@see TestSessionRequestFilter})
|
* - datetime: Mocked SS_DateTime ({@see TestSessionRequestFilter})
|
||||||
* - mailer: Mocked Email sender ({@see TestSessionRequestFilter})
|
* - mailer: Mocked Email sender ({@see TestSessionRequestFilter})
|
||||||
* - stubfile: Path to PHP stub file for setup ({@see TestSessionRequestFilter})
|
* - stubfile: Path to PHP stub file for setup ({@see TestSessionRequestFilter})
|
||||||
* Extensions of TestSessionEnvironment can add extra fields in here to be saved and restored on each request.
|
* Extensions of TestSessionEnvironment can add extra fields in here to be saved and restored on each request.
|
||||||
|
*
|
||||||
|
* See {@link $state} for default information stored in the test session.
|
||||||
*/
|
*/
|
||||||
private $state;
|
class TestSessionEnvironment extends Object {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var int Optional identifier for the session.
|
||||||
|
*/
|
||||||
|
protected $id;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @var string The original database name, before we overrode it with our tmpdb.
|
* @var string The original database name, before we overrode it with our tmpdb.
|
||||||
@ -38,13 +45,20 @@ class TestSessionEnvironment extends Object {
|
|||||||
*/
|
*/
|
||||||
private static $test_state_file;
|
private static $test_state_file;
|
||||||
|
|
||||||
public function __construct() {
|
public function __construct($id = null) {
|
||||||
parent::__construct();
|
parent::__construct();
|
||||||
|
|
||||||
if($this->isRunningTests()) {
|
$this->id = $id;
|
||||||
$this->state = json_decode(file_get_contents(Director::getAbsFile($this->config()->test_state_file)));
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return String Absolute path to the file persisting our state.
|
||||||
|
*/
|
||||||
|
public function getFilePath() {
|
||||||
|
if($this->id) {
|
||||||
|
return Director::getAbsFile(sprintf($this->config()->test_state_id_file, $this->id));
|
||||||
} else {
|
} else {
|
||||||
$this->state = new stdClass;
|
return Director::getAbsFile($this->config()->test_state_file);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -52,7 +66,7 @@ class TestSessionEnvironment extends Object {
|
|||||||
* Tests for the existence of the file specified by $this->test_state_file
|
* Tests for the existence of the file specified by $this->test_state_file
|
||||||
*/
|
*/
|
||||||
public function isRunningTests() {
|
public function isRunningTests() {
|
||||||
return(file_exists(Director::getAbsFile($this->config()->test_state_file)));
|
return(file_exists($this->getFilePath()));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -68,7 +82,9 @@ class TestSessionEnvironment extends Object {
|
|||||||
*
|
*
|
||||||
* @param array $state An array of test state options to write.
|
* @param array $state An array of test state options to write.
|
||||||
*/
|
*/
|
||||||
public function startTestSession($state) {
|
public function startTestSession($state, $id = null) {
|
||||||
|
$this->removeStateFile();
|
||||||
|
|
||||||
$extendedState = $this->extend('onBeforeStartTestSession', $state);
|
$extendedState = $this->extend('onBeforeStartTestSession', $state);
|
||||||
|
|
||||||
// $extendedState is now a multi-dimensional array (if extensions exist)
|
// $extendedState is now a multi-dimensional array (if extensions exist)
|
||||||
@ -84,7 +100,6 @@ class TestSessionEnvironment extends Object {
|
|||||||
$state = json_decode($json);
|
$state = json_decode($json);
|
||||||
|
|
||||||
$this->applyState($state);
|
$this->applyState($state);
|
||||||
$this->persistState();
|
|
||||||
|
|
||||||
$this->extend('onAfterStartTestSession');
|
$this->extend('onAfterStartTestSession');
|
||||||
}
|
}
|
||||||
@ -97,7 +112,6 @@ class TestSessionEnvironment extends Object {
|
|||||||
$state = json_decode($json);
|
$state = json_decode($json);
|
||||||
|
|
||||||
$this->applyState($state);
|
$this->applyState($state);
|
||||||
$this->persistState();
|
|
||||||
|
|
||||||
$this->extend('onAfterUpdateTestSession');
|
$this->extend('onAfterUpdateTestSession');
|
||||||
}
|
}
|
||||||
@ -106,7 +120,7 @@ class TestSessionEnvironment extends Object {
|
|||||||
* Assumes the database has already been created in startTestSession(), as this method can be called from
|
* Assumes the database has already been created in startTestSession(), as this method can be called from
|
||||||
* _config.php where we don't yet have a DB connection.
|
* _config.php where we don't yet have a DB connection.
|
||||||
*
|
*
|
||||||
* Does not persist the state to the filesystem, see {@link self::persistState()}.
|
* Persists the state to the filesystem.
|
||||||
*
|
*
|
||||||
* You can extend this by creating an Extension object and implementing either onBeforeApplyState() or
|
* You can extend this by creating an Extension object and implementing either onBeforeApplyState() or
|
||||||
* onAfterApplyState() to add your own test state handling in.
|
* onAfterApplyState() to add your own test state handling in.
|
||||||
@ -120,12 +134,12 @@ class TestSessionEnvironment extends Object {
|
|||||||
$this->extend('onBeforeApplyState', $state);
|
$this->extend('onBeforeApplyState', $state);
|
||||||
|
|
||||||
// Load existing state from $this->state into $state, if there is any
|
// Load existing state from $this->state into $state, if there is any
|
||||||
if($this->state) {
|
$oldState = $this->getState();
|
||||||
foreach($this->state as $k => $v) {
|
if($oldState) {
|
||||||
|
foreach($oldState as $k => $v) {
|
||||||
if(!isset($state->$k)) $state->$k = $v; // Don't overwrite stuff in $state, as that's the new state
|
if(!isset($state->$k)) $state->$k = $v; // Don't overwrite stuff in $state, as that's the new state
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if(isset($state->database) && $state->database) {
|
if(isset($state->database) && $state->database) {
|
||||||
if(!DB::getConn()) {
|
if(!DB::getConn()) {
|
||||||
// No connection, so try and connect to tmpdb if it exists
|
// No connection, so try and connect to tmpdb if it exists
|
||||||
@ -235,7 +249,10 @@ class TestSessionEnvironment extends Object {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->state = $state;
|
file_put_contents(
|
||||||
|
$this->getFilePath(),
|
||||||
|
json_encode($state, JSON_PRETTY_PRINT)
|
||||||
|
);
|
||||||
|
|
||||||
$this->extend('onAfterApplyState');
|
$this->extend('onAfterApplyState');
|
||||||
}
|
}
|
||||||
@ -243,30 +260,20 @@ class TestSessionEnvironment extends Object {
|
|||||||
public function loadFromFile() {
|
public function loadFromFile() {
|
||||||
if($this->isRunningTests()) {
|
if($this->isRunningTests()) {
|
||||||
try {
|
try {
|
||||||
$contents = file_get_contents(Director::getAbsFile($this->config()->test_state_file));
|
$contents = file_get_contents($this->getFilePath());
|
||||||
$json = json_decode($contents);
|
$json = json_decode($contents);
|
||||||
|
|
||||||
$this->applyState($json);
|
$this->applyState($json);
|
||||||
} catch(Exception $e) {
|
} catch(Exception $e) {
|
||||||
throw new \Exception("A test session appears to be in progress, but we can't retrieve the details. "
|
throw new \Exception("A test session appears to be in progress, but we can't retrieve the details. "
|
||||||
. "Try removing the " . Director::getAbsFile($this->config()->test_state_file) . " file. Inner "
|
. "Try removing the " . $this->getFilePath() . " file. Inner "
|
||||||
. "error: " . $e->getMessage());
|
. "error: " . $e->getMessage());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Writes $this->state JSON object into the $this->config()->test_state_file file.
|
|
||||||
*/
|
|
||||||
public function persistState() {
|
|
||||||
file_put_contents(
|
|
||||||
Director::getAbsFile($this->config()->test_state_file),
|
|
||||||
json_encode($this->state, JSON_PRETTY_PRINT)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
private function removeStateFile() {
|
private function removeStateFile() {
|
||||||
$file = Director::getAbsFile($this->config()->test_state_file);
|
$file = $this->getFilePath();
|
||||||
if(file_exists($file)) {
|
if(file_exists($file)) {
|
||||||
if(!unlink($file)) {
|
if(!unlink($file)) {
|
||||||
throw new \Exception('Unable to remove the testsession state file, please remove it manually. File '
|
throw new \Exception('Unable to remove the testsession state file, please remove it manually. File '
|
||||||
@ -322,7 +329,9 @@ class TestSessionEnvironment extends Object {
|
|||||||
$fixture = Injector::inst()->create('YamlFixture', $fixtureFile);
|
$fixture = Injector::inst()->create('YamlFixture', $fixtureFile);
|
||||||
$fixture->writeInto($factory);
|
$fixture->writeInto($factory);
|
||||||
|
|
||||||
$this->state->fixtures[] = $fixtureFile;
|
$state = $this->getState();
|
||||||
|
$state->fixtures[] = $fixtureFile;
|
||||||
|
$this->applyState($state);
|
||||||
|
|
||||||
return $fixture;
|
return $fixture;
|
||||||
}
|
}
|
||||||
@ -342,6 +351,7 @@ class TestSessionEnvironment extends Object {
|
|||||||
* @return stdClass Data as taken from the JSON object in {@link self::loadFromFile()}
|
* @return stdClass Data as taken from the JSON object in {@link self::loadFromFile()}
|
||||||
*/
|
*/
|
||||||
public function getState() {
|
public function getState() {
|
||||||
return $this->state;
|
$path = Director::getAbsFile($this->config()->test_state_file);
|
||||||
|
return (file_exists($path)) ? json_decode(file_get_contents($path)) : new stdClass;
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -52,6 +52,6 @@ class TestSessionRequestFilter {
|
|||||||
// Store PHP session
|
// Store PHP session
|
||||||
$state = $this->testSessionEnvironment->getState();
|
$state = $this->testSessionEnvironment->getState();
|
||||||
$state->session = Session::get_all();
|
$state->session = Session::get_all();
|
||||||
$this->testSessionEnvironment->persistState();
|
$this->testSessionEnvironment->applyState($state);
|
||||||
}
|
}
|
||||||
}
|
}
|
Loading…
Reference in New Issue
Block a user