-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathfriskis.py
455 lines (352 loc) · 14 KB
/
friskis.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
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
import calendar
import json
import locale
import sys
from datetime import datetime, time, timedelta
from pathlib import Path
import click
import requests
from dateutil.parser import parse as fromisoformat
from pytz import timezone, utc
locale.setlocale(locale.LC_TIME, "sv_SE.UTF-8")
API_ENDPOINT = "https://friskissvettis.brpsystems.com/brponline/api/ver3"
BUSINESS_UNITS_URL = f"{API_ENDPOINT}/businessunits"
LOGIN_URL = f"{API_ENDPOINT}/auth/login"
PROJECT_ROOT = Path(__file__).parent
DEFAULT_LOGIN_CREDENTIALS_PATH = PROJECT_ROOT / ".login.json"
DEFAULT_SCHEDULE_PATH = PROJECT_ROOT / ".schedule.json"
STOCKHOLM_TIMEZONE = timezone("Europe/Stockholm")
WEEKDAYS = [day.lower() for day in calendar.day_name]
DEFAULT_HTTP_TIMEOUT = 5
class FriskisException(Exception):
pass
class Unauthorized(FriskisException):
pass
class FriskisAPIError(FriskisException):
pass
def _parse_datetime(s):
return fromisoformat(s).astimezone(utc)
def _format_date(d):
return d.isoformat()
def _format_datetime(dt, delimiter=" ", tz=STOCKHOLM_TIMEZONE, seconds=False):
aware = dt.astimezone(tz)
date_string = aware.date().isoformat()
time_string = aware.strftime("%H:%M:%S" if seconds else "%H:%M")
return f"{date_string}{delimiter}{time_string}"
def _format_name(name):
return name.title().strip()
def _format_location(location):
return location.title()
def _pluralize_weekday(weekday):
return f"{weekday}ar" if not weekday.endswith("ar") else weekday
def _format_weekday(weekday, plural=False):
weekday = weekday.lower()
return _pluralize_weekday(weekday) if plural else weekday
def _format_weekday_plural(weekday):
return _format_weekday(weekday, plural=True)
def _get_formatted_arguments(name, location, weekday, time):
return (
_format_name(name),
_format_location(location),
_format_weekday_plural(weekday),
time,
)
def _strip_weekday_plural(ctx, weekday):
return weekday[:-2] if weekday.endswith("ar") else weekday
def _lowercase(ctx, s):
return s.lower()
def _normalize(ctx, s, formatters):
if len(formatters) == 0:
return s
if len(formatters) == 1:
return formatters[0](ctx, s)
return _normalize(ctx, formatters[0](ctx, s), formatters[1:])
def _normalize_weekday(ctx, weekday):
return _normalize(ctx, weekday, [_lowercase, _strip_weekday_plural])
def _format_list_display(ctx, s):
return _normalize(ctx, s, [lambda cty, v: v.ljust(16)])
def _datetime_to_time_str(ctx, dt):
return str(dt.strftime("%H:%M"))
def _get_weekday_number(weekday):
return WEEKDAYS.index(weekday) + 1
def _get_weekday(weekday_number):
return WEEKDAYS[weekday_number - 1]
def _http_get(url, *args, timeout=DEFAULT_HTTP_TIMEOUT, **kwargs):
return requests.get(url, *args, timeout=timeout, **kwargs)
def _http_post(url, *args, timeout=DEFAULT_HTTP_TIMEOUT, **kwargs):
return requests.post(url, *args, timeout=timeout, **kwargs)
def _get_business_units():
business_units_response = _http_get(BUSINESS_UNITS_URL)
if business_units_response.status_code != 200:
raise click.ClickException(
f"Det gick inte att hämta platser. ({business_units_response.status_code})"
)
return business_units_response.json()
def _get_business_unit(name):
business_units = _get_business_units()
for business_unit in business_units:
if business_unit["name"].lower() == name.lower():
return business_unit
existing = ", ".join(b["name"] for b in business_units)
raise click.ClickException(
f"Kunde inte hitta någon plats med det namnet. Hittade följande: {existing}"
)
def _get_group_activities(business_unit, day):
url = f"{BUSINESS_UNITS_URL}/{business_unit['id']}/groupactivities"
period_start = datetime.combine(day, time())
period_end = period_start + timedelta(days=1)
def datetime_to_string(dt):
return _format_datetime(dt, delimiter="T", tz=utc, seconds=True) + ".000Z"
params = {
"period.start": datetime_to_string(period_start),
"period.end": datetime_to_string(period_end),
}
group_activities_response = _http_get(url, params)
if group_activities_response.status_code != 200:
raise click.ClickException("Det gick inte att hämta schemalagda aktiviteter.")
return group_activities_response.json()
def _get_group_activity(name, day, business_unit, time):
group_activities = _get_group_activities(business_unit, day)
for group_activity in group_activities:
has_matching_name = group_activity["name"].lower().strip() == name.lower()
has_matching_time = (
_parse_datetime(group_activity["duration"]["start"])
.astimezone(STOCKHOLM_TIMEZONE)
.strftime("%H:%M")
== time
)
if has_matching_name and has_matching_time:
return group_activity
def _get_upcoming_group_activity(name, location, weekday_number, time):
today = datetime.now(STOCKHOLM_TIMEZONE).date()
group_activity_date = today + timedelta(days=1)
while group_activity_date.isoweekday() != weekday_number:
group_activity_date += timedelta(days=1)
business_unit = _get_business_unit(location)
group_activity = _get_group_activity(name, group_activity_date, business_unit, time)
return group_activity, group_activity_date
def _get_bookings(authorization):
username = authorization["username"]
url = f"{API_ENDPOINT}/customers/{username}/bookings/groupactivities"
group_activities_response = _authorized_request(
_http_get, url, authorization=authorization
)
if group_activities_response.status_code != 200:
raise click.ClickException(
f"Det gick inte att hämta befintliga bokningar. ({group_activities_response.status_code})"
)
return group_activities_response.json()
def _get_login_credentials(login_credentials_path):
with open(login_credentials_path) as f:
return json.load(f)
def _get_schedule(schedule_path):
if not Path(schedule_path).exists():
return []
with open(schedule_path) as f:
return json.load(f)
def _set_schedule(schedule, schedule_path):
with open(schedule_path, "w") as f:
json.dump(schedule, f)
def _login(login_credentials_path):
params = _get_login_credentials(login_credentials_path)
login_response = _http_post(LOGIN_URL, json=params)
if login_response.status_code == 200:
return login_response.json()
elif login_response.status_code == 401:
raise click.ClickException(
"Det gick inte att logga in med angivna inloggningsuppgifter."
)
raise click.ClickException(
f"Det gick inte att logga in. ({login_response.status_code})"
)
def _authorized_request(request_method, *request_args, authorization, **request_kwargs):
token_type = authorization["token_type"]
access_token = authorization["access_token"]
headers = {
**request_kwargs.pop("headers", {}),
"authorization": f"{token_type} {access_token}",
}
return request_method(*request_args, **request_kwargs, headers=headers)
def _book_group_activity(group_activity, authorization):
username = authorization["username"]
url = f"{API_ENDPOINT}/customers/{username}/bookings/groupactivities"
params = {
"groupActivity": group_activity["id"],
"allowWaitingList": False,
}
attend_group_activity_response = _authorized_request(
_http_post, url, json=params, authorization=authorization
)
if attend_group_activity_response.status_code == 201:
return attend_group_activity_response.json()
return {}
def _stdout(message):
click.echo(message)
def _stderr(message):
click.echo(message, file=sys.stderr)
schedule_path_option = click.option(
"--schedule-path",
required=False,
default=DEFAULT_SCHEDULE_PATH,
)
login_path_option = click.option(
"--login-path",
required=False,
default=DEFAULT_LOGIN_CREDENTIALS_PATH,
)
@click.group()
def friskis():
pass
@friskis.command("list")
@schedule_path_option
@click.pass_context
def list_schedule(ctx, schedule_path):
for event in sorted(_get_schedule(schedule_path), key=lambda e: e["weekday"]):
name = event["name"]
weekday = _get_weekday(event["weekday"])
time = event["time"]
click.echo(
"\t\t".join(
_format_list_display(ctx, column)
for column in [
name,
event["location"],
f"{weekday}ar".title(),
f"kl. {time}",
]
)
)
@friskis.command()
@click.argument("name", callback=_lowercase)
@click.argument("location", callback=_lowercase)
@click.argument("weekday", callback=_normalize_weekday)
@click.argument(
"time", type=click.DateTime(formats=["%H:%M"]), callback=_datetime_to_time_str
)
@schedule_path_option
def add(name, location, weekday, time, schedule_path):
formatted_name, formatted_location, formatted_weekday, formatted_time = (
_get_formatted_arguments(name, location, weekday, time)
)
schedule = _get_schedule(schedule_path)
weekday_number = _get_weekday_number(weekday)
for event in schedule:
if (
name == event["name"]
and location == event["location"]
and weekday_number == event["weekday"]
and time == event["time"]
):
raise click.ClickException(
f"{formatted_name} på {formatted_location} på {formatted_weekday} kl. {formatted_time} finns redan i schemat."
)
group_activity, group_activity_date = _get_upcoming_group_activity(
name, location, weekday_number, time
)
if not group_activity:
formatted_group_activity_date = _format_date(group_activity_date)
raise click.ClickException(
f"{formatted_name} är inte schemalagt {formatted_group_activity_date} kl. {formatted_time} på {formatted_location}."
)
_set_schedule(
[
*schedule,
{
"name": name,
"location": location,
"weekday": weekday_number,
"time": time,
},
],
schedule_path,
)
click.echo(
f"Lade till {formatted_name} på {formatted_location} på {formatted_weekday} kl. {formatted_time} i schemat."
)
@friskis.command()
@click.argument("name", callback=_lowercase)
@click.argument("location", callback=_lowercase)
@click.argument("weekday", callback=_normalize_weekday)
@click.argument(
"time", type=click.DateTime(formats=["%H:%M"]), callback=_datetime_to_time_str
)
@schedule_path_option
def remove(name, schedule_path, location, weekday, time):
formatted_name, formatted_location, formatted_weekday, formatted_time = (
_get_formatted_arguments(name, location, weekday, time)
)
schedule = _get_schedule(schedule_path)
weekday_number = _get_weekday_number(weekday)
matches = []
for event in schedule:
if (
name.lower() in event["name"].lower()
and location.lower() == event["location"].lower()
and weekday_number == event["weekday"]
and time == event["time"]
):
matches.append(event)
if len(matches) == 0:
raise click.ClickException(
f"{name}, {location}, {weekday} och {time} matchade inte något i schemat."
)
_set_schedule([e for e in schedule if e not in matches], schedule_path)
_stdout(
f"Tog bort {formatted_name} på {formatted_location} på {formatted_weekday} kl. {formatted_time} ur schemat."
)
@login_path_option
@schedule_path_option
@friskis.command()
def book(login_path, schedule_path):
now = datetime.now(STOCKHOLM_TIMEZONE)
authorization = _login(login_path)
existing_bookings = _get_bookings(authorization)
for event in _get_schedule(schedule_path):
group_activity_name = event["name"]
group_activity_weekday = event["weekday"]
group_activity_time = event["time"]
location = event["location"]
formatted_name, formatted_location = (
_format_name(group_activity_name),
_format_location(location),
)
group_activity, group_activity_date = _get_upcoming_group_activity(
group_activity_name, location, group_activity_weekday, group_activity_time
)
formatted_group_activity_date = group_activity_date.isoformat()
if not group_activity:
_stderr(
f"{formatted_name} är inte schemalagt på {formatted_location} {formatted_group_activity_date} kl. {group_activity_time}."
)
continue
if group_activity["cancelled"]:
_stderr(
f"{formatted_name} på {formatted_location} är inställt {formatted_group_activity_date} kl. {group_activity_time}"
)
bookable_earliest = _parse_datetime(group_activity["bookableEarliest"])
already_booked = group_activity["id"] in [
booking["groupActivity"]["id"] for booking in existing_bookings
]
if (
now < bookable_earliest
or already_booked
or now > bookable_earliest + timedelta(days=1)
):
continue
slots = group_activity["slots"]
slots_left = slots["leftToBook"]
if slots_left == 0:
waiting_list_length = slots["inWaitingList"]
_stderr(
f"{formatted_name} på {formatted_location} {formatted_group_activity_date} kl. {group_activity_time} är fullbokat. "
f"Det är {waiting_list_length} {'personer' if waiting_list_length > 1 else 'person'} på reservplats.",
)
continue
group_activity_booking = _book_group_activity(group_activity, authorization)
if not group_activity_booking:
continue
_stdout(
f"{formatted_name} på {formatted_location} {formatted_group_activity_date} kl. {group_activity_time} bokades!"
)
if __name__ == "__main__":
friskis()