function OEmbedSourceTest::testThumbnailUri

Same name and namespace in other branches
  1. 9 core/modules/media/tests/src/Kernel/OEmbedSourceTest.php \Drupal\Tests\media\Kernel\OEmbedSourceTest::testThumbnailUri()
  2. 10 core/modules/media/tests/src/Kernel/OEmbedSourceTest.php \Drupal\Tests\media\Kernel\OEmbedSourceTest::testThumbnailUri()

Tests that remote thumbnails are downloaded correctly.

@covers ::getLocalThumbnailUri

@dataProvider providerThumbnailUri

Parameters

string $remote_thumbnail_url: The URL of the remote thumbnail. This will be wired up to a mocked response containing the data from core/misc/druplicon.png.

array[] $thumbnail_headers: If the thumbnail's file extension cannot be determined from its URL, an attempt will be made to derive the extension from the response's Content-Type header. This array contains the headers that should be returned with the thumbnail response, where the keys are header names and the values are arrays of strings.

string $expected_extension: The extension that the downloaded thumbnail should have.

File

core/modules/media/tests/src/Kernel/OEmbedSourceTest.php, line 110

Class

OEmbedSourceTest
@coversDefaultClass <a href="/api/drupal/core%21modules%21media%21src%21Plugin%21media%21Source%21OEmbed.php/class/OEmbed/11.x" title="Provides a media source plugin for oEmbed resources." class="local">\Drupal\media\Plugin\media\Source\OEmbed</a>

Namespace

Drupal\Tests\media\Kernel

Code

public function testThumbnailUri(string $remote_thumbnail_url, array $thumbnail_headers, string $expected_extension) : void {
    // Create a fake resource with the given thumbnail URL.
    $resource = Resource::rich('<html></html>', 16, 16, NULL, 'Test resource', NULL, NULL, NULL, $remote_thumbnail_url, 16, 16);
    $thumbnail_url = $resource->getThumbnailUrl()
        ->toString();
    // There's no need to resolve the resource URL in this test; we just need
    // to fetch the resource.
    $this->container
        ->set('media.oembed.url_resolver', $this->prophesize(UrlResolverInterface::class)
        ->reveal());
    // Mock the resource fetcher so that it will return our fake resource.
    $resource_fetcher = $this->prophesize(ResourceFetcherInterface::class);
    $resource_fetcher->fetchResource(Argument::any())
        ->willReturn($resource);
    $this->container
        ->set('media.oembed.resource_fetcher', $resource_fetcher->reveal());
    // The source plugin will try to fetch the remote thumbnail, so mock the
    // HTTP client to ensure that request returns a response with some valid
    // image data.
    $data = Utils::tryFopen($this->getDrupalRoot() . '/core/misc/druplicon.png', 'r');
    $response = new Response(200, $thumbnail_headers, Utils::streamFor($data));
    $handler = new MockHandler([
        $response,
    ]);
    $client = new Client([
        'handler' => new HandlerStack($handler),
    ]);
    $this->container
        ->set('http_client', $client);
    $media_type = $this->createMediaType('oembed:video');
    $source = $media_type->getSource();
    // Add some HTML to the global site slogan, and use the site:slogan token in
    // the thumbnail path, in order to prove that the final thumbnail path is
    // stripped of HTML tags, and XML entities are decoded.
    $this->config('system.site')
        ->set('slogan', '<h1>this&amp;that</h1>')
        ->save();
    $configuration = $source->getConfiguration();
    $configuration['thumbnails_directory'] .= '/[site:slogan]';
    $source->setConfiguration($configuration);
    $media_type->save();
    $media = Media::create([
        'bundle' => $media_type->id(),
        $source->getSourceFieldDefinition($media_type)
            ->getName() => $this->randomString(),
    ]);
    $media->save();
    // The thumbnail directory should include the current date, as per the
    // default configuration of the oEmbed source plugin.
    $date = date('Y-m', $this->container
        ->get('datetime.time')
        ->getRequestTime());
    // The thumbnail should have a file extension, even if it wasn't in the URL.
    $expected_uri = "public://oembed_thumbnails/{$date}/this&that/" . Crypt::hashBase64($thumbnail_url) . ".{$expected_extension}";
    $this->assertSame($expected_uri, $source->getMetadata($media, 'thumbnail_uri'));
    // Even if we get the thumbnail_uri more than once, it should only be
    // downloaded once. The HTTP client will throw an exception if we try to
    // do another request without setting up another response.
    $source->getMetadata($media, 'thumbnail_uri');
    // The downloaded thumbnail should be usable by the image toolkit.
    $this->assertFileExists($expected_uri);
    
    /** @var \Drupal\Core\Image\Image $image */
    $image = $this->container
        ->get('image.factory')
        ->get($expected_uri);
    $this->assertTrue($image->isValid());
    // Check that the current date token as per the default configuration of the
    // oEmbed source plugin doesn't make a render context uncacheable.
    $context = new RenderContext();
    \Drupal::service('renderer')->executeInRenderContext($context, function () use ($source, $media) {
        return $source->getMetadata($media, 'thumbnail_uri');
    });
    
    /** @var \Drupal\Core\Render\BubbleableMetadata $bubbleable_metadata */
    $bubbleable_metadata = $context->pop();
    $this->assertSame(Cache::PERMANENT, $bubbleable_metadata->getCacheMaxAge());
}

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