"""Favro API Client. Implements methods for interacting with the [Favro API](https://favro.com/developer/). """ import dataclasses from typing import Any import requests_cache import datetime from collections.abc import Iterator from logging import getLogger import requests import requests_util from .favro_data_model import ( Card, CardId, Collection, CollectionId, CustomFieldId, CustomFieldInfo, OrganizationId, SeqId, TagId, TagInfo, UserId, UserInfo, ) from .favro_markdown import CardContents logger = getLogger(__name__) # Endpoints URL_API_ROOT = 'https://favro.com/api/v1' URL_GET_CARDS = URL_API_ROOT + '/cards' URL_UPDATE_CARD = URL_API_ROOT + '/cards/{card_id}' URL_GET_USERS = URL_API_ROOT + '/users/' URL_GET_USER = URL_API_ROOT + '/users/{user_id}' URL_GET_TAG = URL_API_ROOT + '/tags/{tag_id}' URL_GET_CUSTOM_FIELD = URL_API_ROOT + '/customfields/{custom_field_id}' URL_GET_TASKS = URL_API_ROOT + '/tasks' URL_GET_COLLECTIONS = URL_API_ROOT + '/collections' class CardCache: def __init__(self): self.cards = [] def add_card(self, card: Card) -> None: self.remove(card.card_id) self.cards.append(card) def get_card_by_card_id(self, card_id: CardId) -> Card | None: for card in reversed(self.cards): if card.card_id == card_id: return card return None def get_card_by_seq_id(self, seq_id: SeqId) -> Card | None: for card in reversed(self.cards): if card.seq_id == seq_id: return card return None def remove(self, card_id: CardId) -> Card | None: while card := self.get_card_by_card_id(card_id): self.cards.remove(card) return card return None class FavroClient: """Favro API Client. Implements methods for interacting with the [Favro API](https://favro.com/developer/). """ def __init__( self, *, favro_org_id: OrganizationId, favro_username: str, favro_password: str, session: requests.Session | None = None, read_only=True, ): # Check arguments if not isinstance(favro_org_id, OrganizationId): msg = f'favro_org_id must be str, but was {type(favro_org_id)}' raise TypeError(msg) if not isinstance(favro_username, str): msg = f'favro_username must be str, but was {type(favro_username)}' raise TypeError(msg) if not isinstance(favro_password, str): msg = f'favro_password must be str, but was {type(favro_password)}' raise TypeError(msg) # Setup session self.session = session or requests.Session() self.session.auth = (favro_username, favro_password) self.session.headers.update( { 'organizationId': favro_org_id.raw_id, 'content-type': 'application/json', }, ) self.read_only = read_only self.card_cache = CardCache() self.user_cache = None # Setup caching requests_util.setup_limiter( self.session, URL_API_ROOT, datetime.timedelta(days=7), ) requests_util.setup_limiter( self.session, URL_GET_CARDS, datetime.timedelta(minutes=10), ) requests_util.setup_limiter( self.session, URL_GET_TASKS, datetime.timedelta(minutes=10), ) requests_util.setup_limiter( self.session, URL_GET_CUSTOM_FIELD, datetime.timedelta(days=30), ) def check_logged_in(self) -> None: next(self.get_todo_list_cards()) def get_todo_list_cards(self) -> Iterator[Card]: yield from self.get_cards(todo_list=True) def get_cards( self, *, seq_id: SeqId | None = None, collection_id: CollectionId | None = None, todo_list=False, ) -> Iterator[Card]: logger.info('Getting cards: seq_id=%s, collection_id=%s, todo_list=%s', seq_id, collection_id, todo_list) request = self._get_cards_request( seq_id=seq_id, todo_list=todo_list, collection_id=collection_id, ) for card in self._get_paginated(request, Card.from_json): self.card_cache.add_card(card) yield card def get_collections(self) -> Iterator[Collection]: request = requests.Request('GET', URL_GET_COLLECTIONS) yield from self._get_paginated(request, Collection.from_json) def _get_cards_prepared_request( self, **kwargs, ) -> requests.PreparedRequest: request = self._get_cards_request(**kwargs) return self.session.prepare_request(request) def _get_cards_request( self, seq_id: SeqId | None = None, todo_list: bool = False, collection_id: CollectionId | None = None, ) -> requests.Request: params = {'descriptionFormat': 'markdown'} if seq_id is not None: params['cardSequentialId'] = str(seq_id.raw_id) if todo_list is True: params['todoList'] = 'true' if collection_id is not None: params['collectionId'] = str(collection_id.raw_id) return requests.Request('GET', URL_GET_CARDS, params=params) def get_card(self, seq_id: SeqId) -> Card: if card := self.get_card_if_cached(seq_id): return card return next(self.get_cards(seq_id=seq_id)) def get_card_if_cached(self, seq_id: SeqId) -> Card | None: if card := self.card_cache.get_card_by_seq_id(seq_id): return card return None def get_card_by_card_id(self, card_id: CardId) -> Card: json_data = self._get_json(URL_UPDATE_CARD.format(card_id=card_id.raw_id)) return Card.from_json(json_data) def get_user(self, user_id: UserId) -> UserInfo: self._load_users() return self.user_cache[user_id] def _load_users(self) -> None: if self.user_cache is not None: return request = requests.Request('GET', URL_GET_USERS) user_cache = {} for user_info in self._get_paginated(request, UserInfo.from_json): user_cache[user_info.user_id] = user_info self.user_cache = user_cache logger.info('Loaded %s users', len(user_cache)) def get_tag(self, tag_id: TagId) -> TagInfo: json_data = self._get_json(URL_GET_TAG.format(tag_id=tag_id.raw_id)) return TagInfo.from_json(json_data) def get_custom_field(self, custom_field_id: CustomFieldId) -> CustomFieldInfo: json_data = self._get_json( URL_GET_CUSTOM_FIELD.format(custom_field_id=custom_field_id.raw_id), ) return CustomFieldInfo.from_json(json_data) def _invalidate_cache(self, card_id: CardId) -> None: card = self.card_cache.remove(card_id) if card: self.session.cache.delete( requests=[self._get_cards_prepared_request(seq_id=card.seq_id)], ) def update_card_contents_locally( self, card_id: CardId, card_contents: CardContents, ) -> Card | None: if card := self.card_cache.remove(card_id): card = dataclasses.replace( card, detailed_description=card_contents.description, name=card_contents.name, tags=[], # TODO? assignments=[], # TODO? dependencies=[], # TODO ) self.card_cache.add_card(card) return card def update_card_contents( self, card_id: CardId, card_contents: CardContents, ) -> Card | None: """Returns updated Card.""" if self.read_only == 'silent': logger.warning( 'FavroClient is silent read only: Discarding card description update', ) return None # TODO if self.read_only is True: msg = 'FavroClient is read only' raise Exception(msg) json_body = { 'name': card_contents.name, 'detailedDescription': card_contents.description, 'descriptionFormat': 'markdown', 'archived': card_contents.is_archived, } url = URL_UPDATE_CARD.format(card_id=card_id.raw_id) logger.info('Sending PUT: %s', url) response = self.session.put( url, json=json_body, ) response.raise_for_status() self._invalidate_cache(card_id) return self.update_card_contents_locally(card_id, card_contents) def _get_paginated( self, base_request: requests.Request, entity_from_json, ) -> Iterator: page = 0 request_id = None num_pages = 1 while page < num_pages: # Determine params for get_cards base_request.params['page'] = page base_request.params['request_id'] = request_id request = self.session.prepare_request(base_request) # Run query logger.debug('Sending GET: %s', request.url) response = self.session.send(request) response.raise_for_status() if not response.from_cache: logger.warning('Got new result: %s', request.url) json = response.json() for entity_json in json['entities']: entity = entity_from_json(entity_json) yield entity del entity_json, entity # Pagination bookkeeping page = json['page'] + 1 request_id = json['requestId'] num_pages = json['pages'] def _get_json(self, url: str) -> dict[str, Any]: logger.debug('Sending GET: %s', url) response = self.session.get(url) response.raise_for_status() return response.json()