-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathvad-bot.py
133 lines (111 loc) · 4.19 KB
/
vad-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
# Imports
from requests import get, post
from dotenv import load_dotenv
from os import getenv
from time import sleep
from datetime import datetime
from random import randint
# Global Variables
API_SETU_URL = 'https://cdn-api.co-vin.in/api/v2/'
APPOINTMENTS_AVAILABILITY = 'appointment/sessions/public/'
CALENDAR_DISTRICT = 'calendarByDistrict'
CALENDAR_PINCODE = 'calendarByPin'
API_HEADERS = {
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.93 Safari/537.36 Edg/90.0.818.51'
}
# Environment Variables
load_dotenv()
DISCORD_WEBHOOK_URL = getenv('DISCORD_WEBHOOK_URL')
# Functions
def fetchCalendarByDistrict(district_id: str, date: datetime) -> dict:
params = {
'district_id': district_id,
'date': date.strftime("%d-%m-%Y")
}
resp = get(
url = API_SETU_URL+APPOINTMENTS_AVAILABILITY+CALENDAR_DISTRICT,
params = params,
headers= API_HEADERS
)
print(f'{str(date)}\tAPI response: {resp.status_code}')
if(resp.status_code != 200):
return None
return resp.json()
def fetchCalendarByPinCode(pincode: str, date: datetime) -> dict:
params = {
'pincode': pincode,
'date': date.strftime("%d-%m-%Y")
}
resp = get(
url = API_SETU_URL+APPOINTMENTS_AVAILABILITY+CALENDAR_PINCODE,
params = params,
headers= API_HEADERS
)
print(f'{str(date)}\tAPI response: {resp.status_code}')
if(resp.status_code != 200):
print(f'fetchCalendarByPinCode response: {resp.content}')
return None
return resp.json()
def getNewCenters(old_centers: list, new_centers: list) -> list:
x:set = set(old_centers)
y:set = set(new_centers)
return list(y.union(x).difference(x))
def getAvailableCenters(data: dict) -> list:
available_centers = list()
for center in data['centers']:
for session in center['sessions']:
if(session['available_capacity'] > 0):
available_centers.append(f"{center['pincode']} - {center['name']}")
break
return available_centers
def fetchData() -> dict:
dt = datetime.now()
search_by = getenv('SEARCH_BY')
if (search_by == 'DISTRICT'):
data = fetchCalendarByDistrict(district_id=getenv('DISTRICT_ID'), date=dt)
if(not data):
raise Exception('API_Error: fetchCalendarByDistrict returned None')
return data
elif (search_by == 'PINCODE'):
data = {'centers': []}
pincodes = (getenv('LIST_OF_PINCODES')[1:-1]).replace(' ', '').split(',')
for pincode in pincodes:
resp = fetchCalendarByPinCode(pincode, dt)
if(not resp):
raise Exception('API_Error: fetchCalendarByDistrict returned None')
data['centers'].extend(resp['centers'])
return data
else:
error_title = 'Error in fetchData: Invalid SEARCH_BY mode.'
error_body = 'SEARCH_BY variable in .env should be either DISTRICT or PINCODE'
raise Exception(error_title + '\n' + error_body)
def notifyOnDiscord(available_centers: list) -> None:
body = {
'content': '**Available centers:**\n' + '\n'.join(available_centers)
}
webhook_params = {
'Content-Type': 'multipart/form-data'
}
print('Discord Webhook: ', post(url=DISCORD_WEBHOOK_URL, json=body, params=webhook_params))
# Driver code
if __name__ == "__main__":
try:
previous_centers = list()
while(True):
# Fetch & process data
data = fetchData()
centers = getNewCenters(
old_centers=previous_centers,
new_centers=getAvailableCenters(data)
)
# Avoid Repetition of centers
# Current solution repeats centers every alternate cycle
previous_centers = centers
if(centers):
notifyOnDiscord(available_centers = centers)
# APIs are subject to a rate limit of 100 API calls per 5 minutes per IP
delay_duration = randint(20, 40)
print(f'Train will depart after {delay_duration} sec')
sleep(delay_duration)
except Exception as e:
print(str(e))