Compare commits

...

10 Commits

Author SHA1 Message Date
Steve Boyd
dc76ac2877
Merge d7c4b0c420 into 3eb61e353f 2024-09-27 02:59:59 +00:00
Steve Boyd
d7c4b0c420 FIX Ensure primary connection is updated with temporary database 2024-09-27 14:59:48 +12:00
Steve Boyd
8f4771385c ENH Use class name instead of self 2024-09-27 14:59:48 +12:00
Guy Sartorelli
3eb61e353f
API Update API to reflect changes to CLI interaction (#93) 2024-09-26 17:17:06 +12:00
Guy Sartorelli
a8efc75702
API Use new names for renamed classes (#94) 2024-09-23 14:33:08 +12:00
Guy Sartorelli
b966a32e0c
API Standardise extension hooks (#91) 2024-08-27 15:39:29 +12:00
Guy Sartorelli
c1cf2b4106
DEP Limit PHP support for CMS 6 (#92) 2024-08-22 12:14:07 +12:00
Steve Boyd
bc4cb87627 Merge branch '3' into 4 2024-05-07 15:49:25 +12:00
Guy Sartorelli
2a9fb2b1e3
Merge pull request #88 from creative-commoners/pulls/4/module-standardiser-1708317246
DEP Dependencies for CMS 6
2024-02-20 10:44:52 +13:00
Steve Boyd
b87e4d46d0 DEP Dependencies for CMS 6 2024-02-19 17:34:06 +13:00
5 changed files with 54 additions and 28 deletions

View File

@ -12,7 +12,7 @@ TestSessionEnvironment::singleton()->loadFromFile();
*/ */
if (class_exists('Resque_Event') && class_exists('SSResqueRun')) { if (class_exists('Resque_Event') && class_exists('SSResqueRun')) {
Resque_Event::listen('beforeFork', function ($data) { Resque_Event::listen('beforeFork', function ($data) {
$databaseConfig = DB::getConfig(); $databaseConfig = DB::getConfig(DB::CONN_PRIMARY);
// Reconnect to the database - this may connect to the old DB first, but is required because these processes // Reconnect to the database - this may connect to the old DB first, but is required because these processes
// are long-lived, and MySQL connections often get closed in between worker runs. We need to connect before // are long-lived, and MySQL connections often get closed in between worker runs. We need to connect before

View File

@ -15,9 +15,9 @@
} }
], ],
"require": { "require": {
"php": "^8.1", "php": "^8.3",
"composer/installers": "*", "composer/installers": "*",
"silverstripe/framework": "^5", "silverstripe/framework": "^6",
"silverstripe/vendor-plugin": "^2" "silverstripe/vendor-plugin": "^2"
}, },
"require-dev": { "require-dev": {

View File

@ -16,14 +16,14 @@ use SilverStripe\Forms\Form;
use SilverStripe\Forms\FormAction; use SilverStripe\Forms\FormAction;
use SilverStripe\Forms\HiddenField; use SilverStripe\Forms\HiddenField;
use SilverStripe\Forms\TextField; use SilverStripe\Forms\TextField;
use SilverStripe\ORM\ArrayList; use SilverStripe\Model\List\ArrayList;
use SilverStripe\ORM\Connect\TempDatabase; use SilverStripe\ORM\Connect\TempDatabase;
use SilverStripe\ORM\DB; use SilverStripe\ORM\DB;
use SilverStripe\ORM\FieldType\DBHTMLText; use SilverStripe\ORM\FieldType\DBHTMLText;
use SilverStripe\Security\Permission; use SilverStripe\Security\Permission;
use SilverStripe\Security\RandomGenerator; use SilverStripe\Security\RandomGenerator;
use SilverStripe\Security\Security; use SilverStripe\Security\Security;
use SilverStripe\View\ArrayData; use SilverStripe\Model\ArrayData;
use SilverStripe\View\Requirements; use SilverStripe\View\Requirements;
/** /**
@ -67,7 +67,7 @@ class TestSessionController extends Controller
{ {
parent::init(); parent::init();
$this->extend('init'); $this->extend('onInit');
$canAccess = ( $canAccess = (
!Director::isLive() !Director::isLive()
@ -326,7 +326,7 @@ class TestSessionController extends Controller
} }
/** /**
* As with {@link self::start()}, if you want to extend the functionality of this, then look at * As with {@link TestSessionController::start()}, if you want to extend the functionality of this, then look at
* {@link TestSessionEnvironent::endTestSession()} as the extension points have moved to there now that the logic * {@link TestSessionEnvironent::endTestSession()} as the extension points have moved to there now that the logic
* is there. * is there.
*/ */

View File

@ -15,9 +15,10 @@ use SilverStripe\Core\Extensible;
use SilverStripe\Core\Injector\Injectable; use SilverStripe\Core\Injector\Injectable;
use SilverStripe\Core\Injector\Injector; use SilverStripe\Core\Injector\Injector;
use SilverStripe\Dev\FixtureFactory; use SilverStripe\Dev\FixtureFactory;
use SilverStripe\PolyExecution\Command\DbBuild;
use SilverStripe\PolyExecution\PolyOutput;
use SilverStripe\Dev\YamlFixture; use SilverStripe\Dev\YamlFixture;
use SilverStripe\ORM\Connect\TempDatabase; use SilverStripe\ORM\Connect\TempDatabase;
use SilverStripe\ORM\DatabaseAdmin;
use SilverStripe\ORM\DB; use SilverStripe\ORM\DB;
use SilverStripe\ORM\FieldType\DBDatetime; use SilverStripe\ORM\FieldType\DBDatetime;
use SilverStripe\Versioned\Versioned; use SilverStripe\Versioned\Versioned;
@ -62,7 +63,7 @@ class TestSessionEnvironment
/** /**
* @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.
* *
* Used in {@link self::resetDatabaseName()} when we want to restore the normal DB connection. * Used in {@link TestSessionEnvironment::resetDatabaseName()} when we want to restore the normal DB connection.
*/ */
private $oldDatabaseName; private $oldDatabaseName;
@ -136,14 +137,16 @@ class TestSessionEnvironment
/** /**
* Creates a temp database, sets up any extra requirements, and writes the state file. The database will be * 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 * connected to as part of {@link TestSessionEnvironment::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 * 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). * 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 * 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 * extension to TestSessionEnvironment. You can either extend onBeforeStartTestSession() or
* onAfterStartTestSession(). Alternatively, for more fine-grained control, you can also extend * onAfterStartTestSession(). Alternatively, for more fine-grained control, you can also extend
* onBeforeApplyState() and onAfterApplyState(). See the {@link self::applyState()} method for more. * onBeforeApplyState() and onAfterApplyState(). See the {@link TestSessionEnvironment::applyState()}
* method for more.
* *
* @param array $state An array of test state options to write. * @param array $state An array of test state options to write.
* @param mixed $id * @param mixed $id
@ -269,7 +272,7 @@ class TestSessionEnvironment
$this->extend('onBeforeApplyState', $state); $this->extend('onBeforeApplyState', $state);
// back up source // back up source
$databaseConfig = DB::getConfig(); $databaseConfig = DB::getConfig(DB::CONN_PRIMARY);
$this->oldDatabaseName = $databaseConfig['database']; $this->oldDatabaseName = $databaseConfig['database'];
// Load existing state from $this->state into $state, if there is any // Load existing state from $this->state into $state, if there is any
@ -370,9 +373,13 @@ class TestSessionEnvironment
} }
// In case the dump involved CREATE TABLE commands, we need to ensure the schema is still up to date // In case the dump involved CREATE TABLE commands, we need to ensure the schema is still up to date
$dbAdmin = new DatabaseAdmin(); $dbBuild = new DbBuild();
Versioned::set_reading_mode(''); Versioned::set_reading_mode('');
$dbAdmin->doBuild(true, $requireDefaultRecords); $output = new PolyOutput(
Director::is_cli() ? PolyOutput::FORMAT_ANSI : PolyOutput::FORMAT_HTML,
PolyOutput::VERBOSITY_QUIET
);
$dbBuild->doBuild($output, $requireDefaultRecords);
} }
/** /**
@ -380,9 +387,13 @@ class TestSessionEnvironment
*/ */
public function requireDefaultRecords() public function requireDefaultRecords()
{ {
$dbAdmin = new DatabaseAdmin(); $dbBuild = new DbBuild();
Versioned::set_reading_mode(''); Versioned::set_reading_mode('');
$dbAdmin->doBuild(true, true); $output = new PolyOutput(
Director::is_cli() ? PolyOutput::FORMAT_ANSI : PolyOutput::FORMAT_HTML,
PolyOutput::VERBOSITY_QUIET
);
$dbBuild->doBuild($output, true);
} }
/** /**
@ -436,7 +447,8 @@ class TestSessionEnvironment
/** /**
* Cleans up the test session state by restoring the normal database connect (for the rest of this request, if any) * 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. * and removes the {@link TestSessionEnvironment::$test_state_file} so that future requests don't use this
* test state.
* *
* Can be extended by implementing either onBeforeEndTestSession() or onAfterEndTestSession(). * Can be extended by implementing either onBeforeEndTestSession() or onAfterEndTestSession().
* *
@ -505,14 +517,15 @@ class TestSessionEnvironment
} }
/** /**
* Reset the database connection to use the original database. Called by {@link self::endTestSession()}. * Reset the database connection to use the original database.
* Called by {@link TestSessionEnvironment::endTestSession()}.
*/ */
public function resetDatabaseName() public function resetDatabaseName()
{ {
if ($this->oldDatabaseName) { if ($this->oldDatabaseName) {
$databaseConfig = DB::getConfig(); $databaseConfig = DB::getConfig(DB::CONN_PRIMARY);
$databaseConfig['database'] = $this->oldDatabaseName; $databaseConfig['database'] = $this->oldDatabaseName;
DB::setConfig($databaseConfig); DB::setConfig($databaseConfig, DB::CONN_PRIMARY);
$conn = DB::get_conn(); $conn = DB::get_conn();
@ -523,7 +536,7 @@ class TestSessionEnvironment
} }
/** /**
* @return stdClass Data as taken from the JSON object in {@link self::loadFromFile()} * @return stdClass Data as taken from the JSON object in {@link TestSessionEnvironment::loadFromFile()}
*/ */
public function getState() public function getState()
{ {
@ -555,7 +568,7 @@ class TestSessionEnvironment
$state = $this->getState(); $state = $this->getState();
} }
$databaseConfig = DB::getConfig(); $databaseConfig = DB::getConfig(DB::CONN_PRIMARY);
if (isset($state->database) && $state->database) { if (isset($state->database) && $state->database) {
if (!DB::get_conn()) { if (!DB::get_conn()) {
@ -566,14 +579,14 @@ class TestSessionEnvironment
} }
// Connect to database // Connect to database
DB::connect($databaseConfig); $this->connectToDB($databaseConfig);
} else { } else {
// We've already connected to the database, do a fast check to see what database we're currently using // We've already connected to the database, do a fast check to see what database we're currently using
$db = DB::get_conn()->getSelectedDatabase(); $db = DB::get_conn()->getSelectedDatabase();
if (isset($state->database) && $db != $state->database) { if (isset($state->database) && $db != $state->database) {
$this->oldDatabaseName = $databaseConfig['database']; $this->oldDatabaseName = $databaseConfig['database'];
$databaseConfig['database'] = $state->database; $databaseConfig['database'] = $state->database;
DB::connect($databaseConfig); $this->connectToDB($databaseConfig);
} }
} }
} }
@ -609,4 +622,17 @@ class TestSessionEnvironment
return true; return true;
} }
private function connectToDB(array $databaseConfig): void
{
if (method_exists(DB::class, 'hasConfig')) {
// CMS 5.4+ - ensure we connect the primary connection and not a replica
// which can happen if we use the default value of DB::CONN_DYNAMIC
// and there is a replica database configured
DB::connect($databaseConfig, DB::CONN_PRIMARY);
} else {
// CMS 5.3 and below do not support replica connections, use 'default' connection
DB::connect($databaseConfig);
}
}
} }

View File

@ -33,7 +33,7 @@ class TestSessionState extends DataObject
{ {
$schema = DataObject::getSchema(); $schema = DataObject::getSchema();
$update = SQLUpdate::create(sprintf('"%s"', $schema->tableName(self::class))) $update = SQLUpdate::create(sprintf('"%s"', $schema->tableName(TestSessionState::class)))
->addWhere(['ID' => 1]) ->addWhere(['ID' => 1])
->assignSQL('"PendingRequests"', '"PendingRequests" + 1'); ->assignSQL('"PendingRequests"', '"PendingRequests" + 1');
@ -50,10 +50,10 @@ class TestSessionState extends DataObject
{ {
$schema = DataObject::getSchema(); $schema = DataObject::getSchema();
$update = SQLUpdate::create(sprintf('"%s"', $schema->tableName(self::class))) $update = SQLUpdate::create(sprintf('"%s"', $schema->tableName(TestSessionState::class)))
->addWhere(['ID' => 1]) ->addWhere(['ID' => 1])
->assignSQL('"PendingRequests"', '"PendingRequests" - 1') ->assignSQL('"PendingRequests"', '"PendingRequests" - 1')
->assign('"LastResponseTimestamp"', self::millitime()); ->assign('"LastResponseTimestamp"', TestSessionState::millitime());
$update->execute(); $update->execute();
} }