Source code for pywikibot.site._datasite

"""Objects representing API interface to Wikibase site."""
#
# (C) Pywikibot team, 2012-2024
#
# Distributed under the terms of the MIT license.
#
from __future__ import annotations

import datetime
import json
import uuid
from contextlib import suppress
from typing import Any
from warnings import warn

import pywikibot
from pywikibot.backports import Generator, Iterable, batched
from pywikibot.data import api
from pywikibot.exceptions import (
    APIError,
    EntityTypeUnknownError,
    IsRedirectPageError,
    NoPageError,
    NoWikibaseEntityError,
)
from pywikibot.site._apisite import APISite
from pywikibot.site._decorators import need_extension, need_right, need_version
from pywikibot.tools import deprecated, merge_unique_dicts, remove_last_args


__all__ = ('DataSite', )


[docs] class DataSite(APISite): """Wikibase data capable site.""" def __init__(self, *args, **kwargs) -> None: """Initializer.""" super().__init__(*args, **kwargs) self._item_namespace = None self._property_namespace = None self._type_to_class = { 'item': pywikibot.ItemPage, 'property': pywikibot.PropertyPage, 'mediainfo': pywikibot.MediaInfo, 'lexeme': pywikibot.LexemePage, 'form': pywikibot.LexemeForm, 'sense': pywikibot.LexemeSense, }
[docs] def get_repo_for_entity_type(self, entity_type: str) -> DataSite: """Get the data repository for the entity type. When no foreign repository is defined for the entity type, the method returns this repository itself even if it does not support that entity type either. .. seealso:: https://www.mediawiki.org/wiki/Wikibase/Federation .. versionadded:: 8.0 :raises ValueError: when invalid entity type was provided """ if entity_type not in self._type_to_class: raise ValueError(f'Invalid entity type "{entity_type}"') entity_sources = self.entity_sources() if entity_type in entity_sources: return pywikibot.Site( *entity_sources[entity_type], interface='DataSite', user=self.username()) return self
def _cache_entity_namespaces(self) -> None: """Find namespaces for each known wikibase entity type.""" self._entity_namespaces = {} for entity_type in self._type_to_class: for namespace in self.namespaces.values(): if not hasattr(namespace, 'defaultcontentmodel'): continue content_model = namespace.defaultcontentmodel if content_model == ('wikibase-' + entity_type): self._entity_namespaces[entity_type] = namespace break
[docs] def get_namespace_for_entity_type(self, entity_type): """Return namespace for given entity type. :return: corresponding namespace :rtype: Namespace """ if not hasattr(self, '_entity_namespaces'): self._cache_entity_namespaces() if entity_type in self._entity_namespaces: return self._entity_namespaces[entity_type] raise EntityTypeUnknownError( f'{self!r} does not support entity type "{entity_type}" ' " or it doesn't have its own namespace" )
@property def item_namespace(self): """Return namespace for items. :return: item namespace :rtype: Namespace """ if self._item_namespace is None: self._item_namespace = self.get_namespace_for_entity_type('item') return self._item_namespace @property def property_namespace(self): """Return namespace for properties. :return: property namespace :rtype: Namespace """ if self._property_namespace is None: self._property_namespace = self.get_namespace_for_entity_type( 'property') return self._property_namespace
[docs] def get_entity_for_entity_id(self, entity_id): """Return a new instance for given entity id. :raises pywikibot.exceptions.NoWikibaseEntityError: there is no entity with the id :return: a WikibaseEntity subclass :rtype: WikibaseEntity """ for cls in self._type_to_class.values(): if cls.is_valid_id(entity_id): return cls(self, entity_id) entity = pywikibot.page.WikibaseEntity(self, entity_id) raise NoWikibaseEntityError(entity)
@property @need_version('1.28-wmf.3') def sparql_endpoint(self): """Return the sparql endpoint url, if any has been set. :return: sparql endpoint url :rtype: str|None """ return self.siteinfo['general'].get('wikibase-sparql') @property @need_version('1.28-wmf.23') def concept_base_uri(self): """Return the base uri for concepts/entities. :return: concept base uri :rtype: str """ return self.siteinfo['general']['wikibase-conceptbaseuri']
[docs] def geo_shape_repository(self): """Return Site object for the geo-shapes repository e.g. commons.""" url = self.siteinfo['general'].get('wikibase-geoshapestoragebaseurl') if url: return pywikibot.Site(url=url, user=self.username()) return None
[docs] def tabular_data_repository(self): """Return Site object for the tabular-datas repository e.g. commons.""" url = self.siteinfo['general'].get( 'wikibase-tabulardatastoragebaseurl') if url: return pywikibot.Site(url=url, user=self.username()) return None
[docs] def loadcontent(self, identification, *props): """Fetch the current content of a Wikibase item. This is called loadcontent since wbgetentities does not support fetching old revisions. Eventually this will get replaced by an actual loadrevisions. :param identification: Parameters used to identify the page(s) :type identification: dict :param props: the optional properties to fetch. """ params = merge_unique_dicts(identification, action='wbgetentities', # TODO: When props is empty it results in # an empty string ('&props=') but it should # result in a missing entry. props=props if props else False) req = self.simple_request(**params) data = req.submit() if 'success' not in data: raise APIError(data['errors'], '') return data['entities']
[docs] def preload_entities( self, pagelist: Iterable[pywikibot.page.WikibaseEntity | pywikibot.page.Page], groupsize: int = 50 ) -> Generator[pywikibot.page.WikibaseEntity, None, None]: """Yield subclasses of WikibaseEntity's with content prefilled. .. note:: Pages will be iterated in a different order than in the underlying pagelist. :param pagelist: an iterable that yields either WikibaseEntity objects, or Page objects linked to an ItemPage. :param groupsize: how many pages to query at a time """ if not hasattr(self, '_entity_namespaces'): self._cache_entity_namespaces() for batch in batched(pagelist, groupsize): req = {'ids': [], 'titles': [], 'sites': []} for p in batch: if isinstance(p, pywikibot.page.WikibaseEntity): ident = p._defined_by() for key in ident: req[key].append(ident[key]) elif (p.site == self and p.namespace() in self._entity_namespaces.values()): req['ids'].append(p.title(with_ns=False)) else: assert p.site.has_data_repository, \ 'Site must have a data repository' req['sites'].append(p.site.dbName()) req['titles'].append(p._link._text) req = self.simple_request(action='wbgetentities', **req) data = req.submit() for entity in data['entities']: if 'missing' in data['entities'][entity]: continue cls = self._type_to_class[data['entities'][entity]['type']] page = cls(self, entity) # No api call is made because item._content is given page._content = data['entities'][entity] with suppress(IsRedirectPageError): page.get() # cannot provide get_redirect=True (T145971) yield page
[docs] def get_property_type(self, prop: pywikibot.page.Property) -> str: """Obtain the type of a property. This is used specifically because we can cache the value for a much longer time (near infinite). .. versionadded:: 9.5 :raises NoWikibaseEntityError: *prop* does not exist """ params = {'action': 'wbgetentities', 'ids': prop.getID(), 'props': 'datatype'} expiry = datetime.timedelta(days=365 * 100) # Store it for 100 years req = self._request(expiry=expiry, parameters=params) data = req.submit() # the IDs returned from the API can be upper or lowercase, depending # on the version. See bug T55894 for more information. try: entity = data['entities'][prop.getID()] except KeyError: entity = data['entities'][prop.getID().lower()] if 'missing' in entity: raise NoWikibaseEntityError( prop if isinstance(prop, pywikibot.page.WikibaseEntity) else pywikibot.page.WikibaseEntity(self, prop.getID()) ) return entity['datatype']
[docs] @deprecated('get_property_type', since='9.5.0') def getPropertyType(self, prop): """Obtain the type of a property. .. deprecated:: 9.5 Use :meth:`get_property_type` instead. """ try: return self.get_property_type(prop) except NoWikibaseEntityError as exc: raise KeyError(f'{exc.entity.id} does not exist') from None
[docs] @need_right('edit') def editEntity(self, entity: pywikibot.page.WikibaseEntity | dict, data: dict, bot: bool = True, **kwargs) -> dict: """Edit entity. .. note:: This method is unable to create entities other than ``item`` if dict with API parameters was passed to *entity* parameter. .. versionchanged:: 9.4 *tags* keyword argument was added :param entity: Page to edit, or dict with API parameters to use for entity identification. :param data: data updates :param bot: Whether to mark the edit as a bot edit. :keyword int baserevid: The numeric identifier for the revision to base the modification on. This is used for detecting conflicts during save. :keyword bool clear: If set, the complete entity is emptied before proceeding. The entity will not be saved before it is filled with the *data*, possibly with parts excluded. :keyword str summary: Summary for the edit. Will be prepended by an automatically generated comment. The length limit of the autocomment together with the summary is 260 characters. Be aware that everything above that limit will be cut off. :keyword Iterable[str] | str tags: Change tags to apply to the revision. :return: New entity data """ # this changes the reference to a new object data = dict(data) if isinstance(entity, pywikibot.page.WikibaseEntity): params = entity._defined_by(singular=True) if 'id' in params and params['id'] == '-1': del params['id'] if not params: params['new'] = entity.entity_type data_for_new_entity = entity.get_data_for_new_entity() data.update(data_for_new_entity) else: if 'id' in entity and entity['id'] == '-1': del entity['id'] params = dict(entity) if not params: # If no identification was provided params['new'] = 'item' params.update(action='wbeditentity', bot=bot) if kwargs.get('baserevid'): params['baserevid'] = kwargs['baserevid'] params['token'] = self.tokens['csrf'] for arg, param in kwargs.items(): if arg in ['clear', 'summary', 'tags']: params[arg] = param elif arg != 'baserevid': warn(f'Unknown wbeditentity parameter {arg} ignored', UserWarning, 2) params['data'] = json.dumps(data) req = self.simple_request(**params) return req.submit()
[docs] @need_right('edit') def addClaim(self, entity: pywikibot.page.WikibaseEntity, claim: pywikibot.page.Claim, bot: bool = True, summary: str | None = None, tags: str | None = None) -> None: """Add a claim. .. versionchanged:: 9.4 *tags* parameter was added :param entity: Entity to modify :param claim: Claim to be added :param bot: Whether to mark the edit as a bot edit :param summary: Edit summary :param tags: Change tags to apply to the revision """ claim.snak = entity.getID() + '$' + str(uuid.uuid4()) params = { 'action': 'wbsetclaim', 'claim': json.dumps(claim.toJSON()), 'baserevid': entity.latest_revision_id, 'summary': summary, 'tags': tags, 'bot': bot, 'token': self.tokens['csrf'], } req = self.simple_request(**params) data = req.submit() # Update the item if claim.getID() in entity.claims: entity.claims[claim.getID()].append(claim) else: entity.claims[claim.getID()] = [claim] entity.latest_revision_id = data['pageinfo']['lastrevid']
[docs] @need_right('edit') def changeClaimTarget(self, claim: pywikibot.Claim, snaktype: str = 'value', bot: bool = True, summary: str | None = None, tags: str | None = None): """Set the claim target to the value of the provided claim target. .. versionchanged:: 9.4 *tags* parameter was added :param claim: The source of the claim target value :param snaktype: An optional snaktype ('value', 'novalue' or 'somevalue'). Default: 'value' :param bot: Whether to mark the edit as a bot edit :param summary: Edit summary :param tags: Change tags to apply to the revision """ if claim.isReference or claim.isQualifier: raise NotImplementedError if not claim.snak: # We need to already have the snak value raise NoPageError(claim) params = { 'action': 'wbsetclaimvalue', 'claim': claim.snak, 'snaktype': snaktype, 'summary': summary, 'tags': tags, 'bot': bot, 'token': self.tokens['csrf'], } if snaktype == 'value': params['value'] = json.dumps(claim._formatValue()) params['baserevid'] = claim.on_item.latest_revision_id req = self.simple_request(**params) return req.submit()
[docs] @need_right('edit') def save_claim(self, claim: pywikibot.page.Claim, summary: str | None = None, bot: bool = True, tags: str | None = None): """Save the whole claim to the wikibase site. .. versionchanged:: 9.4 *tags* parameter was added :param claim: The claim to save :param bot: Whether to mark the edit as a bot edit :param summary: Edit summary :param tags: Change tags to apply to the revision :raises NoPageError: missing the the snak value :raises NotImplementedError: ``claim.isReference`` or ``claim.isQualifier`` is given """ if claim.isReference or claim.isQualifier: raise NotImplementedError if not claim.snak: # We need to already have the snak value raise NoPageError(claim) params = { 'action': 'wbsetclaim', 'claim': json.dumps(claim.toJSON()), 'baserevid': claim.on_item.latest_revision_id, 'summary': summary, 'tags': tags, 'bot': bot, 'token': self.tokens['csrf'], } req = self.simple_request(**params) data = req.submit() claim.on_item.latest_revision_id = data['pageinfo']['lastrevid'] return data
[docs] @need_right('edit') @remove_last_args(['baserevid']) # since 7.0.0 def editSource(self, claim: pywikibot.Claim, source: pywikibot.Claim, new: bool = False, bot: bool = True, summary: str | None = None, tags: str | None = None): """Create/Edit a source. .. versionchanged:: 7.0 deprecated *baserevid* parameter was removed .. versionchanged:: 9.4 *tags* parameter was added :param claim: A Claim object to add the source to. :param source: A Claim object to be used as a source. :param new: Whether to create a new one if the "source" already exists. :param bot: Whether to mark the edit as a bot edit. :param summary: Edit summary. :param tags: Change tags to apply to the revision. :raises ValueError: The claim cannot have a source. """ if claim.isReference or claim.isQualifier: raise ValueError('The claim cannot have a source.') params = { 'action': 'wbsetreference', 'statement': claim.snak, 'baserevid': claim.on_item.latest_revision_id, 'summary': summary, 'tags': tags, 'bot': bot, 'token': self.tokens['csrf'], } # build up the snak sources = source if isinstance(source, list) else [source] snak = {} for sourceclaim in sources: datavalue = sourceclaim._formatDataValue() valuesnaks = snak.get(sourceclaim.getID(), []) valuesnaks.append({ 'snaktype': 'value', 'property': sourceclaim.getID(), 'datavalue': datavalue, }) snak[sourceclaim.getID()] = valuesnaks # set the hash if the source should be changed. # if present, all claims of one source have the same hash if not new and hasattr(sourceclaim, 'hash'): params['reference'] = sourceclaim.hash params['snaks'] = json.dumps(snak) req = self.simple_request(**params) return req.submit()
[docs] @need_right('edit') @remove_last_args(['baserevid']) # since 7.0.0 def editQualifier(self, claim: pywikibot.Claim, qualifier: pywikibot.Claim, new: bool = False, bot: bool = True, summary: str | None = None, tags: str | None = None): """Create/Edit a qualifier. .. versionchanged:: 7.0 deprecated *baserevid* parameter was removed .. versionchanged:: 9.4 *tags* parameter was added :param claim: A Claim object to add the qualifier to :param qualifier: A Claim object to be used as a qualifier :param new: Whether to create a new one if the qualifier already exists :param bot: Whether to mark the edit as a bot edit :param summary: Edit summary :param tags: Change tags to apply to the revision :raises ValueError: The claim cannot have a qualifier. """ if claim.isReference or claim.isQualifier: raise ValueError('The claim cannot have a qualifier.') params = { 'action': 'wbsetqualifier', 'claim': claim.snak, 'baserevid': claim.on_item.latest_revision_id, 'summary': summary, 'tags': tags, 'bot': bot, 'token': self.tokens['csrf'], } if (not new and hasattr(qualifier, 'hash') and qualifier.hash is not None): params['snakhash'] = qualifier.hash # build up the snak if qualifier.getSnakType() == 'value': params['value'] = json.dumps(qualifier._formatValue()) params['snaktype'] = qualifier.getSnakType() params['property'] = qualifier.getID() req = self.simple_request(**params) return req.submit()
[docs] @need_right('edit') @remove_last_args(['baserevid']) # since 7.0.0 def removeClaims(self, claims: list[pywikibot.Claim], bot: bool = True, summary: str | None = None, tags: str | None = None): """Remove claims. .. versionchanged:: 7.0 deprecated *baserevid* parameter was removed .. versionchanged:: 9.4 *tags* parameter was added :param claims: Claims to be removed :param bot: Whether to mark the edit as a bot edit :param summary: Edit summary :param tags: Change tags to apply to the revision """ # Check on_item for all additional claims items = {claim.on_item for claim in claims if claim.on_item} assert len(items) == 1 baserevid = items.pop().latest_revision_id params = { 'action': 'wbremoveclaims', 'claim': '|'.join(claim.snak for claim in claims), 'baserevid': baserevid, 'summary': summary, 'tags': tags, 'bot': bot, 'token': self.tokens['csrf'], } req = self.simple_request(**params) return req.submit()
[docs] @need_right('edit') @remove_last_args(['baserevid']) # since 7.0.0 def removeSources(self, claim: pywikibot.Claim, sources: list[pywikibot.Claim], bot: bool = True, summary: str | None = None, tags: str | None = None): """Remove sources. .. versionchanged:: 7.0 deprecated `baserevid` parameter was removed .. versionchanged:: 9.4 *tags* parameter was added :param claim: A Claim object to remove the sources from :param sources: A list of Claim objects that are sources :param bot: Whether to mark the edit as a bot edit :param summary: Edit summary :param tags: Change tags to apply to the revision """ params = { 'action': 'wbremovereferences', 'statement': claim.snak, 'references': '|'.join(source.hash for source in sources), 'baserevid': claim.on_item.latest_revision_id, 'summary': summary, 'tags': tags, 'bot': bot, 'token': self.tokens['csrf'], } req = self.simple_request(**params) return req.submit()
[docs] @need_right('edit') @remove_last_args(['baserevid']) # since 7.0.0 def remove_qualifiers(self, claim: pywikibot.Claim, qualifiers: list[pywikibot.Claim], bot: bool = True, summary: str | None = None, tags: str | None = None): """Remove qualifiers. .. versionchanged:: 7.0 deprecated `baserevid` parameter was removed .. versionchanged:: 9.4 *tags* parameter was added :param claim: A Claim object to remove the qualifier from :param qualifiers: Claim objects currently used as a qualifiers :param bot: Whether to mark the edit as a bot edit :param summary: Edit summary :param tags: Change tags to apply to the revision """ params = { 'action': 'wbremovequalifiers', 'claim': claim.snak, 'qualifiers': [qualifier.hash for qualifier in qualifiers], 'baserevid': claim.on_item.latest_revision_id, 'summary': summary, 'tags': tags, 'bot': bot, 'token': self.tokens['csrf'], } req = self.simple_request(**params) return req.submit()
[docs] @need_right('edit') def linkTitles(self, page1: pywikibot.Page, page2: pywikibot.Page, bot: bool = True) -> dict: """Link two pages together. .. versionchanged:: 9.4 *tags* parameter was added :param page1: First page to link :param page2: Second page to link :param bot: Whether to mark the edit as a bot edit :return: dict API output """ params = { 'action': 'wblinktitles', 'tosite': page1.site.dbName(), 'totitle': page1.title(), 'fromsite': page2.site.dbName(), 'fromtitle': page2.title(), 'bot': bot, 'token': self.tokens['csrf'] } req = self.simple_request(**params) return req.submit()
[docs] @need_right('item-merge') def mergeItems(self, from_item: pywikibot.ItemPage, to_item: pywikibot.ItemPage, ignore_conflicts: list[str] | None = None, summary: str | None = None, bot: bool = True, tags: str | None = None) -> dict: """Merge two items together. .. versionchanged:: 9.4 *tags* parameter was added :param from_item: Item to merge from :param to_item: Item to merge into :param ignore_conflicts: Which type of conflicts ('description', 'sitelink', and 'statement') should be ignored :param summary: Edit summary :param bot: Whether to mark the edit as a bot edit :param tags: Change tags to apply to the revision :return: dict API output """ params = { 'action': 'wbmergeitems', 'fromid': from_item.getID(), 'toid': to_item.getID(), 'ignoreconflicts': ignore_conflicts, 'summary': summary, 'tags': tags, 'bot': bot, 'token': self.tokens['csrf'], } req = self.simple_request(**params) return req.submit()
[docs] @need_right('item-merge') @need_extension('WikibaseLexeme') def mergeLexemes(self, from_lexeme, to_lexeme, summary=None, *, bot: bool = True) -> dict: """Merge two lexemes together. :param from_lexeme: Lexeme to merge from :type from_lexeme: pywikibot.LexemePage :param to_lexeme: Lexeme to merge into :type to_lexeme: pywikibot.LexemePage :param summary: Edit summary :type summary: str :keyword bot: Whether to mark the edit as a bot edit :return: dict API output """ params = { 'action': 'wblmergelexemes', 'source': from_lexeme.getID(), 'target': to_lexeme.getID(), 'token': self.tokens['csrf'], 'summary': summary, 'bot': bot, } req = self.simple_request(**params) return req.submit()
[docs] @need_right('item-redirect') def set_redirect_target(self, from_item, to_item, bot: bool = True): """Make a redirect to another item. :param to_item: title of target item. :type to_item: pywikibot.ItemPage :param from_item: Title of the item to be redirected. :type from_item: pywikibot.ItemPage :param bot: Whether to mark the edit as a bot edit """ params = { 'action': 'wbcreateredirect', 'from': from_item.getID(), 'to': to_item.getID(), 'token': self.tokens['csrf'], 'bot': bot, } req = self.simple_request(**params) return req.submit()
[docs] def search_entities(self, search: str, language: str, total: int | None = None, **kwargs): """Search for pages or properties that contain the given text. :param search: Text to find. :param language: Language to search in. :param total: Maximum number of pages to retrieve in total, or None in case of no limit. :return: 'search' list from API output. :rtype: Generator """ lang_codes = self._paraminfo.parameter('wbsearchentities', 'language')['type'] if language not in lang_codes: raise ValueError('Data site used does not support provided ' 'language.') if 'site' in kwargs: if kwargs['site'].sitename != self.sitename: raise ValueError('The site given in the kwargs is different.') warn('search_entities should not get a site via kwargs.', UserWarning, 2) del kwargs['site'] parameters = dict(search=search, language=language, **kwargs) gen = self._generator(api.APIGenerator, type_arg='wbsearchentities', data_name='search', total=total, parameters=parameters) return gen
[docs] def parsevalue(self, datatype: str, values: list[str], options: dict[str, Any] | None = None, language: str | None = None, validate: bool = False) -> list[Any]: """Send data values to the wikibase parser for interpretation. .. versionadded:: 7.5 .. seealso:: `wbparsevalue API <https://www.wikidata.org/w/api.php?action=help&modules=wbparsevalue>`_ :param datatype: datatype of the values being parsed. Refer the API for a valid datatype. :param values: list of values to be parsed :param options: any additional options for wikibase parser (for time, 'precision' should be specified) :param language: code of the language to parse the value in :param validate: whether parser should provide data validation as well as parsing :return: list of parsed values :raises ValueError: parsing failed due to some invalid input values """ params = { 'action': 'wbparsevalue', 'datatype': datatype, 'values': values, 'options': json.dumps(options or {}), 'validate': validate, 'uselang': language or 'en', } req = self.simple_request(**params) try: data = req.submit() except APIError as e: if e.code.startswith('wikibase-parse-error'): for err in e.other['results']: if 'error' in err: pywikibot.error('{error-info} for value {raw!r}, ' '{expected-format!r} format expected' .format_map(err)) raise ValueError(e) from None raise if 'results' not in data: raise RuntimeError( f"Unexpected missing 'results' in query data\n{data}") results = [] for result_hash in data['results']: if 'value' not in result_hash: # There should be an APIError occurred already raise RuntimeError("Unexpected missing 'value' in query data:" f'\n{result_hash}') results.append(result_hash['value']) return results
@need_right('edit') def _wbset_action(self, itemdef, action: str, action_data, **kwargs) -> dict: """Execute wbset{action} on a Wikibase entity. Supported actions are: wbsetaliases, wbsetdescription, wbsetlabel and wbsetsitelink wbsetaliases: dict shall have the following structure: .. code-block:: python { 'language': value (str), 'add': list of language codes (str), 'remove': list of language codes (str), 'set' list of language codes (str), } 'add' and 'remove' are alternative to 'set' wbsetdescription and wbsetlabel: dict shall have keys 'language', 'value' wbsetsitelink: dict shall have keys 'linksite', 'linktitle' and optionally 'badges' :param itemdef: Entity to modify or create :type itemdef: str, WikibaseEntity or Page connected to such item :param action: wbset{action} to perform: 'wbsetaliases', 'wbsetdescription', 'wbsetlabel', 'wbsetsitelink' :param action_data: data to be used in API request, see API help :type action_data: SiteLink or dict :keyword bot: Whether to mark the edit as a bot edit, default is True :type bot: bool :keyword tags: Change tags to apply with the edit :type tags: list of str :return: query result :raises: AssertionError, TypeError """ def format_sitelink(sitelink): """Convert SiteLink to a dict accepted by wbsetsitelink API.""" if isinstance(sitelink, pywikibot.page.SiteLink): _dict = { 'linksite': sitelink._sitekey, 'linktitle': sitelink._rawtitle, 'badges': '|'.join([b.title() for b in sitelink.badges]), } else: _dict = sitelink return _dict def prepare_data(action, data): """Prepare data as expected by API.""" if action == 'wbsetaliases': res = data keys = set(res) assert keys < {'language', 'add', 'remove', 'set'} assert 'language' in keys assert ({'add', 'remove', 'set'} & keys) assert ({'add', 'set'} >= keys) assert ({'remove', 'set'} >= keys) elif action in ('wbsetlabel', 'wbsetdescription'): res = data keys = set(res) assert keys == {'language', 'value'} elif action == 'wbsetsitelink': res = format_sitelink(data) keys = set(res) assert keys >= {'linksite'} assert keys <= {'linksite', 'linktitle', 'badges'} else: raise ValueError('Something has gone wrong ...') return res # Supported actions assert action in ('wbsetaliases', 'wbsetdescription', 'wbsetlabel', 'wbsetsitelink'), \ f'action {action} not supported.' # prefer ID over (site, title) if isinstance(itemdef, str): itemdef = self.get_entity_for_entity_id(itemdef) elif isinstance(itemdef, pywikibot.Page): itemdef = pywikibot.ItemPage.fromPage(itemdef, lazy_load=True) elif not isinstance(itemdef, pywikibot.page.WikibaseEntity): raise TypeError('itemdef shall be str, WikibaseEntity or Page') params = itemdef._defined_by(singular=True) # TODO: support 'new' baserevid = kwargs.pop( 'baserevid', itemdef.latest_revision_id if 'id' in params else 0 ) params.update( {'baserevid': baserevid, 'action': action, 'token': self.tokens['csrf'], 'bot': kwargs.pop('bot', True), }) params.update(prepare_data(action, action_data)) for arg, param in kwargs.items(): if arg in ['summary', 'tags']: params[arg] = param else: warn(f'Unknown parameter {arg} for action {action}, ignored', UserWarning, 2) req = self.simple_request(**params) return req.submit()
[docs] def wbsetaliases(self, itemdef, aliases, **kwargs): """Set aliases for a single Wikibase entity. See self._wbset_action() for parameters """ return self._wbset_action(itemdef, 'wbsetaliases', aliases, **kwargs)
[docs] def wbsetdescription(self, itemdef, description, **kwargs): """Set description for a single Wikibase entity. See self._wbset_action() """ return self._wbset_action(itemdef, 'wbsetdescription', description, **kwargs)
[docs] def wbsetlabel(self, itemdef, label, **kwargs): """Set label for a single Wikibase entity. See self._wbset_action() for parameters """ return self._wbset_action(itemdef, 'wbsetlabel', label, **kwargs)
[docs] @need_right('edit') @need_extension('WikibaseLexeme') def add_form(self, lexeme, form, *, bot: bool = True, baserevid=None) -> dict: """Add a form. :param lexeme: Lexeme to modify :type lexeme: pywikibot.LexemePage :param form: Form to be added :type form: pywikibot.LexemeForm :keyword bot: Whether to mark the edit as a bot edit :keyword baserevid: Base revision id override, used to detect conflicts. :type baserevid: long """ params = { 'action': 'wbladdform', 'lexemeId': lexeme.getID(), 'data': json.dumps(form.toJSON()), 'bot': bot, 'token': self.tokens['csrf'], } if baserevid: params['baserevid'] = baserevid req = self.simple_request(**params) return req.submit()
[docs] @need_right('edit') @need_extension('WikibaseLexeme') def remove_form(self, form, *, bot: bool = True, baserevid=None) -> dict: """Remove a form. :param form: Form to be removed :type form: pywikibot.LexemeForm :keyword bot: Whether to mark the edit as a bot edit :keyword baserevid: Base revision id override, used to detect conflicts. :type baserevid: long """ params = { 'action': 'wblremoveform', 'id': form.getID(), 'bot': bot, 'token': self.tokens['csrf'], } if baserevid: params['baserevid'] = baserevid req = self.simple_request(**params) return req.submit()
[docs] @need_right('edit') @need_extension('WikibaseLexeme') def edit_form_elements(self, form, data, *, bot: bool = True, baserevid=None) -> dict: """Edit lexeme form elements. :param form: Form :type form: pywikibot.LexemeForm :param data: data updates :type data: dict :keyword bot: Whether to mark the edit as a bot edit :keyword baserevid: Base revision id override, used to detect conflicts. :type baserevid: long :return: New form data """ params = { 'action': 'wbleditformelements', 'formId': form.getID(), 'data': json.dumps(data), 'bot': bot, 'token': self.tokens['csrf'], } if baserevid: params['baserevid'] = baserevid req = self.simple_request(**params) return req.submit()