-
Notifications
You must be signed in to change notification settings - Fork 113
/
maginkcal.py
executable file
·112 lines (95 loc) · 6.07 KB
/
maginkcal.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
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
"""
This project is designed for the WaveShare 12.48" eInk display. Modifications will be needed for other displays,
especially the display drivers and how the image is being rendered on the display. Also, this is the first project that
I posted on GitHub so please go easy on me. There are still many parts of the code (especially with timezone
conversions) that are not tested comprehensively, since my calendar/events are largely based on the timezone I'm in.
There will also be work needed to adjust the calendar rendering for different screen sizes, such as modifying of the
CSS stylesheets in the "render" folder.
"""
import datetime as dt
import sys
from pytz import timezone
from gcal.gcal import GcalHelper
from render.render import RenderHelper
from power.power import PowerHelper
import json
import logging
def main():
# Basic configuration settings (user replaceable)
configFile = open('config.json')
config = json.load(configFile)
displayTZ = timezone(config['displayTZ']) # list of timezones - print(pytz.all_timezones)
thresholdHours = config['thresholdHours'] # considers events updated within last 12 hours as recently updated
maxEventsPerDay = config['maxEventsPerDay'] # limits number of events to display (remainder displayed as '+X more')
isDisplayToScreen = config['isDisplayToScreen'] # set to true when debugging rendering without displaying to screen
isShutdownOnComplete = config['isShutdownOnComplete'] # set to true to conserve power, false if in debugging mode
batteryDisplayMode = config['batteryDisplayMode'] # 0: do not show / 1: always show / 2: show when battery is low
weekStartDay = config['weekStartDay'] # Monday = 0, Sunday = 6
dayOfWeekText = config['dayOfWeekText'] # Monday as first item in list
screenWidth = config['screenWidth'] # Width of E-Ink display. Default is landscape. Need to rotate image to fit.
screenHeight = config['screenHeight'] # Height of E-Ink display. Default is landscape. Need to rotate image to fit.
imageWidth = config['imageWidth'] # Width of image to be generated for display.
imageHeight = config['imageHeight'] # Height of image to be generated for display.
rotateAngle = config['rotateAngle'] # If image is rendered in portrait orientation, angle to rotate to fit screen
calendars = config['calendars'] # Google calendar ids
is24hour = config['is24h'] # set 24 hour time
# Create and configure logger
logging.basicConfig(filename="logfile.log", format='%(asctime)s %(levelname)s - %(message)s', filemode='a')
logger = logging.getLogger('maginkcal')
logger.addHandler(logging.StreamHandler(sys.stdout)) # print logger to stdout
logger.setLevel(logging.INFO)
logger.info("Starting daily calendar update")
try:
# Establish current date and time information
# Note: For Python datetime.weekday() - Monday = 0, Sunday = 6
# For this implementation, each week starts on a Sunday and the calendar begins on the nearest elapsed Sunday
# The calendar will also display 5 weeks of events to cover the upcoming month, ending on a Saturday
powerService = PowerHelper()
powerService.sync_time()
currBatteryLevel = powerService.get_battery()
logger.info('Battery level at start: {:.3f}'.format(currBatteryLevel))
currDatetime = dt.datetime.now(displayTZ)
logger.info("Time synchronised to {}".format(currDatetime))
currDate = currDatetime.date()
calStartDate = currDate - dt.timedelta(days=((currDate.weekday() + (7 - weekStartDay)) % 7))
calEndDate = calStartDate + dt.timedelta(days=(5 * 7 - 1))
calStartDatetime = displayTZ.localize(dt.datetime.combine(calStartDate, dt.datetime.min.time()))
calEndDatetime = displayTZ.localize(dt.datetime.combine(calEndDate, dt.datetime.max.time()))
# Using Google Calendar to retrieve all events within start and end date (inclusive)
start = dt.datetime.now()
gcalService = GcalHelper()
eventList = gcalService.retrieve_events(calendars, calStartDatetime, calEndDatetime, displayTZ, thresholdHours)
logger.info("Calendar events retrieved in " + str(dt.datetime.now() - start))
# Populate dictionary with information to be rendered on e-ink display
calDict = {'events': eventList, 'calStartDate': calStartDate, 'today': currDate, 'lastRefresh': currDatetime,
'batteryLevel': currBatteryLevel, 'batteryDisplayMode': batteryDisplayMode,
'dayOfWeekText': dayOfWeekText, 'weekStartDay': weekStartDay, 'maxEventsPerDay': maxEventsPerDay,
'is24hour': is24hour}
renderService = RenderHelper(imageWidth, imageHeight, rotateAngle)
calBlackImage, calRedImage = renderService.process_inputs(calDict)
if isDisplayToScreen:
from display.display import DisplayHelper
displayService = DisplayHelper(screenWidth, screenHeight)
if currDate.weekday() == weekStartDay:
# calibrate display once a week to prevent ghosting
displayService.calibrate(cycles=0) # to calibrate in production
displayService.update(calBlackImage, calRedImage)
displayService.sleep()
currBatteryLevel = powerService.get_battery()
logger.info('Battery level at end: {:.3f}'.format(currBatteryLevel))
except Exception as e:
logger.error(e)
logger.info("Completed daily calendar update")
logger.info("Checking if configured to shutdown safely - Current hour: {}".format(currDatetime.hour))
if isShutdownOnComplete:
# implementing a failsafe so that we don't shutdown when debugging
# checking if it's 6am in the morning, which is the time I've set PiSugar to wake and refresh the calendar
# if it is 6am, shutdown the RPi. if not 6am, assume I'm debugging the code, so do not shutdown
if currDatetime.hour == 6:
logger.info("Shutting down safely.")
import os
os.system("sudo shutdown -h now")
if __name__ == "__main__":
main()