Mercurial > libervia-backend
diff sat/plugins/plugin_xep_0446.py @ 3896:dbf0c7faaf49
plugin XEP-0446: File Metadata implementation:
rel 379
author | Goffi <goffi@goffi.org> |
---|---|
date | Wed, 21 Sep 2022 22:27:28 +0200 |
parents | |
children | 0ff265725489 |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/sat/plugins/plugin_xep_0446.py Wed Sep 21 22:27:28 2022 +0200 @@ -0,0 +1,165 @@ +#!/usr/bin/env python3 + +# Copyright (C) 2009-2022 Jérôme Poisson (goffi@goffi.org) + +# 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. + +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU Affero General Public License for more details. + +# You should have received a copy of the GNU Affero General Public License +# along with this program. If not, see <http://www.gnu.org/licenses/>. + +from logging import exception +from typing import Optional, Union, Tuple, Dict, Any +from pathlib import Path + +from twisted.words.xish import domish + +from sat.core.constants import Const as C +from sat.core.i18n import _ +from sat.core.log import getLogger +from sat.core import exceptions +from sat.tools import utils + +log = getLogger(__name__) + + +PLUGIN_INFO = { + C.PI_NAME: "File Metadata Element", + C.PI_IMPORT_NAME: "XEP-0446", + C.PI_TYPE: "XEP", + C.PI_MODES: C.PLUG_MODE_BOTH, + C.PI_PROTOCOLS: ["XEP-0446"], + C.PI_DEPENDENCIES: ["XEP-0300"], + C.PI_MAIN: "XEP_0446", + C.PI_HANDLER: "no", + C.PI_DESCRIPTION: _("""Implementation of XEP-0446 (File Metadata Element)"""), +} + +NS_FILE_METADATA = "urn:xmpp:file:metadata:0" + + +class XEP_0446: + + def __init__(self, host): + log.info(_("XEP-0446 (File Metadata Element) plugin initialization")) + host.registerNamespace("file-metadata", NS_FILE_METADATA) + self._hash = host.plugins["XEP-0300"] + + def get_file_metadata_elt( + self, + name: Optional[str] = None, + media_type: Optional[str] = None, + desc: Optional[str] = None, + size: Optional[int] = None, + file_hash: Optional[Tuple[str, str]] = None, + date: Optional[Union[float, int]] = None, + width: Optional[int] = None, + height: Optional[int] = None, + length: Optional[int] = None, + thumbnail: Optional[str] = None, + ) -> domish.Element: + """Generate the element describing a file + + @param name: name of the file + @param media_type: media-type + @param desc: description + @param size: size in bytes + @param file_hash: (algo, hash) + @param date: timestamp of the last modification datetime + @param width: image width in pixels + @param height: image height in pixels + @param length: video length in seconds + @param thumbnail: URL to a thumbnail + @return: ``<file/>`` element + """ + if name: + name = Path(name).name + file_elt = domish.Element((NS_FILE_METADATA, "file")) + for name, value in ( + ("name", name), + ("media-type", media_type), + ("desc", desc), + ("size", size), + ("width", width), + ("height", height), + ("length", length), + ): + if value is not None: + file_elt.addElement(name, content=str(value)) + if file_hash is not None: + hash_algo, hash_ = file_hash + file_elt.addChild(self._hash.buildHashElt(hash_, hash_algo)) + if date is not None: + file_elt.addElement("date", utils.xmpp_date(date)) + if thumbnail is not None: + # TODO: implement thumbnails + log.warning("thumbnail is not implemented yet") + return file_elt + + def parse_file_metadata_elt( + self, + file_metadata_elt: domish.Element + ) -> Dict[str, Any]: + """Parse <file-metadata/> element + + @param file_metadata_elt: <file-metadata/> element + a parent element can also be used + @return: file-metadata data. It's a dict whose keys correspond to + [get_file_metadata_elt] parameters + @raise exceptions.NotFound: no <file-metadata/> element has been found + """ + + if file_metadata_elt.name != "file-metadata": + try: + file_metadata_elt = next( + file_metadata_elt.elements(NS_FILE_METADATA, "file-metadata") + ) + except StopIteration: + raise exceptions.NotFound + data: Dict[str, Any] = {} + + for key, type_ in ( + ("name", str), + ("media-type", str), + ("desc", str), + ("size", int), + ("date", "timestamp"), + ("width", int), + ("height", int), + ("length", int), + ): + elt = next(file_metadata_elt.elements(NS_FILE_METADATA, key), None) + if elt is not None: + if type_ in (str, int): + content = str(elt) + if key == "name": + # we avoid malformed names or names containing path elements + content = Path(content).name + elif key == "media-type": + key = "media_type" + data[key] = type_(content) + elif type == "timestamp": + data[key] = utils.parse_xmpp_date(str(elt)) + else: + raise exceptions.InternalError + + try: + algo, hash_ = self._hash.parseHashElt(file_metadata_elt) + except exceptions.NotFound: + pass + except exceptions.DataError: + from sat.tools.xml_tools import pFmtElt + log.warning("invalid <hash/> element:\n{pFmtElt(file_metadata_elt)}") + else: + data["file_hash"] = (algo, hash_.decode()) + + # TODO: thumbnails + + return data