-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.py
254 lines (225 loc) · 10.1 KB
/
main.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
import datetime
import http.client
import json
import random
import sys
import threading
import typing
import urllib.error
import src.givebutter as givebutter
import src.startgg as startgg
import src.irc_client_listener as irc_client_listener
from time import sleep
from typing import Dict, List, Optional, Tuple
from src.givebutter_handlers import GivebutterDonationHandler
from src.givebutter_listeners import GivebutterListener
from src.irc_client import IRCClient
from src.irc_client_handlers import (
ReportTimeoutHandler, LeaveIfNotModdedHandler, PongIfPingedHandler,
TwitchChatLoginFailedHandler)
from src.streambrain import StreamBrain
from src.twitch import (
TwitchHTTPError, TwitchStreamData, TwitchOauthManager, get_streams)
from src.twitch_chat_command_handler import TwitchChatCommandHandler
CURRENT_CHANNELS_PATH = "current_channels.txt"
EVENT_PROMO_OPTOUTS_PATH = "event_promo_optouts.txt"
def get_twitch_streams(
twitch_oauth_manager: TwitchOauthManager, user_ids: List[int]=[],
user_logins: List[str]=[], game_ids: List[int]=[],
stream_type: Optional[str]=None, language: Optional[str]=None,
page_size: Optional[int]=None, max_pages: Optional[int]=None,
after: Optional[str]=None) -> Tuple[List[TwitchStreamData], str]:
already_failed = False
while True:
try:
return get_streams(
twitch_oauth_manager.access_token,
twitch_oauth_manager.client_id, user_ids, user_logins,
game_ids, stream_type, language, page_size, max_pages,
after)
except TwitchHTTPError as e:
if e.code != 401:
raise
if already_failed:
raise Exception(
"get_twitch_streams: Twitch API key is invalid.")
print("Couldn't get live streams. Invalid access token.")
print("Refreshing access token and trying again.")
twitch_oauth_manager.refresh()
already_failed = True
def get_startgg_league_events(
access_token: str, league_slug: str) -> List[Dict]:
return startgg.get_league_events(access_token, league_slug)
class Routine:
def __init__(self, interval_ticks: int, delay_ticks: int=0):
self._interval_ticks = interval_ticks
self._delay_ticks = delay_ticks
self._remaining_ticks = delay_ticks
def run(self):
pass
def increment(self, increment_ticks: int):
if self._remaining_ticks <= 0:
self.run()
self._remaining_ticks = (
self._interval_ticks + self._remaining_ticks)
self._remaining_ticks -= increment_ticks
class Schedule:
def __init__(self):
self.routines = []
self._incrementing_forever = False
def increment(self, increment_ticks: int):
for routine in self.routines:
# diagnostic
try:
routine.increment(increment_ticks)
except Exception as e:
exc_type, exc_object, exc_traceback = sys.exc_info()
exc_filename = exc_traceback.tb_frame.f_code.co_filename
exc_line_no = exc_traceback.tb_lineno
with open("error_that_killed_me.txt", "w") as crashlog_file:
crashlog_file.write(
f"{exc_type.__name__} in {exc_filename} at "
f"line {exc_line_no}: {exc_object}")
raise
def increment_loop(self, sleep_sec: int, tick_sec_ratio: int=1):
self._incrementing_forever = True
while self._incrementing_forever:
self.increment(sleep_sec * tick_sec_ratio)
sleep(sleep_sec)
def stop(self):
self._incrementing_forever = False
class JazzyEventPromoRoutine(Routine):
def __init__(
self, startgg_access_token: str,
twitch_oauth_manager: TwitchOauthManager, irc_client: IRCClient,
interval_sec: int, delay_sec: int=0):
self._startgg_access_token = startgg_access_token
self._twitch_oauth_manager = twitch_oauth_manager
self._irc_client = irc_client
self._plugged_event_ids = []
super().__init__(interval_sec, delay_sec)
def run(self):
# This code is hideous and I'm sorry. I'm gonna refactor it.
with open(EVENT_PROMO_OPTOUTS_PATH) as event_promo_optouts_file:
optouts = event_promo_optouts_file.read().split()
# diagnostic
print("Getting startgg events.")
events = get_startgg_league_events(
self._startgg_access_token, "the-jazzy-circuit-4")
now = datetime.datetime.now()
max_datetime = now + datetime.timedelta(days=45)
upcoming_events = {}
for event in events:
if (
event["startAt"] > now.timestamp()
and event["startAt"] < max_datetime.timestamp()):
upcoming_events[event["id"]] = event
imminent_events = upcoming_events.copy()
for event_id in self._plugged_event_ids:
try:
del imminent_events[event_id]
except KeyError:
pass
if not imminent_events:
imminent_events = upcoming_events
self._plugged_event_ids = []
random_event_id = random.choice(list(imminent_events.keys()))
plug_event = imminent_events[random_event_id]
tournament_name = plug_event["tournament"]["name"]
tournament_city = plug_event["tournament"]["city"]
tournament_state = plug_event["tournament"]["addrState"]
event_ts = plug_event["startAt"]
tournament_datetime = datetime.datetime.fromtimestamp(event_ts)
tournament_day = tournament_datetime.day
if 11 <= tournament_day <= 13:
day_suffix = "th"
else:
day_suffix_map = {1: "st", 2: "nd", 3: "rd"}
day_suffix = day_suffix_map.get(tournament_day % 10, "th")
tournament_date = (
f"{tournament_datetime.strftime('%a, %b ')} "
f"{tournament_day}{day_suffix}")
tournament_url = f"start.gg/{plug_event['tournament']['slug']}"
promo_channels = []
for channel_name in self._irc_client.channels:
if channel_name not in optouts:
promo_channels.append(channel_name)
# diagnostic
print("Checking if Twitch streams are online.")
live_jazzybot_streams = get_twitch_streams(
self._twitch_oauth_manager,
user_logins=self._irc_client.channels)[0]
for stream in live_jazzybot_streams:
self._irc_client.private_message(
stream.user_login,
f"Don't miss \"{tournament_name}\" in "
f"{tournament_city}, {tournament_state} on "
f"{tournament_date}! Learn more at {tournament_url} .")
self._plugged_event_ids.append(random_event_id)
# Load credentials
twitch_username = "botvencabot"
with open("credentials.json") as credentials_file:
credentials = json.loads(credentials_file.read())
twitch_refresh_token = credentials["twitch_refresh_token"]
startgg_access_token = credentials["startgg_access_token"]
twitch_client_id = credentials["twitch_client_id"]
twitch_client_secret = credentials["twitch_client_secret"]
givebutter_api_key = credentials["givebutter_api_key"]
twitch_oauth_manager = TwitchOauthManager(
twitch_client_id, twitch_client_secret, twitch_refresh_token)
twitch_oauth_manager.refresh()
# Givebutter stuff
with open("processed_giving_space_ids.txt") as giving_space_ids_file:
giving_space_id_lines = giving_space_ids_file.readlines()
processed_giving_space_ids = [int(x.strip()) for x in giving_space_id_lines]
# Set up Twitch chat
twitch_chat = IRCClient()
twitch_chat.connect("irc.chat.twitch.tv")
# Set up listeners
twitch_chat_listener = irc_client_listener.IRCClientListener(twitch_chat)
givebutter_listener = GivebutterListener(
givebutter_api_key, processed_giving_space_ids)
# Set up handlers
twitch_chat_command_handler = TwitchChatCommandHandler(
twitch_chat, CURRENT_CHANNELS_PATH, EVENT_PROMO_OPTOUTS_PATH,
startgg_access_token)
leave_if_not_modded_handler = LeaveIfNotModdedHandler(
twitch_chat, CURRENT_CHANNELS_PATH)
pong_if_pinged_handler = PongIfPingedHandler(twitch_chat)
twitch_chat_login_failed_handler = TwitchChatLoginFailedHandler(
twitch_chat, twitch_oauth_manager)
report_timeout_handler = ReportTimeoutHandler()
givebutter_donation_handler = GivebutterDonationHandler(
twitch_oauth_manager, twitch_chat, "processed_giving_space_ids.txt")
# Create StreamBrain
jazzycircuitbot_brain = StreamBrain()
jazzycircuitbot_brain.activate_handler(twitch_chat_command_handler)
jazzycircuitbot_brain.activate_handler(leave_if_not_modded_handler)
jazzycircuitbot_brain.activate_handler(pong_if_pinged_handler)
jazzycircuitbot_brain.activate_handler(twitch_chat_login_failed_handler)
jazzycircuitbot_brain.activate_handler(report_timeout_handler)
jazzycircuitbot_brain.activate_handler(givebutter_donation_handler)
# Login to Twitch chat
twitch_password = f"oauth:{twitch_oauth_manager.access_token}"
twitch_chat.login(twitch_password, twitch_username, None)
twitch_chat.request_capability("twitch.tv/tags")
twitch_chat.request_capability("twitch.tv/membership")
twitch_chat.request_capability("twitch.tv/commands")
with open(CURRENT_CHANNELS_PATH) as current_channels_file:
raw_channel_lines = current_channels_file.readlines()
for raw_channel_line in raw_channel_lines:
twitch_chat.join(raw_channel_line.strip())
# Main loop
jazzycircuitbot_brain.start_listening(twitch_chat_listener)
jazzycircuitbot_brain.start_listening(givebutter_listener)
routine_schedule = Schedule()
promo_routine = JazzyEventPromoRoutine(
startgg_access_token, twitch_oauth_manager, twitch_chat, 1800)
routine_schedule.routines.append(promo_routine)
schedule_thread = threading.Thread(
target=routine_schedule.increment_loop, args=(1,))
schedule_thread.start()
input()
jazzycircuitbot_brain.stop()
routine_schedule.stop()
# We should also save our new Twitch Refresh Token.