PKKPs,==crapipy/__init__.py""" Clash Royale wrapper for cr-api.com """ __version__ = "1.7" from .client import Client from .client_async import AsyncClient from .exceptions import APITimeoutError, APIClientResponseError, APIError from .url import APIURL from .models import Clan, Player, Clans, Players, Tag, Tournament, EndPoints, TournamentsPKΫK杣crapipy/client.py""" cr-api client for Clash Royale. """ import json import logging import os import requests from requests.exceptions import HTTPError from .exceptions import APIError from .models import Clan, Clans, Player, Constants, Tag, Players, Tournament, EndPoints, Tournaments from .url import APIURL logger = logging.getLogger('__name__') logger.setLevel(logging.DEBUG) ch = logging.StreamHandler() ch.setLevel(logging.ERROR) formatter = logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s') ch.setFormatter(formatter) logger.addHandler(ch) class Client: """ API Client. """ def __init__(self, token=None): self._token = token @property def token(self): """Load token from environment if not defined""" if self._token is None: self._token = os.environ.get('TOKEN') return self._token def fetch(self, url, is_json=True): """Fetch URL. :param url: URL :return: Response in JSON """ headers = {'auth': self.token} try: r = requests.get(url, headers=headers) if is_json: data = r.json() else: data = r.text if r.status_code != 200: logger.error( "API Error | HTTP status {status} | url: {url}".format( status=r.status_code, url=url ) ) raise APIError(**data) if isinstance(data, dict): if data.get('error'): raise APIError(**data) except (HTTPError, ConnectionError, json.JSONDecodeError): raise APIError return data def get_clan(self, clan_tag): """Fetch a single clan.""" url = APIURL.clan.format(clan_tag) data = self.fetch(url) return Clan(data) def get_clans(self, clan_tags): """Fetch multiple clans. :param clan_tags: List of clan tags """ url = APIURL.clan.format(','.join(clan_tags)) data = self.fetch(url) return [Clan(d) for d in data] def get_player(self, tag: str): """Get player profile by tag. :param tag: :return: """ ptag = Tag(tag).tag url = APIURL.player.format(ptag) data = self.fetch(url) return Player(data) def get_players(self, tags): """Fetch multiple players from profile API.""" ptags = [Tag(tag).tag for tag in tags] url = APIURL.player.format(','.join(ptags)) data = self.fetch(url) return [Player(d) for d in data] def get_tournament(self, tag): """Get tournament detail.""" url = APIURL.tournaments.format(tag) data = self.fetch(url) return Tournament(data) def get_constants(self, key=None): """Fetch contants. :param key: Optional field. """ url = APIURL.constants data = self.fetch(url) return Constants(data) def get_top_players(self, location=''): """Fetch top players.""" url = APIURL.top_players.format(location) data = self.fetch(url) return Players(data) def get_top_clans(self, location=''): """Fetch top clans.""" url = APIURL.top_clans.format(location) data = self.fetch(url) return Clans(data) def get_endpoints(self): """Endpoints""" url = APIURL.endpoints data = self.fetch(url) return EndPoints(data) def get_version(self): """API verision.""" url = APIURL.version data = self.fetch(url, is_json=False) return data def get_popular_players(self): """Fetch popular players.""" url = APIURL.popular_players data = self.fetch(url) return Players(data) def get_popular_clans(self): """Fetch popular clans.""" url = APIURL.popular_clans data = self.fetch(url) return Clans(data) def get_popular_tournaments(self): """Fetch popular tournaments.""" url = APIURL.popular_tournaments data = self.fetch(url) return Tournaments(data) PKݫKRnr$$crapipy/client_async.py""" cr-api async client for Clash Royale. """ import asyncio import json import logging import os import aiohttp from .exceptions import APIError from .models import Clan, Tag, Player, Constants, Players, Clans, Tournament, EndPoints, Tournaments from .url import APIURL logger = logging.getLogger('__name__') logger.setLevel(logging.DEBUG) ch = logging.StreamHandler() ch.setLevel(logging.ERROR) formatter = logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s') ch.setFormatter(formatter) logger.addHandler(ch) class AsyncClient: """ API AsyncClient. """ def __init__(self, token=None): self._token = token @property def token(self): """Load token from environment if not defined""" if self._token is None: self._token = os.environ.get('TOKEN') return self._token async def fetch(self, url, is_json=True): """Fetch URL. :param url: URL :return: Response in JSON """ headers = {'auth': self.token} try: async with aiohttp.ClientSession() as session: async with session.get(url, headers=headers) as resp: if is_json: data = await resp.json() else: data = await resp.text() if resp.status != 200: logger.error( "API Error | HTTP status {status} | {errmsg} | url: {url}".format( status=resp.status, errmsg=data.get('error'), url=url ) ) raise APIError(**data) except (asyncio.TimeoutError, aiohttp.ClientResponseError, json.JSONDecodeError): raise APIError return data async def get_clan(self, clan_tag): """Fetch a single clan.""" url = APIURL.clan.format(clan_tag) data = await self.fetch(url) if isinstance(data, list): data = data[0] return Clan(data) async def get_clans(self, clan_tags): """Fetch multiple clans. :param clan_tags: List of clan tags """ url = APIURL.clan.format(','.join(clan_tags)) data = await self.fetch(url) return [Clan(d) for d in data] async def get_player(self, tag: str) -> Player: """Get player profile by tag. :param tag: :return: """ ptag = Tag(tag).tag url = APIURL.player.format(ptag) data = await self.fetch(url) return Player(data) async def get_players(self, tags): """Fetch multiple players from profile API.""" ptags = [Tag(tag).tag for tag in tags] url = APIURL.player.format(','.join(ptags)) data = await self.fetch(url) return [Player(d) for d in data] async def get_tournament(self, tag): """Get tournament detail.""" url = APIURL.tournaments.format(tag) data = await self.fetch(url) return Tournament(data) async def get_constants(self, key=None): """Fetch contants. :param key: Optional field. """ url = APIURL.constants data = await self.fetch(url) return Constants(data) async def get_top_players(self, location=''): """Fetch top players.""" url = APIURL.top_players.format(location) data = await self.fetch(url) return Players(data) async def get_top_clans(self, location=''): """Fetch top clans.""" url = APIURL.top_clans.format(location) data = await self.fetch(url) return Clans(data) async def get_endpoints(self): """Endpoints.""" url = APIURL.endpoints data = await self.fetch(url) return EndPoints(data) async def get_version(self): """API verision.""" url = APIURL.version data = await self.fetch(url, is_json=False) return data async def get_popular_players(self): """Fetch popular players.""" url = APIURL.popular_players data = await self.fetch(url) return Players(data) async def get_popular_clans(self): """Fetch popular players.""" url = APIURL.popular_clans data = await self.fetch(url) return Clans(data) async def get_popular_tournaments(self): """Fetch popular tournaments.""" url = APIURL.popular_tournaments data = await self.fetch(url) return Tournaments(data)PKMK~Mcrapipy/exceptions.py""" Wrapper exceptions """ class BaseException(Exception): def __init__(self): super().__init__() class APIError(BaseException): def __init__(self, error=None, status=None, message=None): super().__init__() self.error = error self.status = status self.message = message class APITimeoutError(APIError): def __init__(self, **kwargs): super().__init__(**kwargs) class APIClientResponseError(APIError): def __init__(self, **kwargs): super().__init__(**kwargs) PKīK crapipy/models.py""" Data models """ from box import Box, BoxList class BaseModel(Box): """ Base model. """ def __init__(self, *args, **kwargs): kwargs.update({ "camel_killer_box": True, }) super().__init__(*args, **kwargs) class BaseListModel(BoxList): """ Base model. """ def __init__(self, *args, **kwargs): kwargs.update({ "camel_killer_box": True, }) super().__init__(*args, **kwargs) class Clan(BaseModel): """Clan.""" def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) class Player(BaseModel): """Player profile.""" def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) class Tournament(BaseModel): """Tournament.""" def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) class Players(BaseListModel): """Top Players""" def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) class Clans(BaseListModel): """Top Clans.""" def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) class EndPoints(BaseListModel): """Endpoints.""" def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) class Tournaments(BaseListModel): """Endpoints.""" def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) class Constants(BaseModel): def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) def get_chest_by_index(self, index): """Return chest by index.""" order = self.chest_cycle.order return order[index % len(order)] def get_card(self, key=None, card_key=None, name=None): """Return card by any property""" for card in self.cards: if card.key == key: return card if card.card_key == card_key: return card if card.name == name: return card return None class Tag: """SuperCell tags.""" TAG_CHARACTERS = "0289PYLQGRJCUV" def __init__(self, tag: str): """Init. Remove # if found. Convert to uppercase. Convert Os to 0s if found. """ if tag.startswith('#'): tag = tag[1:] tag = tag.replace('O', '0') tag = tag.upper() self._tag = tag def __str__(self): return self._tag def __repr__(self): return self._tag @property def tag(self): """Return tag as str.""" return self._tag @property def valid(self): """Return true if tag is valid.""" for c in self.tag: if c not in self.TAG_CHARACTERS: return False return True @property def invalid_chars(self): """Return list of invalid characters.""" invalids = [] for c in self.tag: if c not in self.TAG_CHARACTERS: invalids.append(c) return invalids PKK ;K{{crapipy/url.pyclass APIURL: """ API URL """ clan = 'http://api.cr-api.com/clan/{}' player = 'http://api.cr-api.com/player/{}' constants = 'http://api.cr-api.com/constants' top_players = 'http://api.cr-api.com/top/players/{}' top_clans = 'http://api.cr-api.com/top/clans/{}' tournaments = 'http://api.cr-api.com/tournaments/{}' endpoints = 'http://api.cr-api.com/endpoints' version = 'http://api.cr-api.com/version' popular_players = 'http://api.cr-api.com/popular/players' popular_clans = 'http://api.cr-api.com/popular/clans' popular_tournaments = 'http://api.cr-api.com/popular/tournaments' PKL]K#Wcrapipy/util.py""" Utility functions """ from box import Box, BoxList def make_box(data): """Create Box instance from dict.""" return Box(data, camel_killer_box=True) def make_box_list(data): """Create BoxList instance from dict.""" return BoxList(data, camel_killer_box=True) PKаIK` ::!crapipy-1.7.dist-info/LICENSE.txtMIT License Copyright (c) 2017 SML Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. PK!H}0RRcrapipy-1.7.dist-info/WHEEL1 0 RZtMDtPI{w<wUnbaKM*A1ѭ g\Ic c~PK!Hr%crapipy-1.7.dist-info/METADATAuQN0S)Ӑ%a MtCrg lɶ+w k$ Σт-<)*'-!{I߲3+Q-8l8ʨQlȬ ,k0!W 'SyW+;^7ƝtxIч8 FsmAI{OjqѣwQ*4w/@2 Vp:fE_# "u45Ncȸ߯Ya3/$vO$%;ٱMO{[t oLR T8䣱XE3n ul{I7`MˋD0ؓLsn^l7G0c,PK!HSs$*crapipy-1.7.dist-info/RECORDuI@}l1Y4QDQ73  7]UvGx`4f EKo_j;GR5NRB+Yۇ^R9\u(_*ᴡo G]",Vɖ $1"/IA8<[Jq$B" t<2ΆK{$CFEK8%amѵ{Iד.l 4Hwf&XkE ̉ޭ|ճ˭R6U.e[u.1< w1D (0>z# ҷB;#⡝vǶse 4,몱t+}1< f,eqhF'o*ݧEq:GS:+g;;ֱJ8KA_!47Z>d#*쎺]L5c4.n̏yBO4K oWaѬ0!e'<4 T-w0PKKPs,==crapipy/__init__.pyPKΫK杣ncrapipy/client.pyPKݫKRnr$$@crapipy/client_async.pyPKMK~M$crapipy/exceptions.pyPKīK &crapipy/models.pyPKK ;K{{3crapipy/url.pyPKL]K#W5crapipy/util.pyPKаIK` ::!6crapipy-1.7.dist-info/LICENSE.txtPK!H}0RRm;crapipy-1.7.dist-info/WHEELPK!Hr%;crapipy-1.7.dist-info/METADATAPK!HSs$*=crapipy-1.7.dist-info/RECORDPK @