-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathceleste_bot.py
296 lines (276 loc) · 12.2 KB
/
celeste_bot.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
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
"""
celeste_bot.py
"""
import requests
import re
from time import sleep
from datetime import datetime
from typing import Callable
from enum import IntEnum
from data_models import CelesteGameVersion, Credentials, CelesteGames
from random import randint, random
from requests.models import Response
from urllib.parse import ParseResult, urlparse
from twitch import TwitchHelix
from twitch.exceptions import (
TwitchAttributeException,
TwitchOAuthException,
TwitchAuthException,
)
from twitch.constants import OAUTH_SCOPE_ANALYTICS_READ_EXTENSIONS
__version__ = "1.1"
# funny API cache hack, SRC admins hate him
QUERY_TABLE: dict = {
0: "game",
1: "category",
2: "level",
3: "platform",
4: "region",
5: "emulated",
6: "date",
7: "submitted",
8: "status",
9: "verify-date",
}
def print_with_timestamp(out: str) -> None:
print(datetime.today().strftime("%d.%m.%Y %H:%M:%S").ljust(24) + str(out))
class SubmissionErrors(IntEnum):
ERROR_SUBMITTED_RTA = 0
ERROR_NO_VERSION = 1
ERROR_INVALID_IGT = 2
ERROR_INVALID_VERSION = 3
ERROR_BAD_VOD = 4
class CelesteLeaderboardBot:
"""Class for leaderboard bot, interacting with speedrun.com API."""
ACCOUNT_NAME: str = "BadelineBot"
AGENT: str = f"celeste-leaderboard-bot{__version__}"
BASE_REASON: Callable = (
lambda x: f"{CelesteLeaderboardBot.ACCOUNT_NAME} found the following problem{x} with your submission, please edit it accordingly: "
)
REASON_TEXT: dict = {
0: "Your submission has real-time, leave the real-time column empty as shown here: https://i.imgur.com/zHhvKqw.png",
1: "You did not select a version, make sure to select the correct game version",
2: "Your submission has an invalid IGT, check the final time of your run and adjust the submission",
3: "The version you selected does not exist on your platform, please select the correct game version",
4: "The video you submitted is a Twitch past broadcast that will be deleted after a while. Please create a Twitch Highlight of your run before resubmitting.",
}
# regex for twitch videos ids, from my observation they are always 10 digits but leaving some leeway for potential backwards/forward compatibility
TWITCH_ID_MATCH: re.Pattern = re.compile("^\d{9,11}$")
def __init__(
self, credentials: Credentials, games: CelesteGames, timer: float
) -> None:
self.q_counter: int = 0
self.CREDS: Credentials = credentials
self.GAMES: CelesteGames = games
self.TIMER: int = int(timer)
self.TTV_CLIENT: TwitchHelix = TwitchHelix(
scopes=[OAUTH_SCOPE_ANALYTICS_READ_EXTENSIONS],
client_id=self.CREDS.twitch.client,
client_secret=self.CREDS.twitch.secret,
)
def reject_run(self, run_id: str, reason: str = None) -> None:
"""Rejects a run given by ID. Also posts reason if given."""
if reason is None:
reason = "" # avoid mutable defaults footpistol
# make POST request to endpoint used by webinterface
# actual PUT API endpoint is broken (again, suck a phat one, speedrun.com ...)
res = requests.post(
f"https://www.speedrun.com/run/{run_id}/reject",
headers={"Content-Type": "application/x-www-form-urlencoded"},
cookies={"PHPSESSID": self.CREDS.src.session},
data={"csrftoken": self.CREDS.src.csrf, "answer": reason},
)
res.raise_for_status()
@staticmethod
def valid_real_time(run: dict) -> bool:
"""Checks if any RTA is submitted, returns False if so."""
return run["times"]["realtime_t"] == 0
@staticmethod
def valid_default_version(
run: dict, version: CelesteGameVersion, **_kwargs
) -> bool:
"""Checks if the default version is submitted, returns False if so."""
return not (run["values"][version.variable_id] == version.default_ver)
@staticmethod
def valid_in_game_time(run: dict) -> bool:
"""Checks if the submitted IGT is invalid, returns False if so"""
return (round(1000 * run["times"]["ingame_t"]) % 17) == 0
@staticmethod
def valid_existing_version(
run: dict, version: CelesteGameVersion, **_kwargs
) -> bool:
"""Checks if the submitted version is available on the submitted platform, returns False if it isn't."""
try:
return not (
run["values"][version.variable_id]
in version.invalid_ver[run["system"]["platform"]]
)
# compatibility incase new platform gets added
except KeyError:
print_with_timestamp(
f'There was an error with checking for platform of ID {run["system"]["platform"]}'
)
return True
@staticmethod
def valid_persistent_vod(
run: dict, client: TwitchHelix, matcher: re.Pattern
) -> bool:
"""Checks if submitted VOD is a past broadcast, returns False if so."""
try:
link_list: list = run["videos"]["links"]
ttv_index: int = -1
par_res: ParseResult
for i, x in enumerate(link_list):
par: ParseResult = urlparse(x["uri"])
if "twitch.tv" in par.netloc:
ttv_index = i
par_res = par
break
# only check for twitch uri's
if ttv_index == -1:
return True
else:
try:
# comprehend new list of all parts of the URI after the domainname that match the ID regex, then pick the 0th element
# if there is no element, it will throw an IndexError, which is caught below
vid_id: int = int(
[ele for ele in par_res.path.split("/") if matcher.match(ele)][
0
]
)
vid_data: dict = client.get_videos(video_ids=[vid_id])[0]
if vid_data["type"] == "archive":
return False
else:
return True
# catch potential errors with exctracting vid id
except IndexError:
return True
except ValueError:
return True
# just in case there is no video
except KeyError:
return True
# catch httperror locally
except (
TwitchAttributeException,
TwitchOAuthException,
TwitchAuthException,
requests.exceptions.HTTPError,
) as error:
print_with_timestamp(
f"There was an error with a request on Twitch API: {error}"
)
return True
def main(self, cache: list = []) -> list:
"""
Main function.
Checks for the validity of any new submission and rejects them if necessary.
Doesn't check cached runs, given as list of IDs, and returns list of new cached runs of all runs that were cleared.
"""
print_with_timestamp("Executing main method")
new_cache: list = []
# get new oauth
try:
self.TTV_CLIENT.get_oauth()
except (
TwitchAttributeException,
TwitchOAuthException,
TwitchAuthException,
) as error:
print_with_timestamp(
f"There was an error with getting a Twitch OAuth token: {error}"
)
# loop over all games
for game in self.GAMES.games:
faulty_runs_of_game: list = []
# retrieve all new runs, added query params to try and circumvent caching (suck a phat one speedrun.com ...)
try:
rand_d: str = "desc" if random() < 0.5 else "asc"
runs_res: Response = requests.get(
f"https://www.speedrun.com/api/v1/runs?game={game.id}&status=new&direction={rand_d}&orderby={QUERY_TABLE[self.q_counter]}&max={randint(100, 200)}", # they hate me :^)
headers={
"User-Agent": CelesteLeaderboardBot.AGENT,
"Cache-Control": "no-cache",
},
)
# throw error if not 200 OK
if runs_res.status_code != 200:
raise Exception(
f"Response did not complete with 200 but instead with {runs_res.status_code}"
)
new_runs: dict = runs_res.json()["data"]
except Exception as e:
print_with_timestamp(
f"Could not retrieve runs for game {game.name} from API with the following error: {e}"
)
continue
# loop over all new runs of a given game
for this_run in new_runs:
# skip if already cached
if this_run["id"] in cache:
new_cache.append(this_run["id"])
continue
# validity checks
invalid_run: dict = {"id": this_run["id"], "faults": []}
if not CelesteLeaderboardBot.valid_real_time(this_run):
invalid_run["faults"].append(SubmissionErrors.ERROR_SUBMITTED_RTA)
if not CelesteLeaderboardBot.valid_default_version(
this_run, game.version
):
invalid_run["faults"].append(SubmissionErrors.ERROR_NO_VERSION)
if not CelesteLeaderboardBot.valid_in_game_time(this_run):
invalid_run["faults"].append(SubmissionErrors.ERROR_INVALID_IGT)
if not CelesteLeaderboardBot.valid_existing_version(
this_run, game.version
):
invalid_run["faults"].append(SubmissionErrors.ERROR_INVALID_VERSION)
if not CelesteLeaderboardBot.valid_persistent_vod(
this_run, self.TTV_CLIENT, self.TWITCH_ID_MATCH
):
invalid_run["faults"].append(SubmissionErrors.ERROR_BAD_VOD)
# push to list of faulty runs if an error was found, otherwise append to cache
if len(invalid_run["faults"]) > 0:
faulty_runs_of_game.append(invalid_run)
else:
new_cache.append(this_run["id"])
# loop over all invalid runs of this game
for this_run in faulty_runs_of_game:
# build reason string
full_reason: str
x: str = "s" if len(this_run["faults"]) > 1 else ""
full_reason = CelesteLeaderboardBot.BASE_REASON(x) + " || ".join(
[
CelesteLeaderboardBot.REASON_TEXT[fault]
for fault in this_run["faults"]
]
)
print_with_timestamp(
f'Found following problem{x} with run <{this_run["id"]}>: {this_run["faults"]}'
)
# reject run
try:
self.reject_run(this_run["id"], full_reason)
print_with_timestamp(f'Rejected run <{this_run["id"]}>')
except requests.exceptions.RequestException as error:
print_with_timestamp(error)
# increment query counter for anti-cache and return cached runs
self.q_counter = (self.q_counter + 1) % len(QUERY_TABLE.keys())
return new_cache
def start(self) -> None:
"""Start bot, blocking calling thread."""
print_with_timestamp("Started bot")
try:
cached_runs: list = []
while True:
cached_runs = self.main(cached_runs)
# loop sleeps instead of one big sleep or threading solution to allow for easier exiting
# application isn't time or accuracy critical so this type of interval implementation is sufficient
c = 0
while c <= self.TIMER:
sleep(1)
c = c + 1
except KeyboardInterrupt:
print_with_timestamp("Stopped bot")
except Exception as e:
print_with_timestamp(f"Stopped bot because of uncaught error: {e}")