slidge.group
============

.. py:module:: slidge.group

.. autoapi-nested-parse::

   Everything related to groups.



Classes
-------

.. autoapisummary::

   slidge.group.MucType
   slidge.group.LegacyBookmarks
   slidge.group.LegacyParticipant
   slidge.group.LegacyMUC


Package Contents
----------------

.. py:class:: MucType



   The type of group, private, public, anonymous or not.


   .. py:attribute:: GROUP
      :value: 0


      A private group, members-only and non-anonymous, eg a family group.



   .. py:attribute:: CHANNEL
      :value: 1


      A public group, aka an anonymous channel.



   .. py:attribute:: CHANNEL_NON_ANONYMOUS
      :value: 2


      A public group where participants' legacy IDs are visible to everybody.



.. py:class:: LegacyBookmarks(session)



   This is instantiated once per :class:`~slidge.BaseSession`


   .. py:method:: legacy_id_to_jid_username(legacy_id)
      :async:


      The default implementation calls ``str()`` on the legacy_id and
      escape characters according to :xep:`0106`.

      You can override this class and implement a more subtle logic to raise
      an :class:`~slixmpp.exceptions.XMPPError` early

      :param legacy_id:
      :return:



   .. py:method:: jid_username_to_legacy_id(username)
      :async:


      :param username:
      :return:



   .. py:method:: fill()
      :abstractmethod:

      :async:


      Establish a user's known groups.

      This has to be overridden in plugins with group support and at the
      minimum, this should ``await self.by_legacy_id(group_id)`` for all
      the groups a user is part of.

      Slidge internals will call this on successful :meth:`BaseSession.login`




   .. py:method:: remove(muc, reason = 'You left this group from the official client.', kick = True)
      :async:


      Delete everything about a specific group.

      This should be called when the user leaves the group from the official
      app.

      :param muc: The MUC to remove.
      :param reason: Optionally, a reason why this group was removed.
      :param kick: Whether the user should be kicked from this group. Set this
          to False in case you do this somewhere else in your code, eg, on
          receiving the confirmation that the group was deleted.



.. py:class:: LegacyParticipant(muc, stored, is_system = False, contact = None)



   A legacy participant of a legacy group chat.


   .. py:method:: send_initial_presence(full_jid, nick_change = False, presence_id = None)

      Called when the user joins a MUC, as a mechanism
      to indicate to the joining XMPP client the list of "participants".

      Can be called this to trigger a "participant has joined the group" event.

      :param full_jid: Set this to only send to a specific user XMPP resource.
      :param nick_change: Used when the user joins and the MUC renames them (code 210)
      :param presence_id: set the presence ID. used internally by slidge



   .. py:method:: leave()

      Call this when the participant leaves the room



   .. py:method:: kick(reason = None)

      Call this when the participant is kicked from the room



   .. py:method:: ban(reason = None)

      Call this when the participant is banned from the room



   .. py:method:: online(status = None, last_seen = None)

      Send an "online" presence from this contact to the user.

      :param status: Arbitrary text, details of the status, eg: "Listening to Britney Spears"
      :param last_seen: For :xep:`0319`



   .. py:method:: away(status = None, last_seen = None)

      Send an "away" presence from this contact to the user.

      This is a global status, as opposed to :meth:`.LegacyContact.inactive`
      which concerns a specific conversation, ie a specific "chat window"

      :param status: Arbitrary text, details of the status, eg: "Gone to fight capitalism"
      :param last_seen: For :xep:`0319`



   .. py:method:: extended_away(status = None, last_seen = None)

      Send an "extended away" presence from this contact to the user.

      This is a global status, as opposed to :meth:`.LegacyContact.inactive`
      which concerns a specific conversation, ie a specific "chat window"

      :param status: Arbitrary text, details of the status, eg: "Gone to fight capitalism"
      :param last_seen: For :xep:`0319`



   .. py:method:: busy(status = None, last_seen = None)

      Send a "busy" (ie, "dnd") presence from this contact to the user,

      :param status: eg: "Trying to make sense of XEP-0100"
      :param last_seen: For :xep:`0319`



   .. py:method:: offline(status = None, last_seen = None)

      Send an "offline" presence from this contact to the user.

      :param status: eg: "Trying to make sense of XEP-0100"
      :param last_seen: For :xep:`0319`



   .. py:method:: invite_to(muc, reason = None, password = None, **send_kwargs)

      Send an invitation to join a group (:xep:`0249`) from this :term:`XMPP Entity`.

      :param muc: the muc the user is invited to
      :param reason: a text explaining why the user should join this muc
      :param password: maybe this will make sense later? not sure
      :param send_kwargs: additional kwargs to be passed to _send()
          (internal use by slidge)



   .. py:method:: active(**kwargs)

      Send an "active" chat state (:xep:`0085`) from this
      :term:`XMPP Entity`.



   .. py:method:: composing(**kwargs)

      Send a "composing" (ie "typing notification") chat state (:xep:`0085`)
      from this :term:`XMPP Entity`.



   .. py:method:: paused(**kwargs)

      Send a "paused" (ie "typing paused notification") chat state
      (:xep:`0085`) from this :term:`XMPP Entity`.



   .. py:method:: inactive(**kwargs)

      Send an "inactive" (ie "contact has not interacted with the chat session
      interface for an intermediate period of time") chat state (:xep:`0085`)
      from this :term:`XMPP Entity`.



   .. py:method:: gone(**kwargs)

      Send a "gone" (ie "contact has not interacted with the chat session interface,
      system, or device for a relatively long period of time") chat state
      (:xep:`0085`) from this :term:`XMPP Entity`.



   .. py:method:: ack(legacy_msg_id, **kwargs)

      Send an "acknowledged" message marker (:xep:`0333`) from this :term:`XMPP Entity`.

      :param legacy_msg_id: The message this marker refers to



   .. py:method:: received(legacy_msg_id, **kwargs)

      Send a "received" message marker (:xep:`0333`) from this :term:`XMPP Entity`.
      If called on a :class:`LegacyContact`, also send a delivery receipt
      marker (:xep:`0184`).

      :param legacy_msg_id: The message this marker refers to



   .. py:method:: displayed(legacy_msg_id, **kwargs)

      Send a "displayed" message marker (:xep:`0333`) from this :term:`XMPP Entity`.

      :param legacy_msg_id: The message this marker refers to



   .. py:method:: send_file(attachment, legacy_msg_id = None, *, reply_to = None, when = None, thread = None, **kwargs)
      :async:


      Send a single file from this :term:`XMPP Entity`.

      :param attachment: The file to send.
          Ideally, a :class:`.LegacyAttachment` with a unique ``legacy_file_id``
          attribute set, to optimise potential future reuses.
          It can also be:
          - a :class:`pathlib.Path` instance to point to a local file, or
          - a ``str``, representing a fetchable HTTP URL.
      :param legacy_msg_id: If you want to be able to transport read markers from the gateway
          user to the legacy network, specify this
      :param reply_to: Quote another message (:xep:`0461`)
      :param when: when the file was sent, for a "delay" tag (:xep:`0203`)
      :param thread:



   .. py:method:: send_text(body, legacy_msg_id = None, *, when = None, reply_to = None, thread = None, hints = None, carbon = False, archive_only = False, correction = False, correction_event_id = None, link_previews = None, **send_kwargs)

      Send a text message from this :term:`XMPP Entity`.

      :param body: Content of the message
      :param legacy_msg_id: If you want to be able to transport read markers from the gateway
          user to the legacy network, specify this
      :param when: when the message was sent, for a "delay" tag (:xep:`0203`)
      :param reply_to: Quote another message (:xep:`0461`)
      :param hints:
      :param thread:
      :param carbon: (only used if called on a :class:`LegacyContact`)
          Set this to ``True`` if this is actually a message sent **to** the
          :class:`LegacyContact` by the :term:`User`.
          Use this to synchronize outgoing history for legacy official apps.
      :param correction: whether this message is a correction or not
      :param correction_event_id: in the case where an ID is associated with the legacy
          'correction event', specify it here to use it on the XMPP side. If not specified,
          a random ID will be used.
      :param link_previews: A little of sender (or server, or gateway)-generated
          previews of URLs linked in the body.
      :param archive_only: (only in groups) Do not send this message to user,
          but store it in the archive. Meant to be used during ``MUC.backfill()``



   .. py:method:: correct(legacy_msg_id, new_text, *, when = None, reply_to = None, thread = None, hints = None, carbon = False, archive_only = False, correction_event_id = None, link_previews = None, **send_kwargs)

      Modify a message that was previously sent by this :term:`XMPP Entity`.

      Uses last message correction (:xep:`0308`)

      :param new_text: New content of the message
      :param legacy_msg_id: The legacy message ID of the message to correct
      :param when: when the message was sent, for a "delay" tag (:xep:`0203`)
      :param reply_to: Quote another message (:xep:`0461`)
      :param hints:
      :param thread:
      :param carbon: (only in 1:1) Reflect a message sent to this ``Contact`` by the user.
          Use this to synchronize outgoing history for legacy official apps.
      :param archive_only: (only in groups) Do not send this message to user,
          but store it in the archive. Meant to be used during ``MUC.backfill()``
      :param correction_event_id: in the case where an ID is associated with the legacy
          'correction event', specify it here to use it on the XMPP side. If not specified,
          a random ID will be used.
      :param link_previews: A little of sender (or server, or gateway)-generated
          previews of URLs linked in the body.



   .. py:method:: react(legacy_msg_id, emojis = (), thread = None, **kwargs)

      Send a reaction (:xep:`0444`) from this :term:`XMPP Entity`.

      :param legacy_msg_id: The message which the reaction refers to.
      :param emojis: An iterable of emojis used as reactions
      :param thread:



   .. py:method:: retract(legacy_msg_id, thread = None, **kwargs)

      Send a message retraction (:XEP:`0424`) from this :term:`XMPP Entity`.

      :param legacy_msg_id: Legacy ID of the message to delete
      :param thread:



.. py:class:: LegacyMUC(session, stored)



   A room, a.k.a. a Multi-User Chat.

   MUC instances are obtained by calling :py:meth:`slidge.group.bookmarks.LegacyBookmarks`
   on the user's :py:class:`slidge.core.session.BaseSession`.


   .. py:attribute:: STABLE_ARCHIVE
      :value: False


      Because legacy events like reactions, editions, etc. don't all map to a stanza
      with a proper legacy ID, slidge usually cannot guarantee the stability of the archive
      across restarts.

      Set this to True if you know what you're doing, but realistically, this can't
      be set to True until archive is permanently stored on disk by slidge.

      This is just a flag on archive responses that most clients ignore anyway.



   .. py:attribute:: HAS_DESCRIPTION
      :value: True


      Set this to false if the legacy network does not allow setting a description
      for the group. In this case the description field will not be present in the
      room configuration form.



   .. py:attribute:: HAS_SUBJECT
      :value: True


      Set this to false if the legacy network does not allow setting a subject
      (sometimes also called topic) for the group. In this case, as a subject is
      recommended by :xep:`0045` ("SHALL"), the description (or the group name as
      ultimate fallback) will be used as the room subject.
      By setting this to false, an error will be returned when the :term:`User`
      tries to set the room subject.



   .. py:method:: pop_unread_xmpp_ids_up_to(horizon_xmpp_id)

      Return XMPP msg ids sent in this group up to a given XMPP msg id.

      Plugins have no reason to use this, but it is used by slidge core
      for legacy networks that need to mark *all* messages as read (most XMPP
      clients only send a read marker for the latest message).

      This has side effects: all messages up to the horizon XMPP id will be marked
      as read in the DB. If the horizon XMPP id is not found, all messages of this
      MUC will be marked as read.

      :param horizon_xmpp_id: The latest message
      :return: A list of XMPP ids if horizon_xmpp_id was not found



   .. py:method:: update_info()
      :abstractmethod:

      :async:


      Fetch information about this group from the legacy network

      This is awaited on MUC instantiation, and should be overridden to
      update the attributes of the group chat, like title, subject, number
      of participants etc.

      To take advantage of the slidge avatar cache, you can check the .avatar
      property to retrieve the "legacy file ID" of the cached avatar. If there
      is no change, you should not call
      :py:meth:`slidge.core.mixins.avatar.AvatarMixin.set_avatar()` or
      attempt to modify
      the :attr:.avatar property.



   .. py:method:: backfill(after = None, before = None)
      :abstractmethod:

      :async:


      Override this if the legacy network provide server-side group archives.

      In it, send history messages using ``self.get_participant(xxx).send_xxxx``,
      with the ``archive_only=True`` kwarg. This is only called once per slidge
      run for a given group.

      :param after: Fetch messages after this one. If ``None``, it's up to you
          to decide how far you want to go in the archive. If it's not ``None``,
          it means slidge has some messages in this archive and you should really try
          to complete it to avoid "holes" in the history of this group.
      :param before: Fetch messages before this one. If ``None``, fetch all messages
          up to the most recent one



   .. py:method:: fill_participants()
      :async:


      This method should yield the list of all members of this group.

      Typically, use ``participant = self.get_participant()``, self.get_participant_by_contact(),
      of self.get_user_participant(), and update their affiliation, hats, etc.
      before yielding them.



   .. py:method:: get_user_participant(**kwargs)
      :async:


      Get the participant representing the gateway user

      :param kwargs: additional parameters for the :class:`.Participant`
          construction (optional)
      :return:



   .. py:method:: get_participant(nickname: str) -> slidge.util.types.LegacyParticipantType
                  get_participant(*, occupant_id: str) -> slidge.util.types.LegacyParticipantType
                  get_participant(*, occupant_id: str, create: Literal[False]) -> LegacyParticipantType | None
                  get_participant(*, occupant_id: str, create: Literal[True]) -> slidge.util.types.LegacyParticipantType
                  get_participant(nickname: str, *, occupant_id: str) -> slidge.util.types.LegacyParticipantType
                  get_participant(nickname: str, *, create: Literal[False]) -> LegacyParticipantType | None
                  get_participant(nickname: str, *, create: Literal[True]) -> slidge.util.types.LegacyParticipantType
                  get_participant(nickname: str, *, create: Literal[True], is_user: bool, fill_first: bool, store: bool) -> slidge.util.types.LegacyParticipantType
                  get_participant(nickname: str, *, create: Literal[False], is_user: bool, fill_first: bool, store: bool) -> LegacyParticipantType | None
                  get_participant(nickname: str, *, create: bool, fill_first: bool) -> LegacyParticipantType | None
      :async:


      Get a participant by their nickname.

      In non-anonymous groups, you probably want to use
      :meth:`.LegacyMUC.get_participant_by_contact` instead.

      :param nickname: Nickname of the participant (used as resource part in the MUC)
      :param create: By default, a participant is created if necessary. Set this to
          False to return None if participant was not created before.
      :param is_user: Whether this participant is the slidge user.
      :param fill_first: Ensure :meth:`.LegacyMUC.fill_participants()` has been called
          first (internal use by slidge, plugins should not need that)
      :param store: persistently store the user in the list of MUC participants
      :param occupant_id: optionally, specify the unique ID for this participant, cf
          xep:`0421`
      :return: A participant of this room.



   .. py:method:: get_system_participant()

      Get a pseudo-participant, representing the room itself

      Can be useful for events that cannot be mapped to a participant,
      e.g. anonymous moderation events, or announces from the legacy
      service
      :return:



   .. py:method:: get_participant_by_contact(c: LegacyContact[Any]) -> slidge.util.types.LegacyParticipantType
                  get_participant_by_contact(c: LegacyContact[Any], *, occupant_id: str | None = None) -> slidge.util.types.LegacyParticipantType
                  get_participant_by_contact(c: LegacyContact[Any], *, create: Literal[False], occupant_id: str | None) -> LegacyParticipantType | None
                  get_participant_by_contact(c: LegacyContact[Any], *, create: Literal[True], occupant_id: str | None) -> slidge.util.types.LegacyParticipantType
      :async:


      Get a non-anonymous participant.

      This is what should be used in non-anonymous groups ideally, to ensure
      that the Contact jid is associated to this participant

      :param c: The :class:`.LegacyContact` instance corresponding to this contact
      :param create: Creates the participant if it does not exist.
      :param occupant_id: Optionally, specify a unique occupant ID (:xep:`0421`) for
          this participant.
      :return:



   .. py:method:: remove_participant(p, kick = False, ban = False, reason = None)

      Call this when a participant leaves the room

      :param p: The participant
      :param kick: Whether the participant left because they were kicked
      :param ban: Whether the participant left because they were banned
      :param reason: Optionally, a reason why the participant was removed.



   .. py:method:: kick_resource(r)
      :async:


      Kick a XMPP client of the user. (slidge internal use)

      :param r: The resource to kick



   .. py:method:: add_to_bookmarks(auto_join = True, preserve = True, pin = None, notify = None)
      :async:


      Add the MUC to the user's XMPP bookmarks (:xep:`0402')

      This requires that slidge has the IQ privileged set correctly
      on the XMPP server

      :param auto_join: whether XMPP clients should automatically join
          this MUC on startup. In theory, XMPP clients will receive
          a "push" notification when this is called, and they will
          join if they are online.
      :param preserve: preserve auto-join and bookmarks extensions
          set by the user outside slidge
      :param pin: Pin the group chat bookmark :xep:`0469`. Requires privileged entity.
          If set to ``None`` (default), the bookmark pinning status will be untouched.
      :param notify: Chat notification setting: :xep:`0492`. Requires privileged entity.
          If set to ``None`` (default), the setting will be untouched. Only the "global"
          notification setting is supported (ie, per client type is not possible).



   .. py:method:: on_avatar(data, mime)
      :abstractmethod:

      :async:


      Called when the user tries to set the avatar of the room from an XMPP
      client.

      If the set avatar operation is completed, should return a legacy image
      unique identifier. In this case the MUC avatar will be immediately
      updated on the XMPP side.

      If data is not None and this method returns None, then we assume that
      self.set_avatar() will be called elsewhere, eg triggered by a legacy
      room update event.

      :param data: image data or None if the user meant to remove the avatar
      :param mime: the mime type of the image. Since this is provided by
          the XMPP client, there is no guarantee that this is valid or
          correct.
      :return: A unique avatar identifier, which will trigger
          :py:meth:`slidge.group.room.LegacyMUC.set_avatar`. Alternatively, None, if
          :py:meth:`.LegacyMUC.set_avatar` is meant to be awaited somewhere else.



   .. py:method:: on_set_affiliation(contact, affiliation, reason, nickname)
      :abstractmethod:

      :async:


      Triggered when the user requests changing the affiliation of a contact
      for this group.

      Examples: promotion them to moderator, ban (affiliation=outcast).

      :param contact: The contact whose affiliation change is requested
      :param affiliation: The new affiliation
      :param reason: A reason for this affiliation change
      :param nickname:



   .. py:method:: on_kick(contact, reason)
      :abstractmethod:

      :async:


      Triggered when the user requests changing the role of a contact
      to "none" for this group. Action commonly known as "kick".

      :param contact: Contact to be kicked
      :param reason: A reason for this kick



   .. py:method:: on_set_config(name, description)
      :abstractmethod:

      :async:


      Triggered when the user requests changing the room configuration.
      Only title and description can be changed at the moment.

      The legacy module is responsible for updating :attr:`.title` and/or
      :attr:`.description` of this instance.

      If :attr:`.HAS_DESCRIPTION` is set to False, description will always
      be ``None``.

      :param name: The new name of the room.
      :param description: The new description of the room.



   .. py:method:: on_destroy_request(reason)
      :abstractmethod:

      :async:


      Triggered when the user requests room destruction.

      :param reason: Optionally, a reason for the destruction



   .. py:method:: on_set_subject(subject)
      :abstractmethod:

      :async:


      Triggered when the user requests changing the room subject.

      The legacy module is responsible for updating :attr:`.subject` of this
      instance.

      :param subject: The new subject for this room.



   .. py:method:: on_set_thread_subject(thread, subject)
      :abstractmethod:

      :async:


      Triggered when the user requests changing the subject of a specific thread.

      :param thread: Legacy identifier of the thread
      :param subject: The new subject for this thread.



   .. py:method:: get_archived_messages(msg_id)

      Query the slidge archive for messages sent in this group

      :param msg_id: Message ID of the message in question. Can be either a legacy ID
          or an XMPP ID.
      :return: Iterator over messages. A single legacy ID can map to several messages,
          because of multi-attachment messages.



   .. py:property:: avatar
      :type: slidge.util.types.Avatar | None


      This property can be used to set or unset the avatar.

      Unlike the awaitable :method:`.set_avatar`, it schedules the update for
      later execution and is not blocking



   .. py:method:: set_avatar(avatar = None, delete = False)
      :async:


      Set an avatar for this entity

      :param avatar: The avatar. Should ideally come with a legacy network-wide unique
          ID
      :param delete: If the avatar is provided as a Path, whether to delete
          it once used or not.



   .. py:method:: serialize_extra_attributes()

      If you want custom attributes of your instance to be stored persistently
      to the DB, here is where you have to return them as a dict to be used in
      `deserialize_extra_attributes()`.




   .. py:method:: deserialize_extra_attributes(data)

      This is where you get the dict that you passed in
      `serialize_extra_attributes()`.

      ⚠ Since it is serialized as json, dictionary keys are converted to strings!
      Be sure to convert to other types if necessary.



   .. py:method:: available_emojis(legacy_msg_id = None)
      :async:


      Override this to restrict the subset of reactions this recipient
      can handle.

      :return: A set of emojis or None if any emoji is allowed



