2021-09-01 00:37:07 +02:00
|
|
|
# Copyright 2021 The Matrix.org Foundation C.I.C.
|
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
|
|
|
import logging
|
2021-09-21 18:09:57 +02:00
|
|
|
import urllib.parse
|
2021-09-22 15:45:20 +02:00
|
|
|
from typing import TYPE_CHECKING, List, Optional
|
2021-09-01 00:37:07 +02:00
|
|
|
|
|
|
|
import attr
|
|
|
|
|
2021-09-21 18:09:57 +02:00
|
|
|
from synapse.types import JsonDict
|
|
|
|
from synapse.util import json_decoder
|
2021-09-01 00:37:07 +02:00
|
|
|
|
|
|
|
if TYPE_CHECKING:
|
2021-09-22 15:45:20 +02:00
|
|
|
from lxml import etree
|
|
|
|
|
2021-09-01 00:37:07 +02:00
|
|
|
from synapse.server import HomeServer
|
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2021-09-21 18:09:57 +02:00
|
|
|
@attr.s(slots=True, frozen=True, auto_attribs=True)
|
2021-09-01 00:37:07 +02:00
|
|
|
class OEmbedResult:
|
2021-09-21 18:09:57 +02:00
|
|
|
# The Open Graph result (converted from the oEmbed result).
|
|
|
|
open_graph_result: JsonDict
|
2021-09-22 15:45:20 +02:00
|
|
|
# Number of milliseconds to cache the content, according to the oEmbed response.
|
2021-09-21 18:09:57 +02:00
|
|
|
#
|
|
|
|
# This will be None if no cache-age is provided in the oEmbed response (or
|
|
|
|
# if the oEmbed response cannot be turned into an Open Graph response).
|
|
|
|
cache_age: Optional[int]
|
2021-09-01 00:37:07 +02:00
|
|
|
|
|
|
|
|
|
|
|
class OEmbedProvider:
|
|
|
|
"""
|
|
|
|
A helper for accessing oEmbed content.
|
|
|
|
|
|
|
|
It can be used to check if a URL should be accessed via oEmbed and for
|
|
|
|
requesting/parsing oEmbed content.
|
|
|
|
"""
|
|
|
|
|
2021-10-13 13:00:07 +02:00
|
|
|
def __init__(self, hs: "HomeServer"):
|
2021-09-01 00:37:07 +02:00
|
|
|
self._oembed_patterns = {}
|
|
|
|
for oembed_endpoint in hs.config.oembed.oembed_patterns:
|
2021-09-08 13:17:52 +02:00
|
|
|
api_endpoint = oembed_endpoint.api_endpoint
|
|
|
|
|
|
|
|
# Only JSON is supported at the moment. This could be declared in
|
|
|
|
# the formats field. Otherwise, if the endpoint ends in .xml assume
|
|
|
|
# it doesn't support JSON.
|
|
|
|
if (
|
|
|
|
oembed_endpoint.formats is not None
|
|
|
|
and "json" not in oembed_endpoint.formats
|
|
|
|
) or api_endpoint.endswith(".xml"):
|
|
|
|
logger.info(
|
|
|
|
"Ignoring oEmbed endpoint due to not supporting JSON: %s",
|
|
|
|
api_endpoint,
|
|
|
|
)
|
|
|
|
continue
|
|
|
|
|
|
|
|
# Iterate through each URL pattern and point it to the endpoint.
|
2021-09-01 00:37:07 +02:00
|
|
|
for pattern in oembed_endpoint.url_patterns:
|
2021-09-08 13:17:52 +02:00
|
|
|
self._oembed_patterns[pattern] = api_endpoint
|
2021-09-01 00:37:07 +02:00
|
|
|
|
|
|
|
def get_oembed_url(self, url: str) -> Optional[str]:
|
|
|
|
"""
|
|
|
|
Check whether the URL should be downloaded as oEmbed content instead.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
url: The URL to check.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
A URL to use instead or None if the original URL should be used.
|
|
|
|
"""
|
|
|
|
for url_pattern, endpoint in self._oembed_patterns.items():
|
|
|
|
if url_pattern.fullmatch(url):
|
2021-09-21 18:09:57 +02:00
|
|
|
# TODO Specify max height / width.
|
|
|
|
|
|
|
|
# Note that only the JSON format is supported, some endpoints want
|
|
|
|
# this in the URL, others want it as an argument.
|
|
|
|
endpoint = endpoint.replace("{format}", "json")
|
|
|
|
|
|
|
|
args = {"url": url, "format": "json"}
|
|
|
|
query_str = urllib.parse.urlencode(args, True)
|
|
|
|
return f"{endpoint}?{query_str}"
|
2021-09-01 00:37:07 +02:00
|
|
|
|
|
|
|
# No match.
|
|
|
|
return None
|
|
|
|
|
2021-10-08 20:14:42 +02:00
|
|
|
def autodiscover_from_html(self, tree: "etree.Element") -> Optional[str]:
|
|
|
|
"""
|
|
|
|
Search an HTML document for oEmbed autodiscovery information.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
tree: The parsed HTML body.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
The URL to use for oEmbed information, or None if no URL was found.
|
|
|
|
"""
|
|
|
|
# Search for link elements with the proper rel and type attributes.
|
|
|
|
for tag in tree.xpath(
|
|
|
|
"//link[@rel='alternate'][@type='application/json+oembed']"
|
|
|
|
):
|
|
|
|
if "href" in tag.attrib:
|
|
|
|
return tag.attrib["href"]
|
|
|
|
|
|
|
|
# Some providers (e.g. Flickr) use alternative instead of alternate.
|
|
|
|
for tag in tree.xpath(
|
|
|
|
"//link[@rel='alternative'][@type='application/json+oembed']"
|
|
|
|
):
|
|
|
|
if "href" in tag.attrib:
|
|
|
|
return tag.attrib["href"]
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
2021-09-21 18:09:57 +02:00
|
|
|
def parse_oembed_response(self, url: str, raw_body: bytes) -> OEmbedResult:
|
2021-09-01 00:37:07 +02:00
|
|
|
"""
|
2021-09-21 18:09:57 +02:00
|
|
|
Parse the oEmbed response into an Open Graph response.
|
2021-09-01 00:37:07 +02:00
|
|
|
|
|
|
|
Args:
|
2021-09-21 18:09:57 +02:00
|
|
|
url: The URL which is being previewed (not the one which was
|
|
|
|
requested).
|
|
|
|
raw_body: The oEmbed response as JSON encoded as bytes.
|
2021-09-01 00:37:07 +02:00
|
|
|
|
|
|
|
Returns:
|
2021-09-21 18:09:57 +02:00
|
|
|
json-encoded Open Graph data
|
2021-09-01 00:37:07 +02:00
|
|
|
"""
|
2021-09-08 13:17:52 +02:00
|
|
|
|
2021-09-21 18:09:57 +02:00
|
|
|
try:
|
|
|
|
# oEmbed responses *must* be UTF-8 according to the spec.
|
|
|
|
oembed = json_decoder.decode(raw_body.decode("utf-8"))
|
2021-09-01 00:37:07 +02:00
|
|
|
|
2021-10-13 13:00:07 +02:00
|
|
|
# The version is a required string field, but not always provided,
|
|
|
|
# or sometimes provided as a float. Be lenient.
|
|
|
|
oembed_version = oembed.get("version", "1.0")
|
|
|
|
if oembed_version != "1.0" and oembed_version != 1:
|
|
|
|
raise RuntimeError(f"Invalid oEmbed version: {oembed_version}")
|
2021-09-01 00:37:07 +02:00
|
|
|
|
|
|
|
# Ensure the cache age is None or an int.
|
2021-09-21 18:09:57 +02:00
|
|
|
cache_age = oembed.get("cache_age")
|
2021-09-01 00:37:07 +02:00
|
|
|
if cache_age:
|
2021-09-22 15:45:20 +02:00
|
|
|
cache_age = int(cache_age) * 1000
|
2021-09-01 00:37:07 +02:00
|
|
|
|
2021-09-21 18:09:57 +02:00
|
|
|
# The results.
|
2021-09-22 15:45:20 +02:00
|
|
|
open_graph_response = {
|
|
|
|
"og:url": url,
|
|
|
|
}
|
|
|
|
|
|
|
|
# Use either title or author's name as the title.
|
|
|
|
title = oembed.get("title") or oembed.get("author_name")
|
|
|
|
if title:
|
|
|
|
open_graph_response["og:title"] = title
|
|
|
|
|
|
|
|
# Use the provider name and as the site.
|
|
|
|
provider_name = oembed.get("provider_name")
|
|
|
|
if provider_name:
|
|
|
|
open_graph_response["og:site_name"] = provider_name
|
2021-09-01 00:37:07 +02:00
|
|
|
|
2021-09-21 18:09:57 +02:00
|
|
|
# If a thumbnail exists, use it. Note that dimensions will be calculated later.
|
|
|
|
if "thumbnail_url" in oembed:
|
|
|
|
open_graph_response["og:image"] = oembed["thumbnail_url"]
|
|
|
|
|
|
|
|
# Process each type separately.
|
|
|
|
oembed_type = oembed["type"]
|
2021-09-01 00:37:07 +02:00
|
|
|
if oembed_type == "rich":
|
2021-09-21 18:09:57 +02:00
|
|
|
calc_description_and_urls(open_graph_response, oembed["html"])
|
2021-09-01 00:37:07 +02:00
|
|
|
|
2021-09-21 18:09:57 +02:00
|
|
|
elif oembed_type == "photo":
|
|
|
|
# If this is a photo, use the full image, not the thumbnail.
|
|
|
|
open_graph_response["og:image"] = oembed["url"]
|
2021-09-01 00:37:07 +02:00
|
|
|
|
2021-09-22 15:45:20 +02:00
|
|
|
elif oembed_type == "video":
|
|
|
|
open_graph_response["og:type"] = "video.other"
|
|
|
|
calc_description_and_urls(open_graph_response, oembed["html"])
|
|
|
|
open_graph_response["og:video:width"] = oembed["width"]
|
|
|
|
open_graph_response["og:video:height"] = oembed["height"]
|
|
|
|
|
|
|
|
elif oembed_type == "link":
|
|
|
|
open_graph_response["og:type"] = "website"
|
|
|
|
|
2021-09-21 18:09:57 +02:00
|
|
|
else:
|
|
|
|
raise RuntimeError(f"Unknown oEmbed type: {oembed_type}")
|
2021-09-01 00:37:07 +02:00
|
|
|
|
2021-09-21 18:09:57 +02:00
|
|
|
except Exception as e:
|
|
|
|
# Trap any exception and let the code follow as usual.
|
2021-10-12 19:15:42 +02:00
|
|
|
logger.warning("Error parsing oEmbed metadata from %s: %r", url, e)
|
2021-09-21 18:09:57 +02:00
|
|
|
open_graph_response = {}
|
|
|
|
cache_age = None
|
2021-09-01 00:37:07 +02:00
|
|
|
|
2021-09-21 18:09:57 +02:00
|
|
|
return OEmbedResult(open_graph_response, cache_age)
|
2021-09-01 00:37:07 +02:00
|
|
|
|
|
|
|
|
2021-09-22 15:45:20 +02:00
|
|
|
def _fetch_urls(tree: "etree.Element", tag_name: str) -> List[str]:
|
|
|
|
results = []
|
|
|
|
for tag in tree.xpath("//*/" + tag_name):
|
|
|
|
if "src" in tag.attrib:
|
|
|
|
results.append(tag.attrib["src"])
|
|
|
|
return results
|
|
|
|
|
|
|
|
|
2021-09-21 18:09:57 +02:00
|
|
|
def calc_description_and_urls(open_graph_response: JsonDict, html_body: str) -> None:
|
|
|
|
"""
|
|
|
|
Calculate description for an HTML document.
|
|
|
|
|
|
|
|
This uses lxml to convert the HTML document into plaintext. If errors
|
|
|
|
occur during processing of the document, an empty response is returned.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
open_graph_response: The current Open Graph summary. This is updated with additional fields.
|
|
|
|
html_body: The HTML document, as bytes.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
The summary
|
|
|
|
"""
|
|
|
|
# If there's no body, nothing useful is going to be found.
|
|
|
|
if not html_body:
|
|
|
|
return
|
|
|
|
|
|
|
|
from lxml import etree
|
|
|
|
|
|
|
|
# Create an HTML parser. If this fails, log and return no metadata.
|
|
|
|
parser = etree.HTMLParser(recover=True, encoding="utf-8")
|
|
|
|
|
|
|
|
# Attempt to parse the body. If this fails, log and return no metadata.
|
|
|
|
tree = etree.fromstring(html_body, parser)
|
|
|
|
|
|
|
|
# The data was successfully parsed, but no tree was found.
|
|
|
|
if tree is None:
|
|
|
|
return
|
|
|
|
|
2021-09-22 15:45:20 +02:00
|
|
|
# Attempt to find interesting URLs (images, videos, embeds).
|
|
|
|
if "og:image" not in open_graph_response:
|
|
|
|
image_urls = _fetch_urls(tree, "img")
|
|
|
|
if image_urls:
|
|
|
|
open_graph_response["og:image"] = image_urls[0]
|
|
|
|
|
|
|
|
video_urls = _fetch_urls(tree, "video") + _fetch_urls(tree, "embed")
|
|
|
|
if video_urls:
|
|
|
|
open_graph_response["og:video"] = video_urls[0]
|
|
|
|
|
2021-09-21 18:09:57 +02:00
|
|
|
from synapse.rest.media.v1.preview_url_resource import _calc_description
|
|
|
|
|
|
|
|
description = _calc_description(tree)
|
|
|
|
if description:
|
|
|
|
open_graph_response["og:description"] = description
|