You are here

class HtmlTest in Drupal 10

Same name and namespace in other branches
  1. 8 core/tests/Drupal/Tests/Component/Utility/HtmlTest.php \Drupal\Tests\Component\Utility\HtmlTest
  2. 9 core/tests/Drupal/Tests/Component/Utility/HtmlTest.php \Drupal\Tests\Component\Utility\HtmlTest

Tests \Drupal\Component\Utility\Html.

@group Common

@coversDefaultClass \Drupal\Component\Utility\Html

Hierarchy

  • class \Drupal\Tests\Component\Utility\HtmlTest extends \PHPUnit\Framework\TestCase

Expanded class hierarchy of HtmlTest

File

core/tests/Drupal/Tests/Component/Utility/HtmlTest.php, line 18

Namespace

Drupal\Tests\Component\Utility
View source
class HtmlTest extends TestCase {

  /**
   * {@inheritdoc}
   */
  protected function setUp() : void {
    parent::setUp();
    $property = new \ReflectionProperty('Drupal\\Component\\Utility\\Html', 'seenIdsInit');
    $property
      ->setAccessible(TRUE);
    $property
      ->setValue(NULL);
  }

  /**
   * Tests the Html::cleanCssIdentifier() method.
   *
   * @param string $expected
   *   The expected result.
   * @param string $source
   *   The string being transformed to an ID.
   * @param array|null $filter
   *   (optional) An array of string replacements to use on the identifier. If
   *   NULL, no filter will be passed and a default will be used.
   *
   * @dataProvider providerTestCleanCssIdentifier
   *
   * @covers ::cleanCssIdentifier
   */
  public function testCleanCssIdentifier($expected, $source, $filter = NULL) {
    if ($filter !== NULL) {
      $this
        ->assertSame($expected, Html::cleanCssIdentifier($source, $filter));
    }
    else {
      $this
        ->assertSame($expected, Html::cleanCssIdentifier($source));
    }
  }

  /**
   * Provides test data for testCleanCssIdentifier().
   *
   * @return array
   *   Test data.
   */
  public function providerTestCleanCssIdentifier() {
    $id1 = 'abcdefghijklmnopqrstuvwxyz_ABCDEFGHIJKLMNOPQRSTUVWXYZ-0123456789';
    $id2 = '¡¢£¤¥';
    $id3 = 'css__identifier__with__double__underscores';
    return [
      // Verify that no valid ASCII characters are stripped from the identifier.
      [
        $id1,
        $id1,
        [],
      ],
      // Verify that valid UTF-8 characters are not stripped from the identifier.
      [
        $id2,
        $id2,
        [],
      ],
      // Verify that double underscores are not stripped from the identifier.
      [
        $id3,
        $id3,
      ],
      // Verify that invalid characters (including non-breaking space) are
      // stripped from the identifier.
      [
        'invalididentifier',
        'invalid !"#$%&\'()*+,./:;<=>?@[\\]^`{|}~ identifier',
        [],
      ],
      // Verify that an identifier starting with a digit is replaced.
      [
        '_cssidentifier',
        '1cssidentifier',
        [],
      ],
      // Verify that an identifier starting with a hyphen followed by a digit is
      // replaced.
      [
        '__cssidentifier',
        '-1cssidentifier',
        [],
      ],
      // Verify that an identifier starting with two hyphens is replaced.
      [
        '__cssidentifier',
        '--cssidentifier',
        [],
      ],
      // Verify that passing double underscores as a filter is processed.
      [
        '_cssidentifier',
        '__cssidentifier',
        [
          '__' => '_',
        ],
      ],
    ];
  }

  /**
   * Tests that Html::getClass() cleans the class name properly.
   *
   * @coversDefaultClass ::getClass
   */
  public function testHtmlClass() {

    // Verify Drupal coding standards are enforced.
    $this
      ->assertSame('class-name--ü', Html::getClass('CLASS NAME_[Ü]'), 'Enforce Drupal coding standards.');

    // Test Html::getClass() handles Drupal\Component\Render\MarkupInterface
    // input.
    $markup = HtmlTestMarkup::create('CLASS_FROM_OBJECT');
    $this
      ->assertSame('class-from-object', Html::getClass($markup), 'Markup object is converted to CSS class.');
  }

  /**
   * Tests the Html::getUniqueId() method.
   *
   * @param string $expected
   *   The expected result.
   * @param string $source
   *   The string being transformed to an ID.
   * @param bool $reset
   *   (optional) If TRUE, reset the list of seen IDs. Defaults to FALSE.
   *
   * @dataProvider providerTestHtmlGetUniqueId
   *
   * @covers ::getUniqueId
   */
  public function testHtmlGetUniqueId($expected, $source, $reset = FALSE) {
    if ($reset) {
      Html::resetSeenIds();
    }
    $this
      ->assertSame($expected, Html::getUniqueId($source));
  }

  /**
   * Provides test data for testHtmlGetId().
   *
   * @return array
   *   Test data.
   */
  public function providerTestHtmlGetUniqueId() {

    // cSpell:disable
    $id = 'abcdefghijklmnopqrstuvwxyz-0123456789';
    return [
      // Verify that letters, digits, and hyphens are not stripped from the ID.
      [
        $id,
        $id,
      ],
      // Verify that invalid characters are stripped from the ID.
      [
        'invalididentifier',
        'invalid,./:@\\^`{Üidentifier',
      ],
      // Verify Drupal coding standards are enforced.
      [
        'id-name-1',
        'ID NAME_[1]',
      ],
      // Verify that a repeated ID is made unique.
      [
        'test-unique-id',
        'test-unique-id',
        TRUE,
      ],
      [
        'test-unique-id--2',
        'test-unique-id',
      ],
      [
        'test-unique-id--3',
        'test-unique-id',
      ],
    ];

    // cSpell:enable
  }

  /**
   * Tests the Html::getUniqueId() method.
   *
   * @param string $expected
   *   The expected result.
   * @param string $source
   *   The string being transformed to an ID.
   *
   * @dataProvider providerTestHtmlGetUniqueIdWithAjaxIds
   *
   * @covers ::getUniqueId
   */
  public function testHtmlGetUniqueIdWithAjaxIds($expected, $source) {
    Html::setIsAjax(TRUE);
    $id = Html::getUniqueId($source);

    // Note, we truncate two hyphens at the end.
    // @see \Drupal\Component\Utility\Html::getId()
    if (strpos($source, '--') !== FALSE) {
      $random_suffix = substr($id, strlen($source) + 1);
    }
    else {
      $random_suffix = substr($id, strlen($source) + 2);
    }
    $expected = $expected . $random_suffix;
    $this
      ->assertSame($expected, $id);
  }

  /**
   * Provides test data for testHtmlGetId().
   *
   * @return array
   *   Test data.
   */
  public function providerTestHtmlGetUniqueIdWithAjaxIds() {
    return [
      [
        'test-unique-id1--',
        'test-unique-id1',
      ],
      // Note, we truncate two hyphens at the end.
      // @see \Drupal\Component\Utility\Html::getId()
      [
        'test-unique-id1---',
        'test-unique-id1--',
      ],
      [
        'test-unique-id2--',
        'test-unique-id2',
      ],
    ];
  }

  /**
   * Tests the Html::getUniqueId() method.
   *
   * @param string $expected
   *   The expected result.
   * @param string $source
   *   The string being transformed to an ID.
   *
   * @dataProvider providerTestHtmlGetId
   *
   * @covers ::getId
   */
  public function testHtmlGetId($expected, $source) {
    Html::setIsAjax(FALSE);
    $this
      ->assertSame($expected, Html::getId($source));
  }

  /**
   * Provides test data for testHtmlGetId().
   *
   * @return array
   *   Test data.
   */
  public function providerTestHtmlGetId() {

    // cSpell:disable
    $id = 'abcdefghijklmnopqrstuvwxyz-0123456789';
    return [
      // Verify that letters, digits, and hyphens are not stripped from the ID.
      [
        $id,
        $id,
      ],
      // Verify that invalid characters are stripped from the ID.
      [
        'invalididentifier',
        'invalid,./:@\\^`{Üidentifier',
      ],
      // Verify Drupal coding standards are enforced.
      [
        'id-name-1',
        'ID NAME_[1]',
      ],
      // Verify that a repeated ID is made unique.
      [
        'test-unique-id',
        'test-unique-id',
      ],
      [
        'test-unique-id',
        'test-unique-id',
      ],
    ];

    // cSpell:enable
  }

  /**
   * Tests Html::decodeEntities().
   *
   * @dataProvider providerDecodeEntities
   * @covers ::decodeEntities
   */
  public function testDecodeEntities($text, $expected) {
    $this
      ->assertEquals($expected, Html::decodeEntities($text));
  }

  /**
   * Data provider for testDecodeEntities().
   *
   * @see testDecodeEntities()
   */
  public function providerDecodeEntities() {
    return [
      [
        'Drupal',
        'Drupal',
      ],
      [
        '<script>',
        '<script>',
      ],
      [
        '&lt;script&gt;',
        '<script>',
      ],
      [
        '&#60;script&#62;',
        '<script>',
      ],
      [
        '&amp;lt;script&amp;gt;',
        '&lt;script&gt;',
      ],
      [
        '"',
        '"',
      ],
      [
        '&#34;',
        '"',
      ],
      [
        '&amp;#34;',
        '&#34;',
      ],
      [
        '&quot;',
        '"',
      ],
      [
        '&amp;quot;',
        '&quot;',
      ],
      [
        "'",
        "'",
      ],
      [
        '&#39;',
        "'",
      ],
      [
        '&amp;#39;',
        '&#39;',
      ],
      [
        '©',
        '©',
      ],
      [
        '&copy;',
        '©',
      ],
      [
        '&#169;',
        '©',
      ],
      [
        '→',
        '→',
      ],
      [
        '&#8594;',
        '→',
      ],
      [
        '➼',
        '➼',
      ],
      [
        '&#10172;',
        '➼',
      ],
      [
        '&euro;',
        '€',
      ],
    ];
  }

  /**
   * Tests Html::escape().
   *
   * @dataProvider providerEscape
   * @covers ::escape
   */
  public function testEscape($expected, $text) {
    $this
      ->assertEquals($expected, Html::escape($text));
  }

  /**
   * Data provider for testEscape().
   *
   * @see testEscape()
   */
  public function providerEscape() {
    return [
      [
        'Drupal',
        'Drupal',
      ],
      [
        '&lt;script&gt;',
        '<script>',
      ],
      [
        '&amp;lt;script&amp;gt;',
        '&lt;script&gt;',
      ],
      [
        '&amp;#34;',
        '&#34;',
      ],
      [
        '&quot;',
        '"',
      ],
      [
        '&amp;quot;',
        '&quot;',
      ],
      [
        '&#039;',
        "'",
      ],
      [
        '&amp;#039;',
        '&#039;',
      ],
      [
        '©',
        '©',
      ],
      [
        '→',
        '→',
      ],
      [
        '➼',
        '➼',
      ],
      [
        '€',
        '€',
      ],
      [
        'Drup�al',
        "",
      ],
    ];
  }

  /**
   * Tests relationship between escaping and decoding HTML entities.
   *
   * @covers ::decodeEntities
   * @covers ::escape
   */
  public function testDecodeEntitiesAndEscape() {
    $string = "<em>répét&eacute;</em>";
    $escaped = Html::escape($string);
    $this
      ->assertSame('&lt;em&gt;répét&amp;eacute;&lt;/em&gt;', $escaped);
    $decoded = Html::decodeEntities($escaped);
    $this
      ->assertSame('<em>répét&eacute;</em>', $decoded);
    $decoded = Html::decodeEntities($decoded);
    $this
      ->assertSame('<em>répété</em>', $decoded);
    $escaped = Html::escape($decoded);
    $this
      ->assertSame('&lt;em&gt;répété&lt;/em&gt;', $escaped);
  }

  /**
   * Tests Html::serialize().
   *
   * Resolves an issue by where an empty DOMDocument object sent to serialization would
   * cause errors in getElementsByTagName() in the serialization function.
   *
   * @covers ::serialize
   */
  public function testSerialize() {
    $document = new \DOMDocument();
    $result = Html::serialize($document);
    $this
      ->assertSame('', $result);
  }

  /**
   * @covers ::transformRootRelativeUrlsToAbsolute
   * @dataProvider providerTestTransformRootRelativeUrlsToAbsolute
   */
  public function testTransformRootRelativeUrlsToAbsolute($html, $scheme_and_host, $expected_html) {
    $this
      ->assertSame($expected_html ?: $html, Html::transformRootRelativeUrlsToAbsolute($html, $scheme_and_host));
  }

  /**
   * @covers ::transformRootRelativeUrlsToAbsolute
   * @dataProvider providerTestTransformRootRelativeUrlsToAbsoluteAssertion
   */
  public function testTransformRootRelativeUrlsToAbsoluteAssertion($scheme_and_host) {
    $this
      ->expectException(\AssertionError::class);
    Html::transformRootRelativeUrlsToAbsolute('', $scheme_and_host);
  }

  /**
   * Provides test data for testTransformRootRelativeUrlsToAbsolute().
   *
   * @return array
   *   Test data.
   */
  public function providerTestTransformRootRelativeUrlsToAbsolute() {
    $data = [];

    // Random generator.
    $random = new Random();

    // One random tag name.
    $tag_name = strtolower($random
      ->name(8, TRUE));

    // A site installed either in the root of a domain or a subdirectory.
    $base_paths = [
      '/',
      '/subdir/' . $random
        ->name(8, TRUE) . '/',
    ];
    foreach ($base_paths as $base_path) {

      // The only attribute that has more than just a URL as its value, is
      // 'srcset', so special-case it.
      $data += [
        "{$tag_name}, srcset, {$base_path}: root-relative" => [
          "<{$tag_name} srcset=\"http://example.com{$base_path}already-absolute 200w, {$base_path}root-relative 300w\">root-relative test</{$tag_name}>",
          'http://example.com',
          "<{$tag_name} srcset=\"http://example.com{$base_path}already-absolute 200w, http://example.com{$base_path}root-relative 300w\">root-relative test</{$tag_name}>",
        ],
        "{$tag_name}, srcset, {$base_path}: protocol-relative" => [
          "<{$tag_name} srcset=\"http://example.com{$base_path}already-absolute 200w, //example.com{$base_path}protocol-relative 300w\">protocol-relative test</{$tag_name}>",
          'http://example.com',
          FALSE,
        ],
        "{$tag_name}, srcset, {$base_path}: absolute" => [
          "<{$tag_name} srcset=\"http://example.com{$base_path}already-absolute 200w, http://example.com{$base_path}absolute 300w\">absolute test</{$tag_name}>",
          'http://example.com',
          FALSE,
        ],
      ];
      foreach ([
        'href',
        'poster',
        'src',
        'cite',
        'data',
        'action',
        'formaction',
        'about',
      ] as $attribute) {
        $data += [
          "{$tag_name}, {$attribute}, {$base_path}: root-relative" => [
            "<{$tag_name} {$attribute}=\"{$base_path}root-relative\">root-relative test</{$tag_name}>",
            'http://example.com',
            "<{$tag_name} {$attribute}=\"http://example.com{$base_path}root-relative\">root-relative test</{$tag_name}>",
          ],
          "{$tag_name}, {$attribute}, {$base_path}: protocol-relative" => [
            "<{$tag_name} {$attribute}=\"//example.com{$base_path}protocol-relative\">protocol-relative test</{$tag_name}>",
            'http://example.com',
            FALSE,
          ],
          "{$tag_name}, {$attribute}, {$base_path}: absolute" => [
            "<{$tag_name} {$attribute}=\"http://example.com{$base_path}absolute\">absolute test</{$tag_name}>",
            'http://example.com',
            FALSE,
          ],
        ];
      }
    }
    return $data;
  }

  /**
   * Provides test data for testTransformRootRelativeUrlsToAbsoluteAssertion().
   *
   * @return array
   *   Test data.
   */
  public function providerTestTransformRootRelativeUrlsToAbsoluteAssertion() {
    return [
      'only relative path' => [
        'llama',
      ],
      'only root-relative path' => [
        '/llama',
      ],
      'host and path' => [
        'example.com/llama',
      ],
      'scheme, host and path' => [
        'http://example.com/llama',
      ],
    ];
  }

}

Members

Namesort descending Modifiers Type Description Overrides
HtmlTest::providerDecodeEntities public function Data provider for testDecodeEntities().
HtmlTest::providerEscape public function Data provider for testEscape().
HtmlTest::providerTestCleanCssIdentifier public function Provides test data for testCleanCssIdentifier().
HtmlTest::providerTestHtmlGetId public function Provides test data for testHtmlGetId().
HtmlTest::providerTestHtmlGetUniqueId public function Provides test data for testHtmlGetId().
HtmlTest::providerTestHtmlGetUniqueIdWithAjaxIds public function Provides test data for testHtmlGetId().
HtmlTest::providerTestTransformRootRelativeUrlsToAbsolute public function Provides test data for testTransformRootRelativeUrlsToAbsolute().
HtmlTest::providerTestTransformRootRelativeUrlsToAbsoluteAssertion public function Provides test data for testTransformRootRelativeUrlsToAbsoluteAssertion().
HtmlTest::setUp protected function
HtmlTest::testCleanCssIdentifier public function Tests the Html::cleanCssIdentifier() method.
HtmlTest::testDecodeEntities public function Tests Html::decodeEntities().
HtmlTest::testDecodeEntitiesAndEscape public function Tests relationship between escaping and decoding HTML entities.
HtmlTest::testEscape public function Tests Html::escape().
HtmlTest::testHtmlClass public function Tests that Html::getClass() cleans the class name properly.
HtmlTest::testHtmlGetId public function Tests the Html::getUniqueId() method.
HtmlTest::testHtmlGetUniqueId public function Tests the Html::getUniqueId() method.
HtmlTest::testHtmlGetUniqueIdWithAjaxIds public function Tests the Html::getUniqueId() method.
HtmlTest::testSerialize public function Tests Html::serialize().
HtmlTest::testTransformRootRelativeUrlsToAbsolute public function @covers ::transformRootRelativeUrlsToAbsolute @dataProvider providerTestTransformRootRelativeUrlsToAbsolute
HtmlTest::testTransformRootRelativeUrlsToAbsoluteAssertion public function @covers ::transformRootRelativeUrlsToAbsolute @dataProvider providerTestTransformRootRelativeUrlsToAbsoluteAssertion