summaryrefslogtreecommitdiff
path: root/modules
diff options
context:
space:
mode:
authorDavid Rothstein <drothstein@gmail.com>2012-07-29 15:49:20 -0400
committerDavid Rothstein <drothstein@gmail.com>2012-07-29 15:49:20 -0400
commit568611ec397a2fb663988b04cf9105af68d4a1ba (patch)
tree6f4a62e660cdf636553b1b408c6fcdc2ad6e43e2 /modules
parent22e259c9d9d067583ea7bcfb815a082fb6dbd46a (diff)
downloadbrdo-568611ec397a2fb663988b04cf9105af68d4a1ba.tar.gz
brdo-568611ec397a2fb663988b04cf9105af68d4a1ba.tar.bz2
Issue #1541958 by sun, effulgentsia, David_Rothstein, alexpott, tim.plunkett, Berdir: Split setUp() into specific sub-methods to fix testing framework regressions.
Diffstat (limited to 'modules')
-rw-r--r--modules/simpletest/drupal_web_test_case.php185
-rw-r--r--modules/simpletest/tests/upgrade/upgrade.test134
2 files changed, 177 insertions, 142 deletions
diff --git a/modules/simpletest/drupal_web_test_case.php b/modules/simpletest/drupal_web_test_case.php
index 251c5c1bc..abb0b41f9 100644
--- a/modules/simpletest/drupal_web_test_case.php
+++ b/modules/simpletest/drupal_web_test_case.php
@@ -85,6 +85,10 @@ abstract class DrupalTestCase {
*/
protected $setup = FALSE;
+ protected $setupDatabasePrefix = FALSE;
+
+ protected $setupEnvironment = FALSE;
+
/**
* Constructor for DrupalTestCase.
*
@@ -1251,29 +1255,46 @@ class DrupalWebTestCase extends DrupalTestCase {
}
/**
- * Generates a random database prefix, runs the install scripts on the
- * prefixed database and enable the specified modules. After installation
- * many caches are flushed and the internal browser is setup so that the
- * page requests will run on the new prefix. A temporary files directory
- * is created with the same name as the database prefix.
+ * Generates a database prefix for running tests.
*
- * @param ...
- * List of modules to enable for the duration of the test. This can be
- * either a single array or a variable number of string arguments.
+ * The generated database table prefix is used for the Drupal installation
+ * being performed for the test. It is also used as user agent HTTP header
+ * value by the cURL-based browser of DrupalWebTestCase, which is sent
+ * to the Drupal installation of the test. During early Drupal bootstrap, the
+ * user agent HTTP header is parsed, and if it matches, all database queries
+ * use the database table prefix that has been generated here.
+ *
+ * @see DrupalWebTestCase::curlInitialize()
+ * @see drupal_valid_test_ua()
+ * @see DrupalWebTestCase::setUp()
*/
- protected function setUp() {
- global $user, $language, $conf;
-
- // Generate a temporary prefixed database to ensure that tests have a clean starting point.
+ protected function prepareDatabasePrefix() {
$this->databasePrefix = 'simpletest' . mt_rand(1000, 1000000);
+
+ // As soon as the database prefix is set, the test might start to execute.
+ // All assertions as well as the SimpleTest batch operations are associated
+ // with the testId, so the database prefix has to be associated with it.
db_update('simpletest_test_id')
->fields(array('last_prefix' => $this->databasePrefix))
->condition('test_id', $this->testId)
->execute();
+ }
- // Reset all statics and variables to perform tests in a clean environment.
- $conf = array();
- drupal_static_reset();
+ /**
+ * Changes the database connection to the prefixed one.
+ *
+ * @see DrupalWebTestCase::setUp()
+ */
+ protected function changeDatabasePrefix() {
+ if (empty($this->databasePrefix)) {
+ $this->prepareDatabasePrefix();
+ // If $this->prepareDatabasePrefix() failed to work, return without
+ // setting $this->setupDatabasePrefix to TRUE, so setUp() methods will
+ // know to bail out.
+ if (empty($this->databasePrefix)) {
+ return;
+ }
+ }
// Clone the current connection and replace the current prefix.
$connection_info = Database::getConnectionInfo('default');
@@ -1285,22 +1306,43 @@ class DrupalWebTestCase extends DrupalTestCase {
}
Database::addConnectionInfo('default', 'default', $connection_info['default']);
+ // Indicate the database prefix was set up correctly.
+ $this->setupDatabasePrefix = TRUE;
+ }
+
+ /**
+ * Prepares the current environment for running the test.
+ *
+ * Backups various current environment variables and resets them, so they do
+ * not interfere with the Drupal site installation in which tests are executed
+ * and can be restored in tearDown().
+ *
+ * Also sets up new resources for the testing environment, such as the public
+ * filesystem and configuration directories.
+ *
+ * @see DrupalWebTestCase::setUp()
+ * @see DrupalWebTestCase::tearDown()
+ */
+ protected function prepareEnvironment() {
+ global $user, $language, $conf;
+
// Store necessary current values before switching to prefixed database.
$this->originalLanguage = $language;
$this->originalLanguageDefault = variable_get('language_default');
$this->originalFileDirectory = variable_get('file_public_path', conf_path() . '/files');
$this->originalProfile = drupal_get_profile();
- $clean_url_original = variable_get('clean_url', 0);
+ $this->originalCleanUrl = variable_get('clean_url', 0);
+ $this->originalUser = $user;
// Set to English to prevent exceptions from utf8_truncate() from t()
// during install if the current language is not 'en'.
// The following array/object conversion is copied from language_default().
$language = (object) array('language' => 'en', 'name' => 'English', 'native' => 'English', 'direction' => 0, 'enabled' => 1, 'plurals' => 0, 'formula' => '', 'domain' => '', 'prefix' => '', 'weight' => 0, 'javascript' => '');
- // Save and clean shutdown callbacks array because it static cached and
- // will be changed by the test run. If we don't, then it will contain
- // callbacks from both environments. So testing environment will try
- // to call handlers from original environment.
+ // Save and clean the shutdown callbacks array because it is static cached
+ // and will be changed by the test run. Otherwise it will contain callbacks
+ // from both environments and the testing environment will try to call the
+ // handlers defined by the original one.
$callbacks = &drupal_register_shutdown_function();
$this->originalShutdownCallbacks = $callbacks;
$callbacks = array();
@@ -1308,25 +1350,75 @@ class DrupalWebTestCase extends DrupalTestCase {
// Create test directory ahead of installation so fatal errors and debug
// information can be logged during installation process.
// Use temporary files directory with the same prefix as the database.
- $public_files_directory = $this->originalFileDirectory . '/simpletest/' . substr($this->databasePrefix, 10);
- $private_files_directory = $public_files_directory . '/private';
- $temp_files_directory = $private_files_directory . '/temp';
+ $this->public_files_directory = $this->originalFileDirectory . '/simpletest/' . substr($this->databasePrefix, 10);
+ $this->private_files_directory = $this->public_files_directory . '/private';
+ $this->temp_files_directory = $this->private_files_directory . '/temp';
// Create the directories
- file_prepare_directory($public_files_directory, FILE_CREATE_DIRECTORY | FILE_MODIFY_PERMISSIONS);
- file_prepare_directory($private_files_directory, FILE_CREATE_DIRECTORY);
- file_prepare_directory($temp_files_directory, FILE_CREATE_DIRECTORY);
+ file_prepare_directory($this->public_files_directory, FILE_CREATE_DIRECTORY | FILE_MODIFY_PERMISSIONS);
+ file_prepare_directory($this->private_files_directory, FILE_CREATE_DIRECTORY);
+ file_prepare_directory($this->temp_files_directory, FILE_CREATE_DIRECTORY);
$this->generatedTestFiles = FALSE;
// Log fatal errors.
ini_set('log_errors', 1);
- ini_set('error_log', $public_files_directory . '/error.log');
+ ini_set('error_log', $this->public_files_directory . '/error.log');
// Set the test information for use in other parts of Drupal.
$test_info = &$GLOBALS['drupal_test_info'];
$test_info['test_run_id'] = $this->databasePrefix;
$test_info['in_child_site'] = FALSE;
+ // Indicate the environment was set up correctly.
+ $this->setupEnvironment = TRUE;
+ }
+
+ /**
+ * Sets up a Drupal site for running functional and integration tests.
+ *
+ * Generates a random database prefix and installs Drupal with the specified
+ * installation profile in DrupalWebTestCase::$profile into the
+ * prefixed database. Afterwards, installs any additional modules specified by
+ * the test.
+ *
+ * After installation all caches are flushed and several configuration values
+ * are reset to the values of the parent site executing the test, since the
+ * default values may be incompatible with the environment in which tests are
+ * being executed.
+ *
+ * @param ...
+ * List of modules to enable for the duration of the test. This can be
+ * either a single array or a variable number of string arguments.
+ *
+ * @see DrupalWebTestCase::prepareDatabasePrefix()
+ * @see DrupalWebTestCase::changeDatabasePrefix()
+ * @see DrupalWebTestCase::prepareEnvironment()
+ */
+ protected function setUp() {
+ global $user, $language, $conf;
+
+ // Create the database prefix for this test.
+ $this->prepareDatabasePrefix();
+
+ // Prepare the environment for running tests.
+ $this->prepareEnvironment();
+ if (!$this->setupEnvironment) {
+ return FALSE;
+ }
+
+ // Reset all statics and variables to perform tests in a clean environment.
+ $conf = array();
+ drupal_static_reset();
+
+ // Change the database prefix.
+ // All static variables need to be reset before the database prefix is
+ // changed, since DrupalCacheArray implementations attempt to
+ // write back to persistent caches when they are destructed.
+ $this->changeDatabasePrefix();
+ if (!$this->setupDatabasePrefix) {
+ return FALSE;
+ }
+
// Preset the 'install_profile' system variable, so the first call into
// system_rebuild_module_data() (in drupal_install_system()) will register
// the test's profile as a module. Without this, the installation profile of
@@ -1334,15 +1426,16 @@ class DrupalWebTestCase extends DrupalTestCase {
// profile's hook_install() and other hook implementations are never invoked.
$conf['install_profile'] = $this->profile;
+ // Perform the actual Drupal installation.
include_once DRUPAL_ROOT . '/includes/install.inc';
drupal_install_system();
$this->preloadRegistry();
// Set path variables.
- variable_set('file_public_path', $public_files_directory);
- variable_set('file_private_path', $private_files_directory);
- variable_set('file_temporary_path', $temp_files_directory);
+ variable_set('file_public_path', $this->public_files_directory);
+ variable_set('file_private_path', $this->private_files_directory);
+ variable_set('file_temporary_path', $this->temp_files_directory);
// Set the 'simpletest_parent_profile' variable to add the parent profile's
// search path to the child site's search paths.
@@ -1385,18 +1478,20 @@ class DrupalWebTestCase extends DrupalTestCase {
// the installation process.
drupal_cron_run();
- // Log in with a clean $user.
- $this->originalUser = $user;
+ // Ensure that the session is not written to the new environment and replace
+ // the global $user session with uid 1 from the new test site.
drupal_save_session(FALSE);
+ // Login as uid 1.
$user = user_load(1);
// Restore necessary variables.
variable_set('install_task', 'done');
- variable_set('clean_url', $clean_url_original);
+ variable_set('clean_url', $this->originalCleanUrl);
variable_set('site_mail', 'simpletest@example.com');
variable_set('date_default_timezone', date_default_timezone_get());
+
// Set up English language.
- unset($GLOBALS['conf']['language_default']);
+ unset($conf['language_default']);
$language = language_default();
// Use the test mail class instead of the default mail handler class.
@@ -1506,10 +1601,21 @@ class DrupalWebTestCase extends DrupalTestCase {
// Delete temporary files directory.
file_unmanaged_delete_recursive($this->originalFileDirectory . '/simpletest/' . substr($this->databasePrefix, 10));
- // Remove all prefixed tables (all the tables in the schema).
- $schema = drupal_get_schema(NULL, TRUE);
- foreach ($schema as $name => $table) {
- db_drop_table($name);
+ // Remove all prefixed tables.
+ $tables = db_find_tables($this->databasePrefix . '%');
+ $connection_info = Database::getConnectionInfo('default');
+ $tables = db_find_tables($connection_info['default']['prefix']['default'] . '%');
+ if (empty($tables)) {
+ $this->fail('Failed to find test tables to drop.');
+ }
+ $prefix_length = strlen($connection_info['default']['prefix']['default']);
+ foreach ($tables as $table) {
+ if (db_drop_table(substr($table, $prefix_length))) {
+ unset($tables[$table]);
+ }
+ }
+ if (!empty($tables)) {
+ $this->fail('Failed to drop all prefixed tables.');
}
// Get back to the original connection.
@@ -1540,6 +1646,9 @@ class DrupalWebTestCase extends DrupalTestCase {
// Rebuild caches.
$this->refreshVariables();
+ // Reset public files directory.
+ $GLOBALS['conf']['file_public_path'] = $this->originalFileDirectory;
+
// Reset language.
$language = $this->originalLanguage;
if ($this->originalLanguageDefault) {
diff --git a/modules/simpletest/tests/upgrade/upgrade.test b/modules/simpletest/tests/upgrade/upgrade.test
index 172f30e69..9df8ec779 100644
--- a/modules/simpletest/tests/upgrade/upgrade.test
+++ b/modules/simpletest/tests/upgrade/upgrade.test
@@ -71,7 +71,11 @@ abstract class UpgradePathTestCase extends DrupalWebTestCase {
}
/**
- * Override of DrupalWebTestCase::setUp() specialized for upgrade testing.
+ * Overrides DrupalWebTestCase::setUp() for upgrade testing.
+ *
+ * @see DrupalWebTestCase::prepareDatabasePrefix()
+ * @see DrupalWebTestCase::changeDatabasePrefix()
+ * @see DrupalWebTestCase::prepareEnvironment()
*/
protected function setUp() {
// We are going to set a missing zlib requirement property for usage
@@ -93,55 +97,33 @@ abstract class UpgradePathTestCase extends DrupalWebTestCase {
$this->loadedModules = module_list();
- // Generate a temporary prefixed database to ensure that tests have a clean starting point.
- $this->databasePrefix = 'simpletest' . mt_rand(1000, 1000000);
- db_update('simpletest_test_id')
- ->fields(array('last_prefix' => $this->databasePrefix))
- ->condition('test_id', $this->testId)
- ->execute();
+ // Create the database prefix for this test.
+ $this->prepareDatabasePrefix();
- // Clone the current connection and replace the current prefix.
- $connection_info = Database::getConnectionInfo('default');
- Database::renameConnection('default', 'simpletest_original_default');
- foreach ($connection_info as $target => $value) {
- $connection_info[$target]['prefix'] = array(
- 'default' => $value['prefix']['default'] . $this->databasePrefix,
- );
+ // Prepare the environment for running tests.
+ $this->prepareEnvironment();
+ if (!$this->setupEnvironment) {
+ return FALSE;
}
- Database::addConnectionInfo('default', 'default', $connection_info['default']);
- // Store necessary current values before switching to prefixed database.
- $this->originalLanguage = $language;
- $this->originalLanguageDefault = variable_get('language_default');
- $this->originalFileDirectory = variable_get('file_public_path', conf_path() . '/files');
- $this->originalProfile = drupal_get_profile();
- $clean_url_original = variable_get('clean_url', 0);
+ // Reset all statics and variables to perform tests in a clean environment.
+ $conf = array();
+ drupal_static_reset();
+
+ // Change the database prefix.
+ // All static variables need to be reset before the database prefix is
+ // changed, since DrupalCacheArray implementations attempt to
+ // write back to persistent caches when they are destructed.
+ $this->changeDatabasePrefix();
+ if (!$this->setupDatabasePrefix) {
+ return FALSE;
+ }
// Unregister the registry.
// This is required to make sure that the database layer works properly.
spl_autoload_unregister('drupal_autoload_class');
spl_autoload_unregister('drupal_autoload_interface');
- // Create test directories ahead of installation so fatal errors and debug
- // information can be logged during installation process.
- // Use mock files directories with the same prefix as the database.
- $public_files_directory = $this->originalFileDirectory . '/simpletest/' . substr($this->databasePrefix, 10);
- $private_files_directory = $public_files_directory . '/private';
- $temp_files_directory = $private_files_directory . '/temp';
-
- // Create the directories.
- file_prepare_directory($public_files_directory, FILE_CREATE_DIRECTORY | FILE_MODIFY_PERMISSIONS);
- file_prepare_directory($private_files_directory, FILE_CREATE_DIRECTORY);
- file_prepare_directory($temp_files_directory, FILE_CREATE_DIRECTORY);
- $this->generatedTestFiles = FALSE;
-
- // Log fatal errors.
- ini_set('log_errors', 1);
- ini_set('error_log', $public_files_directory . '/error.log');
-
- // Reset all statics and variables to perform tests in a clean environment.
- $conf = array();
-
// Load the database from the portable PHP dump.
// The files may be gzipped.
foreach ($this->databaseDumpFiles as $file) {
@@ -152,22 +134,23 @@ abstract class UpgradePathTestCase extends DrupalWebTestCase {
}
// Set path variables.
- $this->variable_set('file_public_path', $public_files_directory);
- $this->variable_set('file_private_path', $private_files_directory);
- $this->variable_set('file_temporary_path', $temp_files_directory);
+ $this->variable_set('file_public_path', $this->public_files_directory);
+ $this->variable_set('file_private_path', $this->private_files_directory);
+ $this->variable_set('file_temporary_path', $this->temp_files_directory);
$this->pass('Finished loading the dump.');
- // Load user 1.
- $this->originalUser = $user;
+ // Ensure that the session is not written to the new environment and replace
+ // the global $user session with uid 1 from the new test site.
drupal_save_session(FALSE);
+ // Login as uid 1.
$user = db_query('SELECT * FROM {users} WHERE uid = :uid', array(':uid' => 1))->fetchObject();
// Generate and set a D6-compatible session cookie.
$this->prepareD7Session();
// Restore necessary variables.
- $this->variable_set('clean_url', $clean_url_original);
+ $this->variable_set('clean_url', $this->originalCleanUrl);
$this->variable_set('site_mail', 'simpletest@example.com');
drupal_set_time_limit($this->timeLimit);
@@ -175,63 +158,6 @@ abstract class UpgradePathTestCase extends DrupalWebTestCase {
}
/**
- * Override of DrupalWebTestCase::tearDown() specialized for upgrade testing.
- */
- protected function tearDown() {
- global $user, $language;
-
- if (!$this->zlibInstalled) {
- parent::tearDown();
- return;
- }
-
- // In case a fatal error occurred that was not in the test process read the
- // log to pick up any fatal errors.
- simpletest_log_read($this->testId, $this->databasePrefix, get_class($this), TRUE);
-
- // Delete temporary files directory.
- file_unmanaged_delete_recursive($this->originalFileDirectory . '/simpletest/' . substr($this->databasePrefix, 10));
-
- // Get back to the original connection.
- Database::removeConnection('default');
- Database::renameConnection('simpletest_original_default', 'default');
-
- // Remove all prefixed tables.
- $tables = db_find_tables($this->databasePrefix . '%');
- foreach ($tables as $table) {
- db_drop_table($table);
- }
-
- // Return the user to the original one.
- $user = $this->originalUser;
- drupal_save_session(TRUE);
-
- // Ensure that internal logged in variable and cURL options are reset.
- $this->loggedInUser = FALSE;
- $this->additionalCurlOptions = array();
-
- // Reload module list and implementations to ensure that test module hooks
- // aren't called after tests.
- module_list(TRUE);
- module_implements('', FALSE, TRUE);
-
- // Reset the Field API.
- field_cache_clear();
-
- // Rebuild caches.
- parent::refreshVariables();
-
- // Reset language.
- $language = $this->originalLanguage;
- if ($this->originalLanguageDefault) {
- $GLOBALS['conf']['language_default'] = $this->originalLanguageDefault;
- }
-
- // Close the CURL handler.
- $this->curlClose();
- }
-
- /**
* Specialized variable_set() that works even if the child site is not upgraded.
*
* @param $name