From c7a9e903e946aec4271078af4e990b336be74998 Mon Sep 17 00:00:00 2001 From: "Ralph J. Smit" <59207045+ralphjsmit@users.noreply.github.com> Date: Sat, 15 Jun 2024 16:21:44 +0200 Subject: [PATCH] fix: do not escape image URLs for OG or Twitter --- src/Support/OpenGraphTag.php | 5 +++-- src/Support/TwitterCardTag.php | 5 +++-- src/Tags/OpenGraphTags.php | 3 ++- src/Tags/TwitterCard/Summary.php | 3 ++- src/Tags/TwitterCard/SummaryLargeImage.php | 3 ++- tests/Feature/Tags/OpenGraphTagsTest.php | 18 ++++++++++++++++ .../Tags/TwitterCardSummaryTagsTest.php | 21 +++++++++++++++++++ 7 files changed, 51 insertions(+), 7 deletions(-) diff --git a/src/Support/OpenGraphTag.php b/src/Support/OpenGraphTag.php index 1cd2a2e..f975866 100644 --- a/src/Support/OpenGraphTag.php +++ b/src/Support/OpenGraphTag.php @@ -3,6 +3,7 @@ namespace RalphJSmit\Laravel\SEO\Support; use Illuminate\Support\Collection; +use Illuminate\Support\HtmlString; class OpenGraphTag extends Tag { @@ -10,13 +11,13 @@ class OpenGraphTag extends Tag public function __construct( string $property, - string $content, + string | HtmlString $content, ) { $this->attributes['property'] = $property; $this->attributes['content'] = $content; $this->attributesPipeline[] = function (Collection $collection) { - return $collection->mapWithKeys(function ($value, $key) { + return $collection->mapWithKeys(function (mixed $value, string $key) { if ($key === 'property') { $value = 'og:' . $value; } diff --git a/src/Support/TwitterCardTag.php b/src/Support/TwitterCardTag.php index a8e8da6..1c73c22 100644 --- a/src/Support/TwitterCardTag.php +++ b/src/Support/TwitterCardTag.php @@ -3,6 +3,7 @@ namespace RalphJSmit\Laravel\SEO\Support; use Illuminate\Support\Collection; +use Illuminate\Support\HtmlString; class TwitterCardTag extends Tag { @@ -10,13 +11,13 @@ class TwitterCardTag extends Tag public function __construct( string $name, - string $content, + string | HtmlString $content, ) { $this->attributes['name'] = $name; $this->attributes['content'] = $content; $this->attributesPipeline[] = function (Collection $collection) { - return $collection->mapWithKeys(function ($value, $key) { + return $collection->mapWithKeys(function (mixed $value, string $key) { if ($key === 'name') { $value = 'twitter:' . $value; } diff --git a/src/Tags/OpenGraphTags.php b/src/Tags/OpenGraphTags.php index a172d72..1c2e4e9 100644 --- a/src/Tags/OpenGraphTags.php +++ b/src/Tags/OpenGraphTags.php @@ -4,6 +4,7 @@ use Illuminate\Contracts\Support\Renderable; use Illuminate\Support\Collection; +use Illuminate\Support\HtmlString; use RalphJSmit\Laravel\SEO\Support\MetaContentTag; use RalphJSmit\Laravel\SEO\Support\OpenGraphTag; use RalphJSmit\Laravel\SEO\Support\RenderableCollection; @@ -32,7 +33,7 @@ public static function initialize(SEOData $SEOData): static } if ($SEOData->image) { - $collection->push(new OpenGraphTag('image', $SEOData->image)); + $collection->push(new OpenGraphTag('image', new HtmlString($SEOData->image))); if ($SEOData->imageMeta) { $collection diff --git a/src/Tags/TwitterCard/Summary.php b/src/Tags/TwitterCard/Summary.php index f8dd680..e5440f1 100644 --- a/src/Tags/TwitterCard/Summary.php +++ b/src/Tags/TwitterCard/Summary.php @@ -4,6 +4,7 @@ use Illuminate\Contracts\Support\Renderable; use Illuminate\Support\Collection; +use Illuminate\Support\HtmlString; use RalphJSmit\Laravel\SEO\Support\RenderableCollection; use RalphJSmit\Laravel\SEO\Support\SEOData; use RalphJSmit\Laravel\SEO\Support\TwitterCardTag; @@ -37,7 +38,7 @@ public static function initialize(SEOData $SEOData): static $collection->push(new TwitterCardTag('card', 'summary')); if ($SEOData->image) { - $collection->push(new TwitterCardTag('image', $SEOData->image)); + $collection->push(new TwitterCardTag('image', new HtmlString($SEOData->image))); if ($SEOData->imageMeta) { $collection diff --git a/src/Tags/TwitterCard/SummaryLargeImage.php b/src/Tags/TwitterCard/SummaryLargeImage.php index fba8032..1e64aba 100644 --- a/src/Tags/TwitterCard/SummaryLargeImage.php +++ b/src/Tags/TwitterCard/SummaryLargeImage.php @@ -4,6 +4,7 @@ use Illuminate\Contracts\Support\Renderable; use Illuminate\Support\Collection; +use Illuminate\Support\HtmlString; use RalphJSmit\Laravel\SEO\Support\RenderableCollection; use RalphJSmit\Laravel\SEO\Support\SEOData; use RalphJSmit\Laravel\SEO\Support\TwitterCardTag; @@ -37,7 +38,7 @@ public static function initialize(SEOData $SEOData): static $collection->push(new TwitterCardTag('card', 'summary_large_image')); if ($SEOData->image) { - $collection->push(new TwitterCardTag('image', $SEOData->image)); + $collection->push(new TwitterCardTag('image', new HtmlString($SEOData->image))); if ($SEOData->imageMeta) { $collection diff --git a/tests/Feature/Tags/OpenGraphTagsTest.php b/tests/Feature/Tags/OpenGraphTagsTest.php index a0697e5..48f815d 100644 --- a/tests/Feature/Tags/OpenGraphTagsTest.php +++ b/tests/Feature/Tags/OpenGraphTagsTest.php @@ -153,3 +153,21 @@ get(route('seo.test-page', ['page' => $page])) ->assertSee('', false); }); + +it('will not escape the image URL query parameters', function () { + config()->set('seo.title.suffix', ' | Laravel SEO'); + config()->set('seo.description.fallback', 'Fallback description'); + + $page = Page::create(); + + $page::$overrides = [ + 'title' => 'Test Page', + 'image' => $url = 'https://website.test/images/xSVtl6ZF7fNuZIoXkZbzI2EzoAD.jpg?h=800&fit=contain&q=80&fm=webp', + ]; + + get(route('seo.test-page', ['page' => $page])) + ->assertSee('', false) + ->assertDontSee('assertDontSee('assertDontSee('twitter:site'); // We should not display an empty '@' username. +}); diff --git a/tests/Feature/Tags/TwitterCardSummaryTagsTest.php b/tests/Feature/Tags/TwitterCardSummaryTagsTest.php index 1d35130..e075a66 100644 --- a/tests/Feature/Tags/TwitterCardSummaryTagsTest.php +++ b/tests/Feature/Tags/TwitterCardSummaryTagsTest.php @@ -79,6 +79,27 @@ ['summary_large_image', 'images/twitter-3597x1799.jpg', '3597', '1799'], ]); +it('will not escape the image URL query parameters', function () { + config()->set('seo.title.suffix', ' | Laravel SEO'); + config()->set('seo.description.fallback', 'Fallback description'); + + $page = Page::create(); + + $page::$overrides = [ + 'title' => 'Test Page', + 'image' => $url = 'https://website.test/images/xSVtl6ZF7fNuZIoXkZbzI2EzoAD.jpg?h=800&fit=contain&q=80&fm=webp', + ]; + + get(route('seo.test-page', ['page' => $page])) + ->assertSee('', false) + ->assertSee('', false) + ->assertSee('', false) + ->assertSee('', false) + ->assertDontSee('assertDontSee('assertDontSee('twitter:site'); // We should not display an empty '@' username. +}); + it('will not render the Twitter Card summary_large_image for too large or small images', function (string $imagePath) { $page = Page::create();