EntityFieldTest.php

Definition of Drupal\Core\Entity\Tests\EntityFieldTest.

Namespace

Drupal\system\Tests\Entity

File

drupal/core/modules/system/lib/Drupal/system/Tests/Entity/EntityFieldTest.php
View source
<?php

/**
 * @file
 * Definition of Drupal\Core\Entity\Tests\EntityFieldTest.
 */
namespace Drupal\system\Tests\Entity;

use Drupal\Core\Entity\EntityInterface;
use Drupal\Core\Entity\Field\FieldInterface;
use Drupal\Core\Entity\Field\FieldItemInterface;
use Drupal\Core\TypedData\TypedDataInterface;
use Drupal\simpletest\WebTestBase;

/**
 * Tests Entity API base functionality.
 */
class EntityFieldTest extends WebTestBase {

  /**
   * Modules to enable.
   *
   * @var array
   */
  public static $modules = array(
    'entity_test',
  );
  public static function getInfo() {
    return array(
      'name' => 'Entity Field API',
      'description' => 'Tests the Entity Field API',
      'group' => 'Entity API',
    );
  }

  /**
   * Creates a test entity.
   *
   * @return \Drupal\Core\Entity\EntityInterface
   */
  protected function createTestEntity() {
    $this->entity_name = $this
      ->randomName();
    $this->entity_user = $this
      ->drupalCreateUser();
    $this->entity_field_text = $this
      ->randomName();

    // Pass in the value of the name field when creating. With the user
    // field we test setting a field after creation.
    $entity = entity_create('entity_test', array());
    $entity->user_id->value = $this->entity_user->uid;
    $entity->name->value = $this->entity_name;

    // Set a value for the test field.
    $entity->field_test_text->value = $this->entity_field_text;
    return $entity;
  }

  /**
   * Tests reading and writing properties and field items.
   */
  public function testReadWrite() {
    $entity = $this
      ->createTestEntity();

    // Access the name field.
    $this
      ->assertTrue($entity->name instanceof FieldInterface, 'Field implements interface');
    $this
      ->assertTrue($entity->name[0] instanceof FieldItemInterface, 'Field item implements interface');
    $this
      ->assertEqual($this->entity_name, $entity->name->value, 'Name value can be read.');
    $this
      ->assertEqual($this->entity_name, $entity->name[0]->value, 'Name value can be read through list access.');
    $this
      ->assertEqual($entity->name
      ->getValue(), array(
      0 => array(
        'value' => $this->entity_name,
      ),
    ), 'Plain field value returned.');

    // Change the name.
    $new_name = $this
      ->randomName();
    $entity->name->value = $new_name;
    $this
      ->assertEqual($new_name, $entity->name->value, 'Name can be updated and read.');
    $this
      ->assertEqual($entity->name
      ->getValue(), array(
      0 => array(
        'value' => $new_name,
      ),
    ), 'Plain field value reflects the update.');
    $new_name = $this
      ->randomName();
    $entity->name[0]->value = $new_name;
    $this
      ->assertEqual($new_name, $entity->name->value, 'Name can be updated and read through list access.');

    // Access the user field.
    $this
      ->assertTrue($entity->user_id instanceof FieldInterface, 'Field implements interface');
    $this
      ->assertTrue($entity->user_id[0] instanceof FieldItemInterface, 'Field item implements interface');
    $this
      ->assertEqual($this->entity_user->uid, $entity->user_id->value, 'User id can be read.');
    $this
      ->assertEqual($this->entity_user->name, $entity->user_id->entity->name, 'User name can be read.');

    // Change the assigned user by entity.
    $new_user = $this
      ->drupalCreateUser();
    $entity->user_id->entity = $new_user;
    $this
      ->assertEqual($new_user->uid, $entity->user_id->value, 'Updated user id can be read.');
    $this
      ->assertEqual($new_user->name, $entity->user_id->entity->name, 'Updated user name value can be read.');

    // Change the assigned user by id.
    $new_user = $this
      ->drupalCreateUser();
    $entity->user_id->value = $new_user->uid;
    $this
      ->assertEqual($new_user->uid, $entity->user_id->value, 'Updated user id can be read.');
    $this
      ->assertEqual($new_user->name, $entity->user_id->entity->name, 'Updated user name value can be read.');

    // Try unsetting a field.
    $entity->name->value = NULL;
    $entity->user_id->value = NULL;
    $this
      ->assertNull($entity->name->value, 'Name field is not set.');
    $this
      ->assertNull($entity->user_id->value, 'User ID field is not set.');
    $this
      ->assertNull($entity->user_id->entity, 'User entity field is not set.');

    // Test using isset(), empty() and unset().
    $entity->name->value = 'test unset';
    unset($entity->name->value);
    $this
      ->assertFalse(isset($entity->name->value), 'Name is not set.');
    $this
      ->assertFalse(isset($entity->name[0]->value), 'Name is not set.');
    $this
      ->assertTrue(empty($entity->name->value), 'Name is empty.');
    $this
      ->assertTrue(empty($entity->name[0]->value), 'Name is empty.');
    $entity->name->value = 'a value';
    $this
      ->assertTrue(isset($entity->name->value), 'Name is set.');
    $this
      ->assertTrue(isset($entity->name[0]->value), 'Name is set.');
    $this
      ->assertFalse(empty($entity->name->value), 'Name is not empty.');
    $this
      ->assertFalse(empty($entity->name[0]->value), 'Name is not empty.');
    $this
      ->assertTrue(isset($entity->name[0]), 'Name string item is set.');
    $this
      ->assertFalse(isset($entity->name[1]), 'Second name string item is not set as it does not exist');
    $this
      ->assertTrue(isset($entity->name), 'Name field is set.');
    $this
      ->assertFalse(isset($entity->nameInvalid), 'Not existing field is not set.');
    unset($entity->name[0]);
    $this
      ->assertFalse(isset($entity->name[0]), 'Name field item is not set.');
    $this
      ->assertFalse(isset($entity->name[0]->value), 'Name is not set.');
    $this
      ->assertFalse(isset($entity->name->value), 'Name is not set.');
    $entity->name->value = 'a value';
    $this
      ->assertTrue(isset($entity->name->value), 'Name is set.');
    unset($entity->name);
    $this
      ->assertFalse(isset($entity->name), 'Name field is not set.');
    $this
      ->assertFalse(isset($entity->name[0]), 'Name field item is not set.');
    $this
      ->assertFalse(isset($entity->name[0]->value), 'Name is not set.');
    $this
      ->assertFalse(isset($entity->name->value), 'Name is not set.');

    // Access the language field.
    $this
      ->assertEqual(LANGUAGE_NOT_SPECIFIED, $entity->langcode->value, 'Language code can be read.');
    $this
      ->assertEqual(language_load(LANGUAGE_NOT_SPECIFIED), $entity->langcode->language, 'Language object can be read.');

    // Change the language by code.
    $entity->langcode->value = language_default()->langcode;
    $this
      ->assertEqual(language_default()->langcode, $entity->langcode->value, 'Language code can be read.');
    $this
      ->assertEqual(language_default(), $entity->langcode->language, 'Language object can be read.');

    // Revert language by code then try setting it by language object.
    $entity->langcode->value = LANGUAGE_NOT_SPECIFIED;
    $entity->langcode->language = language_default();
    $this
      ->assertEqual(language_default()->langcode, $entity->langcode->value, 'Language code can be read.');
    $this
      ->assertEqual(language_default(), $entity->langcode->language, 'Language object can be read.');

    // Access the text field and test updating.
    $this
      ->assertEqual($entity->field_test_text->value, $this->entity_field_text, 'Text field can be read.');
    $new_text = $this
      ->randomName();
    $entity->field_test_text->value = $new_text;
    $this
      ->assertEqual($entity->field_test_text->value, $new_text, 'Updated text field can be read.');

    // Test creating the entity by passing in plain values.
    $this->entity_name = $this
      ->randomName();
    $name_item[0]['value'] = $this->entity_name;
    $this->entity_user = $this
      ->drupalCreateUser();
    $user_item[0]['value'] = $this->entity_user->uid;
    $this->entity_field_text = $this
      ->randomName();
    $text_item[0]['value'] = $this->entity_field_text;
    $entity = entity_create('entity_test', array(
      'name' => $name_item,
      'user_id' => $user_item,
      'field_test_text' => $text_item,
    ));
    $this
      ->assertEqual($this->entity_name, $entity->name->value, 'Name value can be read.');
    $this
      ->assertEqual($this->entity_user->uid, $entity->user_id->value, 'User id can be read.');
    $this
      ->assertEqual($this->entity_user->name, $entity->user_id->entity->name, 'User name can be read.');
    $this
      ->assertEqual($this->entity_field_text, $entity->field_test_text->value, 'Text field can be read.');

    // Test copying field values.
    $entity2 = $this
      ->createTestEntity();
    $entity2->name = $entity->name;
    $entity2->user_id = $entity->user_id;
    $entity2->field_test_text = $entity->field_test_text;
    $this
      ->assertTrue($entity->name !== $entity2->name, 'Copying properties results in a different field object.');
    $this
      ->assertEqual($entity->name->value, $entity2->name->value, 'Name field copied.');
    $this
      ->assertEqual($entity->user_id->value, $entity2->user_id->value, 'User id field copied.');
    $this
      ->assertEqual($entity->field_test_text->value, $entity2->field_test_text->value, 'Text field copied.');

    // Tests adding a value to a field item list.
    $entity->name[] = 'Another name';
    $this
      ->assertEqual($entity->name[1]->value == 'Another name', 'List item added via [].');
    $entity->name[2]->value = 'Third name';
    $this
      ->assertEqual($entity->name[2]->value == 'Third name', 'List item added by a accessing not yet created item.');

    // Test removing and empty-ing list items.
    $this
      ->assertEqual(count($entity->name), 3, 'List has 3 items.');
    unset($entity->name[1]);
    $this
      ->assertEqual(count($entity->name), 2, 'Second list item has been removed.');
    $entity->name[2] = NULL;
    $this
      ->assertEqual(count($entity->name), 2, 'Assigning NULL does not reduce array count.');
    $this
      ->assertTrue($entity->name[2]
      ->isEmpty(), 'Assigning NULL empties the item.');

    // Test using isEmpty().
    unset($entity->name[2]);
    $this
      ->assertFalse($entity->name[0]
      ->isEmpty(), 'Name item is not empty.');
    $entity->name->value = NULL;
    $this
      ->assertTrue($entity->name[0]
      ->isEmpty(), 'Name item is empty.');
    $this
      ->assertTrue($entity->name
      ->isEmpty(), 'Name field is empty.');
    $this
      ->assertEqual(count($entity->name), 1, 'Empty item is considered when counting.');
    $this
      ->assertEqual(count(iterator_to_array($entity->name
      ->getIterator())), count($entity->name), 'Count matches iterator count.');
    $this
      ->assertTrue($entity->name
      ->getValue() === array(
      0 => NULL,
    ), 'Name field value contains a NULL value.');

    // Test removing all list items by assigning an empty array.
    $entity->name = array();
    $this
      ->assertIdentical(count($entity->name), 0, 'Name field contains no items.');
    $this
      ->assertIdentical($entity->name
      ->getValue(), array(), 'Name field value is an empty array.');

    // Test removing all list items by setting it to NULL.
    $entity->name = NULL;
    $this
      ->assertIdentical(count($entity->name), 0, 'Name field contains no items.');
    $this
      ->assertIdentical($entity->name
      ->getValue(), array(), 'Name field value is an empty array.');

    // Test get and set field values.
    $entity->name = 'foo';
    $this
      ->assertEqual($entity->name[0]
      ->getPropertyValues(), array(
      'value' => 'foo',
    ), 'Field value has been retrieved via getPropertyValue()');
    $entity->name[0]
      ->setPropertyValues(array(
      'value' => 'bar',
    ));
    $this
      ->assertEqual($entity->name->value, 'bar', 'Field value has been set via setPropertyValue()');
    $values = $entity
      ->getPropertyValues();
    $this
      ->assertEqual($values['name'], array(
      0 => array(
        'value' => 'bar',
      ),
    ), 'Field value has been retrieved via getPropertyValue() from an entity.');
    $entity
      ->setPropertyValues(array(
      'name' => 'foo',
    ));
    $this
      ->assertEqual($entity->name->value, 'foo', 'Field value has been set via setPropertyValue() on an entity.');

    // Make sure the user id can be set to zero.
    $user_item[0]['value'] = 0;
    $entity = entity_create('entity_test', array(
      'name' => $name_item,
      'user_id' => $user_item,
      'field_test_text' => $text_item,
    ));
    $this
      ->assertNotNull($entity->user_id->value, 'User id is not NULL');
    $this
      ->assertIdentical($entity->user_id->value, 0, 'User id has been set to 0');

    // Test setting the ID with the value only.
    $entity = entity_create('entity_test', array(
      'name' => $name_item,
      'user_id' => 0,
      'field_test_text' => $text_item,
    ));
    $this
      ->assertNotNull($entity->user_id->value, 'User id is not NULL');
    $this
      ->assertIdentical($entity->user_id->value, 0, 'User id has been set to 0');
  }

  /**
   * Tries to save and load an entity again.
   */
  public function testSave() {
    $entity = $this
      ->createTestEntity();
    $entity
      ->save();
    $this
      ->assertTrue((bool) $entity
      ->id(), 'Entity has received an id.');
    $entity = entity_load('entity_test', $entity
      ->id());
    $this
      ->assertTrue((bool) $entity
      ->id(), 'Entity loaded.');

    // Access the name field.
    $this
      ->assertEqual(1, $entity->id->value, 'ID value can be read.');
    $this
      ->assertTrue(is_string($entity->uuid->value), 'UUID value can be read.');
    $this
      ->assertEqual(LANGUAGE_NOT_SPECIFIED, $entity->langcode->value, 'Language code can be read.');
    $this
      ->assertEqual(language_load(LANGUAGE_NOT_SPECIFIED), $entity->langcode->language, 'Language object can be read.');
    $this
      ->assertEqual($this->entity_user->uid, $entity->user_id->value, 'User id can be read.');
    $this
      ->assertEqual($this->entity_user->name, $entity->user_id->entity->name, 'User name can be read.');
    $this
      ->assertEqual($this->entity_field_text, $entity->field_test_text->value, 'Text field can be read.');
  }

  /**
   * Tests introspection and getting metadata upfront.
   */
  public function testIntrospection() {

    // Test getting metadata upfront, i.e. without having an entity object.
    $definition = array(
      'type' => 'entity',
      'constraints' => array(
        'entity type' => 'entity_test',
      ),
      'label' => t('Test entity'),
    );
    $wrapped_entity = typed_data()
      ->create($definition);
    $definitions = $wrapped_entity
      ->getPropertyDefinitions($definition);
    $this
      ->assertEqual($definitions['name']['type'], 'string_field', 'Name field found.');
    $this
      ->assertEqual($definitions['user_id']['type'], 'entityreference_field', 'User field found.');
    $this
      ->assertEqual($definitions['field_test_text']['type'], 'text_field', 'Test-text-field field found.');

    // Test introspecting an entity object.
    // @todo: Add bundles and test bundles as well.
    $entity = entity_create('entity_test', array());
    $definitions = $entity
      ->getPropertyDefinitions();
    $this
      ->assertEqual($definitions['name']['type'], 'string_field', 'Name field found.');
    $this
      ->assertEqual($definitions['user_id']['type'], 'entityreference_field', 'User field found.');
    $this
      ->assertEqual($definitions['field_test_text']['type'], 'text_field', 'Test-text-field field found.');
    $name_properties = $entity->name
      ->getPropertyDefinitions();
    $this
      ->assertEqual($name_properties['value']['type'], 'string', 'String value property of the name found.');
    $userref_properties = $entity->user_id
      ->getPropertyDefinitions();
    $this
      ->assertEqual($userref_properties['value']['type'], 'integer', 'Entity id property of the user found.');
    $this
      ->assertEqual($userref_properties['entity']['type'], 'entity', 'Entity reference property of the user found.');
    $textfield_properties = $entity->field_test_text
      ->getPropertyDefinitions();
    $this
      ->assertEqual($textfield_properties['value']['type'], 'string', 'String value property of the test-text field found.');
    $this
      ->assertEqual($textfield_properties['format']['type'], 'string', 'String format field of the test-text field found.');
    $this
      ->assertEqual($textfield_properties['processed']['type'], 'string', 'String processed property of the test-text field found.');

    // @todo: Once the user entity has definitions, continue testing getting
    // them from the $userref_values['entity'] property.
  }

  /**
   * Tests iterating over properties.
   */
  public function testIterator() {
    $entity = $this
      ->createTestEntity();
    foreach ($entity as $name => $field) {
      $this
        ->assertTrue($field instanceof FieldInterface, "Field {$name} implements interface.");
      foreach ($field as $delta => $item) {
        $this
          ->assertTrue($field[0] instanceof FieldItemInterface, "Item {$delta} of field {$name} implements interface.");
        foreach ($item as $value_name => $value_property) {
          $this
            ->assertTrue($value_property instanceof TypedDataInterface, "Value {$value_name} of item {$delta} of field {$name} implements interface.");
          $value = $value_property
            ->getValue();
          $this
            ->assertTrue(!isset($value) || is_scalar($value) || $value instanceof EntityInterface, "Value {$value_name} of item {$delta} of field {$name} is a primitive or an entity.");
        }
      }
    }
    $properties = $entity
      ->getProperties();
    $this
      ->assertEqual(array_keys($properties), array_keys($entity
      ->getPropertyDefinitions()), 'All properties returned.');
    $this
      ->assertEqual($properties, iterator_to_array($entity
      ->getIterator()), 'Entity iterator iterates over all properties.');
  }

  /**
   * Tests working with entity properties based upon data structure and data
   * list interfaces.
   */
  public function testDataStructureInterfaces() {
    $entity = $this
      ->createTestEntity();
    $entity
      ->save();
    $entity_definition = array(
      'type' => 'entity',
      'constraints' => array(
        'entity type' => 'entity_test',
      ),
      'label' => t('Test entity'),
    );
    $wrapped_entity = typed_data()
      ->create($entity_definition, $entity);

    // For the test we navigate through the tree of contained properties and get
    // all contained strings, limited by a certain depth.
    $strings = array();
    $this
      ->getContainedStrings($wrapped_entity, 0, $strings);

    // @todo: Once the user entity has defined properties this should contain
    // the user name and other user entity strings as well.
    $target_strings = array(
      $entity->uuid->value,
      LANGUAGE_NOT_SPECIFIED,
      $this->entity_name,
      $this->entity_field_text,
      // Field format.
      NULL,
    );
    $this
      ->assertEqual($strings, $target_strings, 'All contained strings found.');
  }

  /**
   * Recursive helper for getting all contained strings,
   * i.e. properties of type string.
   */
  public function getContainedStrings(TypedDataInterface $wrapper, $depth, array &$strings) {
    if ($wrapper
      ->getType() == 'string') {
      $strings[] = $wrapper
        ->getValue();
    }

    // Recurse until a certain depth is reached if possible.
    if ($depth < 7) {
      if ($wrapper instanceof \Drupal\Core\TypedData\ListInterface) {
        foreach ($wrapper as $item) {
          $this
            ->getContainedStrings($item, $depth + 1, $strings);
        }
      }
      elseif ($wrapper instanceof \Drupal\Core\TypedData\ComplexDataInterface) {
        foreach ($wrapper as $name => $property) {
          $this
            ->getContainedStrings($property, $depth + 1, $strings);
        }
      }
    }
  }

  /**
   * Tests getting processed property values via a computed property.
   */
  public function testComputedProperties() {

    // Make the test text field processed.
    $instance = field_info_instance('entity_test', 'field_test_text', 'entity_test');
    $instance['settings']['text_processing'] = 1;
    field_update_instance($instance);
    $entity = $this
      ->createTestEntity();
    $entity->field_test_text->value = "The <strong>text</strong> text to filter.";
    $entity->field_test_text->format = filter_default_format();
    $target = "<p>The &lt;strong&gt;text&lt;/strong&gt; text to filter.</p>\n";
    $this
      ->assertEqual($entity->field_test_text->processed, $target, 'Text is processed with the default filter.');

    // Save and load entity and make sure it still works.
    $entity
      ->save();
    $entity = entity_load('entity_test', $entity
      ->id());
    $this
      ->assertEqual($entity->field_test_text->processed, $target, 'Text is processed with the default filter.');
  }

}

Classes

Namesort descending Description
EntityFieldTest Tests Entity API base functionality.