2014-02-04 23:38:22 +01:00
|
|
|
<?php
|
|
|
|
|
2017-04-21 01:58:27 +02:00
|
|
|
namespace SilverStripe\TestSession;
|
|
|
|
|
2018-03-07 00:04:11 +01:00
|
|
|
use DirectoryIterator;
|
2017-06-13 05:00:32 +02:00
|
|
|
use Exception;
|
|
|
|
use InvalidArgumentException;
|
|
|
|
use LogicException;
|
2018-03-07 00:04:11 +01:00
|
|
|
use SilverStripe\Assets\Filesystem;
|
2017-10-27 03:37:12 +02:00
|
|
|
use SilverStripe\Core\Environment;
|
2016-08-29 06:17:53 +02:00
|
|
|
use SilverStripe\Control\Director;
|
2017-06-08 08:01:46 +02:00
|
|
|
use SilverStripe\Control\HTTPRequest;
|
2017-04-21 01:58:27 +02:00
|
|
|
use SilverStripe\Core\Config\Configurable;
|
|
|
|
use SilverStripe\Core\Extensible;
|
|
|
|
use SilverStripe\Core\Injector\Injectable;
|
2016-08-29 06:17:53 +02:00
|
|
|
use SilverStripe\Core\Injector\Injector;
|
|
|
|
use SilverStripe\Dev\FixtureFactory;
|
2017-07-03 05:09:05 +02:00
|
|
|
use SilverStripe\Dev\YamlFixture;
|
2017-06-21 11:04:10 +02:00
|
|
|
use SilverStripe\ORM\Connect\TempDatabase;
|
2016-06-27 04:51:15 +02:00
|
|
|
use SilverStripe\ORM\DatabaseAdmin;
|
2017-06-13 05:00:32 +02:00
|
|
|
use SilverStripe\ORM\DB;
|
2017-02-07 05:52:29 +01:00
|
|
|
use SilverStripe\ORM\FieldType\DBDatetime;
|
2017-04-21 01:58:27 +02:00
|
|
|
use SilverStripe\Versioned\Versioned;
|
|
|
|
use stdClass;
|
2016-08-29 06:17:53 +02:00
|
|
|
|
2014-02-04 23:38:22 +01:00
|
|
|
/**
|
2014-02-09 06:38:58 +01:00
|
|
|
* Responsible for starting and finalizing test sessions.
|
|
|
|
* Since these session span across multiple requests, session information is persisted
|
|
|
|
* in a file. This file is stored in the webroot by default, and the test session
|
|
|
|
* is considered "in progress" as long as this file exists.
|
2014-02-04 23:38:22 +01:00
|
|
|
*
|
2014-02-09 06:38:58 +01:00
|
|
|
* This allows for cross-request, cross-client sharing of the same testsession,
|
|
|
|
* for example: Behat CLI starts a testsession, then opens a web browser which
|
|
|
|
* makes a separate request picking up the same testsession.
|
|
|
|
*
|
2014-03-01 06:28:34 +01:00
|
|
|
* An environment can have an optional identifier ({@link id}), which allows
|
|
|
|
* multiple environments to exist at the same time in the same webroot.
|
2014-07-21 00:10:54 +02:00
|
|
|
* This enables parallel testing with (mostly) isolated state.
|
2014-03-01 06:28:34 +01:00
|
|
|
*
|
|
|
|
* 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)
|
|
|
|
* It can optionally contain other details that should be passed through many separate requests:
|
|
|
|
* - datetime: Mocked SS_DateTime ({@see TestSessionRequestFilter})
|
|
|
|
* - mailer: Mocked Email sender ({@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.
|
|
|
|
*
|
2014-02-09 06:38:58 +01:00
|
|
|
* See {@link $state} for default information stored in the test session.
|
2014-02-04 23:38:22 +01:00
|
|
|
*/
|
2017-04-21 01:58:27 +02:00
|
|
|
class TestSessionEnvironment
|
2015-12-17 19:17:16 +01:00
|
|
|
{
|
2017-04-21 01:58:27 +02:00
|
|
|
use Injectable;
|
|
|
|
use Configurable;
|
|
|
|
use Extensible;
|
2015-12-17 19:17:16 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @var int Optional identifier for the session.
|
|
|
|
*/
|
|
|
|
protected $id;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var string The original database name, before we overrode it with our tmpdb.
|
|
|
|
*
|
|
|
|
* Used in {@link self::resetDatabaseName()} when we want to restore the normal DB connection.
|
|
|
|
*/
|
|
|
|
private $oldDatabaseName;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @config
|
|
|
|
* @var string Path (from web-root) to the test state file that indicates a testsession is in progress.
|
|
|
|
* Defaults to value stored in testsession/_config/_config.yml
|
|
|
|
*/
|
|
|
|
private static $test_state_file = 'TESTS_RUNNING.json';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @config
|
|
|
|
* @var [type]
|
|
|
|
*/
|
|
|
|
private static $test_state_id_file = 'TESTS_RUNNING-%s.json';
|
|
|
|
|
|
|
|
public function __construct($id = null)
|
|
|
|
{
|
|
|
|
if ($id) {
|
|
|
|
$this->id = $id;
|
2017-06-08 08:01:46 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function init(HTTPRequest $request)
|
|
|
|
{
|
|
|
|
if (!$this->id) {
|
2017-06-27 00:55:09 +02:00
|
|
|
$request->getSession()->init($request);
|
2015-12-17 19:17:16 +01:00
|
|
|
// $_SESSION != Session::get() in some execution paths, suspect Controller->pushCurrent()
|
|
|
|
// as part of the issue, easiest resolution is to use session directly for now
|
2017-06-12 04:59:20 +02:00
|
|
|
$this->id = $request->getSession()->get('TestSessionId');
|
2015-12-17 19:17:16 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-06-13 05:00:32 +02:00
|
|
|
* @return string Absolute path to the file persisting our state.
|
2015-12-17 19:17:16 +01:00
|
|
|
*/
|
|
|
|
public function getFilePath()
|
|
|
|
{
|
|
|
|
if ($this->id) {
|
2022-04-13 07:40:59 +02:00
|
|
|
$path = Director::getAbsFile(sprintf($this->config()->get('test_state_id_file') ?? '', $this->id));
|
2015-12-17 19:17:16 +01:00
|
|
|
} else {
|
2017-06-27 00:55:09 +02:00
|
|
|
$path = Director::getAbsFile($this->config()->get('test_state_file'));
|
2015-12-17 19:17:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return $path;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Tests for the existence of the file specified by $this->test_state_file
|
|
|
|
*/
|
|
|
|
public function isRunningTests()
|
|
|
|
{
|
2022-04-13 07:40:59 +02:00
|
|
|
return (file_exists($this->getFilePath() ?? ''));
|
2015-12-17 19:17:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param String $id
|
|
|
|
*/
|
|
|
|
public function setId($id)
|
|
|
|
{
|
|
|
|
$this->id = $id;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return String
|
|
|
|
*/
|
|
|
|
public function getId()
|
|
|
|
{
|
|
|
|
return $this->id;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a temp database, sets up any extra requirements, and writes the state file. The database will be
|
|
|
|
* connected to as part of {@link self::applyState()}, so if you're continuing script execution after calling this
|
|
|
|
* method, be aware that the database will be different - so various things may break (e.g. administrator logins
|
|
|
|
* using the SS_DEFAULT_USERNAME / SS_DEFAULT_PASSWORD constants).
|
|
|
|
*
|
|
|
|
* If something isn't explicitly handled here, and needs special handling, then it should be taken care of by an
|
|
|
|
* extension to TestSessionEnvironment. You can either extend onBeforeStartTestSession() or
|
|
|
|
* onAfterStartTestSession(). Alternatively, for more fine-grained control, you can also extend
|
|
|
|
* onBeforeApplyState() and onAfterApplyState(). See the {@link self::applyState()} method for more.
|
|
|
|
*
|
|
|
|
* @param array $state An array of test state options to write.
|
2016-08-29 06:17:53 +02:00
|
|
|
* @param mixed $id
|
2015-12-17 19:17:16 +01:00
|
|
|
*/
|
|
|
|
public function startTestSession($state = null, $id = null)
|
|
|
|
{
|
|
|
|
if (!$state) {
|
|
|
|
$state = array();
|
|
|
|
}
|
|
|
|
$this->removeStateFile();
|
|
|
|
$this->id = $id;
|
|
|
|
|
|
|
|
// Assumes state will be modified by reference
|
|
|
|
$this->extend('onBeforeStartTestSession', $state);
|
|
|
|
|
|
|
|
// Convert to JSON and back so we can share the applyState() code between this and ->loadFromFile()
|
|
|
|
$json = json_encode($state, JSON_FORCE_OBJECT);
|
2022-04-13 07:40:59 +02:00
|
|
|
$state = json_decode($json ?? '');
|
2015-12-17 19:17:16 +01:00
|
|
|
|
|
|
|
$this->applyState($state);
|
|
|
|
|
2018-03-07 00:04:11 +01:00
|
|
|
// Back up /assets folder
|
|
|
|
$this->backupAssets();
|
|
|
|
|
2015-12-17 19:17:16 +01:00
|
|
|
$this->extend('onAfterStartTestSession');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function updateTestSession($state)
|
|
|
|
{
|
|
|
|
$this->extend('onBeforeUpdateTestSession', $state);
|
|
|
|
|
|
|
|
// Convert to JSON and back so we can share the appleState() code between this and ->loadFromFile()
|
|
|
|
$json = json_encode($state, JSON_FORCE_OBJECT);
|
2022-04-13 07:40:59 +02:00
|
|
|
$state = json_decode($json ?? '');
|
2015-12-17 19:17:16 +01:00
|
|
|
|
|
|
|
$this->applyState($state);
|
|
|
|
|
|
|
|
$this->extend('onAfterUpdateTestSession');
|
|
|
|
}
|
|
|
|
|
2018-03-07 00:04:11 +01:00
|
|
|
/**
|
|
|
|
* Backup all assets from /assets to /assets_backup.
|
|
|
|
* Note: Only does file move, no files ever duplicated / deleted
|
|
|
|
*/
|
|
|
|
protected function backupAssets()
|
|
|
|
{
|
|
|
|
// Ensure files backed up to assets dir
|
|
|
|
$backupFolder = $this->getAssetsBackupfolder();
|
2022-04-13 07:40:59 +02:00
|
|
|
if (!is_dir($backupFolder ?? '')) {
|
2018-03-07 00:04:11 +01:00
|
|
|
Filesystem::makeFolder($backupFolder);
|
|
|
|
}
|
|
|
|
$this->moveRecursive(ASSETS_PATH, $backupFolder, ['.htaccess', 'web.config', '.protected']);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Restore all assets to /assets folder.
|
|
|
|
* Note: Only does file move, no files ever duplicated / deleted
|
|
|
|
*/
|
|
|
|
public function restoreAssets()
|
|
|
|
{
|
|
|
|
// Ensure files backed up to assets dir
|
|
|
|
$backupFolder = $this->getAssetsBackupfolder();
|
2022-04-13 07:40:59 +02:00
|
|
|
if (is_dir($backupFolder ?? '')) {
|
2018-03-07 00:04:11 +01:00
|
|
|
// Move all files
|
|
|
|
Filesystem::makeFolder(ASSETS_PATH);
|
|
|
|
$this->moveRecursive($backupFolder, ASSETS_PATH);
|
|
|
|
Filesystem::removeFolder($backupFolder);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Recursively move files from one directory to another
|
|
|
|
*
|
|
|
|
* @param string $src Source of files being moved
|
|
|
|
* @param string $dest Destination of files being moved
|
|
|
|
* @param array $ignore List of files to not move
|
|
|
|
*/
|
|
|
|
protected function moveRecursive($src, $dest, $ignore = [])
|
|
|
|
{
|
|
|
|
// If source is not a directory stop processing
|
2022-04-13 07:40:59 +02:00
|
|
|
if (!is_dir($src ?? '')) {
|
2018-03-07 00:04:11 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the destination directory does not exist create it
|
2022-04-13 07:40:59 +02:00
|
|
|
if (!is_dir($dest ?? '') && !mkdir($dest ?? '')) {
|
2018-03-07 00:04:11 +01:00
|
|
|
// If the destination directory could not be created stop processing
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Open the source directory to read in files
|
|
|
|
$iterator = new DirectoryIterator($src);
|
|
|
|
foreach ($iterator as $file) {
|
|
|
|
if ($file->isFile()) {
|
2022-04-13 07:40:59 +02:00
|
|
|
if (!in_array($file->getFilename(), $ignore ?? [])) {
|
|
|
|
rename($file->getRealPath() ?? '', $dest . DIRECTORY_SEPARATOR . $file->getFilename());
|
2018-03-07 00:04:11 +01:00
|
|
|
}
|
|
|
|
} elseif (!$file->isDot() && $file->isDir()) {
|
|
|
|
// If a dir is ignored, still move children but don't remove self
|
|
|
|
$this->moveRecursive($file->getRealPath(), $dest . DIRECTORY_SEPARATOR . $file);
|
2022-04-13 07:40:59 +02:00
|
|
|
if (!in_array($file->getFilename(), $ignore ?? [])) {
|
2018-03-07 00:04:11 +01:00
|
|
|
Filesystem::removeFolder($file->getRealPath());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-17 19:17:16 +01:00
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* Persists the state to the filesystem.
|
|
|
|
*
|
|
|
|
* You can extend this by creating an Extension object and implementing either onBeforeApplyState() or
|
|
|
|
* onAfterApplyState() to add your own test state handling in.
|
|
|
|
*
|
2016-08-29 06:17:53 +02:00
|
|
|
* @param mixed $state
|
2015-12-17 19:17:16 +01:00
|
|
|
* @throws LogicException
|
|
|
|
* @throws InvalidArgumentException
|
|
|
|
*/
|
|
|
|
public function applyState($state)
|
|
|
|
{
|
|
|
|
$this->extend('onBeforeApplyState', $state);
|
|
|
|
|
|
|
|
// back up source
|
2017-06-21 11:04:10 +02:00
|
|
|
$databaseConfig = DB::getConfig();
|
2015-12-17 19:17:16 +01:00
|
|
|
$this->oldDatabaseName = $databaseConfig['database'];
|
|
|
|
|
|
|
|
// Load existing state from $this->state into $state, if there is any
|
|
|
|
$oldState = $this->getState();
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ensure we have a connection to the database
|
2019-01-24 20:22:29 +01:00
|
|
|
$this->connectToDatabase($state);
|
2015-12-17 19:17:16 +01:00
|
|
|
|
|
|
|
// Database
|
|
|
|
if (!$this->isRunningTests()) {
|
|
|
|
$dbName = (isset($state->database)) ? $state->database : null;
|
|
|
|
|
|
|
|
if ($dbName) {
|
|
|
|
$dbExists = DB::get_conn()->databaseExists($dbName);
|
|
|
|
} else {
|
|
|
|
$dbExists = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$dbExists) {
|
|
|
|
// Create a new one with a randomized name
|
2017-06-21 11:04:10 +02:00
|
|
|
$tempDB = new TempDatabase();
|
|
|
|
$dbName = $tempDB->build();
|
2015-12-17 19:17:16 +01:00
|
|
|
|
|
|
|
$state->database = $dbName; // In case it's changed by the call to SapphireTest::create_temp_db();
|
|
|
|
|
|
|
|
// Set existing one, assumes it already has been created
|
2017-10-27 03:37:12 +02:00
|
|
|
$prefix = Environment::getEnv('SS_DATABASE_PREFIX') ?: 'ss_';
|
2022-04-13 07:40:59 +02:00
|
|
|
$pattern = strtolower(sprintf('#^%stmpdb.*#', preg_quote($prefix ?? '', '#')));
|
|
|
|
if (!preg_match($pattern ?? '', $dbName ?? '')) {
|
2015-12-17 19:17:16 +01:00
|
|
|
throw new InvalidArgumentException("Invalid database name format");
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->oldDatabaseName = $databaseConfig['database'];
|
|
|
|
$databaseConfig['database'] = $dbName; // Instead of calling DB::set_alternative_db_name();
|
|
|
|
|
|
|
|
// Connect to the new database, overwriting the old DB connection (if any)
|
|
|
|
DB::connect($databaseConfig);
|
|
|
|
}
|
2018-12-18 02:24:59 +01:00
|
|
|
|
|
|
|
TestSessionState::create()->write(); // initialize the session state
|
2015-12-17 19:17:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Mailer
|
|
|
|
$mailer = (isset($state->mailer)) ? $state->mailer : null;
|
|
|
|
|
|
|
|
if ($mailer) {
|
2022-04-13 07:40:59 +02:00
|
|
|
if (!class_exists($mailer ?? '') || !is_subclass_of($mailer, 'SilverStripe\\Control\\Email\\Mailer')) {
|
2015-12-17 19:17:16 +01:00
|
|
|
throw new InvalidArgumentException(sprintf(
|
|
|
|
'Class "%s" is not a valid class, or subclass of Mailer',
|
|
|
|
$mailer
|
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Date and time
|
|
|
|
if (isset($state->datetime)) {
|
2017-02-07 05:52:29 +01:00
|
|
|
$formatter = DBDatetime::singleton()->getFormatter();
|
|
|
|
$formatter->setPattern(DBDatetime::ISO_DATETIME);
|
2015-12-17 19:17:16 +01:00
|
|
|
// Convert DatetimeField format
|
2017-02-07 05:52:29 +01:00
|
|
|
if ($formatter->parse($state->datetime) === false) {
|
2015-12-17 19:17:16 +01:00
|
|
|
throw new LogicException(sprintf(
|
|
|
|
'Invalid date format "%s", use yyyy-MM-dd HH:mm:ss',
|
|
|
|
$state->datetime
|
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->saveState($state);
|
2018-12-18 02:24:59 +01:00
|
|
|
|
2015-12-17 19:17:16 +01:00
|
|
|
$this->extend('onAfterApplyState');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Import the database
|
|
|
|
*
|
|
|
|
* @param String $path Absolute path to a SQL dump (include DROP TABLE commands)
|
2016-08-29 06:17:53 +02:00
|
|
|
* @param bool $requireDefaultRecords
|
2015-12-17 19:17:16 +01:00
|
|
|
*/
|
|
|
|
public function importDatabase($path, $requireDefaultRecords = false)
|
|
|
|
{
|
2022-04-13 07:40:59 +02:00
|
|
|
$sql = file_get_contents($path ?? '');
|
2015-12-17 19:17:16 +01:00
|
|
|
|
|
|
|
// Split into individual query commands, removing comments
|
2017-04-21 01:58:27 +02:00
|
|
|
$sqlCmds = array_filter(preg_split(
|
|
|
|
'/;\n/',
|
2022-04-13 07:40:59 +02:00
|
|
|
preg_replace(array('/^$\n/m', '/^(\/|#).*$\n/m'), '', $sql ?? '') ?? ''
|
|
|
|
) ?? []);
|
2015-12-17 19:17:16 +01:00
|
|
|
|
|
|
|
// Execute each query
|
|
|
|
foreach ($sqlCmds as $sqlCmd) {
|
|
|
|
DB::query($sqlCmd);
|
|
|
|
}
|
|
|
|
|
|
|
|
// In case the dump involved CREATE TABLE commands, we need to ensure the schema is still up to date
|
|
|
|
$dbAdmin = new DatabaseAdmin();
|
|
|
|
Versioned::set_reading_mode('');
|
|
|
|
$dbAdmin->doBuild(true, $requireDefaultRecords);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Build the database with default records, see {@link DataObject->requireDefaultRecords()}.
|
|
|
|
*/
|
|
|
|
public function requireDefaultRecords()
|
|
|
|
{
|
|
|
|
$dbAdmin = new DatabaseAdmin();
|
|
|
|
Versioned::set_reading_mode('');
|
|
|
|
$dbAdmin->doBuild(true, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sliented as if the file already exists by another process, we don't want
|
|
|
|
* to modify.
|
2016-08-29 06:17:53 +02:00
|
|
|
*
|
|
|
|
* @param mixed $state
|
2015-12-17 19:17:16 +01:00
|
|
|
*/
|
|
|
|
public function saveState($state)
|
|
|
|
{
|
|
|
|
if (defined('JSON_PRETTY_PRINT')) {
|
|
|
|
$content = json_encode($state, JSON_PRETTY_PRINT);
|
|
|
|
} else {
|
|
|
|
$content = json_encode($state);
|
|
|
|
}
|
|
|
|
$old = umask(0);
|
2022-04-13 07:40:59 +02:00
|
|
|
file_put_contents($this->getFilePath() ?? '', $content, LOCK_EX);
|
2015-12-17 19:17:16 +01:00
|
|
|
umask($old);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function loadFromFile()
|
|
|
|
{
|
|
|
|
if ($this->isRunningTests()) {
|
|
|
|
try {
|
2022-04-13 07:40:59 +02:00
|
|
|
$contents = file_get_contents($this->getFilePath() ?? '');
|
|
|
|
$json = json_decode($contents ?? '');
|
2015-12-17 19:17:16 +01:00
|
|
|
|
|
|
|
$this->applyState($json);
|
|
|
|
} catch (Exception $e) {
|
2017-06-21 11:04:10 +02:00
|
|
|
throw new Exception(
|
|
|
|
"A test session appears to be in progress, but we can't retrieve the details.\n"
|
|
|
|
. "Try removing the " . $this->getFilePath() . " file.\n"
|
|
|
|
. "Inner error: " . $e->getMessage() . "\n"
|
|
|
|
. "Stacktrace: " . $e->getTraceAsString()
|
|
|
|
);
|
2015-12-17 19:17:16 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private function removeStateFile()
|
|
|
|
{
|
|
|
|
$file = $this->getFilePath();
|
|
|
|
|
2022-04-13 07:40:59 +02:00
|
|
|
if (file_exists($file ?? '')) {
|
|
|
|
if (!unlink($file ?? '')) {
|
2015-12-17 19:17:16 +01:00
|
|
|
throw new \Exception('Unable to remove the testsession state file, please remove it manually. File '
|
|
|
|
. 'path: ' . $file);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Cleans up the test session state by restoring the normal database connect (for the rest of this request, if any)
|
|
|
|
* and removes the {@link self::$test_state_file} so that future requests don't use this test state.
|
|
|
|
*
|
|
|
|
* Can be extended by implementing either onBeforeEndTestSession() or onAfterEndTestSession().
|
|
|
|
*
|
|
|
|
* This should implement itself cleanly in case it is called twice (e.g. don't throw errors when the state file
|
|
|
|
* doesn't exist anymore because it's already been cleaned up etc.) This is because during behat test runs where
|
|
|
|
* a queueing system (e.g. silverstripe-resque) is used, the behat module may call this first, and then the forked
|
|
|
|
* worker will call it as well - but there is only one state file that is created.
|
|
|
|
*/
|
|
|
|
public function endTestSession()
|
|
|
|
{
|
|
|
|
$this->extend('onBeforeEndTestSession');
|
|
|
|
|
2018-03-07 00:04:11 +01:00
|
|
|
// Restore assets
|
|
|
|
$this->restoreAssets();
|
|
|
|
|
|
|
|
// Reset DB
|
2017-06-21 11:04:10 +02:00
|
|
|
$tempDB = new TempDatabase();
|
|
|
|
if ($tempDB->isUsed()) {
|
2015-12-18 03:32:20 +01:00
|
|
|
$state = $this->getState();
|
2016-08-29 06:17:53 +02:00
|
|
|
$dbConn = DB::get_schema();
|
2015-12-18 03:32:20 +01:00
|
|
|
$dbExists = $dbConn->databaseExists($state->database);
|
2017-04-21 01:58:27 +02:00
|
|
|
if ($dbExists) {
|
2015-12-18 03:32:20 +01:00
|
|
|
// Clean up temp database
|
2016-08-29 06:17:53 +02:00
|
|
|
$dbConn->dropDatabase($state->database);
|
2015-12-18 03:32:20 +01:00
|
|
|
file_put_contents('php://stdout', "Deleted temp database: $state->database" . PHP_EOL);
|
|
|
|
}
|
|
|
|
// End test session mode
|
2015-12-17 19:17:16 +01:00
|
|
|
$this->resetDatabaseName();
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->removeStateFile();
|
|
|
|
|
|
|
|
$this->extend('onAfterEndTestSession');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Loads a YAML fixture into the database as part of the {@link TestSessionController}.
|
|
|
|
*
|
|
|
|
* @param string $fixtureFile The .yml file to load
|
|
|
|
* @return FixtureFactory The loaded fixture
|
|
|
|
* @throws LogicException
|
|
|
|
*/
|
|
|
|
public function loadFixtureIntoDb($fixtureFile)
|
|
|
|
{
|
2018-03-07 00:04:11 +01:00
|
|
|
$realFile = realpath(BASE_PATH . '/' . $fixtureFile);
|
2022-04-13 07:40:59 +02:00
|
|
|
$baseDir = realpath(Director::baseFolder() ?? '');
|
|
|
|
if (!$realFile || !file_exists($realFile ?? '')) {
|
2015-12-17 19:17:16 +01:00
|
|
|
throw new LogicException("Fixture file doesn't exist");
|
2022-04-13 07:40:59 +02:00
|
|
|
} elseif (substr($realFile ?? '', 0, strlen($baseDir ?? '')) != $baseDir) {
|
2015-12-17 19:17:16 +01:00
|
|
|
throw new LogicException("Fixture file must be inside $baseDir");
|
2022-04-13 07:40:59 +02:00
|
|
|
} elseif (substr($realFile ?? '', -4) != '.yml') {
|
2015-12-17 19:17:16 +01:00
|
|
|
throw new LogicException("Fixture file must be a .yml file");
|
2022-04-13 07:40:59 +02:00
|
|
|
} elseif (!preg_match('/^([^\/.][^\/]+)\/tests\//', $fixtureFile ?? '')) {
|
2015-12-17 19:17:16 +01:00
|
|
|
throw new LogicException("Fixture file must be inside the tests subfolder of one of your modules.");
|
|
|
|
}
|
|
|
|
|
2017-07-03 05:09:05 +02:00
|
|
|
$factory = Injector::inst()->create(FixtureFactory::class);
|
|
|
|
$fixture = Injector::inst()->create(YamlFixture::class, $fixtureFile);
|
2015-12-17 19:17:16 +01:00
|
|
|
$fixture->writeInto($factory);
|
|
|
|
|
|
|
|
$state = $this->getState();
|
|
|
|
$state->fixtures[] = $fixtureFile;
|
|
|
|
$this->applyState($state);
|
|
|
|
|
|
|
|
return $fixture;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reset the database connection to use the original database. Called by {@link self::endTestSession()}.
|
|
|
|
*/
|
|
|
|
public function resetDatabaseName()
|
|
|
|
{
|
|
|
|
if ($this->oldDatabaseName) {
|
2017-06-22 05:15:17 +02:00
|
|
|
$databaseConfig = DB::getConfig();
|
2015-12-17 19:17:16 +01:00
|
|
|
$databaseConfig['database'] = $this->oldDatabaseName;
|
2017-06-22 05:15:17 +02:00
|
|
|
DB::setConfig($databaseConfig);
|
2015-12-17 19:17:16 +01:00
|
|
|
|
|
|
|
$conn = DB::get_conn();
|
|
|
|
|
|
|
|
if ($conn) {
|
|
|
|
$conn->selectDatabase($this->oldDatabaseName, false, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return stdClass Data as taken from the JSON object in {@link self::loadFromFile()}
|
|
|
|
*/
|
|
|
|
public function getState()
|
|
|
|
{
|
|
|
|
$path = Director::getAbsFile($this->getFilePath());
|
2022-04-13 07:40:59 +02:00
|
|
|
return (file_exists($path ?? '')) ? json_decode(file_get_contents($path)) : new stdClass;
|
2015-12-17 19:17:16 +01:00
|
|
|
}
|
2018-03-07 00:04:11 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Path where assets should be backed up during testing
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
protected function getAssetsBackupfolder()
|
|
|
|
{
|
|
|
|
return PUBLIC_PATH . DIRECTORY_SEPARATOR . 'assets_backup';
|
|
|
|
}
|
2018-12-18 02:24:59 +01:00
|
|
|
|
2019-01-24 20:22:29 +01:00
|
|
|
/**
|
|
|
|
* Ensure that there is a connection to the database
|
2022-04-13 07:40:59 +02:00
|
|
|
*
|
2019-01-24 20:22:29 +01:00
|
|
|
* @param mixed $state
|
|
|
|
*/
|
2022-04-13 07:40:59 +02:00
|
|
|
public function connectToDatabase($state = null)
|
|
|
|
{
|
2019-01-24 20:22:29 +01:00
|
|
|
if ($state == null) {
|
|
|
|
$state = $this->getState();
|
|
|
|
}
|
|
|
|
|
|
|
|
$databaseConfig = DB::getConfig();
|
|
|
|
|
|
|
|
if (isset($state->database) && $state->database) {
|
|
|
|
if (!DB::get_conn()) {
|
|
|
|
// No connection, so try and connect to tmpdb if it exists
|
|
|
|
if (isset($state->database)) {
|
|
|
|
$this->oldDatabaseName = $databaseConfig['database'];
|
|
|
|
$databaseConfig['database'] = $state->database;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Connect to database
|
|
|
|
DB::connect($databaseConfig);
|
|
|
|
} else {
|
|
|
|
// We've already connected to the database, do a fast check to see what database we're currently using
|
|
|
|
$db = DB::get_conn()->getSelectedDatabase();
|
|
|
|
if (isset($state->database) && $db != $state->database) {
|
|
|
|
$this->oldDatabaseName = $databaseConfig['database'];
|
|
|
|
$databaseConfig['database'] = $state->database;
|
|
|
|
DB::connect($databaseConfig);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-12-18 02:24:59 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Wait for pending requests
|
|
|
|
*
|
|
|
|
* @param int $await Time to wait (in ms) after the last response (to allow the browser react)
|
|
|
|
* @param int $timeout For how long (in ms) do we wait before giving up
|
|
|
|
*
|
|
|
|
* @return bool Whether there are no more pending requests
|
|
|
|
*/
|
2019-01-08 04:36:20 +01:00
|
|
|
public function waitForPendingRequests($await = 700, $timeout = 10000)
|
2018-12-18 02:24:59 +01:00
|
|
|
{
|
2019-01-10 03:30:39 +01:00
|
|
|
$timeout = TestSessionState::millitime() + $timeout;
|
2019-01-08 04:36:20 +01:00
|
|
|
$interval = max(300, $await);
|
2019-01-09 23:48:59 +01:00
|
|
|
|
2018-12-18 02:24:59 +01:00
|
|
|
do {
|
2019-01-10 03:30:39 +01:00
|
|
|
$now = TestSessionState::millitime();
|
2019-01-09 23:48:59 +01:00
|
|
|
|
|
|
|
if ($timeout < $now) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-12-18 02:24:59 +01:00
|
|
|
$model = TestSessionState::get()->byID(1);
|
|
|
|
|
|
|
|
$pendingRequests = $model->PendingRequests > 0;
|
2019-01-09 23:48:59 +01:00
|
|
|
$lastRequestAwait = ($model->LastResponseTimestamp + $await) > $now;
|
2018-12-18 02:24:59 +01:00
|
|
|
|
|
|
|
$pending = $pendingRequests || $lastRequestAwait;
|
|
|
|
} while ($pending && (usleep($interval * 1000) || true));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2014-06-12 04:44:55 +02:00
|
|
|
}
|