This repository was archived by the owner on Apr 28, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhandler.go
159 lines (145 loc) · 4.45 KB
/
handler.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
package main
import (
"fmt"
"regexp"
"strings"
"time"
"github.com/PuerkitoBio/goquery"
)
type Handler interface {
Handle(message Message, act Actions) (err error)
}
type SecurityAlertHandler struct {
}
func (h SecurityAlertHandler) Handle(msg Message, act Actions) (err error) {
if !strings.Contains(msg.Subject, "Security alert") || !strings.EqualFold(msg.From[0].Address, "no-reply@accounts.google.com") {
return nil
}
act.Print("This email appears to be a security alert from Google.")
if act.GetInput("Do you want to delete it? (Y/N)") != "Y" {
return nil
}
act.Delete(msg)
return nil
}
type FailedMessageSendHandler struct {
}
func (h FailedMessageSendHandler) Handle(msg Message, act Actions) (err error) {
if !strings.EqualFold(msg.From[0].Name, "Mail Delivery Subsystem") {
return nil
}
act.Print("This email appears to be from the Mail Delivery Subsystem. This might be from a failed email send.")
if act.GetInput("Do you want to delete it? (Y/N)") != "Y" {
return nil
}
act.Delete(msg)
return nil
}
type CalendarHandler struct {
}
var calendarSubjectRegex = regexp.MustCompile(`.*@.*\(GMT\) \(.*\)`)
func (h CalendarHandler) Handle(msg Message, act Actions) (err error) {
for _, attachment := range msg.Attachments {
if !strings.HasSuffix(attachment.FileName, ".ics") {
continue
}
matches := calendarSubjectRegex.MatchString(msg.Subject)
if matches {
act.Print("This message is a google calendar invite. Skipping ICS installation.")
continue
}
if act.GetInput("This email has a calendar file, do you want to install it into Google Calendar? (Y/N)") == "Y" {
link, err := act.ImportCalendarEvent(string(attachment.Body))
if err != nil {
return err
}
act.Print(fmt.Sprintf("Calendar event created: %v", link))
}
}
return
}
type ExpiredEventHandler struct {
}
func (h ExpiredEventHandler) Handle(msg Message, act Actions) (err error) {
doc, err := goquery.NewDocumentFromReader(strings.NewReader(msg.Body))
if err != nil {
act.Print(fmt.Sprintf("expiredeventhandler: not a HTML message: %w", err))
}
// Get the start time.
ts, _ := doc.Find("time").First().Attr("datetime")
layout := "20060102T150405Z"
if ts == "" {
act.Print("No event start time found for this message.")
return nil
}
// Format the time string into a variable.
startTime, err := time.Parse(layout, ts)
if err != nil {
return fmt.Errorf("expiredeventhandler: invalid start time format: %w", err)
}
// Get a tag name "time", then get the last occurence, then find the unformatted string containing the time.
te, _ := doc.Find("time").Last().Attr("datetime")
if te == "" {
act.Print("No event end time found for this message.")
return nil
}
// Format the time string into a variable.
endTime, err := time.Parse(layout, te)
if err != nil {
return fmt.Errorf("expiredeventhandler: invalid end time format: %w", err)
}
act.Print(fmt.Sprintf("Event found: Start time: %v, end time: %v", startTime, endTime))
// If the end date of the event has already passed.
if endTime.Before(time.Now()) {
if act.GetInput("The event in this email is from the past, do you want to delete the email? (Y/N)") == "Y" {
act.Delete(msg)
return
}
} else if strings.Contains(msg.Body, "https://calendar.google.com/calendar/event?action=RESPOND") {
h.respond(msg, doc, act)
return
}
act.Print("No way to send a responce found for this message.")
return
}
func (h ExpiredEventHandler) respond(msg Message, doc *goquery.Document, act Actions) {
response := act.GetInput("This email has a calendar event do you want to, respond with [ Yes (Y) ], [ No (N) ], [ Maybe (M) ], [ View some Details (D) ], [ Ignore the email and move on (I) ] or [ Delete the email (X) ]?")
if response == "I" {
return
}
if response == "X" {
act.Delete(msg)
return
}
if strings.EqualFold(response, "D") {
doc.Find("a").Each(func(i int, ul *goquery.Selection) {
link, _ := ul.Attr("href")
if strings.Contains(link, "event?action=VIEW") {
act.OpenBrowser(link)
} else {
act.Print("Could not find any details.")
}
h.respond(msg, doc, act)
return
})
}
var rst string
switch response {
case "Y":
rst = "1"
case "M":
rst = "3"
case "N":
rst = "2"
default:
act.Print("Invalid input!")
h.respond(msg, doc, act)
return
}
doc.Find("a").Each(func(i int, ul *goquery.Selection) {
link, _ := ul.Attr("href")
if strings.Contains(link, "event?action=RESPOND") && strings.Contains(link, "rst="+rst) {
act.OpenBrowserLater(link)
}
})
}