kpsync/app/application.go
2025-08-19 17:21:15 +02:00

199 lines
4.8 KiB
Go

package app
import (
"fmt"
"os"
"os/signal"
"sync"
"syscall"
"time"
"fyne.io/systray"
"git.blackforestbytes.com/BlackForestBytes/goext/syncext"
"git.blackforestbytes.com/BlackForestBytes/goext/termext"
"mikescher.com/kpsync/assets"
)
type Application struct {
masterLock sync.Mutex
config Config
trayReady *syncext.AtomicBool
uploadRunning *syncext.AtomicBool
syncLoopRunning *syncext.AtomicBool
keepassRunning *syncext.AtomicBool
sigKPExitChan chan bool // keepass exited
sigManualStopChan chan bool // manual stop
sigErrChan chan error // fatal error
sigSyncLoopStopChan chan bool // stop sync loop
sigTermKeepassChan chan bool // stop keepass
dbFile string
stateFile string
}
func NewApplication() *Application {
app := &Application{
masterLock: sync.Mutex{},
uploadRunning: syncext.NewAtomicBool(false),
trayReady: syncext.NewAtomicBool(false),
syncLoopRunning: syncext.NewAtomicBool(false),
keepassRunning: syncext.NewAtomicBool(false),
sigKPExitChan: make(chan bool, 128),
sigManualStopChan: make(chan bool, 128),
sigErrChan: make(chan error, 128),
sigSyncLoopStopChan: make(chan bool, 128),
sigTermKeepassChan: make(chan bool, 128),
}
app.LogInfo("Starting kpsync...")
app.LogDebug(fmt.Sprintf("SupportsColors := %v", termext.SupportsColors()))
app.LogLine()
return app
}
func (app *Application) Run() {
var configPath string
app.config, configPath = app.loadConfig()
app.LogInfo(fmt.Sprintf("Loaded config from %s", configPath))
app.LogDebug(fmt.Sprintf("WebDAVURL := '%s'", app.config.WebDAVURL))
app.LogDebug(fmt.Sprintf("WebDAVUser := '%s'", app.config.WebDAVUser))
app.LogDebug(fmt.Sprintf("WebDAVPass := '%s'", app.config.WebDAVPass))
app.LogDebug(fmt.Sprintf("LocalFallback := '%s'", app.config.LocalFallback))
app.LogDebug(fmt.Sprintf("WorkDir := '%s'", app.config.WorkDir))
app.LogDebug(fmt.Sprintf("ForceColors := %v", app.config.ForceColors))
app.LogDebug(fmt.Sprintf("Debounce := %d", app.config.Debounce))
app.LogDebug(fmt.Sprintf("ForceColors := %v", app.config.ForceColors))
app.LogLine()
go func() { app.initTray() }()
go func() {
app.syncLoopRunning = syncext.NewAtomicBool(true)
defer app.syncLoopRunning.Set(false)
isr, err := app.initSync()
if err != nil {
app.sigErrChan <- err
return
}
if isr == InitSyncResponseAbort {
app.sigManualStopChan <- true
return
} else if isr == InitSyncResponseOkay {
go func() {
app.keepassRunning = syncext.NewAtomicBool(true)
defer app.keepassRunning.Set(false)
app.runKeepass(false)
}()
time.Sleep(1 * time.Second)
app.setTrayStateDirect("Sleeping...", assets.IconDefault)
err = app.runSyncLoop()
if err != nil {
app.sigErrChan <- err
return
}
} else if isr == InitSyncResponseFallback {
app.LogInfo(fmt.Sprintf("Starting KeepassXC with local fallback database (without sync loop!)"))
app.LogDebug(fmt.Sprintf("DB-Path := '%s'", app.config.LocalFallback))
go func() {
app.keepassRunning = syncext.NewAtomicBool(true)
defer app.keepassRunning.Set(false)
app.runKeepass(true)
}()
app.setTrayStateDirect("Sleeping...", assets.IconDefault)
} else {
app.LogError("Unknown InitSyncResponse: "+string(isr), nil)
app.sigErrChan <- fmt.Errorf("unknown InitSyncResponse: %s", isr)
return
}
}()
sigTerm := make(chan os.Signal, 1)
signal.Notify(sigTerm, os.Interrupt, syscall.SIGTERM)
select {
case <-sigTerm: // kpsync received SIGTERM
app.LogInfo("Stopping application (received SIGTERM signal)")
app.stopBackgroundRoutines()
// TODO try final sync (?)
case err := <-app.sigErrChan: // fatal error
app.LogInfo("Stopping application (received ERROR)")
app.stopBackgroundRoutines()
app.LogError("Stopped due to error: "+err.Error(), nil)
return
case <-app.sigManualStopChan: // manual
app.LogInfo("Stopping application (manual)")
app.stopBackgroundRoutines()
return
case _ = <-app.sigKPExitChan: // keepass exited
app.LogInfo("Stopping application (received STOP)")
app.stopBackgroundRoutines()
// TODO try final sync
}
}
func (app *Application) stopBackgroundRoutines() {
app.LogInfo("Stopping go-routines...")
app.LogDebug("Stopping systray...")
systray.Quit()
app.trayReady.Wait(false)
app.LogDebug("Stopped systray.")
if app.uploadRunning.Get() {
app.LogInfo("Waiting for active upload...")
app.uploadRunning.Wait(false)
app.LogInfo("Upload finished.")
}
app.LogDebug("Stopping sync-loop...")
app.sigSyncLoopStopChan <- true
app.syncLoopRunning.Wait(false)
app.LogDebug("Stopped sync-loop.")
app.LogDebug("Stopping keepass...")
app.sigTermKeepassChan <- true
app.keepassRunning.Wait(false)
app.LogDebug("Stopped keepass.")
app.LogLine()
}