annotate core/tests/Drupal/FunctionalTests/Installer/InstallerTestBase.php @ 5:12f9dff5fda9 tip

Update to Drupal core 8.7.1
author Chris Cannam
date Thu, 09 May 2019 15:34:47 +0100
parents a9cd425dd02b
children
rev   line source
Chris@0 1 <?php
Chris@0 2
Chris@0 3 namespace Drupal\FunctionalTests\Installer;
Chris@0 4
Chris@0 5 use Drupal\Core\DrupalKernel;
Chris@0 6 use Drupal\Core\Language\Language;
Chris@0 7 use Drupal\Core\Session\UserSession;
Chris@0 8 use Drupal\Core\Site\Settings;
Chris@0 9 use Drupal\Core\Test\HttpClientMiddleware\TestHttpClientMiddleware;
Chris@0 10 use Drupal\Tests\BrowserTestBase;
Chris@5 11 use Drupal\Tests\RequirementsPageTrait;
Chris@0 12 use GuzzleHttp\HandlerStack;
Chris@0 13 use Symfony\Component\DependencyInjection\ContainerBuilder;
Chris@0 14 use Symfony\Component\DependencyInjection\Reference;
Chris@0 15 use Symfony\Component\HttpFoundation\Request;
Chris@0 16 use Symfony\Component\HttpFoundation\RequestStack;
Chris@0 17
Chris@0 18 /**
Chris@0 19 * Base class for testing the interactive installer.
Chris@0 20 */
Chris@0 21 abstract class InstallerTestBase extends BrowserTestBase {
Chris@0 22
Chris@5 23 use RequirementsPageTrait;
Chris@5 24
Chris@0 25 /**
Chris@0 26 * Custom settings.php values to write for a test run.
Chris@0 27 *
Chris@0 28 * @var array
Chris@0 29 * An array of settings to write out, in the format expected by
Chris@0 30 * drupal_rewrite_settings().
Chris@0 31 */
Chris@0 32 protected $settings = [];
Chris@0 33
Chris@0 34 /**
Chris@0 35 * The language code in which to install Drupal.
Chris@0 36 *
Chris@0 37 * @var string
Chris@0 38 */
Chris@0 39 protected $langcode = 'en';
Chris@0 40
Chris@0 41 /**
Chris@0 42 * The installation profile to install.
Chris@0 43 *
Chris@0 44 * @var string
Chris@0 45 */
Chris@0 46 protected $profile = 'testing';
Chris@0 47
Chris@0 48 /**
Chris@0 49 * Additional parameters to use for installer screens.
Chris@0 50 *
Chris@0 51 * @see FunctionalTestSetupTrait::installParameters()
Chris@0 52 *
Chris@0 53 * @var array
Chris@0 54 */
Chris@0 55 protected $parameters = [];
Chris@0 56
Chris@0 57 /**
Chris@0 58 * A string translation map used for translated installer screens.
Chris@0 59 *
Chris@0 60 * Keys are English strings, values are translated strings.
Chris@0 61 *
Chris@0 62 * @var array
Chris@0 63 */
Chris@0 64 protected $translations = [
Chris@0 65 'Save and continue' => 'Save and continue',
Chris@0 66 ];
Chris@0 67
Chris@0 68 /**
Chris@0 69 * Whether the installer has completed.
Chris@0 70 *
Chris@0 71 * @var bool
Chris@0 72 */
Chris@0 73 protected $isInstalled = FALSE;
Chris@0 74
Chris@0 75 /**
Chris@0 76 * {@inheritdoc}
Chris@0 77 */
Chris@0 78 protected function setUp() {
Chris@0 79 $this->isInstalled = FALSE;
Chris@0 80
Chris@0 81 $this->setupBaseUrl();
Chris@0 82
Chris@0 83 $this->prepareDatabasePrefix();
Chris@0 84
Chris@0 85 // Install Drupal test site.
Chris@0 86 $this->prepareEnvironment();
Chris@0 87
Chris@0 88 // Define information about the user 1 account.
Chris@0 89 $this->rootUser = new UserSession([
Chris@0 90 'uid' => 1,
Chris@0 91 'name' => 'admin',
Chris@0 92 'mail' => 'admin@example.com',
Chris@0 93 'pass_raw' => $this->randomMachineName(),
Chris@0 94 ]);
Chris@0 95
Chris@0 96 // If any $settings are defined for this test, copy and prepare an actual
Chris@0 97 // settings.php, so as to resemble a regular installation.
Chris@0 98 if (!empty($this->settings)) {
Chris@0 99 // Not using File API; a potential error must trigger a PHP warning.
Chris@0 100 copy(DRUPAL_ROOT . '/sites/default/default.settings.php', DRUPAL_ROOT . '/' . $this->siteDirectory . '/settings.php');
Chris@0 101 $this->writeSettings($this->settings);
Chris@0 102 }
Chris@0 103
Chris@0 104 // Note that FunctionalTestSetupTrait::installParameters() returns form
Chris@0 105 // input values suitable for a programmed
Chris@0 106 // \Drupal::formBuilder()->submitForm().
Chris@0 107 // @see InstallerTestBase::translatePostValues()
Chris@0 108 $this->parameters = $this->installParameters();
Chris@0 109
Chris@0 110 // Set up a minimal container (required by BrowserTestBase). Set cookie and
Chris@0 111 // server information so that XDebug works.
Chris@0 112 // @see install_begin_request()
Chris@0 113 $request = Request::create($GLOBALS['base_url'] . '/core/install.php', 'GET', [], $_COOKIE, [], $_SERVER);
Chris@0 114 $this->container = new ContainerBuilder();
Chris@0 115 $request_stack = new RequestStack();
Chris@0 116 $request_stack->push($request);
Chris@0 117 $this->container
Chris@0 118 ->set('request_stack', $request_stack);
Chris@0 119 $this->container
Chris@0 120 ->setParameter('language.default_values', Language::$defaultValues);
Chris@0 121 $this->container
Chris@0 122 ->register('language.default', 'Drupal\Core\Language\LanguageDefault')
Chris@0 123 ->addArgument('%language.default_values%');
Chris@0 124 $this->container
Chris@0 125 ->register('string_translation', 'Drupal\Core\StringTranslation\TranslationManager')
Chris@0 126 ->addArgument(new Reference('language.default'));
Chris@0 127 $this->container
Chris@0 128 ->register('http_client', 'GuzzleHttp\Client')
Chris@0 129 ->setFactory('http_client_factory:fromOptions');
Chris@0 130 $this->container
Chris@0 131 ->register('http_client_factory', 'Drupal\Core\Http\ClientFactory')
Chris@0 132 ->setArguments([new Reference('http_handler_stack')]);
Chris@0 133 $handler_stack = HandlerStack::create();
Chris@0 134 $test_http_client_middleware = new TestHttpClientMiddleware();
Chris@0 135 $handler_stack->push($test_http_client_middleware(), 'test.http_client.middleware');
Chris@0 136 $this->container
Chris@0 137 ->set('http_handler_stack', $handler_stack);
Chris@0 138
Chris@0 139 $this->container
Chris@0 140 ->set('app.root', DRUPAL_ROOT);
Chris@0 141 \Drupal::setContainer($this->container);
Chris@0 142
Chris@0 143 // Setup Mink.
Chris@0 144 $this->initMink();
Chris@0 145
Chris@4 146 // Set up the browser test output file.
Chris@4 147 $this->initBrowserOutputFile();
Chris@4 148
Chris@0 149 $this->visitInstaller();
Chris@0 150
Chris@0 151 // Select language.
Chris@0 152 $this->setUpLanguage();
Chris@0 153
Chris@0 154 // Select profile.
Chris@0 155 $this->setUpProfile();
Chris@0 156
Chris@0 157 // Address the requirements problem screen, if any.
Chris@0 158 $this->setUpRequirementsProblem();
Chris@0 159
Chris@0 160 // Configure settings.
Chris@0 161 $this->setUpSettings();
Chris@0 162
Chris@0 163 // @todo Allow test classes based on this class to act on further installer
Chris@0 164 // screens.
Chris@0 165
Chris@0 166 // Configure site.
Chris@0 167 $this->setUpSite();
Chris@0 168
Chris@0 169 if ($this->isInstalled) {
Chris@0 170 // Import new settings.php written by the installer.
Chris@0 171 $request = Request::createFromGlobals();
Chris@0 172 $class_loader = require $this->container->get('app.root') . '/autoload.php';
Chris@0 173 Settings::initialize($this->container->get('app.root'), DrupalKernel::findSitePath($request), $class_loader);
Chris@0 174 foreach ($GLOBALS['config_directories'] as $type => $path) {
Chris@0 175 $this->configDirectories[$type] = $path;
Chris@0 176 }
Chris@0 177
Chris@0 178 // After writing settings.php, the installer removes write permissions
Chris@0 179 // from the site directory. To allow drupal_generate_test_ua() to write
Chris@0 180 // a file containing the private key for drupal_valid_test_ua(), the site
Chris@0 181 // directory has to be writable.
Chris@0 182 // BrowserTestBase::tearDown() will delete the entire test site directory.
Chris@0 183 // Not using File API; a potential error must trigger a PHP warning.
Chris@0 184 chmod($this->container->get('app.root') . '/' . $this->siteDirectory, 0777);
Chris@0 185 $this->kernel = DrupalKernel::createFromRequest($request, $class_loader, 'prod', FALSE);
Chris@0 186 $this->kernel->prepareLegacyRequest($request);
Chris@0 187 $this->container = $this->kernel->getContainer();
Chris@0 188
Chris@0 189 // Manually configure the test mail collector implementation to prevent
Chris@0 190 // tests from sending out emails and collect them in state instead.
Chris@0 191 $this->container->get('config.factory')
Chris@0 192 ->getEditable('system.mail')
Chris@0 193 ->set('interface.default', 'test_mail_collector')
Chris@0 194 ->save();
Chris@0 195 }
Chris@0 196 }
Chris@0 197
Chris@0 198 /**
Chris@0 199 * {@inheritdoc}
Chris@0 200 */
Chris@0 201 protected function initFrontPage() {
Chris@0 202 // We don't want to visit the front page with the installer when
Chris@0 203 // initializing Mink, so we do nothing here.
Chris@0 204 }
Chris@0 205
Chris@0 206 /**
Chris@0 207 * Visits the interactive installer.
Chris@0 208 */
Chris@0 209 protected function visitInstaller() {
Chris@0 210 $this->drupalGet($GLOBALS['base_url'] . '/core/install.php');
Chris@0 211 }
Chris@0 212
Chris@0 213 /**
Chris@0 214 * Installer step: Select language.
Chris@0 215 */
Chris@0 216 protected function setUpLanguage() {
Chris@0 217 $edit = [
Chris@0 218 'langcode' => $this->langcode,
Chris@0 219 ];
Chris@0 220 $this->drupalPostForm(NULL, $edit, $this->translations['Save and continue']);
Chris@0 221 }
Chris@0 222
Chris@0 223 /**
Chris@0 224 * Installer step: Select installation profile.
Chris@0 225 */
Chris@0 226 protected function setUpProfile() {
Chris@0 227 $edit = [
Chris@0 228 'profile' => $this->profile,
Chris@0 229 ];
Chris@0 230 $this->drupalPostForm(NULL, $edit, $this->translations['Save and continue']);
Chris@0 231 }
Chris@0 232
Chris@0 233 /**
Chris@0 234 * Installer step: Configure settings.
Chris@0 235 */
Chris@0 236 protected function setUpSettings() {
Chris@0 237 $edit = $this->translatePostValues($this->parameters['forms']['install_settings_form']);
Chris@0 238 $this->drupalPostForm(NULL, $edit, $this->translations['Save and continue']);
Chris@0 239 }
Chris@0 240
Chris@0 241 /**
Chris@0 242 * Installer step: Requirements problem.
Chris@0 243 *
Chris@0 244 * Override this method to test specific requirements warnings or errors
Chris@0 245 * during the installer.
Chris@0 246 *
Chris@0 247 * @see system_requirements()
Chris@0 248 */
Chris@0 249 protected function setUpRequirementsProblem() {
Chris@5 250 // Do nothing.
Chris@0 251 }
Chris@0 252
Chris@0 253 /**
Chris@0 254 * Final installer step: Configure site.
Chris@0 255 */
Chris@0 256 protected function setUpSite() {
Chris@0 257 $edit = $this->translatePostValues($this->parameters['forms']['install_configure_form']);
Chris@0 258 $this->drupalPostForm(NULL, $edit, $this->translations['Save and continue']);
Chris@0 259 // If we've got to this point the site is installed using the regular
Chris@0 260 // installation workflow.
Chris@0 261 $this->isInstalled = TRUE;
Chris@0 262 }
Chris@0 263
Chris@0 264 /**
Chris@0 265 * {@inheritdoc}
Chris@0 266 *
Chris@0 267 * FunctionalTestSetupTrait::refreshVariables() tries to operate on persistent
Chris@0 268 * storage, which is only available after the installer completed.
Chris@0 269 */
Chris@0 270 protected function refreshVariables() {
Chris@0 271 if ($this->isInstalled) {
Chris@0 272 parent::refreshVariables();
Chris@0 273 }
Chris@0 274 }
Chris@0 275
Chris@0 276 }