forked from iamkroot/erp-gcal
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathgcal.py
160 lines (134 loc) · 5.4 KB
/
gcal.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
import os
from datetime import datetime as dt
from datetime import timedelta as td
import httplib2
from apiclient.discovery import build
from googleapiclient.errors import HttpError
from oauth2client import client, tools
from oauth2client.file import Storage
from utils import combine_dt, find_entity
SCOPES = "https://www.googleapis.com/auth/calendar"
CLIENT_SECRET_FILE = "client_secret.json"
APPLICATION_NAME = "ERP to Google Calendar"
def get_credentials(new_creds=False):
home_dir = os.path.expanduser("~")
credential_dir = os.path.join(home_dir, ".local", "google-creds")
if not os.path.exists(credential_dir):
os.makedirs(credential_dir)
credential_path = os.path.join(credential_dir, "erp-gcal-creds.json")
store = Storage(credential_path)
credentials = None if new_creds else store.get()
if not credentials or credentials.invalid:
flow = client.flow_from_clientsecrets(CLIENT_SECRET_FILE, SCOPES)
flow.user_agent = APPLICATION_NAME
args = tools.argparser.parse_known_args()[0]
credentials = tools.run_flow(flow, store, args)
print("Storing credentials to " + credential_path)
print("Pass '-n' next time if you want to run with another account.")
return credentials
def create_cal_serv(new_creds=False):
credentials = get_credentials(new_creds)
http = credentials.authorize(httplib2.Http())
return build("calendar", "v3", http=http)
class GCal:
def __init__(self, new_creds=False, cal_id="primary"):
self.cal_id = cal_id
self.service = create_cal_serv(new_creds)
def get_all_entities(self, entity_name, verb="list", **kwargs):
entity_serv = getattr(self.service, entity_name)
assert entity_serv, "Invalid entity name"
assert getattr(entity_serv(), verb), "Invalid verb"
page_token = None
while True:
entities = getattr(entity_serv(), verb)(
pageToken=page_token, **kwargs
).execute()
for entity in entities["items"]:
yield entity
page_token = entities.get("nextPageToken")
if not page_token:
break
def get_all_cals(self):
return self.get_all_entities("calendarList")
def get_all_events(self, **params):
return self.get_all_entities("events", calendarId=self.cal_id, **params)
def delete_event(self, event):
return (
self.service.events()
.delete(calendarId=self.cal_id, eventId=event["id"])
.execute()
)
def create_event(self, event):
return (
self.service.events().insert(calendarId=self.cal_id, body=event).execute()
)
def patch_event(self, event, data):
return (
self.service.events()
.patch(calendarId=self.cal_id, eventId=event["id"], body=data)
.execute()
)
def get_event_instances(self, event, **params):
return self.get_all_entities(
"events", "instances", calendarId=self.cal_id, eventId=event["id"], **params
)
def create_cal(self, calendar):
return self.service.calendars().insert(body=calendar).execute()
def find_event(self, event):
events = self.get_all_events()
fields = ("summary", "description", "location")
return find_entity(event, events, fields)
def find_cal(self, cal):
cals = self.get_all_cals()
fields = ("summary", "description")
return find_entity(cal, cals, fields, thresh=1) # name should be exact match
def set_cal(self, summary, description=None):
calendar = {"summary": summary}
if description:
calendar["description"] = description
cal = self.find_cal(calendar)
if cal:
self.cal_id = cal["id"]
return True
cal = self.create_cal(calendar)
if not cal:
print("Failed to find or create calendar", calendar)
return
self.cal_id = cal["id"]
def clear_cal(self):
for event in self.get_all_events():
self.print_event(event, "Deleting")
try:
self.delete_event(event)
except HttpError as err:
if err.resp.status != 410:
raise err
def clear_day(self, date: dt, only_recurring=True):
"""Clear all event instances of a particular date"""
assert isinstance(date, dt), "Argument should be a datetime object"
date = combine_dt(date.date())
next_day = date + td(days=1)
min_date, max_date = date.isoformat(), next_day.isoformat()
for event in self.get_all_events(
timeMin=min_date, timeMax=max_date, singleEvents=True
):
if only_recurring and "recurringEventId" not in event:
continue
self.print_event(event, "Deleting")
self.delete_event(event)
print("Cleared all events for", date.strftime("%a, %d/%m/%Y"))
@staticmethod
def print_event(event, prefix="", suffix=""):
if prefix:
print(prefix, end=" ")
print(event.get("summary", event), end=" " if suffix else "\n")
if suffix:
print(suffix)
if __name__ == "__main__":
from utils import get_cal_name
gcal = GCal()
gcal.set_cal(get_cal_name())
for event in gcal.get_all_events():
gcal.print_event(event)
date = dt(2019, 10, 17)
gcal.clear_day(date)