From f2154d585e78f2015b74b5c4cdb9abacb3978b7e Mon Sep 17 00:00:00 2001
From: xjm <xjm@65776.no-reply.drupal.org>
Date: Thu, 12 Jan 2023 13:29:47 -0600
Subject: [PATCH] Issue #3174924 by joachim, xjm, smustgrave, sourabhjain,
 ranjith_kumar_k_u, kiwimind, catch, bnjmnm: Throw a better exception when a
 reference field can't find the target entity type

---
 .../Drupal/Core/Field/FieldItemInterface.php  |  3 +++
 .../Field/FieldType/EntityReferenceItem.php   | 12 ++++++++++-
 .../Core/Entity/EntityReferenceFieldTest.php  | 21 +++++++++++++++++++
 3 files changed, 35 insertions(+), 1 deletion(-)

diff --git a/core/lib/Drupal/Core/Field/FieldItemInterface.php b/core/lib/Drupal/Core/Field/FieldItemInterface.php
index aee45492f287..d6ef1557e689 100644
--- a/core/lib/Drupal/Core/Field/FieldItemInterface.php
+++ b/core/lib/Drupal/Core/Field/FieldItemInterface.php
@@ -84,6 +84,9 @@ public static function mainPropertyName();
    *     stored in SQL. Also, the possible usage is limited, as you cannot
    *     specify another field as related, only existing SQL tables,
    *     such as {taxonomy_term_data}.
+   *
+   * @throws \Drupal\Core\Field\FieldException
+   *   Throws an exception if the schema is invalid.
    */
   public static function schema(FieldStorageDefinitionInterface $field_definition);
 
diff --git a/core/lib/Drupal/Core/Field/Plugin/Field/FieldType/EntityReferenceItem.php b/core/lib/Drupal/Core/Field/Plugin/Field/FieldType/EntityReferenceItem.php
index 146b9fd68e32..43cd540bb2ab 100644
--- a/core/lib/Drupal/Core/Field/Plugin/Field/FieldType/EntityReferenceItem.php
+++ b/core/lib/Drupal/Core/Field/Plugin/Field/FieldType/EntityReferenceItem.php
@@ -6,6 +6,7 @@
 use Drupal\Component\Utility\NestedArray;
 use Drupal\Core\Entity\ContentEntityStorageInterface;
 use Drupal\Core\Entity\EntityInterface;
+use Drupal\Component\Plugin\Exception\PluginNotFoundException;
 use Drupal\Core\Entity\EntityTypeInterface;
 use Drupal\Core\Entity\FieldableEntityInterface;
 use Drupal\Core\Entity\TypedData\EntityDataDefinition;
@@ -121,7 +122,16 @@ public static function mainPropertyName() {
    */
   public static function schema(FieldStorageDefinitionInterface $field_definition) {
     $target_type = $field_definition->getSetting('target_type');
-    $target_type_info = \Drupal::entityTypeManager()->getDefinition($target_type);
+    try {
+      $target_type_info = \Drupal::entityTypeManager()->getDefinition($target_type);
+    }
+    catch (PluginNotFoundException $e) {
+      throw new FieldException(sprintf("Field '%s' on entity type '%s' references a target entity type '%s' which does not exist.",
+        $field_definition->getName(),
+        $field_definition->getTargetEntityTypeId(),
+        $target_type
+      ));
+    }
     $properties = static::propertyDefinitions($field_definition)['target_id'];
     if ($target_type_info->entityClassImplements(FieldableEntityInterface::class) && $properties->getDataType() === 'integer') {
       $columns = [
diff --git a/core/tests/Drupal/KernelTests/Core/Entity/EntityReferenceFieldTest.php b/core/tests/Drupal/KernelTests/Core/Entity/EntityReferenceFieldTest.php
index c3005afd879d..9b58c8d0a6b7 100644
--- a/core/tests/Drupal/KernelTests/Core/Entity/EntityReferenceFieldTest.php
+++ b/core/tests/Drupal/KernelTests/Core/Entity/EntityReferenceFieldTest.php
@@ -6,6 +6,7 @@
 use Drupal\Core\Entity\EntityInterface;
 use Drupal\Core\Entity\EntityStorageException;
 use Drupal\Core\Field\BaseFieldDefinition;
+use Drupal\Core\Field\FieldException;
 use Drupal\Core\Field\FieldStorageDefinitionInterface;
 use Drupal\field\Entity\FieldConfig;
 use Drupal\field\Entity\FieldStorageConfig;
@@ -368,6 +369,26 @@ protected function assertUserRoleAutocreate(EntityInterface $entity, callable $s
     $this->assertEquals($entity->user_role->target_id, $role->id());
   }
 
+  /**
+   * Tests exception thrown with a missing target entity type.
+   */
+  public function testTargetEntityTypeMissing() {
+    // Setup a test entity type with an entity reference field to an entity type
+    // that doesn't exist.
+    $definitions = [
+      'bad_reference' => BaseFieldDefinition::create('entity_reference')
+        ->setSetting('target_type', 'made_up_entity_type')
+        ->setSetting('handler', 'default'),
+    ];
+    $this->state->set('entity_test.additional_base_field_definitions', $definitions);
+    $this->entityTypeManager->clearCachedDefinitions();
+
+    $this->expectException(FieldException::class);
+    $this->expectExceptionMessage("Field 'bad_reference' on entity type 'entity_test' references a target entity type 'made_up_entity_type' which does not exist.");
+
+    $this->installEntitySchema('entity_test');
+  }
+
   /**
    * Tests that the target entity is not unnecessarily loaded.
    */
-- 
GitLab