2021-10-13 13:00:07 +02:00
|
|
|
|
#
|
2023-11-21 21:29:58 +01:00
|
|
|
|
# This file is licensed under the Affero General Public License (AGPL) version 3.
|
|
|
|
|
#
|
2024-01-23 12:26:48 +01:00
|
|
|
|
# Copyright 2021 The Matrix.org Foundation C.I.C.
|
2023-11-21 21:29:58 +01:00
|
|
|
|
# Copyright (C) 2023 New Vector, Ltd
|
|
|
|
|
#
|
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
|
|
# it under the terms of the GNU Affero General Public License as
|
|
|
|
|
# published by the Free Software Foundation, either version 3 of the
|
|
|
|
|
# License, or (at your option) any later version.
|
|
|
|
|
#
|
|
|
|
|
# See the GNU Affero General Public License for more details:
|
|
|
|
|
# <https://www.gnu.org/licenses/agpl-3.0.html>.
|
|
|
|
|
#
|
|
|
|
|
# Originally licensed under the Apache License, Version 2.0:
|
|
|
|
|
# <http://www.apache.org/licenses/LICENSE-2.0>.
|
|
|
|
|
#
|
|
|
|
|
# [This file includes modifications made by New Vector Limited]
|
2021-10-13 13:00:07 +02:00
|
|
|
|
#
|
|
|
|
|
#
|
|
|
|
|
|
|
|
|
|
import json
|
2024-10-17 17:05:00 +02:00
|
|
|
|
from typing import Any
|
2021-10-13 13:00:07 +02:00
|
|
|
|
|
2022-10-07 15:29:43 +02:00
|
|
|
|
from parameterized import parameterized
|
|
|
|
|
|
2021-10-13 13:00:07 +02:00
|
|
|
|
from twisted.test.proto_helpers import MemoryReactor
|
|
|
|
|
|
2023-02-27 14:26:05 +01:00
|
|
|
|
from synapse.media.oembed import OEmbedProvider, OEmbedResult
|
2021-10-13 13:00:07 +02:00
|
|
|
|
from synapse.server import HomeServer
|
|
|
|
|
from synapse.types import JsonDict
|
|
|
|
|
from synapse.util import Clock
|
|
|
|
|
|
|
|
|
|
from tests.unittest import HomeserverTestCase
|
|
|
|
|
|
2022-10-07 15:29:43 +02:00
|
|
|
|
try:
|
|
|
|
|
import lxml
|
|
|
|
|
except ImportError:
|
2023-05-31 19:06:57 +02:00
|
|
|
|
lxml = None # type: ignore[assignment]
|
2022-10-07 15:29:43 +02:00
|
|
|
|
|
2021-10-13 13:00:07 +02:00
|
|
|
|
|
|
|
|
|
class OEmbedTests(HomeserverTestCase):
|
2022-10-07 15:29:43 +02:00
|
|
|
|
if not lxml:
|
|
|
|
|
skip = "url preview feature requires lxml"
|
|
|
|
|
|
2022-03-03 17:05:44 +01:00
|
|
|
|
def prepare(self, reactor: MemoryReactor, clock: Clock, hs: HomeServer) -> None:
|
|
|
|
|
self.oembed = OEmbedProvider(hs)
|
2021-10-13 13:00:07 +02:00
|
|
|
|
|
2022-03-03 17:05:44 +01:00
|
|
|
|
def parse_response(self, response: JsonDict) -> OEmbedResult:
|
2021-10-13 13:00:07 +02:00
|
|
|
|
return self.oembed.parse_oembed_response(
|
|
|
|
|
"https://test", json.dumps(response).encode("utf-8")
|
|
|
|
|
)
|
|
|
|
|
|
2022-03-03 17:05:44 +01:00
|
|
|
|
def test_version(self) -> None:
|
2021-10-13 13:00:07 +02:00
|
|
|
|
"""Accept versions that are similar to 1.0 as a string or int (or missing)."""
|
2024-10-17 17:05:00 +02:00
|
|
|
|
version: Any
|
2021-10-13 13:00:07 +02:00
|
|
|
|
for version in ("1.0", 1.0, 1):
|
2022-10-07 15:29:43 +02:00
|
|
|
|
result = self.parse_response({"version": version})
|
2021-10-13 13:00:07 +02:00
|
|
|
|
# An empty Open Graph response is an error, ensure the URL is included.
|
|
|
|
|
self.assertIn("og:url", result.open_graph_result)
|
|
|
|
|
|
|
|
|
|
# A missing version should be treated as 1.0.
|
|
|
|
|
result = self.parse_response({"type": "link"})
|
|
|
|
|
self.assertIn("og:url", result.open_graph_result)
|
|
|
|
|
|
|
|
|
|
# Invalid versions should be rejected.
|
|
|
|
|
for version in ("2.0", "1", 1.1, 0, None, {}, []):
|
|
|
|
|
result = self.parse_response({"version": version, "type": "link"})
|
|
|
|
|
# An empty Open Graph response is an error, ensure the URL is included.
|
|
|
|
|
self.assertEqual({}, result.open_graph_result)
|
2022-10-07 15:29:43 +02:00
|
|
|
|
|
|
|
|
|
def test_cache_age(self) -> None:
|
|
|
|
|
"""Ensure a cache-age is parsed properly."""
|
2024-10-17 17:05:00 +02:00
|
|
|
|
cache_age: Any
|
2022-10-07 15:29:43 +02:00
|
|
|
|
# Correct-ish cache ages are allowed.
|
|
|
|
|
for cache_age in ("1", 1.0, 1):
|
|
|
|
|
result = self.parse_response({"cache_age": cache_age})
|
|
|
|
|
self.assertEqual(result.cache_age, 1000)
|
|
|
|
|
|
|
|
|
|
# Invalid cache ages are ignored.
|
|
|
|
|
for cache_age in ("invalid", {}):
|
|
|
|
|
result = self.parse_response({"cache_age": cache_age})
|
|
|
|
|
self.assertIsNone(result.cache_age)
|
|
|
|
|
|
|
|
|
|
# Cache age is optional.
|
|
|
|
|
result = self.parse_response({})
|
|
|
|
|
self.assertIsNone(result.cache_age)
|
|
|
|
|
|
|
|
|
|
@parameterized.expand(
|
|
|
|
|
[
|
|
|
|
|
("title", "title"),
|
|
|
|
|
("provider_name", "site_name"),
|
|
|
|
|
("thumbnail_url", "image"),
|
|
|
|
|
],
|
|
|
|
|
name_func=lambda func, num, p: f"{func.__name__}_{p.args[0]}",
|
|
|
|
|
)
|
|
|
|
|
def test_property(self, oembed_property: str, open_graph_property: str) -> None:
|
|
|
|
|
"""Test properties which must be strings."""
|
|
|
|
|
result = self.parse_response({oembed_property: "test"})
|
|
|
|
|
self.assertIn(f"og:{open_graph_property}", result.open_graph_result)
|
|
|
|
|
self.assertEqual(result.open_graph_result[f"og:{open_graph_property}"], "test")
|
|
|
|
|
|
|
|
|
|
result = self.parse_response({oembed_property: 1})
|
|
|
|
|
self.assertNotIn(f"og:{open_graph_property}", result.open_graph_result)
|
|
|
|
|
|
|
|
|
|
def test_author_name(self) -> None:
|
|
|
|
|
"""Test the author_name property."""
|
|
|
|
|
result = self.parse_response({"author_name": "test"})
|
|
|
|
|
self.assertEqual(result.author_name, "test")
|
|
|
|
|
|
|
|
|
|
result = self.parse_response({"author_name": 1})
|
|
|
|
|
self.assertIsNone(result.author_name)
|
|
|
|
|
|
|
|
|
|
def test_rich(self) -> None:
|
|
|
|
|
"""Test a type of rich."""
|
|
|
|
|
result = self.parse_response({"html": "test<img src='foo'>", "type": "rich"})
|
|
|
|
|
self.assertIn("og:description", result.open_graph_result)
|
|
|
|
|
self.assertIn("og:image", result.open_graph_result)
|
|
|
|
|
self.assertEqual(result.open_graph_result["og:description"], "test")
|
|
|
|
|
self.assertEqual(result.open_graph_result["og:image"], "foo")
|
|
|
|
|
|
|
|
|
|
result = self.parse_response({"type": "rich"})
|
|
|
|
|
self.assertNotIn("og:description", result.open_graph_result)
|
|
|
|
|
|
|
|
|
|
result = self.parse_response({"html": 1, "type": "rich"})
|
|
|
|
|
self.assertNotIn("og:description", result.open_graph_result)
|
|
|
|
|
|
|
|
|
|
def test_photo(self) -> None:
|
|
|
|
|
"""Test a type of photo."""
|
|
|
|
|
result = self.parse_response({"url": "test", "type": "photo"})
|
|
|
|
|
self.assertIn("og:image", result.open_graph_result)
|
|
|
|
|
self.assertEqual(result.open_graph_result["og:image"], "test")
|
|
|
|
|
|
|
|
|
|
result = self.parse_response({"type": "photo"})
|
|
|
|
|
self.assertNotIn("og:image", result.open_graph_result)
|
|
|
|
|
|
|
|
|
|
result = self.parse_response({"url": 1, "type": "photo"})
|
|
|
|
|
self.assertNotIn("og:image", result.open_graph_result)
|
|
|
|
|
|
|
|
|
|
def test_video(self) -> None:
|
|
|
|
|
"""Test a type of video."""
|
|
|
|
|
result = self.parse_response({"html": "test", "type": "video"})
|
|
|
|
|
self.assertIn("og:type", result.open_graph_result)
|
|
|
|
|
self.assertEqual(result.open_graph_result["og:type"], "video.other")
|
|
|
|
|
self.assertIn("og:description", result.open_graph_result)
|
|
|
|
|
self.assertEqual(result.open_graph_result["og:description"], "test")
|
|
|
|
|
|
|
|
|
|
result = self.parse_response({"type": "video"})
|
|
|
|
|
self.assertIn("og:type", result.open_graph_result)
|
|
|
|
|
self.assertEqual(result.open_graph_result["og:type"], "video.other")
|
|
|
|
|
self.assertNotIn("og:description", result.open_graph_result)
|
|
|
|
|
|
|
|
|
|
result = self.parse_response({"url": 1, "type": "video"})
|
|
|
|
|
self.assertIn("og:type", result.open_graph_result)
|
|
|
|
|
self.assertEqual(result.open_graph_result["og:type"], "video.other")
|
|
|
|
|
self.assertNotIn("og:description", result.open_graph_result)
|
|
|
|
|
|
|
|
|
|
def test_link(self) -> None:
|
|
|
|
|
"""Test type of link."""
|
|
|
|
|
result = self.parse_response({"type": "link"})
|
|
|
|
|
self.assertIn("og:type", result.open_graph_result)
|
|
|
|
|
self.assertEqual(result.open_graph_result["og:type"], "website")
|
2023-01-09 15:22:02 +01:00
|
|
|
|
|
|
|
|
|
def test_title_html_entities(self) -> None:
|
|
|
|
|
"""Test HTML entities in title"""
|
|
|
|
|
result = self.parse_response(
|
|
|
|
|
{"title": "Why JSON isn’t a Good Configuration Language"}
|
|
|
|
|
)
|
|
|
|
|
self.assertEqual(
|
|
|
|
|
result.open_graph_result["og:title"],
|
|
|
|
|
"Why JSON isn’t a Good Configuration Language",
|
|
|
|
|
)
|