Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

User live score #78

Open
wants to merge 12 commits into
base: master
Choose a base branch
from
110 changes: 72 additions & 38 deletions fpl/fpl.py
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,8 @@
import itertools
import os

import requests

from .constants import API_URLS
from .models.classic_league import ClassicLeague
from .models.fixture import Fixture
Expand All @@ -44,6 +46,14 @@ class FPL:

def __init__(self, session):
self.session = session
static = requests.get(API_URLS["static"]).json() # use synchronous request
for k, v in static.items():
try:
v = {w["id"]: w for w in v}
except (KeyError, TypeError):
pass
setattr(self, k, v)
setattr(self, "current_gameweek", next(event for event in static["events"] if event["is_current"])['id'])
johnmadden86 marked this conversation as resolved.
Show resolved Hide resolved

async def get_user(self, user_id=None, return_json=False):
"""Returns the user with the given ``user_id``.
Expand Down Expand Up @@ -91,19 +101,19 @@ async def get_teams(self, team_ids=None, return_json=False):
:type return_json: bool
:rtype: list
"""
url = API_URLS["static"]
teams = await fetch(self.session, url)
teams = teams["teams"]
teams = getattr(self, "teams")

if team_ids:
team_ids = set(team_ids)
teams = [team for team in teams if team["id"] in team_ids]
teams = [team for team in teams.values() if team["id"] in team_ids]
else:
teams = [team for team in teams.values()]

if return_json:
return teams

return [Team(team_information, self.session)
for team_information in teams]
return {team_information["id"]: Team(team_information, self.session)
amosbastian marked this conversation as resolved.
Show resolved Hide resolved
for team_information in teams}

async def get_team(self, team_id, return_json=False):
"""Returns the team with the given ``team_id``.
Expand Down Expand Up @@ -145,9 +155,8 @@ async def get_team(self, team_id, return_json=False):
"""
assert 0 < int(
team_id) < 21, "Team ID must be a number between 1 and 20."
url = API_URLS["static"]
teams = await fetch(self.session, url)
team = next(team for team in teams["teams"]
teams = getattr(self, "teams")
team = next(team for team in teams.values()
if team["id"] == int(team_id))

if return_json:
Expand Down Expand Up @@ -177,6 +186,7 @@ async def get_player_summary(self, player_id, return_json=False):

return PlayerSummary(player_summary)

# not used
async def get_player_summaries(self, player_ids, return_json=False):
"""Returns a list of summaries of players whose ID are
in the ``player_ids`` list.
Expand Down Expand Up @@ -206,14 +216,15 @@ async def get_player_summaries(self, player_ids, return_json=False):
return [PlayerSummary(player_summary)
for player_summary in player_summaries]

async def get_player(self, player_id, players=None, include_summary=False,
async def get_player(self, player_id, players=None, gameweek=None, include_summary=False,
return_json=False):
"""Returns the player with the given ``player_id``.

Information is taken from e.g.:
https://fantasy.premierleague.com/api/bootstrap-static/
https://fantasy.premierleague.com/api/element-summary/1/ (optional)

:param gameweek: the current gameweek data (for applying live scores)
:param player_id: A player's ID.
:type player_id: string or int
:param list players: (optional) A list of players.
Expand All @@ -226,11 +237,10 @@ async def get_player(self, player_id, players=None, include_summary=False,
:raises ValueError: Player with ``player_id`` not found
"""
if not players:
players = await fetch(self.session, API_URLS["static"])
players = players["elements"]
players = getattr(self, "elements")

try:
player = next(player for player in players
player = next(player for player in players.values()
if player["id"] == player_id)
except StopIteration:
raise ValueError(f"Player with ID {player_id} not found")
Expand All @@ -240,12 +250,19 @@ async def get_player(self, player_id, players=None, include_summary=False,
player["id"], return_json=True)
player.update(player_summary)

player["image_url"] = f'https://platform-static-files.s3.amazonaws.com' \
johnmadden86 marked this conversation as resolved.
Show resolved Hide resolved
f'/premierleague/photos/players/110x140/p{player["code"]}.png'

if gameweek:
player["live_score"] = gameweek.elements[player_id]["stats"]["total_points"]
player["did_not_play"] = gameweek.elements[player_id]["did_not_play"]

if return_json:
return player

return Player(player, self.session)

async def get_players(self, player_ids=None, include_summary=False,
async def get_players(self, player_ids=None, include_summary=False, include_live=None,
return_json=False):
"""Returns either a list of *all* players, or a list of players whose
IDs are in the given ``player_ids`` list.
Expand All @@ -254,6 +271,7 @@ async def get_players(self, player_ids=None, include_summary=False,
https://fantasy.premierleague.com/api/bootstrap-static/
https://fantasy.premierleague.com/api/element-summary/1/ (optional)

:param include_live: (optional) include a player's live score
:param list player_ids: (optional) A list of player IDs
:param boolean include_summary: (optional) Includes a player's summary
if ``True``.
Expand All @@ -263,19 +281,25 @@ async def get_players(self, player_ids=None, include_summary=False,
:type return_json: bool
:rtype: list
"""
players = await fetch(self.session, API_URLS["static"])
players = players["elements"]
players = getattr(self, "elements")
gameweek = None

if not player_ids:
player_ids = [player["id"] for player in players]
player_ids = [player["id"] for player in players.values()]

if include_live:
gameweek = await self.get_gameweek(getattr(self, "current_gameweek"), include_live=True)

tasks = [asyncio.ensure_future(
self.get_player(
player_id, players, include_summary, return_json))
player_id, players, gameweek, include_summary, return_json))
for player_id in player_ids]
players = await asyncio.gather(*tasks)

return players
if return_json:
return list(filter(lambda p: p["id"] in player_ids, players))
amosbastian marked this conversation as resolved.
Show resolved Hide resolved

return {player.id: player for player in players}

async def get_fixture(self, fixture_id, return_json=False):
"""Returns the fixture with the given ``fixture_id``.
Expand Down Expand Up @@ -352,7 +376,7 @@ async def get_fixtures_by_id(self, fixture_ids, return_json=False):
if return_json:
return fixtures

return [Fixture(fixture) for fixture in fixtures]
return {fixture["id"]: Fixture(fixture) for fixture in fixtures}

async def get_fixtures_by_gameweek(self, gameweek, return_json=False):
"""Returns a list of all fixtures of the given ``gameweek``.
Expand All @@ -375,7 +399,7 @@ async def get_fixtures_by_gameweek(self, gameweek, return_json=False):
if return_json:
return fixtures

return [Fixture(fixture) for fixture in fixtures]
return {fixture["id"]: Fixture(fixture) for fixture in fixtures}

async def get_fixtures(self, return_json=False):
"""Returns a list of *all* fixtures.
Expand All @@ -402,18 +426,17 @@ async def get_fixtures(self, return_json=False):
if return_json:
return fixtures

return [Fixture(fixture) for fixture in fixtures]
return {fixture["id"]: Fixture(fixture) for fixture in fixtures}

async def get_gameweek(self, gameweek_id, include_live=False,
return_json=False):
async def get_gameweek(self, gameweek_id, include_live=False, return_json=False):
"""Returns the gameweek with the ID ``gameweek_id``.

Information is taken from e.g.:
https://fantasy.premierleague.com/api/bootstrap-static/
https://fantasy.premierleague.com/api/event/1/live/

:param int gameweek_id: A gameweek's ID.
:param bool include_summary: (optional) Includes a gameweek's live data
:param bool include_live: (optional) Includes a gameweek's live data
if ``True``.
:param return_json: (optional) Boolean. If ``True`` returns a ``dict``,
if ``False`` returns a :class:`Gameweek` object. Defaults to
Expand All @@ -422,11 +445,10 @@ async def get_gameweek(self, gameweek_id, include_live=False,
:rtype: :class:`Gameweek` or ``dict``
"""

static_gameweeks = await fetch(self.session, API_URLS["static"])
static_gameweeks = static_gameweeks["events"]
static_gameweeks = getattr(self, "events")

try:
static_gameweek = next(gameweek for gameweek in static_gameweeks if
static_gameweek = next(gameweek for gameweek in static_gameweeks.values() if
gameweek["id"] == gameweek_id)
except StopIteration:
raise ValueError(f"Gameweek with ID {gameweek_id} not found")
Expand All @@ -441,16 +463,25 @@ async def get_gameweek(self, gameweek_id, include_live=False,
# include live bonus points
if not static_gameweek['finished']:
fixtures = await self.get_fixtures_by_gameweek(gameweek_id)
fixtures = filter(lambda f: not f.finished, fixtures)
fixtures_not_finished = filter(lambda f: not f.finished, fixtures.values())
johnmadden86 marked this conversation as resolved.
Show resolved Hide resolved
bonus_for_gameweek = []
for fixture in fixtures:
for fixture in fixtures_not_finished:
bonus = fixture.get_bonus(provisional=True)
bonus_for_gameweek.extend(bonus['a'] + bonus['h'])
bonus_for_gameweek = {b['element']: b['value'] for b in bonus_for_gameweek}
for player_id, bonus_points in bonus_for_gameweek:
if live_gameweek["elements"][player_id]["bonus"] == 0:
live_gameweek["elements"][player_id]["bonus"] += bonus_points
live_gameweek["elements"][player_id]["total_points"] += bonus_points
for player_id, bonus_points in bonus_for_gameweek.items():
if live_gameweek["elements"][player_id]["stats"]["bonus"] == 0:
live_gameweek["elements"][player_id]["stats"]["bonus"] += bonus_points
live_gameweek["elements"][player_id]["stats"]["total_points"] += bonus_points

# mark players that did not play
fixtures_started = filter(lambda f: f.started, fixtures.values())
fixtures_started = list(map(lambda f: f.id, fixtures_started))
johnmadden86 marked this conversation as resolved.
Show resolved Hide resolved
for element in live_gameweek["elements"].values():
player_id = element["id"]
no_minutes = element["stats"]["minutes"] == 0
game_started = any([game["fixture"] in fixtures_started for game in element["explain"]])
live_gameweek["elements"][player_id]["did_not_play"] = no_minutes and game_started

static_gameweek.update(live_gameweek)

Expand All @@ -459,8 +490,7 @@ async def get_gameweek(self, gameweek_id, include_live=False,

return Gameweek(static_gameweek)

async def get_gameweeks(self, gameweek_ids=None, include_live=False,
return_json=False):
async def get_gameweeks(self, gameweek_ids=None, return_json=False):
"""Returns either a list of *all* gamweeks, or a list of gameweeks
whose IDs are in the ``gameweek_ids`` list.

Expand All @@ -480,11 +510,15 @@ async def get_gameweeks(self, gameweek_ids=None, include_live=False,
gameweek_ids = range(1, 39)

tasks = [asyncio.ensure_future(
self.get_gameweek(gameweek_id, include_live, return_json))
self.get_gameweek(gameweek_id, False, return_json))
for gameweek_id in gameweek_ids]

gameweeks = await asyncio.gather(*tasks)
return gameweeks

if return_json:
return gameweeks

return {gameweek.id: gameweek for gameweek in gameweeks}

async def get_classic_league(self, league_id, return_json=False):
"""Returns the classic league with the given ``league_id``. Requires
Expand Down
Loading