-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathapp.go
176 lines (148 loc) · 4.68 KB
/
app.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
package main
import (
"context"
"errors"
"fmt"
"regexp"
"strings"
"sync"
"time"
"github.com/wailsapp/wails/v2/pkg/runtime"
"gopkg.in/gomail.v2"
)
type LogType string
var emailRegexp *regexp.Regexp
// add a new log type for neutral log messages and set info to success
const (
LogTypeInfo LogType = "info"
LogTypeWarning LogType = "warning"
LogTypeError LogType = "error"
)
// LogEntry represents a log entry
type LogEntry struct {
Timestamp string `json:"timestamp"`
Message string `json:"message"`
Type LogType `json:"type"`
}
// App struct
type App struct {
ctx context.Context
logs []LogEntry
logsMutex sync.RWMutex
subscriptions map[string]chan LogEntry
subMutex sync.RWMutex
}
type recipient struct {
Surname string `json:"surname"`
OtherNames string `json:"otherNames"`
Email string `json:"email"`
}
// NewApp creates a new App application struct
func NewApp() *App {
return &App{
logs: make([]LogEntry, 0),
subscriptions: make(map[string]chan LogEntry),
}
}
// startup is called at application startup
func (a *App) startup(ctx context.Context) {
// Perform your setup here
a.ctx = ctx
}
// domReady is called after front-end resources have been loaded
func (a *App) domReady(ctx context.Context) {
// Add your action here
}
// beforeClose is called when the application is about to quit,
// either by clicking the window close button or calling runtime.Quit.
// Returning true will cause the application to continue, false will continue shutdown as normal.
func (a *App) beforeClose(ctx context.Context) (prevent bool) {
return false
}
// shutdown is called at application termination
func (a *App) shutdown(ctx context.Context) {
// Perform your teardown here
}
func (a *App) SendMail(sender string, password string, subject string, messageBody string, csvData []recipient) error {
if sender == "" || password == "" || subject == "" || messageBody == "" {
a.AddLog("Invalid input: One of the required fields is empty", LogTypeError)
return errors.New("error: check logs")
}
isBadCsv := false
// send logs to frontend to indicate check as started
a.AddLog("Checking if recipients are valid", LogTypeInfo)
// check if any row has both first name and last name empty
for idx, r := range csvData {
if r.Surname == "" && r.OtherNames == "" {
isBadCsv = true
a.AddLog(fmt.Sprintf("Entry at index: %d is invalid: No names object %v", idx+1, r), LogTypeError)
}
}
// check if all emails are valid
for idx, r := range csvData {
if !emailRegexp.MatchString(r.Email) {
isBadCsv = true
a.AddLog(fmt.Sprintf("Entry at index: %d is invalid: Email is invalid %v", idx+1, r), LogTypeError)
}
}
if isBadCsv {
return errors.New("recipient entries contain errors: see logs for more details")
}
a.AddLog("Check was successful", LogTypeInfo)
d := gomail.NewDialer("smtp.gmail.com", 465, sender, password)
// Loop through each recipient and send email
for idx, r := range csvData {
// Create a new message for each recipient
m := gomail.NewMessage()
m.SetHeader("From", sender)
m.SetHeader("To", r.Email)
// replace escape sequence (new line) to html tag and add name to EmailBody object
messageBody = strings.ReplaceAll(messageBody, "\n", "<br>")
bodyContent := "Dear " + r.Surname + " " + r.OtherNames + ",<br><br>" + messageBody
m.SetHeader("Subject", subject)
// add signature to the end
m.SetBody("text/html", bodyContent)
// Send the email
if err := d.DialAndSend(m); err != nil {
if idx != 0 {
a.AddLog(fmt.Sprintf("Email sent to %d recipient(s)", idx+1), LogTypeError)
}
a.AddLog(fmt.Sprintf("sending mail to %s failed at entry number: %d. The whole process has been terminated", r.Email, idx+1), LogTypeError)
return errors.New("error: check logs")
}
a.AddLog(fmt.Sprintf("Email sent to %s successfully", r.Email), LogTypeInfo)
}
return nil
}
// GetLogs returns all stored logs
func (a *App) GetLogs() []LogEntry {
a.logsMutex.RLock()
defer a.logsMutex.RUnlock()
return a.logs
}
// AddLog adds a new log entry and notifies all subscribers
func (a *App) AddLog(message string, logType LogType) {
newLog := LogEntry{
Timestamp: time.Now().Format(time.RFC3339),
Message: message,
Type: logType,
}
a.logsMutex.Lock()
a.logs = append(a.logs, newLog)
a.logsMutex.Unlock()
// Notify all subscribers
a.subMutex.RLock()
for _, ch := range a.subscriptions {
select {
case ch <- newLog:
default:
// If channel is full, we skip this subscriber
}
}
a.subMutex.RUnlock()
// Emit an event to notify the frontend
runtime.EventsEmit(a.ctx, "newLog", newLog)
}
func init() {
emailRegexp = regexp.MustCompile(`^([a-z0-9_\.-]+)@([\da-z\.-]+)\.([a-z\.]{2,6})$`)
}