poezio/poezio/contact.py

218 lines
6 KiB
Python
Raw Normal View History

2011-09-06 00:45:53 +00:00
# Copyright 2010-2011 Florent Le Coz <louiz@louiz.org>
2010-09-16 17:57:27 +00:00
#
# This file is part of Poezio.
#
# Poezio is free software: you can redistribute it and/or modify
# it under the terms of the GPL-3.0+ license. See the COPYING file.
"""
2010-11-15 11:59:09 +00:00
Defines the Resource and Contact classes, which are used in
2012-05-21 00:14:25 +00:00
the roster.
"""
2010-11-15 11:59:09 +00:00
2018-07-22 12:23:39 +00:00
from collections import defaultdict
2010-11-10 21:15:08 +00:00
import logging
from typing import (
Any,
Dict,
Iterator,
List,
Optional,
Union,
)
2010-09-16 17:57:27 +00:00
from slixmpp import InvalidJID, JID
2018-07-22 12:23:39 +00:00
log = logging.getLogger(__name__)
2010-11-15 11:59:09 +00:00
2017-11-12 14:03:09 +00:00
class Resource:
2010-09-16 17:57:27 +00:00
"""
Defines a roster item.
It's a precise resource.
2010-09-16 17:57:27 +00:00
"""
2017-11-12 14:03:09 +00:00
def __init__(self, jid, data):
2012-05-21 00:14:25 +00:00
"""
data: the dict to use as a source
"""
2018-07-22 12:23:39 +00:00
# Full JID
self._jid: str = jid
self._data: Dict[str, Union[str, int]] = data
2011-11-09 21:00:38 +00:00
@property
2018-07-22 12:23:39 +00:00
def jid(self) -> str:
2010-09-16 17:57:27 +00:00
return self._jid
2011-11-09 21:00:38 +00:00
@property
2018-07-22 12:23:39 +00:00
def priority(self) -> int:
2021-03-14 21:31:22 +00:00
try:
return int(self._data.get('priority', 0))
except Exception:
return 0
2011-11-09 21:00:38 +00:00
@property
2018-07-22 12:23:39 +00:00
def presence(self) -> str:
2021-03-14 21:31:22 +00:00
return str(self._data.get('show')) or ''
2011-11-09 21:00:38 +00:00
@property
2018-07-22 12:23:39 +00:00
def status(self) -> str:
2021-03-14 21:31:22 +00:00
return str(self._data.get('status')) or ''
2018-07-22 12:23:39 +00:00
def __repr__(self) -> str:
return '<%s>' % self._jid
2018-07-22 12:23:39 +00:00
def __eq__(self, value: object) -> bool:
if not isinstance(value, Resource):
return False
return self.jid == value.jid and self._data == value._data
2017-11-12 14:03:09 +00:00
class Contact:
"""
This a way to gather multiple resources from the same bare JID.
This class contains zero or more Resource object and useful methods
to get the resource with the highest priority, etc
"""
2017-11-12 14:03:09 +00:00
def __init__(self, item):
"""
2014-07-24 00:07:08 +00:00
item: a slixmpp RosterItem pointing to that contact
"""
self.__item = item
self.folded_states: Dict[str, bool] = defaultdict(lambda: True)
self._name = ''
self.avatar = None
self.error = None
self.rich_presence: Dict[str, Any] = defaultdict(lambda: None)
2011-11-09 21:00:38 +00:00
@property
2018-07-22 12:23:39 +00:00
def groups(self) -> List[str]:
"""Name of the groups the contact is in"""
return self.__item['groups'] or ['none']
2011-10-01 12:26:19 +00:00
2011-11-09 21:00:38 +00:00
@property
2018-07-22 12:23:39 +00:00
def bare_jid(self) -> JID:
2012-05-21 00:14:25 +00:00
"""The bare jid of the contact"""
return self.__item.jid
@property
def name(self):
2012-05-21 00:14:25 +00:00
"""The name of the contact or an empty string."""
return self.__item['name'] or self._name or ''
@name.setter
def name(self, value):
"""Set the name of the contact with user nickname"""
self._name = value
@property
def ask(self):
if self.__item['pending_out']:
return 'asked'
@property
def pending_in(self):
2012-05-21 00:14:25 +00:00
"""We received a subscribe stanza from this contact."""
return self.__item['pending_in']
@pending_in.setter
def pending_in(self, value):
self.__item['pending_in'] = value
2011-11-09 21:00:38 +00:00
@property
def pending_out(self):
2012-05-21 00:14:25 +00:00
"""We sent a subscribe stanza to this contact."""
return self.__item['pending_out']
@pending_out.setter
def pending_out(self, value):
self.__item['pending_out'] = value
2011-11-09 21:00:38 +00:00
@property
2018-07-22 12:23:39 +00:00
def resources(self) -> Iterator[Resource]:
"""List of the available resources as Resource objects"""
2017-11-12 14:03:09 +00:00
return (Resource('%s%s' % (self.bare_jid, ('/' + key)
if key else ''), self.__item.resources[key])
for key in self.__item.resources.keys())
2011-11-09 21:00:38 +00:00
@property
2018-07-22 12:23:39 +00:00
def subscription(self) -> str:
return self.__item['subscription']
def __contains__(self, value):
try:
resource = JID(value).resource
except InvalidJID:
resource = None
return value in self.__item.resources or \
(resource is not None and resource in self.__item.resources)
2011-11-09 21:00:38 +00:00
2018-07-22 12:23:39 +00:00
def __len__(self) -> int:
"""Number of resources"""
return len(self.__item.resources)
2018-07-22 12:23:39 +00:00
def __bool__(self) -> bool:
"""This contact exists even when he has no resources"""
return True
2018-07-22 12:23:39 +00:00
def __getitem__(self, key) -> Optional[Resource]:
"""Return the corresponding Resource object, or None"""
try:
res = JID(key).resource
except InvalidJID:
return None
resources = self.__item.resources
item = resources.get(res, None) or resources.get(key, None)
return Resource(key, item) if item else None
def subscribe(self):
"""Subscribe to this JID"""
self.__item.subscribe()
def authorize(self):
"""Authorize this JID"""
self.__item.authorize()
def unauthorize(self):
"""Unauthorize this JID"""
self.__item.unauthorize()
def unsubscribe(self):
"""Unsubscribe from this JID"""
self.__item.unsubscribe()
2018-07-22 12:23:39 +00:00
def get(self, key: str,
default: Optional[Resource] = None) -> Optional[Resource]:
"""Same as __getitem__, but with a configurable default"""
return self[key] or default
2018-07-22 12:23:39 +00:00
def get_resources(self) -> List[Resource]:
"""Return all resources, sorted by priority """
compare_resources = lambda x: x.priority
return sorted(self.resources, key=compare_resources, reverse=True)
2018-07-22 12:23:39 +00:00
def get_highest_priority_resource(self) -> Optional[Resource]:
"""Return the resource with the highest priority"""
resources = self.get_resources()
if resources:
return resources[0]
return None
2018-07-22 12:23:39 +00:00
def folded(self, group_name='none') -> bool:
"""
Return the Folded state of a contact for this group
"""
return self.folded_states[group_name]
def toggle_folded(self, group='none'):
"""
Fold if it's unfolded, and vice versa
"""
self.folded_states[group] = not self.folded_states[group]
2018-07-22 12:23:39 +00:00
def __repr__(self) -> str:
ret = '<Contact: %s' % self.bare_jid
for resource in self.resources:
ret += '\n\t\t%s' % resource
return ret + ' />\n'