Source code for pywikibot.page._user

"""Object representing a Wiki user."""
#
# (C) Pywikibot team, 2009-2024
#
# Distributed under the terms of the MIT license.
#
from __future__ import annotations

import pywikibot
from pywikibot.backports import Generator
from pywikibot.exceptions import (
    APIError,
    AutoblockUserError,
    NoRenameTargetError,
    NotEmailableError,
    UserRightsError,
)
from pywikibot.page._links import Link
from pywikibot.page._page import Page
from pywikibot.page._revision import Revision
from pywikibot.tools import deprecated, is_ip_address, is_ip_network


__all__ = ('User', )


[docs] class User(Page): """ A class that represents a Wiki user. This class also represents the Wiki page User:<username> """ def __init__(self, source, title: str = '') -> None: """ Initializer for a User object. All parameters are the same as for Page() Initializer. """ self._isAutoblock = True if title.startswith('#'): title = title[1:] elif ':#' in title: title = title.replace(':#', ':') else: self._isAutoblock = False super().__init__(source, title, ns=2) if self.namespace() != 2: raise ValueError(f"'{self.title()}' is not in the user namespace!") if self._isAutoblock: # This user is probably being queried for purpose of lifting # an autoblock. pywikibot.info( 'This is an autoblock ID, you can only use to unblock it.') @property def username(self) -> str: """ The username. Convenience method that returns the title of the page with namespace prefix omitted, which is the username. """ if self._isAutoblock: return '#' + self.title(with_ns=False) return self.title(with_ns=False)
[docs] def isRegistered(self, force: bool = False) -> bool: # noqa: N802 """ Determine if the user is registered on the site. It is possible to have a page named User:xyz and not have a corresponding user with username xyz. The page does not need to exist for this method to return True. :param force: if True, forces reloading the data from API """ # T135828: the registration timestamp may be None but the key exists return (not self.isAnonymous() and 'registration' in self.getprops(force))
[docs] def isAnonymous(self) -> bool: # noqa: N802 """Determine if the user is editing as an IP address.""" return is_ip_address(self.username)
[docs] def is_CIDR(self) -> bool: # noqa: N802 """Determine if the input refers to a range of IP addresses.""" return is_ip_network(self.username)
[docs] def getprops(self, force: bool = False) -> dict: """ Return a properties about the user. :param force: if True, forces reloading the data from API """ if force and hasattr(self, '_userprops'): del self._userprops if not hasattr(self, '_userprops'): self._userprops = list(self.site.users([self.username]))[0] if self.isAnonymous() or self.is_CIDR(): r = list(self.site.blocks(iprange=self.username, total=1)) if r: self._userprops['blockedby'] = r[0]['by'] self._userprops['blockreason'] = r[0]['reason'] return self._userprops
[docs] def registration(self, force: bool = False) -> pywikibot.Timestamp | None: """ Fetch registration date for this user. :param force: if True, forces reloading the data from API """ if not self.isAnonymous(): reg = self.getprops(force).get('registration') if reg: return pywikibot.Timestamp.fromISOformat(reg) return None
[docs] def editCount(self, force: bool = False) -> int: # noqa: N802 """ Return edit count for a registered user. Always returns 0 for 'anonymous' users. :param force: if True, forces reloading the data from API """ return self.getprops(force).get('editcount', 0)
[docs] def is_blocked(self, force: bool = False) -> bool: """Determine whether the user is currently blocked. .. versionchanged:: 7.0 renamed from :meth:`isBlocked` method, can also detect range blocks. :param force: if True, forces reloading the data from API """ return 'blockedby' in self.getprops(force)
[docs] @deprecated('is_blocked', since='7.0.0') def isBlocked(self, force: bool = False) -> bool: # noqa: N802 """Determine whether the user is currently blocked. .. deprecated:: 7.0 use :meth:`is_blocked` instead :param force: if True, forces reloading the data from API """ return self.is_blocked(force)
[docs] def is_locked(self, force: bool = False) -> bool: """Determine whether the user is currently locked globally. .. versionadded:: 7.0 :param force: if True, forces reloading the data from API """ return self.site.is_locked(self.username, force)
[docs] def isEmailable(self, force: bool = False) -> bool: # noqa: N802 """ Determine whether emails may be send to this user through MediaWiki. :param force: if True, forces reloading the data from API """ return not self.isAnonymous() and 'emailable' in self.getprops(force)
[docs] def groups(self, force: bool = False) -> list: """ Return a list of groups to which this user belongs. The list of groups may be empty. :param force: if True, forces reloading the data from API :return: groups property """ return self.getprops(force).get('groups', [])
[docs] def gender(self, force: bool = False) -> str: """Return the gender of the user. :param force: if True, forces reloading the data from API :return: return 'male', 'female', or 'unknown' """ if self.isAnonymous(): return 'unknown' return self.getprops(force).get('gender', 'unknown')
[docs] def rights(self, force: bool = False) -> list: """Return user rights. :param force: if True, forces reloading the data from API :return: return user rights """ return self.getprops(force).get('rights', [])
[docs] def getUserPage(self, subpage: str = '') -> Page: # noqa: N802 """ Return a Page object relative to this user's main page. :param subpage: subpage part to be appended to the main page title (optional) :return: Page object of user page or user subpage """ if self._isAutoblock: # This user is probably being queried for purpose of lifting # an autoblock, so has no user pages per se. raise AutoblockUserError( 'This is an autoblock ID, you can only use to unblock it.') if subpage: subpage = '/' + subpage return Page(Link(self.title() + subpage, self.site))
[docs] def getUserTalkPage(self, subpage: str = '') -> Page: # noqa: N802 """ Return a Page object relative to this user's main talk page. :param subpage: subpage part to be appended to the main talk page title (optional) :return: Page object of user talk page or user talk subpage """ if self._isAutoblock: # This user is probably being queried for purpose of lifting # an autoblock, so has no user talk pages per se. raise AutoblockUserError( 'This is an autoblock ID, you can only use to unblock it.') if subpage: subpage = '/' + subpage return Page(Link(self.username + subpage, self.site, default_namespace=3))
[docs] def send_email(self, subject: str, text: str, ccme: bool = False) -> bool: """ Send an email to this user via MediaWiki's email interface. :param subject: the subject header of the mail :param text: mail body :param ccme: if True, sends a copy of this email to the bot :raises NotEmailableError: the user of this User is not emailable :raises UserRightsError: logged in user does not have 'sendemail' right :return: operation successful indicator """ if not self.isEmailable(): raise NotEmailableError(self) if not self.site.has_right('sendemail'): raise UserRightsError("You don't have permission to send mail") params = { 'action': 'emailuser', 'target': self.username, 'token': self.site.tokens['csrf'], 'subject': subject, 'text': text, } if ccme: params['ccme'] = 1 mailrequest = self.site.simple_request(**params) maildata = mailrequest.submit() return ('emailuser' in maildata and maildata['emailuser']['result'] == 'Success')
[docs] def block(self, *args, **kwargs): """ Block user. Refer :py:obj:`APISite.blockuser` method for parameters. :return: None """ try: self.site.blockuser(self, *args, **kwargs) except APIError as err: if err.code == 'invalidrange': raise ValueError(f'{self.username} is not a valid IP range.') raise
[docs] def unblock(self, reason: str | None = None) -> None: """ Remove the block for the user. :param reason: Reason for the unblock. """ self.site.unblockuser(self, reason)
[docs] def logevents(self, **kwargs): """Yield user activities. :keyword logtype: only iterate entries of this type (see mediawiki api documentation for available types) :type logtype: str :keyword page: only iterate entries affecting this page :type page: Page or str :keyword namespace: namespace to retrieve logevents from :type namespace: int or Namespace :keyword start: only iterate entries from and after this Timestamp :type start: Timestamp or ISO date string :keyword end: only iterate entries up to and through this Timestamp :type end: Timestamp or ISO date string :keyword reverse: if True, iterate oldest entries first (default: newest) :type reverse: bool :keyword tag: only iterate entries tagged with this tag :type tag: str :keyword total: maximum number of events to iterate :type total: int :rtype: iterable """ return self.site.logevents(user=self.username, **kwargs)
@property def last_event(self): """Return last user activity. :return: last user log entry :rtype: LogEntry or None """ return next(self.logevents(total=1), None)
[docs] def contributions( self, total: int | None = 500, **kwargs ) -> Generator[ tuple[Page, int, pywikibot.Timestamp, str | None], None, None ]: """Yield tuples describing this user edits. Each tuple is composed of a pywikibot.Page object, the revision id, the edit timestamp and the comment. Pages returned are not guaranteed to be unique. Example: >>> site = pywikibot.Site('wikipedia:test') >>> user = pywikibot.User(site, 'pywikibot-test') >>> contrib = next(user.contributions(reverse=True)) >>> len(contrib) 4 >>> contrib[0].title() 'User:John Vandenberg/appendtext test' >>> contrib[1] 504588 >>> str(contrib[2]) '2022-03-04T17:36:02Z' >>> contrib[3] '' .. seealso:: :meth:`Site.usercontribs() <pywikibot.site._generators.GeneratorsMixin.usercontribs>` :param total: limit result to this number of pages :keyword start: Iterate contributions starting at this Timestamp :keyword end: Iterate contributions ending at this Timestamp :keyword reverse: Iterate oldest contributions first (default: newest) :keyword namespaces: only iterate pages in these namespaces :type namespaces: iterable of str or Namespace key, or a single instance of those types. May be a '|' separated list of namespace identifiers. :keyword showMinor: if True, iterate only minor edits; if False and not None, iterate only non-minor edits (default: iterate both) :keyword top_only: if True, iterate only edits which are the latest revision (default: False) :return: tuple of pywikibot.Page, revid, pywikibot.Timestamp, comment """ for contrib in self.site.usercontribs( user=self.username, total=total, **kwargs): ts = pywikibot.Timestamp.fromISOformat(contrib['timestamp']) yield (Page(self.site, contrib['title'], contrib['ns']), contrib['revid'], ts, contrib.get('comment'))
@property def first_edit( self ) -> tuple[Page, int, pywikibot.Timestamp, str | None] | None: """Return first user contribution. :return: first user contribution entry :return: tuple of pywikibot.Page, revid, pywikibot.Timestamp, comment """ return next(self.contributions(reverse=True, total=1), None) @property def last_edit( self ) -> tuple[Page, int, pywikibot.Timestamp, str | None] | None: """Return last user contribution. :return: last user contribution entry :return: tuple of pywikibot.Page, revid, pywikibot.Timestamp, comment """ return next(self.contributions(total=1), None)
[docs] def deleted_contributions( self, *, total: int | None = 500, **kwargs, ) -> Generator[tuple[Page, Revision], None, None]: """Yield tuples describing this user's deleted edits. .. versionadded:: 5.5 :param total: Limit results to this number of pages :keyword start: Iterate contributions starting at this Timestamp :keyword end: Iterate contributions ending at this Timestamp :keyword reverse: Iterate oldest contributions first (default: newest) :keyword namespaces: Only iterate pages in these namespaces """ for data in self.site.alldeletedrevisions(user=self.username, total=total, **kwargs): page = Page(self.site, data['title'], data['ns']) for contrib in data['revisions']: yield page, Revision(**contrib)
[docs] def uploadedImages(self, total: int = 10): # noqa: N802 """ Yield tuples describing files uploaded by this user. Each tuple is composed of a pywikibot.Page, the timestamp (str in ISO8601 format), comment (str) and a bool for pageid > 0. Pages returned are not guaranteed to be unique. :param total: limit result to this number of pages """ if not self.isRegistered(): return for item in self.logevents(logtype='upload', total=total): yield (item.page(), str(item.timestamp()), item.comment(), item.pageid() > 0)
@property def is_thankable(self) -> bool: """ Determine if the user has thanks notifications enabled. .. note:: This doesn't accurately determine if thanks is enabled for user. Privacy of thanks preferences is under discussion, please see :phab:`T57401#2216861` and :phab:`T120753#1863894`. """ return self.isRegistered() and 'bot' not in self.groups()
[docs] def renamed_target(self) -> User: """Return a User object for the target this user was renamed to. If this user was not renamed, it will raise a :exc:`NoRenameTargetError`. **Usage:** >>> site = pywikibot.Site('wikipedia:de') >>> user = pywikibot.User(site, 'Foo') >>> user.isRegistered() False >>> target = user.renamed_target() >>> target.isRegistered() True >>> target.title(with_ns=False) 'Foo~dewiki' >>> target.renamed_target() Traceback (most recent call last): ... pywikibot.exceptions.NoRenameTargetError: Rename target user ... .. seealso:: * :meth:`BasePage.moved_target` * :meth:`BasePage.getRedirectTarget` .. versionadded:: 9.4 :raises NoRenameTargetError: user was not renamed """ gen = iter(self.site.logevents(logtype='renameuser', page=self, total=1)) try: renamed = next(gen) except StopIteration: raise NoRenameTargetError(self) return User(self.site, renamed.params['newuser'])