-
Notifications
You must be signed in to change notification settings - Fork 14
/
Copy pathconfig.go
339 lines (317 loc) · 9.93 KB
/
config.go
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
// Copyright 2024 Google Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// This file manages reading the user configuration file.
package main
import (
"encoding/json"
"fmt"
"log"
"os"
"strings"
"time"
)
// Configuration file:
// JSON file with the following structure:
// {
// excludes: [ "event", "names", "to", "ignore"],
// excludePrefixes: [ "prefixes", "to", "ignore"],
// startTime: "hh:mm (24 hr format) to start blinking at every day",
// endTime: "hh:mm (24 hr format) to stop blinking at every day",
// skipDays: [ "weekdays", "to", "skip"],
// pollInterval: 30
// calendar: "calendar"
// responseState: "all"
// deviceFailureRetries: 10
// showDots: true
// multiEvent: true
// priorityFlashSide: 1
//}
// Notes on items:
// Calendar is the calendar ID - the email address of the calendar. For a person's calendar, that's their email.
// For a secondary calendar, it's the base64 string @group.calendar.google.com on the calendar details page. "primary"
// is a magic string that means "the logged-in user's primary calendar".
// SkipDays may be localized.
// Excludes is exact string matches only.
// ExcludePrefixes will exclude all events starting with the given prefix.
// ResponseState can be one of: "all" (all events whatever their response status), "accepted" (only accepted events),
// "notRejected" (any events that are not rejected). Default is notRejected.
// DeviceFailureRetries is the number of consecutive failures to initialize the device before the program quits. Default is 10.
// ShowDots indicates whether to show dots and similar marks to indicate that the program has completed an update cycle.
// MultiEvent indicates whether to show two events if there are multiple events in the time range.
// userPrefs is a struct that manages the user preferences as set by the config file and command line.
type UserPrefs struct {
Excludes map[string]bool
ExcludePrefixes []string
StartTime *time.Time
EndTime *time.Time
SkipDays [7]bool
PollInterval int
Calendars []string
ResponseState ResponseState
DeviceFailureRetries int
ShowDots bool
MultiEvent bool
PriorityFlashSide int
WorkingLocations []WorkSite
}
// Struct used for decoding the JSON
type prefLayout struct {
Excludes []string
ExcludePrefixes []string
StartTime string
EndTime string
SkipDays []string
PollInterval int64
Calendar string
Calendars []string
ResponseState string
DeviceFailureRetries int64
ShowDots string
MultiEvent string
PriorityFlashSide int64
WorkingLocations []string
}
// responseState is an enumerated list of event response states, used to control which events will activate the blink(1).
type ResponseState string
const (
ResponseStateAll = ResponseState("all")
ResponseStateAccepted = ResponseState("accepted")
ResponseStateNotRejected = ResponseState("notRejected")
)
// checkStatus returns true if the given event status is one that should activate the blink(1) in the given responseState.
func (state ResponseState) CheckStatus(status string) bool {
switch state {
case ResponseStateAll:
return true
case ResponseStateAccepted:
return (status == "accepted")
case ResponseStateNotRejected:
return (status != "declined")
}
return false
}
func (state ResponseState) isValidState() bool {
switch state {
case ResponseStateAll:
return true
case ResponseStateAccepted:
return true
case ResponseStateNotRejected:
return true
}
return false
}
// Work site information
type WorkSiteType int
const (
WorkSiteHome WorkSiteType = iota
WorkSiteOffice
WorkSiteCustom
)
func makeWorkSiteType(location string) WorkSiteType {
switch location {
case "officeLocation", "office":
return WorkSiteOffice
case "customLocation", "custom":
return WorkSiteCustom
}
return WorkSiteHome
}
func (siteType WorkSiteType) toString() string {
switch siteType {
case WorkSiteHome:
return "Home"
case WorkSiteOffice:
return "Office"
case WorkSiteCustom:
return "Custom"
}
return ""
}
// workSite is a struct that holds a working location. If name is unset, should match
// all sites of the given type.
type WorkSite struct {
SiteType WorkSiteType
Name string
}
// Converts a string into a workSite structure. Returns an unset structure if the string is invalid.
func makeWorkSite(location string) WorkSite {
split := strings.SplitN(location, ":", 2)
siteType := makeWorkSiteType(split[0])
name := ""
if len(split) > 1 {
name = split[1]
}
fmt.Fprintf(debugOut, "Work Site: type %v, name %v", siteType, name)
return WorkSite{SiteType: siteType, Name: name}
}
// User preferences methods
func readUserPrefs() *UserPrefs {
userPrefs := &UserPrefs{}
// Set defaults from command line
userPrefs.PollInterval = *pollIntervalFlag
userPrefs.Calendars = []string{*calNameFlag}
userPrefs.ResponseState = ResponseState(*responseStateFlag)
userPrefs.DeviceFailureRetries = *deviceFailureRetriesFlag
userPrefs.ShowDots = *showDotsFlag
file, err := os.Open(*configFileFlag)
defer file.Close()
if err != nil {
// Lack of a config file is not a fatal error.
fmt.Fprintf(debugOut, "Unable to read config file %v : %v\n", *configFileFlag, err)
return userPrefs
}
prefs := prefLayout{}
decoder := json.NewDecoder(file)
decoder.DisallowUnknownFields()
err = decoder.Decode(&prefs)
fmt.Fprintf(debugOut, "Decoded prefs: %v\n", prefs)
if err != nil {
log.Fatalf("Unable to parse config file %v", err)
}
if prefs.StartTime != "" {
startTime, err := time.Parse("15:04", prefs.StartTime)
if err != nil {
log.Fatalf("Invalid start time %v : %v", prefs.StartTime, err)
}
userPrefs.StartTime = &startTime
}
if prefs.EndTime != "" {
endTime, err := time.Parse("15:04", prefs.EndTime)
if err != nil {
log.Fatalf("Invalid end time %v : %v", prefs.EndTime, err)
}
userPrefs.EndTime = &endTime
}
userPrefs.Excludes = make(map[string]bool)
for _, item := range prefs.Excludes {
fmt.Fprintf(debugOut, "Excluding item %v\n", item)
userPrefs.Excludes[item] = true
}
userPrefs.ExcludePrefixes = prefs.ExcludePrefixes
weekdays := make(map[string]int)
for i := 0; i < 7; i++ {
weekdays[time.Weekday(i).String()] = i
}
for _, day := range prefs.SkipDays {
i, ok := weekdays[day]
if ok {
userPrefs.SkipDays[i] = true
} else {
log.Fatalf("Invalid day in skipdays: %v", day)
}
}
if prefs.Calendar != "" {
userPrefs.Calendars = []string{prefs.Calendar}
}
if len(prefs.Calendars) > 0 {
userPrefs.Calendars = prefs.Calendars
}
if prefs.PollInterval != 0 {
userPrefs.PollInterval = int(prefs.PollInterval)
}
if prefs.ResponseState != "" {
userPrefs.ResponseState = ResponseState(prefs.ResponseState)
if !userPrefs.ResponseState.isValidState() {
log.Fatalf("Invalid response state %v", prefs.ResponseState)
}
}
if prefs.DeviceFailureRetries != 0 {
userPrefs.DeviceFailureRetries = int(prefs.DeviceFailureRetries)
}
if prefs.ShowDots != "" {
userPrefs.ShowDots = (prefs.ShowDots == "true")
}
userPrefs.MultiEvent = (prefs.MultiEvent == "true")
if prefs.PriorityFlashSide != 0 {
userPrefs.PriorityFlashSide = int(prefs.PriorityFlashSide)
}
for _, location := range prefs.WorkingLocations {
userPrefs.WorkingLocations = append(userPrefs.WorkingLocations, makeWorkSite(location))
}
fmt.Fprintf(debugOut, "User prefs: %v\n", userPrefs)
return userPrefs
}
func printStartInfo(userPrefs *UserPrefs) {
fmt.Printf("Running with %v second intervals\n", userPrefs.PollInterval)
if len(userPrefs.Calendars) == 1 {
fmt.Printf("Monitoring calendar ID %v\n", userPrefs.Calendars[0])
} else {
fmt.Println("Monitoring calendar IDs:")
for _, item := range userPrefs.Calendars {
fmt.Printf(" %v\n", item)
}
}
switch userPrefs.ResponseState {
case ResponseStateAll:
fmt.Println("All events shown, regardless of accepted/rejected status.")
case ResponseStateAccepted:
fmt.Println("Only accepted events shown.")
case ResponseStateNotRejected:
fmt.Println("Rejected events not shown.")
}
if len(userPrefs.WorkingLocations) > 0 {
fmt.Println("Working Locations:")
for _, item := range userPrefs.WorkingLocations {
if item.SiteType == WorkSiteHome {
fmt.Printf(" Home\n")
} else {
fmt.Printf(" %v: %v\n", item.SiteType.toString(), item.Name)
}
}
}
if len(userPrefs.Excludes) > 0 {
fmt.Println("Excluded events:")
for item := range userPrefs.Excludes {
fmt.Printf(" %v\n", item)
}
}
if len(userPrefs.ExcludePrefixes) > 0 {
fmt.Println("Excluded event prefixes:")
for _, item := range userPrefs.ExcludePrefixes {
fmt.Printf(" %v\n", item)
}
}
skipDays := ""
join := ""
for i, val := range userPrefs.SkipDays {
if val {
skipDays += join
skipDays += time.Weekday(i).String()
join = ", "
}
}
if len(skipDays) > 0 {
fmt.Println("Skip days: " + skipDays)
}
timeString := ""
if userPrefs.StartTime != nil {
timeString += fmt.Sprintf("Time restrictions: after %02d:%02d", userPrefs.StartTime.Hour(), userPrefs.StartTime.Minute())
}
if userPrefs.EndTime != nil {
endTimeString := fmt.Sprintf("until %02d:%02d", userPrefs.EndTime.Hour(), userPrefs.EndTime.Minute())
if len(timeString) > 0 {
timeString += " and "
} else {
timeString += "Time restrictions: "
}
timeString += endTimeString
}
if len(timeString) > 0 {
fmt.Println(timeString)
}
if userPrefs.MultiEvent {
fmt.Println("Multievent is active.")
}
}