Add Schema.org ItemList support to YouTube index page
All checks were successful
Deploy to Server / deploy (push) Successful in 12s

This commit is contained in:
Joakim Hellsén 2026-03-16 23:47:07 +01:00
commit 60ac907163
Signed by: Joakim Hellsén
SSH key fingerprint: SHA256:/9h/CsExpFp+PRhsfA0xznFx2CGfTT5R/kpuFfUgEQk
2 changed files with 75 additions and 1 deletions

View file

@ -1,4 +1,6 @@
import json
from typing import TYPE_CHECKING
from typing import Any
from django.test import TestCase
from django.urls import reverse
@ -41,6 +43,53 @@ class YouTubeIndexViewTest(TestCase):
'Blizzard, Fortnite, Riot Games, and more." />'
) in content
def test_index_schema_data_is_valid_itemlist(self) -> None:
"""The page should include a valid Schema.org ItemList in the JSON-LD context."""
response: _MonkeyPatchedWSGIResponse = self.client.get(reverse("youtube:index"))
assert response.context is not None
assert "schema_data" in response.context
schema: dict[str, Any] = json.loads(response.context["schema_data"])
assert schema["@context"] == "https://schema.org"
assert schema["@type"] == "ItemList"
assert schema["name"] == "YouTube channels with rewards"
assert "itemListElement" in schema
items: list[dict[str, Any]] = schema["itemListElement"]
assert len(items) > 0
# Every entry must be a ListItem wrapping an Organization
for item in items:
assert item["@type"] == "ListItem"
assert "position" in item
org: dict[str, Any] = item["item"]
assert org["@type"] == "Organization"
assert "name" in org
assert isinstance(org["sameAs"], list)
assert len(org["sameAs"]) > 0
def test_index_schema_data_includes_known_orgs(self) -> None:
"""The Schema.org ItemList should contain entries for known organizations."""
response: _MonkeyPatchedWSGIResponse = self.client.get(reverse("youtube:index"))
schema: dict[str, Any] = json.loads(response.context["schema_data"]) # type: ignore[index]
org_names: list[str] = [
item["item"]["name"] for item in schema["itemListElement"]
]
assert "Activision (Call of Duty)" in org_names
assert "Battle.net / Blizzard" in org_names
assert "Riot Games" in org_names
assert "Epic Games" in org_names
def test_index_schema_data_org_same_as_are_youtube_urls(self) -> None:
"""Each Organization's sameAs values should be YouTube URLs."""
response: _MonkeyPatchedWSGIResponse = self.client.get(reverse("youtube:index"))
schema: dict[str, Any] = json.loads(response.context["schema_data"]) # type: ignore[index]
for list_item in schema["itemListElement"]:
for url in list_item["item"]["sameAs"]:
assert url.startswith("https://www.youtube.com/")
def test_index_returns_200(self) -> None:
"""The YouTube index page should return HTTP 200."""
response: _MonkeyPatchedWSGIResponse = self.client.get(reverse("youtube:index"))

View file

@ -1,5 +1,7 @@
import json
from collections import defaultdict
from typing import TYPE_CHECKING
from typing import Any
from django.shortcuts import render
@ -116,9 +118,32 @@ def index(request: HttpRequest) -> HttpResponse:
"channels": sorted_items,
})
context: dict[str, str | list[dict[str, str | list[dict[str, str]]]]] = {
list_items: list[dict[str, Any]] = [
{
"@type": "ListItem",
"position": position,
"item": {
"@type": "Organization",
"name": group["organization"],
"sameAs": [ch["url"] for ch in group["channels"]], # type: ignore[index]
},
}
for position, group in enumerate(organization_groups, start=1)
]
schema: dict[str, Any] = {
"@context": "https://schema.org",
"@type": "ItemList",
"name": PAGE_TITLE,
"description": PAGE_DESCRIPTION,
"url": request.build_absolute_uri(),
"itemListElement": list_items,
}
context: dict[str, Any] = {
"page_title": PAGE_TITLE,
"page_description": PAGE_DESCRIPTION,
"organization_groups": organization_groups,
"schema_data": json.dumps(schema),
}
return render(request=request, template_name="youtube/index.html", context=context)