function PathAliasToEntityUpdateTest::testConversionToEntities

Tests the conversion of path aliases to entities.

@dataProvider providerConversionToEntities

Parameters

bool $perform_check: Whether a schema check should be performed on "path_alias" save.

See also

system_update_8803()

system_update_8804()

File

core/modules/system/tests/src/Functional/Update/PathAliasToEntityUpdateTest.php, line 42

Class

PathAliasToEntityUpdateTest
Tests the conversion of path aliases to entities.

Namespace

Drupal\Tests\system\Functional\Update

Code

public function testConversionToEntities($perform_check) {
    $database = \Drupal::database();
    $schema = $database->schema();
    $this->assertTrue($schema->tableExists('url_alias'));
    $query = $database->select('url_alias');
    $query->addField('url_alias', 'pid', 'id');
    $query->addField('url_alias', 'source', 'path');
    $query->addField('url_alias', 'alias');
    $query->addField('url_alias', 'langcode');
    // Path aliases did not have a 'status' value before the conversion to
    // entities, but we're adding it here to ensure that the field was installed
    // and populated correctly.
    $query->addExpression('1', 'status');
    $original_records = $query->execute()
        ->fetchAllAssoc('id');
    // drupal-8.filled.standard.php.gz contains one URL alias and
    // drupal-8.convert-path-aliases-to-entities-2336597.php adds another four.
    $url_alias_count = 5;
    $this->assertCount($url_alias_count, $original_records);
    // Enable or disable the "path_alias" save schema check.
    drupal_rewrite_settings([
        'settings' => [
            'system.path_alias_schema_check' => (object) [
                'value' => $perform_check,
                'required' => TRUE,
            ],
        ],
    ]);
    // Enable our test module in a way that does not affect the subsequent
    // updates run (::rebuildAll() would).
    
    /** @var \Drupal\Core\Extension\ModuleInstallerInterface $module_installer */
    $module_installer = $this->container
        ->get('module_installer');
    $module_installer->install([
        'system_test',
    ]);
    $this->container = \Drupal::getContainer();
    // Trigger a path alias save during the update.
    
    /** @var \Drupal\Core\State\StateInterface $state */
    $state = $this->container
        ->get('state');
    $state->set('system_test.path_alias_save', TRUE);
    // If the schema check is not performed, the path alias will be saved and we
    // will get an integrity exception, so no point in checking for failed
    // updates. If the schema check is performed, a logic exception will be
    // thrown, which will be caught in our test code, so the update is expected
    // to complete successfully. In this case we do want to check for failed
    // updates.
    $this->checkFailedUpdates = $perform_check;
    $this->runUpdates();
    if (!$perform_check) {
        $error_message = $this->cssSelect('.failure')[0]
            ->getText();
        // In order to test against multiple database drivers assert that the
        // expected exception occurs and the error contains the expected table
        // name. The specifics of the error depend on the database driver.
        $this->assertContains("Failed: Drupal\\Core\\Database\\IntegrityConstraintViolationException", $error_message);
        $this->assertContains("path_alias", $error_message);
        // Nothing else to assert.
        return;
    }
    // Check that an exception was thrown on "path_alias" save.
    $exception_info = $state->get('system_test.path_alias_save_exception_thrown');
    $this->assertIdentical($exception_info['class'], \LogicException::class);
    $this->assertIdentical($exception_info['message'], 'Path alias "/test" ("/user") could not be saved because the "system_update_8804" database update was not applied yet.');
    
    /** @var \Drupal\Core\Extension\ModuleHandlerInterface $module_handler */
    $module_handler = $this->container
        ->get('module_handler');
    $this->assertTrue($module_handler->moduleExists('path_alias'));
    $entity_type = \Drupal::entityDefinitionUpdateManager()->getEntityType('path_alias');
    $this->assertEquals('path_alias', $entity_type->getProvider());
    $this->assertEquals(PathAlias::class, $entity_type->getClass());
    $field_storage_definitions = \Drupal::service('entity.last_installed_schema.repository')->getLastInstalledFieldStorageDefinitions('path_alias');
    $this->assertEquals('path_alias', $field_storage_definitions['id']->getProvider());
    $this->assertEquals('path_alias', $field_storage_definitions['revision_id']->getProvider());
    $this->assertEquals('path_alias', $field_storage_definitions['langcode']->getProvider());
    $this->assertEquals('path_alias', $field_storage_definitions['uuid']->getProvider());
    $this->assertEquals('path_alias', $field_storage_definitions['status']->getProvider());
    $this->assertEquals('path_alias', $field_storage_definitions['path']->getProvider());
    $this->assertEquals('path_alias', $field_storage_definitions['alias']->getProvider());
    // Check that the 'path_alias' entity tables have been created and the
    // 'url_alias' table has been deleted.
    $this->assertTrue($schema->tableExists('path_alias'));
    $this->assertTrue($schema->tableExists('path_alias_revision'));
    $this->assertFalse($schema->tableExists('url_alias'));
    // Check that we have a backup of the old table.
    $this->assertCount(1, $schema->findTables('old_%_url_alias'));
    $path_alias_count = \Drupal::entityTypeManager()->getStorage('path_alias')
        ->loadMultiple();
    $this->assertCount($url_alias_count, $path_alias_count);
    // Make sure that existing aliases still work.
    $assert_session = $this->assertSession();
    $this->drupalGet('test-article');
    $assert_session->responseContains('/node/1');
    $assert_session->pageTextContains('Test Article - New title');
    $this->drupalGet('test-article-new-alias');
    $assert_session->responseContains('/node/1');
    $assert_session->pageTextContains('Test Article - New title');
    $this->drupalGet('test-alias-for-any-language');
    $assert_session->responseContains('/node/8');
    $assert_session->pageTextContains('Test title');
    $this->drupalGet('test-alias-in-english');
    $assert_session->responseContains('/node/8');
    $assert_session->pageTextContains('Test title');
    $spanish = \Drupal::languageManager()->getLanguage('es');
    $this->drupalGet('test-alias-for-any-language', [
        'language' => $spanish,
    ]);
    $assert_session->responseContains('/es/node/8');
    $assert_session->pageTextContains('Test title Spanish');
    $this->drupalGet('test-alias-in-spanish', [
        'language' => $spanish,
    ]);
    $assert_session->responseContains('/es/node/8');
    $assert_session->pageTextContains('Test title Spanish');
    // Check that correct data was written in both the base and the revision
    // tables.
    $base_table_records = $database->select('path_alias')
        ->fields('path_alias', [
        'id',
        'path',
        'alias',
        'langcode',
        'status',
    ])
        ->execute()
        ->fetchAllAssoc('id');
    $this->assertEquals($original_records, $base_table_records);
    $revision_table_records = $database->select('path_alias_revision')
        ->fields('path_alias_revision', [
        'id',
        'path',
        'alias',
        'langcode',
        'status',
    ])
        ->execute()
        ->fetchAllAssoc('id');
    $this->assertEquals($original_records, $revision_table_records);
}

Buggy or inaccurate documentation? Please file an issue. Need support? Need help programming? Connect with the Drupal community.