'%$FixtureFactory', ); /** * Path to fixture data for this test run. * If passed as an array, multiple fixture files will be loaded. * Please note that you won't be able to refer with "=>" notation * between the fixtures, they act independent of each other. * * @var string|array */ protected static $fixture_file = null; /** * @var FixtureFactory */ protected $fixtureFactory; /** * @var Boolean If set to TRUE, this will force a test database to be generated * in {@link setUp()}. Note that this flag is overruled by the presence of a * {@link $fixture_file}, which always forces a database build. */ protected $usesDatabase = null; protected $originalMemberPasswordValidator; protected $originalRequirements; protected $originalIsRunningTest; protected $originalNestedURLsState; protected $originalMemoryLimit; /** * @var TestMailer */ protected $mailer; /** * Pointer to the manifest that isn't a test manifest */ protected static $regular_manifest; /** * @var boolean */ protected static $is_running_test = false; /** * @var ClassManifest */ protected static $test_class_manifest; /** * By default, setUp() does not require default records. Pass * class names in here, and the require/augment default records * function will be called on them. */ protected $requireDefaultRecordsFrom = array(); /** * A list of extensions that can't be applied during the execution of this run. If they are * applied, they will be temporarily removed and a database migration called. * * The keys of the are the classes that the extensions can't be applied the extensions to, and * the values are an array of illegal extensions on that class. */ protected static $illegal_extensions = []; /** * A list of extensions that must be applied during the execution of this run. If they are * not applied, they will be temporarily added and a database migration called. * * The keys of the are the classes to apply the extensions to, and the values are an array * of required extensions on that class. * * Example: * * array("MyTreeDataObject" => array("Versioned", "Hierarchy")) * */ protected static $required_extensions = []; /** * By default, the test database won't contain any DataObjects that have the interface TestOnly. * This variable lets you define additional TestOnly DataObjects to set up for this test. * Set it to an array of DataObject subclass names. */ protected static $extra_dataobjects = []; /** * List of class names of {@see Controller} objects to register routes for * Controllers must implement Link() method * * @var array */ protected static $extra_controllers = []; /** * We need to disabling backing up of globals to avoid overriding * the few globals SilverStripe relies on, like $lang for the i18n subsystem. * * @see http://sebastian-bergmann.de/archives/797-Global-Variables-and-PHPUnit.html */ protected $backupGlobals = false; /** * Helper arrays for illegal_extensions/required_extensions code */ private static $extensions_to_reapply = []; private static $extensions_to_remove = []; /** * Check flushables on setupBeforeClass() * * @var bool */ protected static $flushedFlushables = false; /** * Determines if unit tests are currently run, flag set during test bootstrap. * This is used as a cheap replacement for fully mockable state * in certain contiditions (e.g. access checks). * Caution: When set to FALSE, certain controllers might bypass * access checks, so this is a very security sensitive setting. * * @return boolean */ public static function is_running_test() { return self::$is_running_test; } public static function set_is_running_test($bool) { self::$is_running_test = $bool; } /** * Set the manifest to be used to look up test classes by helper functions * * @param ClassManifest $manifest */ public static function set_test_class_manifest($manifest) { self::$test_class_manifest = $manifest; } /** * Return the manifest being used to look up test classes by helper functions * * @return ClassManifest */ public static function get_test_class_manifest() { return self::$test_class_manifest; } /** * @return String */ public static function get_fixture_file() { return static::$fixture_file; } protected $model; /** * State of Versioned before this test is run * * @var string */ protected $originalReadingMode = null; protected $originalEnv = null; protected function setUp() { //nest config and injector for each test so they are effectively sandboxed per test Config::nest(); Injector::nest(); $this->originalEnv = Director::get_environment_type(); if (class_exists(Versioned::class)) { $this->originalReadingMode = Versioned::get_reading_mode(); } // We cannot run the tests on this abstract class. if (get_class($this) == __CLASS__) { $this->markTestSkipped(sprintf('Skipping %s ', get_class($this))); return; } // Mark test as being run $this->originalIsRunningTest = self::$is_running_test; self::$is_running_test = true; // i18n needs to be set to the defaults or tests fail i18n::set_locale(i18n::config()->uninherited('default_locale')); // Set default timezone consistently to avoid NZ-specific dependencies date_default_timezone_set('UTC'); // Remove password validation $this->originalMemberPasswordValidator = Member::password_validator(); $this->originalRequirements = Requirements::backend(); Member::set_password_validator(null); Cookie::config()->update('report_errors', false); if (class_exists(RootURLController::class)) { RootURLController::reset(); } // Reset all resettables /** @var Resettable $resettable */ foreach (ClassInfo::implementorsOf(Resettable::class) as $resettable) { $resettable::reset(); } if (Controller::has_curr()) { Controller::curr()->setSession(Session::create(array())); } Security::$database_is_ready = null; // Set up test routes $this->setUpRoutes(); $fixtureFiles = $this->getFixturePaths(); // Todo: this could be a special test model $this->model = DataModel::inst(); // Set up fixture if ($fixtureFiles || $this->usesDatabase) { if (!self::using_temp_db()) { self::create_temp_db(); } DataObject::singleton()->flushCache(); self::empty_temp_db(); foreach ($this->requireDefaultRecordsFrom as $className) { $instance = singleton($className); if (method_exists($instance, 'requireDefaultRecords')) { $instance->requireDefaultRecords(); } if (method_exists($instance, 'augmentDefaultRecords')) { $instance->augmentDefaultRecords(); } } foreach ($fixtureFiles as $fixtureFilePath) { $fixture = YamlFixture::create($fixtureFilePath); $fixture->writeInto($this->getFixtureFactory()); } $this->logInWithPermission("ADMIN"); } // Preserve memory settings $this->originalMemoryLimit = ini_get('memory_limit'); // turn off template debugging SSViewer::config()->update('source_file_comments', false); // Clear requirements Requirements::clear(); // Set up the test mailer $this->mailer = new TestMailer(); Injector::inst()->registerService($this->mailer, Mailer::class); Email::config()->remove('send_all_emails_to'); Email::config()->remove('send_all_emails_from'); Email::config()->remove('cc_all_emails_to'); Email::config()->remove('bcc_all_emails_to'); } /** * Called once per test case ({@link SapphireTest} subclass). * This is different to {@link setUp()}, which gets called once * per method. Useful to initialize expensive operations which * don't change state for any called method inside the test, * e.g. dynamically adding an extension. See {@link teardownAfterClass()} * for tearing down the state again. */ public static function setUpBeforeClass() { static::start(); //nest config and injector for each suite so they are effectively sandboxed Config::nest(); Injector::nest(); $isAltered = false; if (!Director::isDev()) { user_error('Tests can only run in "dev" mode', E_USER_ERROR); } // Remove any illegal extensions that are present foreach (static::$illegal_extensions as $class => $extensions) { if (!class_exists($class)) { continue; } foreach ($extensions as $extension) { if (!class_exists($extension) || !$class::has_extension($extension)) { continue; } if (!isset(self::$extensions_to_reapply[$class])) { self::$extensions_to_reapply[$class] = array(); } self::$extensions_to_reapply[$class][] = $extension; $class::remove_extension($extension); $isAltered = true; } } // Add any required extensions that aren't present foreach (static::$required_extensions as $class => $extensions) { if (!class_exists($class)) { $self = static::class; throw new LogicException("Test {$self} requires class {$class} which doesn't exist"); } self::$extensions_to_remove[$class] = array(); foreach ($extensions as $extension) { $extensionClass = Extension::get_classname_without_arguments($extension); if (!class_exists($extensionClass)) { $self = static::class; throw new LogicException("Test {$self} requires extension {$extension} which doesn't exist"); } if (!$class::has_extension($extension)) { if (!isset(self::$extensions_to_remove[$class])) { self::$extensions_to_reapply[$class] = array(); } self::$extensions_to_remove[$class][] = $extension; $class::add_extension($extension); $isAltered = true; } } } // If we have made changes to the extensions present, then migrate the database schema. if ($isAltered || self::$extensions_to_reapply || self::$extensions_to_remove || static::getExtraDataObjects()) { DataObject::reset(); if (!self::using_temp_db()) { self::create_temp_db(); } static::resetDBSchema(true); } // clear singletons, they're caching old extension info // which is used in DatabaseAdmin->doBuild() Injector::inst()->unregisterAllObjects(); // Set default timezone consistently to avoid NZ-specific dependencies date_default_timezone_set('UTC'); // Flush all flushable records $flush = !empty($_GET['flush']); if (!self::$flushedFlushables && $flush) { self::$flushedFlushables = true; foreach (ClassInfo::implementorsOf(Flushable::class) as $class) { $class::flush(); } } } /** * tearDown method that's called once per test class rather once per test method. */ public static function tearDownAfterClass() { // If we have made changes to the extensions present, then migrate the database schema. if (self::$extensions_to_reapply || self::$extensions_to_remove) { // @todo: This isn't strictly necessary to restore extensions, but only to ensure that // Object::$extra_methods is properly flushed. This should be replaced with a simple // flush mechanism for each $class. // // Remove extensions added for testing foreach (self::$extensions_to_remove as $class => $extensions) { foreach ($extensions as $extension) { $class::remove_extension($extension); } } // Reapply ones removed foreach (self::$extensions_to_reapply as $class => $extensions) { foreach ($extensions as $extension) { $class::add_extension($extension); } } } //unnest injector / config now that the test suite is over // this will reset all the extensions on the object too (see setUpBeforeClass) Injector::unnest(); Config::unnest(); $extraDataObjects = static::getExtraDataObjects(); if (!empty(self::$extensions_to_reapply) || !empty(self::$extensions_to_remove) || !empty($extraDataObjects)) { static::resetDBSchema(); } } /** * @return FixtureFactory */ public function getFixtureFactory() { if (!$this->fixtureFactory) { $this->fixtureFactory = Injector::inst()->create(FixtureFactory::class); } return $this->fixtureFactory; } public function setFixtureFactory(FixtureFactory $factory) { $this->fixtureFactory = $factory; return $this; } /** * Get the ID of an object from the fixture. * * @param string $className The data class or table name, as specified in your fixture file. Parent classes won't work * @param string $identifier The identifier string, as provided in your fixture file * @return int */ protected function idFromFixture($className, $identifier) { $id = $this->getFixtureFactory()->getId($className, $identifier); if (!$id) { user_error(sprintf( "Couldn't find object '%s' (class: %s)", $identifier, $className ), E_USER_ERROR); } return $id; } /** * Return all of the IDs in the fixture of a particular class name. * Will collate all IDs form all fixtures if multiple fixtures are provided. * * @param string $className The data class or table name, as specified in your fixture file * @return array A map of fixture-identifier => object-id */ protected function allFixtureIDs($className) { return $this->getFixtureFactory()->getIds($className); } /** * Get an object from the fixture. * * @param string $className The data class or table name, as specified in your fixture file. Parent classes won't work * @param string $identifier The identifier string, as provided in your fixture file * * @return DataObject */ protected function objFromFixture($className, $identifier) { $obj = $this->getFixtureFactory()->get($className, $identifier); if (!$obj) { user_error(sprintf( "Couldn't find object '%s' (class: %s)", $identifier, $className ), E_USER_ERROR); } return $obj; } /** * Load a YAML fixture file into the database. * Once loaded, you can use idFromFixture() and objFromFixture() to get items from the fixture. * Doesn't clear existing fixtures. * * @param string $fixtureFile The location of the .yml fixture file, relative to the site base dir */ public function loadFixture($fixtureFile) { $fixture = Injector::inst()->create(YamlFixture::class, $fixtureFile); $fixture->writeInto($this->getFixtureFactory()); } /** * Clear all fixtures which were previously loaded through * {@link loadFixture()} */ public function clearFixtures() { $this->getFixtureFactory()->clear(); } /** * Useful for writing unit tests without hardcoding folder structures. * * @return String Absolute path to current class. */ protected function getCurrentAbsolutePath() { $filename = self::$test_class_manifest->getItemPath(get_class($this)); if (!$filename) { throw new LogicException("getItemPath returned null for " . get_class($this)); } return dirname($filename); } /** * @return String File path relative to webroot */ protected function getCurrentRelativePath() { $base = Director::baseFolder(); $path = $this->getCurrentAbsolutePath(); if (substr($path, 0, strlen($base)) == $base) { $path = preg_replace('/^\/*/', '', substr($path, strlen($base))); } return $path; } protected function tearDown() { // Preserve memory settings ini_set('memory_limit', ($this->originalMemoryLimit) ? $this->originalMemoryLimit : -1); // Restore email configuration $this->mailer = null; // Restore password validation if ($this->originalMemberPasswordValidator) { Member::set_password_validator($this->originalMemberPasswordValidator); } // Restore requirements if ($this->originalRequirements) { Requirements::set_backend($this->originalRequirements); } // Mark test as no longer being run - we use originalIsRunningTest to allow for nested SapphireTest calls self::$is_running_test = $this->originalIsRunningTest; $this->originalIsRunningTest = null; // Reset mocked datetime DBDatetime::clear_mock_now(); // Stop the redirection that might have been requested in the test. // Note: Ideally a clean Controller should be created for each test. // Now all tests executed in a batch share the same controller. $controller = Controller::has_curr() ? Controller::curr() : null; if ($controller && ($response = $controller->getResponse()) && $response->getHeader('Location')) { $response->setStatusCode(200); $response->removeHeader('Location'); } Director::set_environment_type($this->originalEnv); if (class_exists(Versioned::class)) { Versioned::set_reading_mode($this->originalReadingMode); } //unnest injector / config now that tests are over Injector::unnest(); Config::unnest(); } public static function assertContains( $needle, $haystack, $message = '', $ignoreCase = false, $checkForObjectIdentity = true, $checkForNonObjectIdentity = false ) { if ($haystack instanceof DBField) { $haystack = (string)$haystack; } parent::assertContains($needle, $haystack, $message, $ignoreCase, $checkForObjectIdentity, $checkForNonObjectIdentity); } public static function assertNotContains( $needle, $haystack, $message = '', $ignoreCase = false, $checkForObjectIdentity = true, $checkForNonObjectIdentity = false ) { if ($haystack instanceof DBField) { $haystack = (string)$haystack; } parent::assertNotContains($needle, $haystack, $message, $ignoreCase, $checkForObjectIdentity, $checkForNonObjectIdentity); } /** * Clear the log of emails sent */ public function clearEmails() { return $this->mailer->clearEmails(); } /** * Search for an email that was sent. * All of the parameters can either be a string, or, if they start with "/", a PREG-compatible regular expression. * @param $to * @param $from * @param $subject * @param $content * @return array Contains keys: 'type', 'to', 'from', 'subject','content', 'plainContent', 'attachedFiles', * 'customHeaders', 'htmlContent', 'inlineImages' */ public function findEmail($to, $from = null, $subject = null, $content = null) { return $this->mailer->findEmail($to, $from, $subject, $content); } /** * Assert that the matching email was sent since the last call to clearEmails() * All of the parameters can either be a string, or, if they start with "/", a PREG-compatible regular expression. * @param $to * @param $from * @param $subject * @param $content * @return array Contains the keys: 'type', 'to', 'from', 'subject', 'content', 'plainContent', 'attachedFiles', * 'customHeaders', 'htmlContent', inlineImages' */ public function assertEmailSent($to, $from = null, $subject = null, $content = null) { $found = (bool)$this->findEmail($to, $from, $subject, $content); $infoParts = ""; $withParts = array(); if ($to) { $infoParts .= " to '$to'"; } if ($from) { $infoParts .= " from '$from'"; } if ($subject) { $withParts[] = "subject '$subject'"; } if ($content) { $withParts[] = "content '$content'"; } if ($withParts) { $infoParts .= " with " . implode(" and ", $withParts); } $this->assertTrue( $found, "Failed asserting that an email was sent$infoParts." ); } /** * Assert that the given {@link SS_List} includes DataObjects matching the given key-value * pairs. Each match must correspond to 1 distinct record. * * @param SS_List|array $matches The patterns to match. Each pattern is a map of key-value pairs. You can * either pass a single pattern or an array of patterns. * @param SS_List $dataObjectSet The {@link SS_List} to test. * * Examples * -------- * Check that $members includes an entry with Email = sam@example.com: * $this->assertDOSContains(array('Email' => '...@example.com'), $members); * * Check that $members includes entries with Email = sam@example.com and with * Email = ingo@example.com: * $this->assertDOSContains(array( * array('Email' => '...@example.com'), * array('Email' => 'i...@example.com'), * ), $members); */ public function assertDOSContains($matches, $dataObjectSet) { $extracted = array(); foreach ($dataObjectSet as $object) { /** @var DataObject $object */ $extracted[] = $object->toMap(); } foreach ($matches as $match) { $matched = false; foreach ($extracted as $i => $item) { if ($this->dataObjectArrayMatch($item, $match)) { // Remove it from $extracted so that we don't get duplicate mapping. unset($extracted[$i]); $matched = true; break; } } // We couldn't find a match - assertion failed $this->assertTrue( $matched, "Failed asserting that the SS_List contains an item matching " . var_export($match, true) . "\n\nIn the following SS_List:\n" . $this->DOSSummaryForMatch($dataObjectSet, $match) ); } } /** * Asserts that no items in a given list appear in the given dataobject list * * @param SS_List|array $matches The patterns to match. Each pattern is a map of key-value pairs. You can * either pass a single pattern or an array of patterns. * @param SS_List $dataObjectSet The {@link SS_List} to test. * * Examples * -------- * Check that $members doesn't have an entry with Email = sam@example.com: * $this->assertNotDOSContains(array('Email' => '...@example.com'), $members); * * Check that $members doesn't have entries with Email = sam@example.com and with * Email = ingo@example.com: * $this->assertNotDOSContains(array( * array('Email' => '...@example.com'), * array('Email' => 'i...@example.com'), * ), $members); */ public function assertNotDOSContains($matches, $dataObjectSet) { $extracted = array(); foreach ($dataObjectSet as $object) { /** @var DataObject $object */ $extracted[] = $object->toMap(); } $matched = []; foreach ($matches as $match) { foreach ($extracted as $i => $item) { if ($this->dataObjectArrayMatch($item, $match)) { $matched[] = $extracted[$i]; break; } } // We couldn't find a match - assertion failed $this->assertEmpty( $matched, "Failed asserting that the SS_List dosn't contain a set of objects. " . "Found objects were: " . var_export($matched, true) ); } } /** * Assert that the given {@link SS_List} includes only DataObjects matching the given * key-value pairs. Each match must correspond to 1 distinct record. * * Example * -------- * Check that *only* the entries Sam Minnee and Ingo Schommer exist in $members. Order doesn't * matter: * $this->assertDOSEquals(array( * array('FirstName' =>'Sam', 'Surname' => 'Minnee'), * array('FirstName' => 'Ingo', 'Surname' => 'Schommer'), * ), $members); * * @param mixed $matches The patterns to match. Each pattern is a map of key-value pairs. You can * either pass a single pattern or an array of patterns. * @param mixed $dataObjectSet The {@link SS_List} to test. */ public function assertDOSEquals($matches, $dataObjectSet) { // Extract dataobjects $extracted = array(); if ($dataObjectSet) { foreach ($dataObjectSet as $object) { /** @var DataObject $object */ $extracted[] = $object->toMap(); } } // Check all matches if ($matches) { foreach ($matches as $match) { $matched = false; foreach ($extracted as $i => $item) { if ($this->dataObjectArrayMatch($item, $match)) { // Remove it from $extracted so that we don't get duplicate mapping. unset($extracted[$i]); $matched = true; break; } } // We couldn't find a match - assertion failed $this->assertTrue( $matched, "Failed asserting that the SS_List contains an item matching " . var_export($match, true) . "\n\nIn the following SS_List:\n" . $this->DOSSummaryForMatch($dataObjectSet, $match) ); } } // If we have leftovers than the DOS has extra data that shouldn't be there $this->assertTrue( (count($extracted) == 0), // If we didn't break by this point then we couldn't find a match "Failed asserting that the SS_List contained only the given items, the " . "following items were left over:\n" . var_export($extracted, true) ); } /** * Assert that the every record in the given {@link SS_List} matches the given key-value * pairs. * * Example * -------- * Check that every entry in $members has a Status of 'Active': * $this->assertDOSAllMatch(array('Status' => 'Active'), $members); * * @param mixed $match The pattern to match. The pattern is a map of key-value pairs. * @param mixed $dataObjectSet The {@link SS_List} to test. */ public function assertDOSAllMatch($match, $dataObjectSet) { $extracted = array(); foreach ($dataObjectSet as $object) { /** @var DataObject $object */ $extracted[] = $object->toMap(); } foreach ($extracted as $i => $item) { $this->assertTrue( $this->dataObjectArrayMatch($item, $match), "Failed asserting that the the following item matched " . var_export($match, true) . ": " . var_export($item, true) ); } } /** * Removes sequences of repeated whitespace characters from SQL queries * making them suitable for string comparison * * @param string $sql * @return string The cleaned and normalised SQL string */ protected function normaliseSQL($sql) { return trim(preg_replace('/\s+/m', ' ', $sql)); } /** * Asserts that two SQL queries are equivalent * * @param string $expectedSQL * @param string $actualSQL * @param string $message * @param float|int $delta * @param integer $maxDepth * @param boolean $canonicalize * @param boolean $ignoreCase */ public function assertSQLEquals( $expectedSQL, $actualSQL, $message = '', $delta = 0, $maxDepth = 10, $canonicalize = false, $ignoreCase = false ) { // Normalise SQL queries to remove patterns of repeating whitespace $expectedSQL = $this->normaliseSQL($expectedSQL); $actualSQL = $this->normaliseSQL($actualSQL); $this->assertEquals($expectedSQL, $actualSQL, $message, $delta, $maxDepth, $canonicalize, $ignoreCase); } /** * Asserts that a SQL query contains a SQL fragment * * @param string $needleSQL * @param string $haystackSQL * @param string $message * @param boolean $ignoreCase * @param boolean $checkForObjectIdentity */ public function assertSQLContains( $needleSQL, $haystackSQL, $message = '', $ignoreCase = false, $checkForObjectIdentity = true ) { $needleSQL = $this->normaliseSQL($needleSQL); $haystackSQL = $this->normaliseSQL($haystackSQL); $this->assertContains($needleSQL, $haystackSQL, $message, $ignoreCase, $checkForObjectIdentity); } /** * Asserts that a SQL query contains a SQL fragment * * @param string $needleSQL * @param string $haystackSQL * @param string $message * @param boolean $ignoreCase * @param boolean $checkForObjectIdentity */ public function assertSQLNotContains( $needleSQL, $haystackSQL, $message = '', $ignoreCase = false, $checkForObjectIdentity = true ) { $needleSQL = $this->normaliseSQL($needleSQL); $haystackSQL = $this->normaliseSQL($haystackSQL); $this->assertNotContains($needleSQL, $haystackSQL, $message, $ignoreCase, $checkForObjectIdentity); } /** * Helper function for the DOS matchers * * @param array $item * @param array $match * @return bool */ private function dataObjectArrayMatch($item, $match) { foreach ($match as $k => $v) { if (!array_key_exists($k, $item) || $item[$k] != $v) { return false; } } return true; } /** * Helper function for the DOS matchers * * @param SS_List|array $dataObjectSet * @param array $match * @return string */ private function DOSSummaryForMatch($dataObjectSet, $match) { $extracted = array(); foreach ($dataObjectSet as $item) { $extracted[] = array_intersect_key($item->toMap(), $match); } return var_export($extracted, true); } /** * Start test environment */ public static function start() { if (!static::is_running_test()) { new FakeController(); static::use_test_manifest(); static::set_is_running_test(true); } } /** * Pushes a class and template manifest instance that include tests onto the * top of the loader stacks. */ protected static function use_test_manifest() { $flush = !empty($_GET['flush']); $classManifest = new ClassManifest( BASE_PATH, true, $flush ); ClassLoader::instance()->pushManifest($classManifest, false); static::set_test_class_manifest($classManifest); ThemeResourceLoader::instance()->addSet('$default', new ThemeManifest( BASE_PATH, project(), true, $flush )); // Once new class loader is registered, push a new uncached config $config = CoreConfigFactory::inst()->createCore(); ConfigLoader::instance()->pushManifest($config); // Invalidate classname spec since the test manifest will now pull out new subclasses for each internal class // (e.g. Member will now have various subclasses of DataObjects that implement TestOnly) DataObject::reset(); } /** * Returns true if we are currently using a temporary database */ public static function using_temp_db() { $dbConn = DB::get_conn(); $prefix = getenv('SS_DATABASE_PREFIX') ?: 'ss_'; return 1 === preg_match(sprintf('/^%stmpdb_[0-9]+_[0-9]+$/i', preg_quote($prefix, '/')), $dbConn->getSelectedDatabase()); } public static function kill_temp_db() { // Delete our temporary database if (self::using_temp_db()) { $dbConn = DB::get_conn(); $dbName = $dbConn->getSelectedDatabase(); if ($dbName && DB::get_conn()->databaseExists($dbName)) { // Some DataExtensions keep a static cache of information that needs to // be reset whenever the database is killed foreach (ClassInfo::subclassesFor(DataExtension::class) as $class) { $toCall = array($class, 'on_db_reset'); if (is_callable($toCall)) { call_user_func($toCall); } } // echo "Deleted temp database " . $dbConn->currentDatabase() . "\n"; $dbConn->dropSelectedDatabase(); } } } /** * Remove all content from the temporary database. */ public static function empty_temp_db() { if (self::using_temp_db()) { DB::get_conn()->clearAllData(); // Some DataExtensions keep a static cache of information that needs to // be reset whenever the database is cleaned out $classes = array_merge(ClassInfo::subclassesFor(DataExtension::class), ClassInfo::subclassesFor(DataObject::class)); foreach ($classes as $class) { $toCall = array($class, 'on_db_reset'); if (is_callable($toCall)) { call_user_func($toCall); } } } } public static function create_temp_db() { // Disable PHPUnit error handling $oldErrorHandler = set_error_handler(null); // Create a temporary database, and force the connection to use UTC for time global $databaseConfig; $databaseConfig['timezone'] = '+0:00'; DB::connect($databaseConfig); $dbConn = DB::get_conn(); $prefix = getenv('SS_DATABASE_PREFIX') ?: 'ss_'; do { $dbname = strtolower(sprintf('%stmpdb_%s_%s', $prefix, time(), rand(1000000, 9999999))); } while ($dbConn->databaseExists($dbname)); $dbConn->selectDatabase($dbname, true); static::resetDBSchema(); // Reinstate PHPUnit error handling set_error_handler($oldErrorHandler); // Ensure test db is killed on exit register_shutdown_function(function () { static::kill_temp_db(); }); return $dbname; } public static function delete_all_temp_dbs() { $prefix = getenv('SS_DATABASE_PREFIX') ?: 'ss_'; foreach (DB::get_schema()->databaseList() as $dbName) { if (1 === preg_match(sprintf('/^%stmpdb_[0-9]+_[0-9]+$/i', preg_quote($prefix, '/')), $dbName)) { DB::get_schema()->dropDatabase($dbName); if (Director::is_cli()) { echo "Dropped database \"$dbName\"" . PHP_EOL; } else { echo "
  • Dropped database \"$dbName\"
  • " . PHP_EOL; } flush(); } } } /** * Reset the testing database's schema. * @param bool $includeExtraDataObjects If true, the extraDataObjects tables will also be included */ public static function resetDBSchema($includeExtraDataObjects = false) { if (self::using_temp_db()) { DataObject::reset(); // clear singletons, they're caching old extension info which is used in DatabaseAdmin->doBuild() Injector::inst()->unregisterAllObjects(); $dataClasses = ClassInfo::subclassesFor(DataObject::class); array_shift($dataClasses); DB::quiet(); $schema = DB::get_schema(); $extraDataObjects = $includeExtraDataObjects ? static::getExtraDataObjects() : null; $schema->schemaUpdate(function () use ($dataClasses, $extraDataObjects) { foreach ($dataClasses as $dataClass) { // Check if class exists before trying to instantiate - this sidesteps any manifest weirdness if (class_exists($dataClass)) { $SNG = singleton($dataClass); if (!($SNG instanceof TestOnly)) { $SNG->requireTable(); } } } // If we have additional dataobjects which need schema, do so here: if ($extraDataObjects) { foreach ($extraDataObjects as $dataClass) { $SNG = singleton($dataClass); if (singleton($dataClass) instanceof DataObject) { $SNG->requireTable(); } } } }); ClassInfo::reset_db_cache(); DataObject::singleton()->flushCache(); } } /** * Create a member and group with the given permission code, and log in with it. * Returns the member ID. * * @param string|array $permCode Either a permission, or list of permissions * @return int Member ID */ public function logInWithPermission($permCode = "ADMIN") { if (is_array($permCode)) { $permArray = $permCode; $permCode = implode('.', $permCode); } else { $permArray = array($permCode); } // Check cached member if (isset($this->cache_generatedMembers[$permCode])) { $member = $this->cache_generatedMembers[$permCode]; } else { // Generate group with these permissions $group = Group::create(); $group->Title = "$permCode group"; $group->write(); // Create each individual permission foreach ($permArray as $permArrayItem) { $permission = Permission::create(); $permission->Code = $permArrayItem; $permission->write(); $group->Permissions()->add($permission); } $member = Member::get()->filter([ 'Email' => "$permCode@example.org", ])->first(); if (!$member) { $member = Member::create(); } $member->FirstName = $permCode; $member->Surname = "User"; $member->Email = "$permCode@example.org"; $member->write(); $group->Members()->add($member); $this->cache_generatedMembers[$permCode] = $member; } $member->logIn(); return $member->ID; } /** * Cache for logInWithPermission() */ protected $cache_generatedMembers = array(); /** * Test against a theme. * * @param string $themeBaseDir themes directory * @param string $theme Theme name * @param callable $callback * @throws Exception */ protected function useTestTheme($themeBaseDir, $theme, $callback) { Config::nest(); if (strpos($themeBaseDir, BASE_PATH) === 0) { $themeBaseDir = substr($themeBaseDir, strlen(BASE_PATH)); } SSViewer::config()->update('theme_enabled', true); SSViewer::set_themes([$themeBaseDir.'/themes/'.$theme, '$default']); $e = null; try { $callback(); } catch (Exception $e) { /* NOP for now, just save $e */ } Config::unnest(); if ($e) { throw $e; } } /** * Get fixture paths for this test * * @return array List of paths */ protected function getFixturePaths() { $fixtureFile = static::get_fixture_file(); if (empty($fixtureFile)) { return []; } $fixtureFiles = (is_array($fixtureFile)) ? $fixtureFile : [$fixtureFile]; return array_map(function ($fixtureFilePath) { return $this->resolveFixturePath($fixtureFilePath); }, $fixtureFiles); } /** * Return all extra objects to scaffold for this test * @return array */ protected static function getExtraDataObjects() { return static::$extra_dataobjects; } /** * Get additional controller classes to register routes for * * @return array */ protected static function getExtraControllers() { return static::$extra_controllers; } /** * Map a fixture path to a physical file * * @param string $fixtureFilePath * @return string */ protected function resolveFixturePath($fixtureFilePath) { // Support fixture paths relative to the test class, rather than relative to webroot // String checking is faster than file_exists() calls. $isRelativeToFile = (strpos('/', $fixtureFilePath) === false) || preg_match('/^(\.){1,2}/', $fixtureFilePath); if ($isRelativeToFile) { $resolvedPath = realpath($this->getCurrentAbsolutePath() . '/' . $fixtureFilePath); if ($resolvedPath) { return $resolvedPath; } } // Check if file exists relative to base dir $resolvedPath = realpath(Director::baseFolder() . '/' . $fixtureFilePath); if ($resolvedPath) { return $resolvedPath; } return $fixtureFilePath; } protected function setUpRoutes() { // Get overridden routes $rules = $this->getExtraRoutes(); // Add all other routes foreach (Director::config()->uninherited('rules') as $route => $rule) { if (!isset($rules[$route])) { $rules[$route] = $rule; } } // Add default catch-all rule $rules['$Controller//$Action/$ID/$OtherID'] = '*'; // Add controller-name auto-routing Director::config()->set('rules', $rules); } /** * Get extra routes to merge into Director.rules * * @return array */ protected function getExtraRoutes() { $rules = []; foreach ($this->getExtraControllers() as $class) { $controllerInst = Controller::singleton($class); $link = Director::makeRelative($controllerInst->Link()); $route = rtrim($link, '/') . '//$Action/$ID/$OtherID'; $rules[$route] = $class; } return $rules; } }