-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathseason.py
251 lines (200 loc) · 7.33 KB
/
season.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
from __future__ import annotations
from data_structures.bset import BSet
from data_structures.referential_array import ArrayR
from dataclasses import dataclass
from team import Team
from typing import Generator, Union
@dataclass
class Game:
"""
Simple container for a game between two teams.
Both teams must be team objects, there cannot be a game without two teams.
Note: Python will automatically generate the init for you.
Use Game(home_team: Team, away_team: Team) to use this class.
See: https://docs.python.org/3/library/dataclasses.html
"""
home_team: Team = None
away_team: Team = None
class WeekOfGames:
"""
Simple container for a week of games.
A fixture must have at least one game.
"""
def __init__(self, week: int, games: ArrayR[Game]) -> None:
"""
Container for a week of games.
Args:
week (int): The week number.
games (ArrayR[Game]): The games for this week.
"""
self.games: ArrayR[Game] = games
self.week: int = week
def get_games(self) -> ArrayR:
"""
Returns the games in a given week.
Returns:
ArrayR: The games in a given week.
Complexity:
Best Case Complexity: O(1)
Worst Case Complexity: O(1)
"""
return self.games
def get_week(self) -> int:
"""
Returns the week number.
Returns:
int: The week number.
Complexity:
Best Case Complexity: O(1)
Worst Case Complexity: O(1)
"""
return self.week
def __iter__(self):
"""
Complexity:
Best Case Complexity:
Worst Case Complexity:
"""
raise NotImplementedError
def __next__(self):
"""
Complexity:
Best Case Complexity:
Worst Case Complexity:
"""
raise NotImplementedError
class Season:
def __init__(self, teams: ArrayR[Team]) -> None:
"""
Initializes the season with a schedule.
Args:
teams (ArrayR[Team]): The teams played in this season.
Complexity:
Best Case Complexity:
Worst Case Complexity:
"""
raise NotImplementedError
def _generate_schedule(self) -> ArrayR[ArrayR[Game]]:
"""
Generates a schedule by generating all possible games between the teams.
Return:
ArrayR[ArrayR[Game]]: The schedule of the season.
The outer array is the weeks in the season.
The inner array is the games for that given week.
Complexity:
Best Case Complexity: O(N^2) where N is the number of teams in the season.
Worst Case Complexity: O(N^2) where N is the number of teams in the season.
"""
num_teams: int = len(self.teams)
weekly_games: list[ArrayR[Game]] = []
flipped_weeks: list[ArrayR[Game]] = []
games: list[Game] = []
# Generate all possible matchups (team1 vs team2, team2 vs team1, etc.)
for i in range(num_teams):
for j in range(i + 1, num_teams):
games.append(Game(self.teams[i], self.teams[j]))
# Allocate games into each week ensuring no team plays more than once in a week
week: int = 0
while games:
current_week: list[Game] = []
flipped_week: list[Game] = []
used_teams: BSet = BSet()
week_game_no: int = 0
for game in games[:]: # Iterate over a copy of the list
if game.home_team.get_number() not in used_teams and game.away_team.get_number() not in used_teams:
current_week.append(game)
used_teams.add(game.home_team.get_number())
used_teams.add(game.away_team.get_number())
flipped_week.append(Game(game.away_team, game.home_team))
games.remove(game)
week_game_no += 1
weekly_games.append(ArrayR.from_list(current_week))
flipped_weeks.append(ArrayR.from_list(flipped_week))
week += 1
return ArrayR.from_list(weekly_games + flipped_weeks)
def simulate_season(self) -> None:
"""
Simulates the season.
Complexity:
Assume simulate_game is O(1)
Remember to define your variables and their complexity.
Best Case Complexity:
Worst Case Complexity:
"""
raise NotImplementedError
def delay_week_of_games(self, orig_week: int, new_week: Union[int, None] = None) -> None:
"""
Delay a week of games from one week to another.
Args:
orig_week (int): The original week to move the games from.
new_week (Union[int, None]): The new week to move the games to. If this is None, it moves the games to the end of the season.
Complexity:
Best Case Complexity:
Worst Case Complexity:
"""
raise NotImplementedError
def get_next_game(self) -> Union[Generator[Game], None]:
"""
Gets the next game in the season.
Returns:
Game: The next game in the season.
or None if there are no more games left.
Complexity:
Best Case Complexity:
Worst Case Complexity:
"""
raise NotImplementedError
def get_leaderboard(self) -> ArrayR[ArrayR[Union[int, str]]]:
"""
Generates the final season leaderboard.
Returns:
ArrayR(ArrayR[ArrayR[Union[int, str]]]):
Outer array represents each team in the leaderboard
Inner array consists of 10 elements:
- Team name (str)
- Games Played (int)
- Points (int)
- Wins (int)
- Draws (int)
- Losses (int)
- Goals For (int)
- Goals Against (int)
- Goal Difference (int)
- Previous Five Results (ArrayR(str)) where result should be WIN LOSS OR DRAW
Complexity:
Best Case Complexity:
Worst Case Complexity:
"""
raise NotImplementedError
def get_teams(self) -> ArrayR[Team]:
"""
Returns:
PlayerPosition (ArrayR(Team)): The teams participating in the season.
Complexity:
Best Case Complexity:
Worst Case Complexity:
"""
raise NotImplementedError
def __len__(self) -> int:
"""
Returns the number of teams in the season.
Complexity:
Best Case Complexity:
Worst Case Complexity:
"""
raise NotImplementedError
def __str__(self) -> str:
"""
Optional but highly recommended.
You may choose to implement this method to help you debug.
However your code must not rely on this method for its functionality.
Returns:
str: The string representation of the season object.
Complexity:
Analysis not required.
"""
return ""
def __repr__(self) -> str:
"""Returns a string representation of the Season object.
Useful for debugging or when the Season is held in another data structure."""
return str(self)