X-Git-Url: http://www.aleph1.co.uk/gitweb/?p=yaffs-website;a=blobdiff_plain;f=web%2Fcore%2Fmodules%2Fcontent_translation%2Ftests%2Fsrc%2FKernel%2FContentTranslationFieldSyncRevisionTest.php;fp=web%2Fcore%2Fmodules%2Fcontent_translation%2Ftests%2Fsrc%2FKernel%2FContentTranslationFieldSyncRevisionTest.php;h=8185182516c7bd4b7942f7f7bf529690d68a879e;hp=0000000000000000000000000000000000000000;hb=af6d1fb995500ae68849458ee10d66abbdcfb252;hpb=680c79a86e3ed402f263faeac92e89fb6d9edcc0 diff --git a/web/core/modules/content_translation/tests/src/Kernel/ContentTranslationFieldSyncRevisionTest.php b/web/core/modules/content_translation/tests/src/Kernel/ContentTranslationFieldSyncRevisionTest.php new file mode 100644 index 000000000..818518251 --- /dev/null +++ b/web/core/modules/content_translation/tests/src/Kernel/ContentTranslationFieldSyncRevisionTest.php @@ -0,0 +1,482 @@ +installEntitySchema($entity_type_id); + $this->installEntitySchema('file'); + $this->installSchema('file', ['file_usage']); + + ConfigurableLanguage::createFromLangcode('it')->save(); + ConfigurableLanguage::createFromLangcode('fr')->save(); + + /** @var \Drupal\field\Entity\FieldStorageConfig $field_storage */ + $field_storage_config = FieldStorageConfig::create([ + 'field_name' => $this->fieldName, + 'type' => 'image', + 'entity_type' => $entity_type_id, + 'cardinality' => 1, + 'translatable' => 1, + ]); + $field_storage_config->save(); + + $field_config = FieldConfig::create([ + 'entity_type' => $entity_type_id, + 'field_name' => $this->fieldName, + 'bundle' => $entity_type_id, + 'label' => 'Synchronized field', + 'translatable' => 1, + ]); + $field_config->save(); + + $property_settings = [ + 'alt' => 'alt', + 'title' => 'title', + 'file' => 0, + ]; + $field_config->setThirdPartySetting('content_translation', 'translation_sync', $property_settings); + $field_config->save(); + + $this->entityManager->clearCachedDefinitions(); + + $this->contentTranslationManager = $this->container->get('content_translation.manager'); + $this->contentTranslationManager->setEnabled($entity_type_id, $entity_type_id, TRUE); + + $this->storage = $this->entityManager->getStorage($entity_type_id); + + foreach ($this->getTestFiles('image') as $file) { + $entity = File::create((array) $file + ['status' => 1]); + $entity->save(); + } + + $this->state->set('content_translation.entity_access.file', ['view' => TRUE]); + + $account = User::create([ + 'name' => $this->randomMachineName(), + 'status' => 1, + ]); + $account->save(); + } + + /** + * Checks that field synchronization works as expected with revisions. + * + * @covers \Drupal\content_translation\Plugin\Validation\Constraint\ContentTranslationSynchronizedFieldsConstraintValidator::create + * @covers \Drupal\content_translation\Plugin\Validation\Constraint\ContentTranslationSynchronizedFieldsConstraintValidator::validate + * @covers \Drupal\content_translation\Plugin\Validation\Constraint\ContentTranslationSynchronizedFieldsConstraintValidator::hasSynchronizedPropertyChanges + * @covers \Drupal\content_translation\FieldTranslationSynchronizer::getFieldSynchronizedProperties + * @covers \Drupal\content_translation\FieldTranslationSynchronizer::synchronizeFields + * @covers \Drupal\content_translation\FieldTranslationSynchronizer::synchronizeItems + */ + public function testFieldSynchronizationAndValidation() { + // Test that when untranslatable field widgets are displayed, synchronized + // field properties can be changed only in default revisions. + $this->setUntranslatableFieldWidgetsDisplay(TRUE); + $entity = $this->saveNewEntity(); + $entity_id = $entity->id(); + $this->assertLatestRevisionFieldValues($entity_id, [1, 1, 1, 'Alt 1 EN']); + + /** @var \Drupal\Core\Entity\ContentEntityInterface $en_revision */ + $en_revision = $this->createRevision($entity, FALSE); + $en_revision->get($this->fieldName)->target_id = 2; + $violations = $en_revision->validate(); + $this->assertViolations($violations); + + $it_translation = $entity->addTranslation('it', $entity->toArray()); + /** @var \Drupal\Core\Entity\ContentEntityInterface $it_revision */ + $it_revision = $this->createRevision($it_translation, FALSE); + $metadata = $this->contentTranslationManager->getTranslationMetadata($it_revision); + $metadata->setSource('en'); + $it_revision->get($this->fieldName)->target_id = 2; + $it_revision->get($this->fieldName)->alt = 'Alt 2 IT'; + $violations = $it_revision->validate(); + $this->assertViolations($violations); + $it_revision->isDefaultRevision(TRUE); + $violations = $it_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($it_revision); + $this->assertLatestRevisionFieldValues($entity_id, [2, 2, 2, 'Alt 1 EN', 'Alt 2 IT']); + + $en_revision = $this->createRevision($en_revision, FALSE); + $en_revision->get($this->fieldName)->alt = 'Alt 3 EN'; + $violations = $en_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($en_revision); + $this->assertLatestRevisionFieldValues($entity_id, [3, 2, 2, 'Alt 3 EN', 'Alt 2 IT']); + + $it_revision = $this->createRevision($it_revision, FALSE); + $it_revision->get($this->fieldName)->alt = 'Alt 4 IT'; + $violations = $it_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($it_revision); + $this->assertLatestRevisionFieldValues($entity_id, [4, 2, 2, 'Alt 1 EN', 'Alt 4 IT']); + + $en_revision = $this->createRevision($en_revision); + $en_revision->get($this->fieldName)->alt = 'Alt 5 EN'; + $violations = $en_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($en_revision); + $this->assertLatestRevisionFieldValues($entity_id, [5, 2, 2, 'Alt 5 EN', 'Alt 2 IT']); + + $en_revision = $this->createRevision($en_revision); + $en_revision->get($this->fieldName)->target_id = 6; + $en_revision->get($this->fieldName)->alt = 'Alt 6 EN'; + $violations = $en_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($en_revision); + $this->assertLatestRevisionFieldValues($entity_id, [6, 6, 6, 'Alt 6 EN', 'Alt 2 IT']); + + $it_revision = $this->createRevision($it_revision); + $it_revision->get($this->fieldName)->alt = 'Alt 7 IT'; + $violations = $it_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($it_revision); + $this->assertLatestRevisionFieldValues($entity_id, [7, 6, 6, 'Alt 6 EN', 'Alt 7 IT']); + + // Test that when untranslatable field widgets are hidden, synchronized + // field properties can be changed only when editing the default + // translation. This may lead to temporarily desynchronized values, when + // saving a pending revision for the default translation that changes a + // synchronized property (see revision 11). + $this->setUntranslatableFieldWidgetsDisplay(FALSE); + $entity = $this->saveNewEntity(); + $entity_id = $entity->id(); + $this->assertLatestRevisionFieldValues($entity_id, [8, 1, 1, 'Alt 1 EN']); + + /** @var \Drupal\Core\Entity\ContentEntityInterface $en_revision */ + $en_revision = $this->createRevision($entity, FALSE); + $en_revision->get($this->fieldName)->target_id = 2; + $en_revision->get($this->fieldName)->alt = 'Alt 2 EN'; + $violations = $en_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($en_revision); + $this->assertLatestRevisionFieldValues($entity_id, [9, 2, 2, 'Alt 2 EN']); + + $it_translation = $entity->addTranslation('it', $entity->toArray()); + /** @var \Drupal\Core\Entity\ContentEntityInterface $it_revision */ + $it_revision = $this->createRevision($it_translation, FALSE); + $metadata = $this->contentTranslationManager->getTranslationMetadata($it_revision); + $metadata->setSource('en'); + $it_revision->get($this->fieldName)->target_id = 3; + $violations = $it_revision->validate(); + $this->assertViolations($violations); + $it_revision->isDefaultRevision(TRUE); + $violations = $it_revision->validate(); + $this->assertViolations($violations); + + $it_revision = $this->createRevision($it_translation); + $metadata = $this->contentTranslationManager->getTranslationMetadata($it_revision); + $metadata->setSource('en'); + $it_revision->get($this->fieldName)->alt = 'Alt 3 IT'; + $violations = $it_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($it_revision); + $this->assertLatestRevisionFieldValues($entity_id, [10, 1, 1, 'Alt 1 EN', 'Alt 3 IT']); + + $en_revision = $this->createRevision($en_revision, FALSE); + $en_revision->get($this->fieldName)->alt = 'Alt 4 EN'; + $violations = $en_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($en_revision); + $this->assertLatestRevisionFieldValues($entity_id, [11, 2, 1, 'Alt 4 EN', 'Alt 3 IT']); + + $it_revision = $this->createRevision($it_revision, FALSE); + $it_revision->get($this->fieldName)->alt = 'Alt 5 IT'; + $violations = $it_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($it_revision); + $this->assertLatestRevisionFieldValues($entity_id, [12, 1, 1, 'Alt 1 EN', 'Alt 5 IT']); + + $en_revision = $this->createRevision($en_revision); + $en_revision->get($this->fieldName)->target_id = 6; + $en_revision->get($this->fieldName)->alt = 'Alt 6 EN'; + $violations = $en_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($en_revision); + $this->assertLatestRevisionFieldValues($entity_id, [13, 6, 6, 'Alt 6 EN', 'Alt 3 IT']); + + $it_revision = $this->createRevision($it_revision); + $it_revision->get($this->fieldName)->target_id = 7; + $violations = $it_revision->validate(); + $this->assertViolations($violations); + + $it_revision = $this->createRevision($it_revision); + $it_revision->get($this->fieldName)->alt = 'Alt 7 IT'; + $violations = $it_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($it_revision); + $this->assertLatestRevisionFieldValues($entity_id, [14, 6, 6, 'Alt 6 EN', 'Alt 7 IT']); + + // Test that creating a default revision starting from a pending revision + // having changes to synchronized properties, without introducing new + // changes works properly. + $this->setUntranslatableFieldWidgetsDisplay(FALSE); + $entity = $this->saveNewEntity(); + $entity_id = $entity->id(); + $this->assertLatestRevisionFieldValues($entity_id, [15, 1, 1, 'Alt 1 EN']); + + $it_translation = $entity->addTranslation('it', $entity->toArray()); + /** @var \Drupal\Core\Entity\ContentEntityInterface $it_revision */ + $it_revision = $this->createRevision($it_translation); + $metadata = $this->contentTranslationManager->getTranslationMetadata($it_revision); + $metadata->setSource('en'); + $it_revision->get($this->fieldName)->alt = 'Alt 2 IT'; + $violations = $it_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($it_revision); + $this->assertLatestRevisionFieldValues($entity_id, [16, 1, 1, 'Alt 1 EN', 'Alt 2 IT']); + + /** @var \Drupal\Core\Entity\ContentEntityInterface $en_revision */ + $en_revision = $this->createRevision($entity); + $en_revision->get($this->fieldName)->target_id = 3; + $en_revision->get($this->fieldName)->alt = 'Alt 3 EN'; + $violations = $en_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($en_revision); + $this->assertLatestRevisionFieldValues($entity_id, [17, 3, 3, 'Alt 3 EN', 'Alt 2 IT']); + + $en_revision = $this->createRevision($entity, FALSE); + $en_revision->get($this->fieldName)->target_id = 4; + $en_revision->get($this->fieldName)->alt = 'Alt 4 EN'; + $violations = $en_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($en_revision); + $this->assertLatestRevisionFieldValues($entity_id, [18, 4, 3, 'Alt 4 EN', 'Alt 2 IT']); + + $en_revision = $this->createRevision($entity); + $violations = $en_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($en_revision); + $this->assertLatestRevisionFieldValues($entity_id, [19, 4, 4, 'Alt 4 EN', 'Alt 2 IT']); + + $it_revision = $this->createRevision($it_revision); + $it_revision->get($this->fieldName)->alt = 'Alt 6 IT'; + $violations = $it_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($it_revision); + $this->assertLatestRevisionFieldValues($entity_id, [20, 4, 4, 'Alt 4 EN', 'Alt 6 IT']); + + // Check that we are not allowed to perform changes to multiple translations + // in pending revisions when synchronized properties are involved. + $this->setUntranslatableFieldWidgetsDisplay(FALSE); + $entity = $this->saveNewEntity(); + $entity_id = $entity->id(); + $this->assertLatestRevisionFieldValues($entity_id, [21, 1, 1, 'Alt 1 EN']); + + $it_translation = $entity->addTranslation('it', $entity->toArray()); + /** @var \Drupal\Core\Entity\ContentEntityInterface $it_revision */ + $it_revision = $this->createRevision($it_translation); + $metadata = $this->contentTranslationManager->getTranslationMetadata($it_revision); + $metadata->setSource('en'); + $it_revision->get($this->fieldName)->alt = 'Alt 2 IT'; + $violations = $it_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($it_revision); + $this->assertLatestRevisionFieldValues($entity_id, [22, 1, 1, 'Alt 1 EN', 'Alt 2 IT']); + + $en_revision = $this->createRevision($entity, FALSE); + $en_revision->get($this->fieldName)->target_id = 2; + $en_revision->getTranslation('it')->get($this->fieldName)->alt = 'Alt 3 IT'; + $violations = $en_revision->validate(); + $this->assertViolations($violations); + + // Test that when saving a new default revision starting from a pending + // revision, outdated synchronized properties do not override more recent + // ones. + $this->setUntranslatableFieldWidgetsDisplay(TRUE); + $entity = $this->saveNewEntity(); + $entity_id = $entity->id(); + $this->assertLatestRevisionFieldValues($entity_id, [23, 1, 1, 'Alt 1 EN']); + + $it_translation = $entity->addTranslation('it', $entity->toArray()); + /** @var \Drupal\Core\Entity\ContentEntityInterface $it_revision */ + $it_revision = $this->createRevision($it_translation, FALSE); + $metadata = $this->contentTranslationManager->getTranslationMetadata($it_revision); + $metadata->setSource('en'); + $it_revision->get($this->fieldName)->alt = 'Alt 2 IT'; + $violations = $it_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($it_revision); + $this->assertLatestRevisionFieldValues($entity_id, [24, 1, 1, 'Alt 1 EN', 'Alt 2 IT']); + + /** @var \Drupal\Core\Entity\ContentEntityInterface $en_revision */ + $en_revision = $this->createRevision($entity); + $en_revision->get($this->fieldName)->target_id = 3; + $en_revision->get($this->fieldName)->alt = 'Alt 3 EN'; + $violations = $en_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($en_revision); + $this->assertLatestRevisionFieldValues($entity_id, [25, 3, 3, 'Alt 3 EN', 'Alt 2 IT']); + + $it_revision = $this->createRevision($it_revision); + $it_revision->get($this->fieldName)->alt = 'Alt 4 IT'; + $violations = $it_revision->validate(); + $this->assertEmpty($violations); + $this->storage->save($it_revision); + $this->assertLatestRevisionFieldValues($entity_id, [26, 3, 3, 'Alt 3 EN', 'Alt 4 IT']); + } + + /** + * Sets untranslatable field widgets' display status. + * + * @param bool $display + * Whether untranslatable field widgets should be displayed. + */ + protected function setUntranslatableFieldWidgetsDisplay($display) { + $entity_type_id = $this->storage->getEntityTypeId(); + $settings = ['untranslatable_fields_hide' => !$display]; + $this->contentTranslationManager->setBundleTranslationSettings($entity_type_id, $entity_type_id, $settings); + /** @var \Drupal\Core\Entity\EntityTypeBundleInfo $bundle_info */ + $bundle_info = $this->container->get('entity_type.bundle.info'); + $bundle_info->clearCachedBundles(); + } + + /** + * @return \Drupal\Core\Entity\ContentEntityInterface + */ + protected function saveNewEntity() { + /** @var \Drupal\Core\Entity\ContentEntityInterface $entity */ + $entity = EntityTestMulRev::create([ + 'uid' => 1, + 'langcode' => 'en', + $this->fieldName => [ + 'target_id' => 1, + 'alt' => 'Alt 1 EN', + ], + ]); + $metadata = $this->contentTranslationManager->getTranslationMetadata($entity); + $metadata->setSource(LanguageInterface::LANGCODE_NOT_SPECIFIED); + $violations = $entity->validate(); + $this->assertEmpty($violations); + $this->storage->save($entity); + return $entity; + } + + /** + * Creates a new revision starting from the latest translation-affecting one. + * + * @param \Drupal\Core\Entity\ContentEntityInterface $translation + * The translation to be revisioned. + * @param bool $default + * (optional) Whether the new revision should be marked as default. Defaults + * to TRUE. + * + * @return \Drupal\Core\Entity\ContentEntityInterface + * An entity revision object. + */ + protected function createRevision(ContentEntityInterface $translation, $default = TRUE) { + if (!$translation->isNewTranslation()) { + $langcode = $translation->language()->getId(); + $revision_id = $this->storage->getLatestTranslationAffectedRevisionId($translation->id(), $langcode); + /** @var \Drupal\Core\Entity\ContentEntityInterface $revision */ + $revision = $this->storage->loadRevision($revision_id); + $translation = $revision->getTranslation($langcode); + } + /** @var \Drupal\Core\Entity\ContentEntityInterface $revision */ + $revision = $this->storage->createRevision($translation, $default); + return $revision; + } + + /** + * Asserts that the expected violations were found. + * + * @param \Drupal\Core\Entity\EntityConstraintViolationListInterface $violations + * A list of violations. + */ + protected function assertViolations(EntityConstraintViolationListInterface $violations) { + $entity_type_id = $this->storage->getEntityTypeId(); + $settings = $this->contentTranslationManager->getBundleTranslationSettings($entity_type_id, $entity_type_id); + $message = !empty($settings['untranslatable_fields_hide']) ? + 'Non-translatable field elements can only be changed when updating the original language.' : + 'Non-translatable field elements can only be changed when updating the current revision.'; + + $list = []; + foreach ($violations as $violation) { + if ((string) $violation->getMessage() === $message) { + $list[] = $violation; + } + } + $this->assertCount(1, $list); + } + + /** + * Asserts that the latest revision has the expected field values. + * + * @param $entity_id + * The entity ID. + * @param array $expected_values + * An array of expected values in the following order: + * - revision ID + * - target ID (en) + * - target ID (it) + * - alt (en) + * - alt (it) + */ + protected function assertLatestRevisionFieldValues($entity_id, array $expected_values) { + /** @var \Drupal\Core\Entity\ContentEntityInterface $entity */ + $entity = $this->storage->loadRevision($this->storage->getLatestRevisionId($entity_id)); + @list($revision_id, $target_id_en, $target_id_it, $alt_en, $alt_it) = $expected_values; + $this->assertEquals($revision_id, $entity->getRevisionId()); + $this->assertEquals($target_id_en, $entity->get($this->fieldName)->target_id); + $this->assertEquals($alt_en, $entity->get($this->fieldName)->alt); + if ($entity->hasTranslation('it')) { + $it_translation = $entity->getTranslation('it'); + $this->assertEquals($target_id_it, $it_translation->get($this->fieldName)->target_id); + $this->assertEquals($alt_it, $it_translation->get($this->fieldName)->alt); + } + } + +}