mirror of https://github.com/hrfee/jfa-go
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
93 lines
2.5 KiB
93 lines
2.5 KiB
5 years ago
|
package main
|
||
|
|
||
|
import (
|
||
|
"encoding/json"
|
||
|
"github.com/fsnotify/fsnotify"
|
||
|
"io/ioutil"
|
||
|
"os"
|
||
|
"strings"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
func (ctx *appContext) StartPWR() {
|
||
|
ctx.info.Println("Starting password reset daemon")
|
||
|
path := ctx.config.Section("password_resets").Key("watch_directory").String()
|
||
|
if _, err := os.Stat(path); os.IsNotExist(err) {
|
||
|
ctx.err.Printf("Failed to start password reset daemon: Directory \"%s\" doesn't exist", path)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
watcher, err := fsnotify.NewWatcher()
|
||
|
if err != nil {
|
||
|
ctx.err.Printf("Couldn't initialise password reset daemon")
|
||
|
return
|
||
|
}
|
||
|
defer watcher.Close()
|
||
|
|
||
|
done := make(chan bool)
|
||
|
go pwrMonitor(ctx, watcher)
|
||
|
err = watcher.Add(path)
|
||
|
if err != nil {
|
||
|
ctx.err.Printf("Failed to start password reset daemon: %s", err)
|
||
|
}
|
||
|
<-done
|
||
|
}
|
||
|
|
||
|
type Pwr struct {
|
||
|
Pin string `json:"Pin"`
|
||
|
Username string `json:"UserName"`
|
||
|
Expiry time.Time `json:"ExpirationDate"`
|
||
|
}
|
||
|
|
||
|
func pwrMonitor(ctx *appContext, watcher *fsnotify.Watcher) {
|
||
|
for {
|
||
|
select {
|
||
|
case event, ok := <-watcher.Events:
|
||
|
if !ok {
|
||
|
return
|
||
|
}
|
||
|
if event.Op&fsnotify.Write == fsnotify.Write && strings.Contains(event.Name, "passwordreset") {
|
||
|
var pwr Pwr
|
||
|
data, err := ioutil.ReadFile(event.Name)
|
||
|
if err != nil {
|
||
|
return
|
||
|
}
|
||
|
err = json.Unmarshal(data, &pwr)
|
||
|
if len(pwr.Pin) == 0 || err != nil {
|
||
|
return
|
||
|
}
|
||
|
ctx.info.Printf("New password reset for user \"%s\"", pwr.Username)
|
||
|
if ct := time.Now(); pwr.Expiry.After(ct) {
|
||
|
user, status, err := ctx.jf.userByName(pwr.Username, false)
|
||
|
if !(status == 200 || status == 204) || err != nil {
|
||
|
ctx.err.Printf("Failed to get users from Jellyfin: Code %d", status)
|
||
|
ctx.debug.Printf("Error: %s", err)
|
||
|
return
|
||
|
}
|
||
|
ctx.storage.loadEmails()
|
||
|
address, ok := ctx.storage.emails[user["Id"].(string)].(string)
|
||
|
if !ok {
|
||
|
ctx.err.Printf("Couldn't find email for user \"%s\". Make sure it's set", pwr.Username)
|
||
|
return
|
||
|
}
|
||
|
if ctx.email.constructReset(pwr, ctx) != nil {
|
||
|
ctx.err.Printf("Failed to construct password reset email for %s", pwr.Username)
|
||
|
} else if ctx.email.send(address, ctx) != nil {
|
||
|
ctx.err.Printf("Failed to send password reset email to \"%s\"", address)
|
||
|
} else {
|
||
|
ctx.info.Printf("Sent password reset email to \"%s\"", address)
|
||
|
}
|
||
|
} else {
|
||
|
ctx.err.Printf("Password reset for user \"%s\" has already expired (%s). Check your time settings.", pwr.Username, pwr.Expiry)
|
||
|
}
|
||
|
|
||
|
}
|
||
|
case err, ok := <-watcher.Errors:
|
||
|
if !ok {
|
||
|
return
|
||
|
}
|
||
|
ctx.err.Printf("Password reset daemon: %s", err)
|
||
|
}
|
||
|
}
|
||
|
}
|