view libervia/backend/plugins/plugin_xep_0103.py @ 4334:111dce64dcb5

plugins XEP-0300, XEP-0446, XEP-0447, XEP0448 and others: Refactoring to use Pydantic: Pydantic models are used more and more in Libervia, for the bridge API, and also to convert `domish.Element` to internal representation. Type hints have also been added in many places. rel 453
author Goffi <goffi@goffi.org>
date Tue, 03 Dec 2024 00:12:38 +0100
parents 0d7bb4df2343
children
line wrap: on
line source

#!/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 typing import Self

from pydantic import BaseModel, Field
from twisted.words.xish import domish

from libervia.backend.core import exceptions
from libervia.backend.core.constants import Const as C
from libervia.backend.core.i18n import _
from libervia.backend.core.log import getLogger

log = getLogger(__name__)

PLUGIN_INFO = {
    C.PI_NAME: "URL Address Information",
    C.PI_IMPORT_NAME: "XEP-0103",
    C.PI_TYPE: "XEP",
    C.PI_MODES: C.PLUG_MODE_BOTH,
    C.PI_PROTOCOLS: ["XEP-0103"],
    C.PI_MAIN: "XEP_0103",
    C.PI_HANDLER: "no",
    C.PI_DESCRIPTION: _("""Implementation of XEP-0103 (URL Address Information)"""),
}
NS_URL_DATA = "http://jabber.org/protocol/url-data"


class Desc(BaseModel):
    """
    Model for the <desc/> element.
    """

    content: str
    xml_lang: str | None = None


class URLData(BaseModel):
    """
    Model for the <url-data/> element.
    """

    url: str
    desc: list[Desc] = Field(default_factory=list)

    @classmethod
    def from_element(cls, element: domish.Element) -> Self:
        """Create a URLData instance from a <url-data> element or its parent.

        @param url_data_elt: The <url-data> element or a parent element.
        @return: URLData instance.
        @raise exceptions.NotFound: If the <url-data> element is not found.
        """
        if element.uri != NS_URL_DATA or element.name != "url-data":
            child_url_data_elt = next(element.elements(NS_URL_DATA, "url-data"), None)
            if child_url_data_elt is None:
                raise exceptions.NotFound("<url-data> element not found")
            else:
                element = child_url_data_elt
        kwargs = {
            "url": element["target"],
            "desc": [
                Desc(content=str(desc_elt), xml_lang=desc_elt.getAttribute("xml:lang"))
                for desc_elt in element.elements(NS_URL_DATA, "desc")
            ],
        }
        return cls(**kwargs)

    def to_element(self) -> domish.Element:
        """Build the <url-data> element from this instance's data.

        @return: <url-data> element.
        """
        url_data_elt = domish.Element((NS_URL_DATA, "url-data"))
        url_data_elt["target"] = self.url
        for desc in self.desc:
            desc_elt = url_data_elt.addElement((NS_URL_DATA, "desc"))
            if desc.xml_lang:
                desc_elt["xml:lang"] = desc.xml_lang
            desc_elt.addContent(desc.content)
        return url_data_elt


class XEP_0103:
    namespace = NS_URL_DATA

    def __init__(self, host):
        log.info(f"plugin {PLUGIN_INFO[C.PI_NAME]!r} initialization")
        host.register_namespace("url-data", NS_URL_DATA)

    def generate_url_data(self, url: str, **kwargs) -> URLData:
        """Generate the element describing the URL

        @param url: URL to use
        @param extra: extra metadata describing how to access the URL
        @return: ``<url-data/>`` element
        """
        url_data = URLData(url=url, **kwargs)
        return url_data

    def parse_url_data_elt(self, url_data_elt: domish.Element) -> URLData:
        """Parse <url-data/> element

        @param url_data_elt: <url-data/> element
            a parent element can also be used
        @return: URLData instance
        @raise exceptions.NotFound: no <url-data/> element has been found
        """
        return URLData.from_element(url_data_elt)